File: vector-unique.c

package info (click to toggle)
util-vserver 0.30.216-pre3054-1
  • links: PTS, VCS
  • area: main
  • in suites: jessie, jessie-kfreebsd
  • size: 10,556 kB
  • ctags: 3,909
  • sloc: ansic: 23,121; sh: 16,905; xml: 1,972; makefile: 383; python: 301; perl: 85; awk: 4
file content (57 lines) | stat: -rw-r--r-- 1,673 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
// $Id$    --*- c -*--

// Copyright (C) 2004 Enrico Scholz <enrico.scholz@informatik.tu-chemnitz.de>
//  
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation; version 2 of the License.
//  
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.
//  
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.


#ifdef HAVE_CONFIG_H
#  include <config.h>
#endif

#include "vector.h"

#include <assert.h>
#include <string.h>

  // TODO: do not iterate from begin to end but in the reverse direction. This should be more
  // effective.
void
Vector_unique(struct Vector *vec, int (*compare)(const void *, const void *))
{
  size_t		idx;
  
  if (vec->count<2) return;

  for (idx=0; idx+1<vec->count; ++idx) {
    char	*ptr      = (char *)(vec->data) + idx*vec->elem_size;
    char	*next_ptr = ptr + vec->elem_size;
    size_t	next_idx  = idx + 1;

    while (next_idx<vec->count &&
	   compare(ptr, next_ptr)==0) {
      ++next_idx;
      next_ptr += vec->elem_size;
    }

    if (next_idx==vec->count)
      vec->count = idx+1;
    else if (next_idx-idx > 1) {
      memmove(ptr + vec->elem_size,
	      next_ptr, (vec->count - next_idx)*vec->elem_size);
      vec->count -= (next_idx-idx-1);
    }
  }
}