merge with 0.30.213
[util-vserver.git] / ensc_vector / vector-unique.c
1 // $Id: vector-unique.c 814 2004-02-06 14:47:18Z ensc $    --*- c -*--
2
3 // Copyright (C) 2004 Enrico Scholz <enrico.scholz@informatik.tu-chemnitz.de>
4 //  
5 // This program is free software; you can redistribute it and/or modify
6 // it under the terms of the GNU General Public License as published by
7 // the Free Software Foundation; version 2 of the License.
8 //  
9 // This program is distributed in the hope that it will be useful,
10 // but WITHOUT ANY WARRANTY; without even the implied warranty of
11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12 // GNU General Public License for more details.
13 //  
14 // You should have received a copy of the GNU General Public License
15 // along with this program; if not, write to the Free Software
16 // Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
17
18
19 #ifdef HAVE_CONFIG_H
20 #  include <config.h>
21 #endif
22
23 #include "vector.h"
24
25 #include <assert.h>
26 #include <string.h>
27
28   // TODO: do not iterate from begin to end but in the reverse direction. This should be more
29   // effective.
30 void
31 Vector_unique(struct Vector *vec, int (*compare)(const void *, const void *))
32 {
33   size_t                idx;
34   
35   if (vec->count<2) return;
36
37   for (idx=0; idx+1<vec->count; ++idx) {
38     char        *ptr      = (char *)(vec->data) + idx*vec->elem_size;
39     char        *next_ptr = ptr + vec->elem_size;
40     size_t      next_idx  = idx + 1;
41
42     while (next_idx<vec->count &&
43            compare(ptr, next_ptr)==0) {
44       ++next_idx;
45       next_ptr += vec->elem_size;
46     }
47
48     if (next_idx==vec->count)
49       vec->count = idx+1;
50     else if (next_idx-idx > 1) {
51       memmove(ptr + vec->elem_size,
52               next_ptr, (vec->count - next_idx)*vec->elem_size);
53       vec->count -= (next_idx-idx-1);
54     }
55   }
56 }
57