SoPlex Doxygen Documentation
svset.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 <assert.h>
17 
18 #include "spxdefines.h"
19 #include "svset.h"
20 
21 namespace soplex
22 {
23 
24 void SVSet::ensureMem(int n)
25 {
26  if (memSize() + n > memMax())
27  {
28  int newMax = int( memFactor * memMax() );
29  if ( memSize() + n > newMax )
30  newMax = memSize() + n;
31  memRemax( newMax );
32  }
33 }
34 
35 void SVSet::add(const SVector svec[], int n)
36 {
37  assert(n >= 0);
38 
39  int i, len;
40  for (i = len = 0; i < n; ++i)
41  len += svec[i].size();
42 
43  ensurePSVec(n);
44  ensureMem(len + n);
45 
46  for (i = 0; i < n; ++i)
47  *create(svec[i].size()) = svec[i];
48 }
49 
50 void SVSet::add(DataKey nkey[], const SVector svec[], int n)
51 {
52  add(svec, n);
53  for (int i = num() - 1; --n; --i)
54  nkey[n] = key(i);
55 }
56 
57 void SVSet::add(const SVSet& pset)
58 {
59  int i, n, len;
60  n = pset.num();
61  for (i = len = 0; i < n; ++i)
62  len += pset[i].size();
63 
64  ensurePSVec(n);
65  ensureMem(len + n);
66 
67  for (i = 0; i < n; ++i)
68  *create(pset[i].size()) = pset[i];
69 }
70 
71 void SVSet::add(DataKey nkey[], const SVSet& pset)
72 {
73  add(pset);
74 
75  int i = size();
76  int n = pset.size();
77 
78  while(n > 0)
79  nkey[--n] = key(--i);
80 }
81 
82 SVector* SVSet::create(int idxmax)
83 {
84  DLPSV* ps;
85 
86  if (list.last())
87  {
88  ps = list.last();
89  removeLast(ps->max() - ps->size());
90  ps->set_max( ps->size() );
91  }
92 
93  if (idxmax < 0)
94  {
95  ensureMem(2);
96  idxmax = memMax() - memSize() - 1;
97  }
98  else
99  ensureMem(idxmax + 1);
100 
101  ensurePSVec(1);
102 
103  assert( memMax() >= memSize() + idxmax + 1 );
104 
105  ps = set.create();
106  list.append(ps);
107  /// resize the dataarray
108  SVSetBase::reSize(memSize() + idxmax + 1);
109 
110  ps->setMem(idxmax + 1, &last() - idxmax);
111  return ps;
112 }
113 
114 SVector* SVSet::create(DataKey& nkey, int idxmax)
115 {
116  SVector* ps = create(idxmax);
117  nkey = key(num() - 1);
118  return ps;
119 }
120 
121 void SVSet::xtend(SVector& svec, int newmax)
122 {
123  if (svec.max() < newmax)
124  {
126  memPack();
127 
128  assert(has(&svec));
129  DLPSV* ps = static_cast<DLPSV*>( & svec );
130 
131  if (ps == list.last())
132  {
133  int sz = ps->size();
134  ensureMem (newmax - ps->max() + 1);
135  insert(memSize(), newmax - ps->max());
136  ps->setMem (newmax + 1, ps->mem());
137  ps->set_size( sz );
138  }
139  else
140  {
141  ensureMem(newmax + 1);
142  SVector newps(newmax + 1, &last() + 1);
143  int sz = ps->size();
144  insert(memSize(), newmax + 1);
145  newps = svec;
146 
147  if (ps != list.first())
148  {
149  SVector* prev = ps->prev();
150  int prevsz = prev->size();
151  prev->setMem (prev->max()
152  + ps->max() + 2, prev->mem());
153  prev->set_size(prevsz);
154 
155  possiblyUnusedMem += ps->max();
156  }
157  list.remove(ps);
158  list.append(ps);
159 
160  ps->setMem(newmax + 1, newps.mem());
161  ps->set_size(sz);
162  }
163  }
164 }
165 
166 void SVSet::add2(SVector &svec, int idx, Real val)
167 {
168  xtend(svec, svec.size() + 1);
169  svec.add(idx, val);
170 }
171 
172 void SVSet::add2(SVector &svec, int n, const int idx[], const Real val[])
173 {
174  xtend(svec, svec.size() + n);
175  svec.add(n, idx, val);
176 }
177 
179 {
180  /* delete last entries, in an SVECTOR and also in an DLPSV there is always the position -1 used for memorizing
181  * the size; this is the reason why we need to delete max()+1 entries
182  */
183  if (list.last() == ps)
184  removeLast(ps->max() + 1);
185  /* merge space of predecessor with position which will be deleted, therefore we do not need to delete any
186  * memory or do an expensive memmove
187  *
188  * @note an SVECTOR and also in an DLPSV memorize the size always on position -1; this is the reason why we
189  * need to set the new mem size to the old combined size + 2
190  */
191  else if (list.first() != ps)
192  {
193  SVector* prev = ps->prev();
194  int sz = prev->size();
195  prev->setMem(prev->max() + ps->max() + 2, prev->mem());
196  prev->set_size(sz);
197  }
198  /* delete the front entries of the first list entry and correct the memory pointers in the vectors */
199  /* @note we do this by merging the first both vectors, move the entries from the second vector up front, and
200  * correct the size
201  */
202  else
203  {
204  SVector* next = ps->next();
205  int sz = next->size();
206  int bothmax = next->max() + ps->max();
207  int offset = 0;
208 
209  /* the first element does not need to start at the beginning of the data array; why ??? */
210  while( &(this->SVSetBase::operator[](offset)) != ps->mem() )
211  {
212  ++offset;
213  assert(offset < size());
214  }
215 
216  /* move all entries of the second vector to the front */
217  for(int j = 0; j <= sz; ++j)
218  {
219  this->SVSetBase::operator[](offset + j) = next->mem()[j];
220  }
221 
222  /* correct the data memmory pointer and the maximal space */
223  next->setMem(bothmax + 2, ps->mem());
224  /* correct size */
225  next->set_size(sz);
226  }
227 
228  list.remove(ps);
229 }
230 
231 void SVSet::remove(const DataKey& removekey)
232 {
233  deleteVec(&set[removekey]);
234  set.remove(removekey);
235 }
236 
237 void SVSet::remove(int perm[])
238 {
239  int j = num();
240 
241  /* due to performance reasons we use a backwards loop to delete entries, because it could result instead of only
242  * decreasing the number of elements j times in memmoving the whole array j times
243  */
244  for (int i = j - 1; i >= 0; --i)
245  {
246  if (perm[i] < 0)
247  {
248  deleteVec(&set[i]);
249  }
250  }
251  set.remove(perm);
252 }
253 
254 void SVSet::remove(const DataKey keys[], int n, int* perm)
255 {
256  for (int i = num() - 1; i >= 0; --i)
257  perm[i] = i;
258  while (n--)
259  perm[ number(*keys++) ] = -1;
260  remove(perm);
261 }
262 
263 void SVSet::remove(const int nums[], int n, int* perm)
264 {
265  for (int i = num() - 1; i >= 0; --i)
266  perm[i] = i;
267  while (n--)
268  perm[ *nums++ ] = -1;
269  remove(perm);
270 }
271 
272 
273 /* \SubSection{Memory Management}
274  */
275 void SVSet::reMax(int newmax)
276 {
277  list.move(set.reMax(newmax));
278 }
279 
280 void SVSet::memRemax(int newmax)
281 {
282  ptrdiff_t delta = DataArray < SVector::Element > ::reMax(newmax);
283 
284  if (delta != 0)
285  {
286  for (DLPSV* ps = list.first(); ps; ps = list.next(ps))
287  {
288  // Extract the size and maximum capacity of the SVector, which for some reason is stored
289  // as the first element.
290  SVector::Element * info = reinterpret_cast<SVector::Element*>(reinterpret_cast<char*>(ps->mem()) + delta);
291  int sz = info->idx;
292  int l_max = int( info->val );
293  assert(l_max >= sz );
294  ps->setMem (l_max + 1, info);
295  ps->set_max (l_max);
296  ps->set_size(sz);
297  }
298  }
299 }
300 
301 /** garbage collection in nonzero memory.
302  *
303  * Pack the svectors together as tightly as possible. This removes all additional unused memory,
304  * i.e., size = max for every svector after the call.
305  *
306  * Note: do *not* call isConsistent() here, because the following might happen: In
307  * SPxLP::doAddRows(const LPRowSet& p_set), when adding rows, the sizes of the vectors for the
308  * columns of the LP are increased (without yet filling in the data) to recieve the additional
309  * entries. This is done by calling xtend() above. xtend() in turn might call this method, which
310  * checks the yet unfilled positions, i.e., isConsistent() is likely to fail. In general,
311  * isConsistent() should not be called within this class, but in classes further up in the
312  * hierarchy.
313  */
315 {
316  int used;
317  int j;
318  DLPSV* ps;
319  for (used = 0, ps = list.first(); ps; ps = list.next(ps))
320  {
321  const int sz = ps->size();
322 
323  if (ps->mem() != &this->SVSetBase::operator[](used))
324  {
325  // cannot use memcpy, because the memory might overlap
326  for (j = 0; j <= sz; ++j)
327  this->SVSetBase::operator[](used + j) = ps->mem()[j];
328  ps->setMem(sz + 1, &this->SVSetBase::operator[](used));
329  ps->set_size(sz);
330  }
331  else
332  ps->set_max(sz);
333  used += sz + 1;
334  }
335  SVSetBase::reSize(used);
336 
337  possiblyUnusedMem = 0;
338 }
339 
341 {
342 #ifdef ENABLE_CONSISTENCY_CHECKS
343  DLPSV* ps;
344  DLPSV* next;
345  for (ps = list.first(); ps; ps = next)
346  {
347  if (!ps->isConsistent())
348  return MSGinconsistent("SVSet");
349  if (ps->mem() > &last())
350  return MSGinconsistent("SVSet");
351  next = list.next(ps);
352  if (next && ps->mem() + ps->max() + 1 != next->mem()) {
353  return MSGinconsistent("SVSet");
354  }
355  }
357  && set.isConsistent() && list.isConsistent();
358 #else
359  return true;
360 #endif
361 }
362 
364 {
365  if (this != &rhs)
366  {
367  clear();
368 
369  if (rhs.size() > 0)
370  {
371  DataArray < SVector::Element > ::operator=(rhs);
372  set = rhs.set;
373 
374  DLPSV* ps;
375  DLPSV* newps;
376 
377  void* delta0 = &(*(static_cast<SVSetBase*>(this)))[0];
378  void* delta1 = &(*(static_cast<SVSetBase*>(
379  const_cast<SVSet*>(&rhs))))[0];
380  ptrdiff_t delta = reinterpret_cast<char*>(
381  delta0) - reinterpret_cast<char*>(delta1);
382 
383  for (ps = rhs.list.first(); ps; ps = rhs.list.next(ps))
384  {
385  newps = & set[ rhs.number(ps) ];
386  list.append(newps);
387  newps->setMem(ps->max() + 1,
388  reinterpret_cast<SVector::Element*>(
389  reinterpret_cast<char*>(ps->mem()) + delta));
390  newps->set_size( ps->size() );
391  }
392  }
393  }
394  assert(isConsistent());
395 
396  return *this;
397 }
398 
399 SVSet::SVSet(const SVSet& old)
400  : DataArray < SVector::Element > ()
401  , possiblyUnusedMem (old.possiblyUnusedMem)
402  , factor (old.factor)
403 {
404  *this = old;
405  assert(SVSet::isConsistent());
406 }
407 } // namespace soplex
408 
409 //-----------------------------------------------------------------------------
410 //Emacs Local Variables:
411 //Emacs mode:c++
412 //Emacs c-basic-offset:3
413 //Emacs tab-width:8
414 //Emacs indent-tabs-mode:nil
415 //Emacs End:
416 //-----------------------------------------------------------------------------