SoPlex Doxygen Documentation
nameset.cpp
Go to the documentation of this file.
1 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2 /* */
3 /* This file is part of the class library */
4 /* SoPlex --- the Sequential object-oriented simPlex. */
5 /* */
6 /* Copyright (C) 1996-2012 Konrad-Zuse-Zentrum */
7 /* fuer Informationstechnik Berlin */
8 /* */
9 /* SoPlex is distributed under the terms of the ZIB Academic Licence. */
10 /* */
11 /* You should have received a copy of the ZIB Academic License */
12 /* along with SoPlex; see the file COPYING. If not email to soplex@zib.de. */
13 /* */
14 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
15 
16 #include <string.h>
17 #include "spxdefines.h"
18 #include "nameset.h"
19 #include "spxalloc.h"
20 
21 namespace soplex
22 {
23 const char NameSet::Name::deflt = '\0';
24 
25 void NameSet::add(const char* str)
26 {
27  DataKey k;
28  add(k, str);
29 }
30 
31 void NameSet::add(DataKey& p_key, const char* str)
32 {
33  const Name nstr (str);
34 
35  if (!hashtab.has(nstr))
36  {
37  if (size() + 1 > max())
38  {
39  assert(factor >= 1);
40  reMax(int(factor*max() + 8));
41  }
42 
43  if (memSize() + int(strlen(str)) >= memMax())
44  {
45  memPack();
46  if (memSize() + int(strlen(str)) >= memMax())
47  {
48  assert(memFactor >= 1);
49  memRemax(int(memFactor*memMax()) + 9 + int(strlen(str)));
50  assert(memSize() + int(strlen(str)) < memMax());
51  }
52  }
53  int idx = memused;
54  char* tmp = &(mem[idx]);
55  memused += int(strlen(str)) + 1;
56 
57  strcpy(tmp, str);
58  *(set.create(p_key)) = idx;
59  Name memname(tmp);
60  hashtab.add(memname, p_key);
61  }
62 }
63 
64 void NameSet::add(const NameSet& p_set)
65 {
66  for (int i = 0; i < p_set.num(); ++i)
67  {
68  Name iname(p_set[i]);
69  if (!hashtab.has(iname))
70  add(p_set[i]);
71  }
72 }
73 
74 void NameSet::add(DataKey p_key[], const NameSet& p_set)
75 {
76  for (int i = 0; i < p_set.num(); ++i)
77  {
78  Name iname = Name(p_set[i]);
79  if (!hashtab.has(iname))
80  add(p_key[i], p_set[i]);
81  }
82 }
83 
84 void NameSet::remove(const char *str)
85 {
86  const Name nam(str);
87  if (hashtab.has(nam))
88  {
89  const DataKey* hkey = hashtab.get(nam);
90  assert(hkey != 0);
91  hashtab.remove(nam);
92  set.remove(*hkey);
93  }
94 }
95 
96 void NameSet::remove(const DataKey& p_key)
97 {
98  assert(has(p_key));
99 
100  hashtab.remove(Name(&mem[set[p_key]]));
101  set.remove(p_key);
102 }
103 
104 void NameSet::remove(const DataKey keys[], int n)
105 {
106  for (int i = 0; i < n; ++i)
107  remove(keys[i]);
108 }
109 
110 void NameSet::remove(const int nums[], int n)
111 {
112  for (int i = 0; i < n; ++i)
113  remove(nums[i]);
114 }
115 
116 void NameSet::remove(int dstat[])
117 {
118  for(int i = 0; i < set.num(); i++)
119  {
120  if (dstat[i] < 0)
121  {
122  const Name nam = &mem[set[i]];
123  hashtab.remove(nam);
124  }
125  }
126  set.remove(dstat);
127 
128  assert(isConsistent());
129 }
130 
132 {
133  set.clear();
134  hashtab.clear();
135  memused = 0;
136 }
137 
138 void NameSet::reMax(int newmax)
139 {
140  hashtab.reMax(newmax);
141  set.reMax(newmax);
142 }
143 
144 void NameSet::memRemax(int newmax)
145 {
146  memmax = (newmax < memSize()) ? memSize() : newmax;
148 
149  hashtab.clear ();
150 
151  for (int i = num() - 1; i >= 0; --i)
152  hashtab.add(Name(&mem[set[key(i)]]), key(i));
153 }
154 
156 {
157  char* newmem = 0;
158  int newlast = 0;
159  int i;
160 
161  hashtab.clear();
162 
163  spx_alloc(newmem, memSize());
164 
165  for(i = 0; i < num(); i++)
166  {
167  const char* t = &mem[set[i]];
168  strcpy(&newmem[newlast], t);
169  set[i] = newlast;
170  newlast += int(strlen(t)) + 1;
171  }
172  memcpy(mem, newmem, static_cast<size_t>(newlast));
173  memused = newlast;
174 
175  assert(memSize() <= memMax());
176 
177  spx_free(newmem);
178 
179  for (i = 0; i < num(); i++)
180  hashtab.add(Name(&mem[set[key(i)]]), key(i));
181 }
182 
183 /// returns the hash value of the name.
185 {
186  unsigned int res = 0;
187  const char* sptr = str->name;
188 
189  while(*sptr != '\0')
190  {
191  res *= 65;
192  res += *sptr++ - int('0');
193  res %= 0x0fffffff;
194  }
195  return res;
196 }
197 
198 #if 0
199 NameSet& NameSet::operator=(const NameSet& rhs)
200 {
201  if (this != &rhs)
202  {
203  if (max() < rhs.size())
204  reMax(rhs.size());
205  if (memMax() < rhs.memSize())
206  memRemax(rhs.memSize());
207 
208  set = rhs.set;
209 
210  hashtab.clear();
211  for (int i = 0; i < set.num(); ++i)
212  {
213  Name iname(set[i].name);
214  DataKey ikey = DataKey(set.key(i));
215  hashtab.add(iname, ikey);
216  }
217  memPack();
218  }
219  return *this;
220 }
221 
222 NameSet::NameSet(const NameSet& org)
223  : set(org.set)
224  , mem(0)
225  , hashtab(org.hashtab)
226  , factor(org.factor)
227  , memFactor(org.memFactor)
228 {
229  memused = 0;
230  memmax = org.memSize();
231  spx_alloc(mem, memmax);
232 
233  list.clear();
234  hashtab.clear();
235  for (int i = 0; i < set.num(); ++i)
236  {
237  list.append(&(set[i]));
238  Name iname = set[i];
239  DataKey k = DataKey(set.key(i));
240  hashtab.add(iname, k);
241  }
242  memPack();
243 }
244 #endif
245 
246 NameSet::NameSet(int p_max, int mmax, Real fac, Real memFac)
247  : set(p_max)
248  , mem(0)
249  , hashtab(NameSetNameHashFunction, set.max(), 0, fac)
250  , factor(fac)
251  , memFactor(memFac)
252 {
253  memused = 0;
254  memmax = (mmax < 1) ? (8 * set.max() + 1) : mmax;
255  spx_alloc(mem, memmax);
256 }
257 
259 {
260  spx_free(mem);
261 }
262 
264 {
265 #ifdef ENABLE_CONSISTENCY_CHECKS
266  if (memused > memmax)
267  return MSGinconsistent("NameSet");
268 
269  int i;
270 
271  for(i = 0; i < num(); i++)
272  {
273  const char* t = &mem[set[i]];
274 
275  if (!has(t))
276  return MSGinconsistent("NameSet");
277 
278  if (strcmp(t, operator[](key(t))))
279  return MSGinconsistent("NameSet");
280  }
281  return set.isConsistent() && hashtab.isConsistent();
282 #else
283  return true;
284 #endif
285 }
286 
287 std::ostream& operator<<(std::ostream& s, const NameSet& nset)
288 {
289  for(int i = 0; i < nset.num(); i++)
290  {
291  s << i << " "
292  << nset.key(i).info << "."
293  << nset.key(i).idx << "= "
294  << nset[i]
295  << std::endl;
296  }
297  return s;
298 }
299 
300 
301 } // namespace soplex
302 
303 //-----------------------------------------------------------------------------
304 //Emacs Local Variables:
305 //Emacs mode:c++
306 //Emacs c-basic-offset:3
307 //Emacs tab-width:8
308 //Emacs indent-tabs-mode:nil
309 //Emacs End:
310 //-----------------------------------------------------------------------------