summaryrefslogtreecommitdiff
path: root/src/traffic-incidents-service/org.genivi.trafficinfo.dbus-java/src/main/java/org/freedesktop/dbus/EfficientMap.java
blob: 67a248486aa3e001b2cb14b82c28afe457f58f2d (plain)
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
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
/*
   D-Bus Java Implementation
   Copyright (c) 2005-2006 Matthew Johnson

   This program is free software; you can redistribute it and/or modify it
   under the terms of either the GNU Lesser General Public License Version 2 or the
   Academic Free Licence Version 2.1.

   Full licence texts are included in the COPYING file with this program.
*/
package org.freedesktop.dbus;

/**
 * Provides a long => MethodCall map which doesn't allocate objects
 * on insertion/removal. Keys must be inserted in ascending order. */
class EfficientMap
{
   private long[] kv;
   private MethodCall[] vv;
   private int start;
   private int end;
   private int init_size;
   public EfficientMap(int initial_size)
   {
      init_size = initial_size;
      shrink();
   }
   private void grow()
   {
      // create new vectors twice as long
      long[] oldkv = kv;
      kv = new long[oldkv.length*2];
      MethodCall[] oldvv = vv;
      vv = new MethodCall[oldvv.length*2];

      // copy start->length to the start of the new vector
      System.arraycopy(oldkv,start,kv,0,oldkv.length-start);
      System.arraycopy(oldvv,start,vv,0,oldvv.length-start);
      // copy 0->end to the next part of the new vector
      if (end != (oldkv.length-1)) {
         System.arraycopy(oldkv,0,kv,oldkv.length-start,end+1);
         System.arraycopy(oldvv,0,vv,oldvv.length-start,end+1);
      }
      // reposition pointers
      start = 0;
      end = oldkv.length;
   }
   // create a new vector with just the valid keys in and return it
   public long[] getKeys()
   {
      int size;
      if (start < end) size = end-start;
      else size = kv.length-(start-end);
      long[] lv = new long[size];
      int copya;
      if (size > kv.length-start) copya = kv.length-start;
      else copya = size;
      System.arraycopy(kv,start,lv,0,copya);
      if (copya < size) {
         System.arraycopy(kv,0,lv,copya,size-copya);
      }
      return lv;
   }
   private void shrink()
   {
      if (null != kv && kv.length == init_size) return;
      // reset to original size
      kv = new long[init_size];
      vv = new MethodCall[init_size];
      start = 0;
      end = 0;
   }
   public void put(long l, MethodCall m)
   {
      // put this at the end
      kv[end] = l;
      vv[end] = m;
      // move the end
      if (end == (kv.length-1)) end = 0; else end++;
      // if we are out of space, grow.
      if (end == start) grow();
   }
   public MethodCall remove(long l)
   {
      // find the item
      int pos = find(l);
      // if we don't have it return null
      if (-1 == pos) return null;
      // get the value
      MethodCall m = vv[pos];
      // set it as unused
      vv[pos] = null;
      kv[pos] = -1;
      // move the pointer to the first full element
      while (-1 == kv[start]) {
         if (start == (kv.length-1)) start = 0; else start++;
         // if we have emptied the list, shrink it
         if (start == end) { shrink(); break; }
      }
      return m;
   }
   public boolean contains(long l)
   {
      // check if find succeeds
      return -1 != find(l);
   }   
   /* could binary search, but it's probably the first one */
   private int find(long l)
   {
      int i = start;
      while (i != end && kv[i] != l)
         if (i == (kv.length-1)) i = 0; else i++;
      if (i == end) return -1;
      return i;
   }
}