SoPlex Doxygen Documentation
mpsinput.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 mpsinput.cpp
17  * @brief Read MPS format files.
18  */
19 //#define DEBUGGING 1 // Setting this generates a lot of output
20 
21 #include <assert.h>
22 #include <ctype.h>
23 #include <string.h>
24 
25 #include "spxdefines.h"
26 #include "mpsinput.h"
27 #include "spxout.h"
28 
29 #define PATCH_CHAR '_'
30 #define BLANK ' '
31 
32 namespace soplex
33 {
34 
35 /// fill the line from \p pos up to column 80 with blanks.
36 static void clear_from(char* buf, int pos)
37 {
38  for(int i = pos; i < 80; i++)
39  buf[i] = BLANK;
40  buf[80] = '\0';
41 }
42 
43 /// change all blanks inside a field to #PATCH_CHAR.
44 static void patch_field(char* buf, int beg, int end)
45 {
46  while((beg <= end) && (buf[end] == BLANK))
47  end--;
48 
49  while((beg <= end) && (buf[beg] == BLANK))
50  beg++;
51 
52  for(int i = beg; i <= end; i++)
53  if (buf[i] == BLANK)
54  buf[i] = PATCH_CHAR;
55 }
56 
57 /// read a MPS format data line and parse the fields.
59 {
60  int len;
61  int space;
62  char* s;
63  bool is_marker;
64 
65  do
66  {
67  m_f0 = m_f1 = m_f2 = m_f3 = m_f4 = m_f5 = 0;
68  is_marker = false;
69 
70  // Read until we have a not comment line.
71  do
72  {
73  if (m_input.getline(m_buf, sizeof(m_buf)) == 0)
74  return false;
75  m_lineno++;
76 
77  MSG_DEBUG( spxout << "DMPSIN01 Line " << m_lineno
78  << " " << m_buf << std::endl; )
79  }
80  while(*m_buf == '*');
81 
82  /* Normalize line
83  */
84  len = int(strlen(m_buf));
85 
86  for(int i = 0; i < len; i++)
87  if ((m_buf[i] == '\t') || (m_buf[i] == '\n') || (m_buf[i] == '\r'))
88  m_buf[i] = BLANK;
89 
90  if (len < 80)
91  clear_from(m_buf, len);
92 
93  assert(strlen(m_buf) >= 80);
94 
95  /* Look for new section
96  */
97  if (*m_buf != BLANK)
98  {
99  m_f0 = strtok(&m_buf[0], " ");
100 
101  assert(m_f0 != 0);
102 
103  m_f1 = strtok(0, " ");
104 
105  return true;
106  }
107 
108  if (!m_is_new_format)
109  {
110  /* Test for fixed format comments
111  */
112  if ((m_buf[14] == '$') && (m_buf[13] == ' '))
113  clear_from(m_buf, 14);
114  else if ((m_buf[39] == '$') && (m_buf[38] == ' '))
115  clear_from(m_buf, 39);
116 
117  /* Test for fixed format
118  */
119  space = m_buf[12] | m_buf[13]
120  | m_buf[22] | m_buf[23]
121  | m_buf[36] | m_buf[37] | m_buf[38]
122  | m_buf[47] | m_buf[48]
123  | m_buf[61] | m_buf[62] | m_buf[63];
124 
125  if (space == BLANK || len < 13)
126  {
127  /* Now we have space at the right positions.
128  * But are there also the non space where they
129  * should be ?
130  */
131  bool number = isdigit(m_buf[24]) || isdigit(m_buf[25])
132  || isdigit(m_buf[26]) || isdigit(m_buf[27])
133  || isdigit(m_buf[28]) || isdigit(m_buf[29])
134  || isdigit(m_buf[30]) || isdigit(m_buf[31])
135  || isdigit(m_buf[32]) || isdigit(m_buf[33])
136  || isdigit(m_buf[34]) || isdigit(m_buf[35]);
137 
138  /* len < 13 is handle ROW lines with embedded spaces
139  * in the names correctly
140  */
141  if (number || len < 13)
142  {
143  /* Now we assume fixed format, so we patch possible embedded spaces.
144  */
145  patch_field(m_buf, 4, 12);
146  patch_field(m_buf, 14, 22);
147  patch_field(m_buf, 39, 47);
148  }
149  else
150  {
151  if ( m_section == COLUMNS || m_section == RHS
152  || m_section == RANGES || m_section == BOUNDS)
153  m_is_new_format = true;
154  }
155  }
156  else
157  {
158  m_is_new_format = true;
159  }
160  }
161  s = &m_buf[1];
162 
163  /* At this point it is not clear if we have a indicator field.
164  * If there is none (e.g. empty) f1 will be the first name field.
165  * If there is one, f2 will be the first name field.
166  *
167  * Initially comment marks '$' ar only allowed in the beginning
168  * of the 2nd and 3rd name field. We test all fields but the first.
169  * This makes no difference, since if the $ is at the start of a value
170  * field, the line will be errornous anyway.
171  */
172  do
173  {
174  if (0 == (m_f1 = strtok(s, " ")))
175  break;
176 
177  if ((0 == (m_f2 = strtok(0, " "))) || (*m_f2 == '$'))
178  {
179  m_f2 = 0;
180  break;
181  }
182  if (!strcmp(m_f2, "'MARKER'"))
183  is_marker = true;
184 
185  if ((0 == (m_f3 = strtok(0, " "))) || (*m_f3 == '$'))
186  {
187  m_f3 = 0;
188  break;
189  }
190  if (is_marker)
191  {
192  if (!strcmp(m_f3, "'INTORG'"))
193  m_is_integer = true;
194  else if (!strcmp(m_f3, "'INTEND'"))
195  m_is_integer = false;
196  else
197  break; // unknown marker
198  }
199 
200  if (!strcmp(m_f3, "'MARKER'"))
201  is_marker = true;
202 
203  if ((0 == (m_f4 = strtok(0, " "))) || (*m_f4 == '$'))
204  {
205  m_f4 = 0;
206  break;
207  }
208  if (is_marker)
209  {
210  if (!strcmp(m_f4, "'INTORG'"))
211  m_is_integer = true;
212  else if (!strcmp(m_f4, "'INTEND'"))
213  m_is_integer = false;
214  else
215  break; // unknown marker
216  }
217 
218  if ((0 == (m_f5 = strtok(0, " "))) || (*m_f5 == '$'))
219  m_f5 = 0;
220  }
221  while(false);
222  }
223  while(is_marker);
224 
225  MSG_DEBUG(
226  spxout << "DMPSIN02 -----------------------------------------------"
227  << std::endl
228  << "DMPSIN03 f0=" << ((m_f0 == 0) ? "nil" : m_f0) << std::endl
229  << "DMPSIN04 f1=" << ((m_f1 == 0) ? "nil" : m_f1) << std::endl
230  << "DMPSIN05 f2=" << ((m_f2 == 0) ? "nil" : m_f2) << std::endl
231  << "DMPSIN06 f3=" << ((m_f3 == 0) ? "nil" : m_f3) << std::endl
232  << "DMPSIN07 f4=" << ((m_f4 == 0) ? "nil" : m_f4) << std::endl
233  << "DMPSIN08 f5=" << ((m_f5 == 0) ? "nil" : m_f5) << std::endl
234  << "DMPSIN09 -----------------------------------------------"
235  << std::endl;
236  )
237 
238  return true;
239 }
240 
241 /// Insert \p name as field 1 and shift all other fields up.
242 void MPSInput::insertName(const char* name, bool second)
243 {
244  m_f5 = m_f4;
245  m_f4 = m_f3;
246  m_f3 = m_f2;
247 
248  if (second)
249  m_f2 = name;
250  else
251  {
252  m_f2 = m_f1;
253  m_f1 = name;
254  }
255 }
256 
257 } // namespace soplex
258 
259 //-----------------------------------------------------------------------------
260 //Emacs Local Variables:
261 //Emacs mode:c++
262 //Emacs c-basic-offset:3
263 //Emacs tab-width:8
264 //Emacs indent-tabs-mode:nil
265 //Emacs End:
266 //-----------------------------------------------------------------------------