SoPlex Doxygen Documentation
spxlpfread.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 /**@file spxlpfread.cpp
17  * @brief Read LP format files.
18  */
19 // #define DEBUGGING 1
20 
21 #include <assert.h>
22 #include <stdio.h>
23 #include <ctype.h>
24 #include <iostream>
25 
26 #include "spxdefines.h"
27 #include "spxlp.h"
28 #include "spxout.h"
29 #include "exceptions.h"
30 
31 /* The manual says the maximum allowed line length is 255 characters,
32  * but CPLEX does not complain if the lines are longer.
33  */
34 #define MAX_LINE_LEN 8192 ///< maximum length of a line (8190 + \\n + \\0)
35 
36 namespace soplex
37 {
38 /// Is \p c a \c space, \c tab, \c nl or \c cr ?
39 static inline bool isSpace(int c)
40 {
41  return (c == ' ') || (c == '\t') || (c == '\n') || (c == '\r');
42 }
43 
44 /// Is there a number at the beginning of \p s ?
45 static bool isValue(const char* s)
46 {
47  return ((*s >= '0') && (*s <= '9'))
48  || (*s == '+') || (*s == '-') || (*s == '.');
49 }
50 
51 /// Is there a possible column name at the beginning of \p s ?
52 static bool isColName(const char* s)
53 {
54  // strchr() gives a true for the null char.
55  if (*s == '\0')
56  return false;
57 
58  return ((*s >= 'A') && (*s <= 'Z'))
59  || ((*s >= 'a') && (*s <= 'z'))
60  || (strchr("!\"#$%&()/,;?@_'`{}|~", *s) != 0);
61 }
62 
63 /// Is there a comparison operator at the beginning of \p s ?
64 static bool isSense(const char* s)
65 {
66  return (*s == '<') || (*s == '>') || (*s == '=');
67 }
68 
69 static bool isInfinity(const char* s)
70 {
71  return ((s[0] == '-') || (s[0] == '+'))
72  && (tolower(s[1]) == 'i')
73  && (tolower(s[2]) == 'n')
74  && (tolower(s[3]) == 'f');
75 }
76 
77 static bool isFree(const char* s)
78 {
79  return (tolower(s[0]) == 'f')
80  && ( tolower(s[1]) == 'r')
81  && ( tolower(s[2]) == 'e')
82  && ( tolower(s[3]) == 'e');
83 }
84 
85 /// Read the next number and advance \p pos.
86 /** If only a sign is encountered, the number is assumed to
87  * be \c sign * 1.0.
88  * This routine will not catch malformatted numbers like .e10 !
89  */
90 static Real readValue(char*& pos)
91 {
92  assert(isValue(pos));
93 
94  char tmp[MAX_LINE_LEN];
95  const char* s = pos;
96  char* t;
97  Real value = 1.0;
98  bool has_digits = false;
99  bool has_emptyexponent = false;
100 
101  // 1. sign
102  if ((*s == '+') || (*s == '-'))
103  s++;
104 
105  // 2. Digits before the decimal dot
106  while((*s >= '0') && (*s <= '9'))
107  {
108  has_digits = true;
109  s++;
110  }
111  // 3. Decimal dot
112  if (*s == '.')
113  {
114  s++;
115 
116  // 4. If there was a dot, possible digit behind it
117  while((*s >= '0') && (*s <= '9'))
118  {
119  has_digits = true;
120  s++;
121  }
122  }
123  // 5. Exponent
124  if (tolower(*s) == 'e')
125  {
126  has_emptyexponent = true;
127  s++;
128 
129  // 6. Exponent sign
130  if ((*s == '+') || (*s == '-'))
131  s++;
132 
133  // 7. Exponent digits
134  while((*s >= '0') && (*s <= '9'))
135  {
136  has_emptyexponent = false;
137  s++;
138  }
139  }
140  assert(s != pos);
141 
142  if( has_emptyexponent )
143  {
144  MSG_WARNING( spxout << "WLPFRD01 Warning: found empty exponent in LP file - check for forbidden variable names with initial 'e' or 'E'\n"; )
145  }
146 
147  if (!has_digits)
148  value = (*pos == '-') ? -1.0 : 1.0;
149  else
150  {
151  for(t = tmp; pos != s; pos++)
152  *t++ = *pos;
153  *t = '\0';
154  value = atof(tmp);
155  }
156  pos += s - pos;
157 
158  assert(pos == s);
159 
160  MSG_DEBUG( spxout << "DLPFRD01 readValue = " << value << std::endl; )
161 
162  if (isSpace(*pos))
163  pos++;
164 
165  return value;
166 }
167 
168 /// Read the next column name from the input.
169 /** The name read is looked up and if not found \p emptycol
170  * is added to \p colset. \p pos is advanced behind the name.
171  * @return The Index of the named column.
172  */
173 static int readColName(
174  char*& pos, NameSet* colnames, LPColSet& colset, const LPCol* emptycol)
175 {
176  assert(isColName(pos));
177  assert(colnames != 0);
178 
179  char name[MAX_LINE_LEN];
180  const char* s = pos;
181  int i;
182  int colidx;
183 
184  // This are the characters that are not allowed in a column name.
185  while((strchr("+-.<>= ", *s) == 0) && (*s != '\0'))
186  s++;
187 
188  for(i = 0; pos != s; i++, pos++)
189  name[i] = *pos;
190 
191  name[i] = '\0';
192 
193  if ((colidx = colnames->number(name)) < 0)
194  {
195  // We only add the name if we got an empty column.
196  if (emptycol == 0)
197  MSG_WARNING( spxout << "WLPFRD02 Unknown variable \"" << name << "\" "; )
198  else
199  {
200  colidx = colnames->num();
201  colnames->add(name);
202  colset.add(*emptycol);
203  }
204  }
205  MSG_DEBUG( spxout << "DLPFRD03 readColName [" << name << "] = "
206  << colidx << std::endl; )
207 
208  if (isSpace(*pos))
209  pos++;
210 
211  return colidx;
212 }
213 
214 /// Read the next <,>,=,==,<=,=<,>=,=> and advance \p pos.
215 static int readSense(char*& pos)
216 {
217  assert(isSense(pos));
218 
219  int sense = *pos++;
220 
221  if ((*pos == '<') || (*pos == '>'))
222  sense = *pos++;
223  else if (*pos == '=')
224  pos++;
225 
226  MSG_DEBUG( spxout << "DLPFRD04 readSense = " << static_cast<char>(sense)
227  << std::endl; )
228 
229  if (isSpace(*pos))
230  pos++;
231 
232  return sense;
233 }
234 
235 /// Is the \p keyword present in \p buf ? If yes, advance \p pos.
236 /** \p keyword should be lower case. It can contain optional sections
237  * which are enclosed in '[' ']' like "min[imize]".
238  */
239 static bool hasKeyword(char*& pos, const char* keyword)
240 {
241  int i;
242  int k;
243 
244  assert(keyword != 0);
245 
246  for(i = 0, k = 0; keyword[i] != '\0'; i++, k++)
247  {
248  if (keyword[i] == '[')
249  {
250  i++;
251 
252  // Here we assumed that we have a ']' for the '['.
253  while((tolower(pos[k]) == keyword[i]) && (pos[k] != '\0'))
254  {
255  k++;
256  i++;
257  }
258  while(keyword[i] != ']')
259  i++;
260  --k;
261  }
262  else
263  {
264  if (keyword[i] != tolower(pos[k]))
265  break;
266  }
267  }
268  // we have to be at the end of the keyword and the word
269  // found on the line also has to end here.
270  // Attention: The isSense is a kludge to allow hasKeyword also to process
271  // Inf[inity] keywords in the bounds section.
272  if (keyword[i] == '\0' && (pos[k] == '\0' || isSpace(pos[k]) || isSense(&pos[k])))
273  {
274  pos += k;
275 
276  MSG_DEBUG( spxout << "DLPFRD05 hasKeyword: " << keyword << std::endl; )
277  return true;
278  }
279  return false;
280 }
281 
282 /// If \p buf start with "name:" store the name in \p rownames
283 /// and advance \p pos.
284 static bool hasRowName(char*& pos, NameSet* rownames)
285 {
286  const char* s = strchr(pos, ':');
287 
288  if (s == 0)
289  return false;
290 
291  int dcolpos = int(s - pos);
292 
293  int end, srt;
294 
295  for(end = dcolpos-1; end >= 0; end--) // skip spaces between name and ":"
296  if( pos[end] != ' ')
297  break;
298 
299  if( end < 0 ) // are there only spaces in front of the ":" ?
300  {
301  pos = &(pos[dcolpos+1]);
302  return false;
303  }
304 
305  for(srt = end-1; srt >= 0; srt--) // skip spaces in front of name
306  if (pos[srt] == ' ')
307  break;
308 
309  srt++; // go back to the non-space character
310 
311  assert( srt <= end && pos[srt] != ' ' );
312 
313  char name[MAX_LINE_LEN];
314  int i;
315  int k = 0;
316 
317  for(i = srt; i <= end; i++)
318  name[k++] = pos[i];
319 
320  name[k] = '\0';
321 
322  if (rownames != 0)
323  rownames->add(name);
324 
325  pos = &(pos[dcolpos+1]);
326 
327  return true;
328 }
329 
330 static Real readInfinity(char*& pos)
331 {
332  assert(isInfinity(pos));
333 
334  Real sense = (*pos == '-') ? -1.0 : 1.0;
335 
336  (void) hasKeyword(++pos, "inf[inity]");
337 
338  return sense * infinity;
339 }
340 
341 /// Read LP in "CPLEX LP File Format".
342 /**
343  * The specification is taken from the
344  * ILOG CPLEX 7.0 Reference Manual, Appendix E, Page 527.
345  *
346  * This routine should read (most?) valid LP format files.
347  * What it will not do, is find all cases where a file is ill formed.
348  * If this happens it may complain and read nothing or read "something".
349  *
350  * Problem: A line ending in '+' or '-' followed by a line starting
351  * with a number, will be regarded as an error.
352  *
353  * The reader will accept the keyword INT[egers] as a synonym for
354  * GEN[erals] which is an undocumented feature in CPLEX.
355  *
356  * A difference to the CPLEX reader, is that no name for the objective
357  * row is required.
358  *
359  * @return true if the file was read correctly
360  */
362  std::istream& p_input, ///< input stream.
363  NameSet* p_rnames, ///< row names.
364  NameSet* p_cnames, ///< column names.
365  DIdxSet* p_intvars) ///< integer variables.
366 {
367  enum
368  {
369  START, OBJECTIVE, CONSTRAINTS, BOUNDS, INTEGERS, BINARIES
370  } section = START;
371 
372  NameSet* rnames; ///< row names.
373  NameSet* cnames; ///< column names.
374 
375  LPColSet cset; ///< the set of columns read.
376  LPRowSet rset; ///< the set of rows read.
377  LPCol emptycol; ///< reusable empty column.
378  LPRow row; ///< last assembled row.
379  DSVector vec; ///< last assembled vector (from row).
380  Real val = 1.0;
381  int colidx;
382  int sense = 0;
383 
384  char buf [MAX_LINE_LEN];
385  char tmp [MAX_LINE_LEN];
386  char line[MAX_LINE_LEN];
387  int lineno = 0;
388  bool unnamed = true;
389  bool finished = false;
390  bool other;
391  bool have_value = true;
392  int i;
393  int k;
394  char* s;
395  char* pos;
396  char* pos_old = 0;
397 
398  cnames = (p_cnames != 0)
399  ? p_cnames : new NameSet();
400 
401  cnames->clear();
402 
403  try
404  {
405  rnames = (p_rnames != 0)
406  ? p_rnames : new NameSet();
407  }catch(std::bad_alloc& x)
408  {
409  if(p_cnames == 0)
410  delete cnames;
411  throw x;
412  }
413  rnames->clear();
414 
415  SPxLP::clear(); // clear the LP.
416 
417  //--------------------------------------------------------------------------
418  //--- Main Loop
419  //--------------------------------------------------------------------------
420  for(;;)
421  {
422  // 0. Read a line from the file.
423  if (!p_input.getline(buf, sizeof(buf)))
424  {
425  if (strlen(buf) == MAX_LINE_LEN - 1)
426  {
427  MSG_ERROR( spxout << "ELPFRD06 Line exceeds " << MAX_LINE_LEN - 2
428  << " characters" << std::endl; )
429  }
430  else
431  {
432  MSG_ERROR( spxout << "ELPFRD07 No 'End' marker found" << std::endl; )
433  finished = true;
434  }
435  break;
436  }
437  lineno++;
438  i = 0;
439  pos = buf;
440 
441  MSG_DEBUG( spxout << "DLPFRD08 Reading line " << lineno
442  << " (pos=" << pos << ")" << std::endl; )
443 
444  // 1. Remove comments.
445  if (0 != (s = strchr(buf, '\\')))
446  *s = '\0';
447 
448  // 2. Look for keywords.
449  if (section == START)
450  {
451  if (hasKeyword(pos, "max[imize]"))
452  {
454  section = OBJECTIVE;
455  }
456  else if (hasKeyword(pos, "min[imize]"))
457  {
459  section = OBJECTIVE;
460  }
461  }
462  else if (section == OBJECTIVE)
463  {
464  if (hasKeyword(pos, "s[ubject][ ]t[o]")
465  || hasKeyword(pos, "s[uch][ ]t[hat]")
466  || hasKeyword(pos, "s[.][ ]t[.]")
467  || hasKeyword(pos, "lazy con[straints]"))
468  {
469  // store objective vector
470  for(int j = vec.size() - 1; j >= 0; --j)
471  cset.maxObj_w(vec.index(j)) = vec.value(j);
472  // multiplication with -1 for minimization is done below
473  vec.clear();
474  have_value = true;
475  val = 1.0;
476  section = CONSTRAINTS;
477  }
478  }
479  else if (section == CONSTRAINTS &&
480  (hasKeyword(pos, "s[ubject][ ]t[o]")
481  || hasKeyword(pos, "s[uch][ ]t[hat]")
482  || hasKeyword(pos, "s[.][ ]t[.]")))
483  {
484  have_value = true;
485  val = 1.0;
486  }
487  else
488  {
489  if (hasKeyword(pos, "lazy con[straints]"))
490  ;
491  else if (hasKeyword(pos, "bound[s]"))
492  section = BOUNDS;
493  else if (hasKeyword(pos, "bin[ary]"))
494  section = BINARIES;
495  else if (hasKeyword(pos, "bin[aries]"))
496  section = BINARIES;
497  else if (hasKeyword(pos, "gen[erals]"))
498  section = INTEGERS;
499  else if (hasKeyword(pos, "int[egers]")) // this is undocumented
500  section = INTEGERS;
501  else if (hasKeyword(pos, "end"))
502  {
503  finished = true;
504  break;
505  }
506  else if (hasKeyword(pos, "s[ubject][ ]t[o]") // second time
507  || hasKeyword(pos, "s[uch][ ]t[hat]")
508  || hasKeyword(pos, "s[.][ ]t[.]")
509  || hasKeyword(pos, "lazy con[straints]"))
510  {
511  // In principle this has to checked for all keywords above,
512  // otherwise we just ignore any half finished constraint
513  if (have_value)
514  goto syntax_error;
515 
516  have_value = true;
517  val = 1.0;
518  }
519  }
520 
521  // 3a. Look for row names in objective and drop it.
522  if (section == OBJECTIVE)
523  hasRowName(pos, 0);
524 
525  // 3b. Look for row name in constraint and store it.
526  if (section == CONSTRAINTS)
527  if (hasRowName(pos, rnames))
528  unnamed = false;
529 
530  // 4a. Remove initial spaces.
531  while(isSpace(pos[i]))
532  i++;
533 
534  // 4b. remove spaces if they do not appear before the name of a vaiable.
535  for(k = 0; pos[i] != '\0'; i++)
536  if (!isSpace(pos[i]) || isColName(&pos[i + 1]))
537  tmp[k++] = pos[i];
538 
539  tmp[k] = '\0';
540 
541  // 5. Is this a empty line ?
542  if (tmp[0] == '\0')
543  continue;
544 
545  // 6. Collapse sequences of '+' and '-'. e.g ++---+ => -
546  for(i = 0, k = 0; tmp[i] != '\0'; i++)
547  {
548  while(((tmp[i] == '+') || (tmp[i] == '-'))
549  && ((tmp[i + 1] == '+') || (tmp[i + 1] == '-')))
550  {
551  if (tmp[i++] == '-')
552  tmp[i] = (tmp[i] == '-') ? '+' : '-';
553  }
554  line[k++] = tmp[i];
555  }
556  line[k] = '\0';
557 
558  //-----------------------------------------------------------------------
559  //--- Line processing loop
560  //-----------------------------------------------------------------------
561  pos = line;
562 
563  MSG_DEBUG( spxout << "DLPFRD09 pos=" << pos << std::endl; )
564 
565  // 7. We have something left to process.
566  while((pos != 0) && (*pos != '\0'))
567  {
568  // remember our position, so we are sure we make progress.
569  pos_old = pos;
570 
571  // Now process the sections
572  switch(section)
573  {
574  case OBJECTIVE :
575  if (isValue(pos))
576  {
577  Real pre_sign = 1.0;
578 
579  /* Allready having here a value could only result from
580  * being the first number in a constraint, or a sign
581  * '+' or '-' as last token on the previous line.
582  */
583  if (have_value)
584  {
585  if (NE(fabs(val), 1.0))
586  goto syntax_error;
587 
588  if (EQ(val, -1.0))
589  pre_sign = val;
590  }
591  have_value = true;
592  val = readValue(pos) * pre_sign;
593  }
594  if (*pos == '\0')
595  continue;
596 
597  if (!have_value || !isColName(pos))
598  goto syntax_error;
599 
600  have_value = false;
601  colidx = readColName(pos, cnames, cset, &emptycol);
602  vec.add(colidx, val);
603  break;
604  case CONSTRAINTS :
605  if (isValue(pos))
606  {
607  Real pre_sign = 1.0;
608 
609  /* Allready having here a value could only result from
610  * being the first number in a constraint, or a sign
611  * '+' or '-' as last token on the previous line.
612  */
613  if (have_value)
614  {
615  if (NE(fabs(val), 1.0))
616  goto syntax_error;
617 
618  if (EQ(val, -1.0))
619  pre_sign = val;
620  }
621 
622  have_value = true;
623  val = readValue(pos) * pre_sign;
624 
625  if (sense != 0)
626  {
627  if (sense == '<')
628  {
629  row.setLhs(-infinity);
630  row.setRhs(val);
631  }
632  else if (sense == '>')
633  {
634  row.setLhs(val);
635  row.setRhs(infinity);
636  }
637  else
638  {
639  assert(sense == '=');
640 
641  row.setLhs(val);
642  row.setRhs(val);
643  }
644  row.setRowVector(vec);
645  rset.add(row);
646  vec.clear();
647 
648  if (!unnamed)
649  unnamed = true;
650  else
651  {
652  char name[16];
653  sprintf(name, "C%d", rset.num());
654  rnames->add(name);
655  }
656  have_value = true;
657  val = 1.0;
658  sense = 0;
659  pos = 0;
660  // next line
661  continue;
662  }
663  }
664  if (*pos == '\0')
665  continue;
666 
667  if (have_value)
668  {
669  if (isColName(pos))
670  {
671  colidx = readColName(pos, cnames, cset, &emptycol);
672 
673  if (val != 0.0)
674  {
675  // Do we have this index allready in the row ?
676  int n = vec.number(colidx);
677 
678  // No! So add it.
679  if (n < 0)
680  vec.add(colidx, val);
681  else
682  {
683  /* Yes. So we add them up and remove the element
684  * if it amounts to zero.
685  */
686  assert(vec.index(n) == colidx);
687 
688  val += vec.value(n);
689 
690  if (val == 0.0)
691  vec.remove(n);
692  else
693  vec.value(n) = val;
694 
695  assert(cnames->has(colidx));
696 
697  MSG_WARNING( spxout << "WLPFRD10 Duplicate index "
698  << (*cnames)[colidx]
699  << " in line " << lineno
700  << std::endl; )
701  }
702  }
703  have_value = false;
704  }
705  else
706  {
707  /* We have a row like c1: <= 5 with no variables in it.
708  * We can not handle 10 <= 5; issue a syntax error.
709  */
710  if (val != 1.0)
711  goto syntax_error;
712 
713  // If the next thing is not the sense we give up also.
714  if (!isSense(pos))
715  goto syntax_error;
716 
717  have_value = false;
718  }
719  }
720  assert(!have_value);
721 
722  if (isSense(pos))
723  sense = readSense(pos);
724  break;
725  case BOUNDS :
726  other = false;
727  sense = 0;
728 
729  if (isValue(pos))
730  {
731  val = isInfinity(pos) ? readInfinity(pos) : readValue(pos);
732 
733  if (!isSense(pos))
734  goto syntax_error;
735 
736  sense = readSense(pos);
737  other = true;
738  }
739  if (!isColName(pos))
740  goto syntax_error;
741 
742  if ((colidx = readColName(pos, cnames, cset, 0)) < 0)
743  {
744  MSG_WARNING( spxout << "WLPFRD11 in Bounds section line "
745  << lineno << " ignored" << std::endl; )
746  *pos = '\0';
747  continue;
748  }
749  if (sense)
750  {
751  if (sense == '<')
752  cset.lower_w(colidx) = val;
753  else if (sense == '>')
754  cset.upper_w(colidx) = val;
755  else
756  {
757  assert(sense == '=');
758  cset.lower_w(colidx) = val;
759  cset.upper_w(colidx) = val;
760  }
761  }
762  if (isFree(pos))
763  {
764  cset.lower_w(colidx) = -infinity;
765  cset.upper_w(colidx) = infinity;
766  other = true;
767  pos += 4; // set position after the word "free"
768  }
769  else if (isSense(pos))
770  {
771  sense = readSense(pos);
772  other = true;
773 
774  if (!isValue(pos))
775  goto syntax_error;
776 
777  val = isInfinity(pos) ? readInfinity(pos) : readValue(pos);
778 
779  if (sense == '<')
780  cset.upper_w(colidx) = val;
781  else if (sense == '>')
782  cset.lower_w(colidx) = val;
783  else
784  {
785  assert(sense == '=');
786  cset.lower_w(colidx) = val;
787  cset.upper_w(colidx) = val;
788  }
789  }
790  /* Do we have only a single column name in the input line?
791  * We could ignore this savely, but it is probably a sign
792  * of some other error.
793  */
794  if (!other)
795  goto syntax_error;
796  break;
797  case BINARIES :
798  case INTEGERS :
799  if ((colidx = readColName(pos, cnames, cset, 0)) < 0)
800  {
801  MSG_WARNING( spxout << "WLPFRD12 in Binary/General section line "
802  << lineno << " ignored" << std::endl; )
803  }
804  else
805  {
806  if (section == BINARIES)
807  {
808  if (cset.lower(colidx) < 0.0)
809  {
810  cset.lower_w(colidx) = 0.0;
811  }
812  if (cset.upper(colidx) > 1.0)
813  {
814  cset.upper_w(colidx) = 1.0;
815  }
816  }
817  if (p_intvars != 0)
818  p_intvars->addIdx(colidx);
819  }
820  break;
821  case START :
822  MSG_ERROR( spxout << "ELPFRD13 This seems to be no LP format file"
823  << std::endl; )
824  goto syntax_error;
825  default :
826  throw SPxInternalCodeException("XLPFRD01 This should never happen.");
827  }
828  if (pos == pos_old)
829  goto syntax_error;
830  }
831  }
832  //--------------------------------------------------------------------------
833 
834  assert(isConsistent());
835 
836  addCols(cset);
837  assert(isConsistent());
838 
839  addRows(rset);
840  assert(isConsistent());
841 
842 syntax_error:
843  if (finished)
844  {
845  MSG_INFO2( spxout << "ILPFRD14 Finished reading " << lineno
846  << " lines" << std::endl; )
847  }
848  else
849  MSG_ERROR( spxout << "ELPFRD15 Syntax error in line " << lineno << std::endl; )
850 
851  if (p_cnames == 0)
852  delete cnames;
853  if (p_rnames == 0)
854  delete rnames;
855 
856  MSG_DEBUG( spxout << "DLPFRD16\n" << *this; )
857 
858  return finished;
859 }
860 } // namespace soplex