src/glpmpl04.c
changeset 1 c445c931472f
equal deleted inserted replaced
-1:000000000000 0:989d2bbca641
       
     1 /* glpmpl04.c */
       
     2 
       
     3 /***********************************************************************
       
     4 *  This code is part of GLPK (GNU Linear Programming Kit).
       
     5 *
       
     6 *  Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008,
       
     7 *  2009, 2010 Andrew Makhorin, Department for Applied Informatics,
       
     8 *  Moscow Aviation Institute, Moscow, Russia. All rights reserved.
       
     9 *  E-mail: <mao@gnu.org>.
       
    10 *
       
    11 *  GLPK is free software: you can redistribute it and/or modify it
       
    12 *  under the terms of the GNU General Public License as published by
       
    13 *  the Free Software Foundation, either version 3 of the License, or
       
    14 *  (at your option) any later version.
       
    15 *
       
    16 *  GLPK is distributed in the hope that it will be useful, but WITHOUT
       
    17 *  ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
       
    18 *  or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public
       
    19 *  License for more details.
       
    20 *
       
    21 *  You should have received a copy of the GNU General Public License
       
    22 *  along with GLPK. If not, see <http://www.gnu.org/licenses/>.
       
    23 ***********************************************************************/
       
    24 
       
    25 #define _GLPSTD_ERRNO
       
    26 #define _GLPSTD_STDIO
       
    27 #include "glpmpl.h"
       
    28 #define xfault xerror
       
    29 #define dmp_create_poolx(size) dmp_create_pool()
       
    30 
       
    31 /**********************************************************************/
       
    32 /* * *              GENERATING AND POSTSOLVING MODEL              * * */
       
    33 /**********************************************************************/
       
    34 
       
    35 /*----------------------------------------------------------------------
       
    36 -- alloc_content - allocate content arrays for all model objects.
       
    37 --
       
    38 -- This routine allocates content arrays for all existing model objects
       
    39 -- and thereby finalizes creating model.
       
    40 --
       
    41 -- This routine must be called immediately after reading model section,
       
    42 -- i.e. before reading data section or generating model. */
       
    43 
       
    44 void alloc_content(MPL *mpl)
       
    45 {     STATEMENT *stmt;
       
    46       /* walk through all model statements */
       
    47       for (stmt = mpl->model; stmt != NULL; stmt = stmt->next)
       
    48       {  switch (stmt->type)
       
    49          {  case A_SET:
       
    50                /* model set */
       
    51                xassert(stmt->u.set->array == NULL);
       
    52                stmt->u.set->array = create_array(mpl, A_ELEMSET,
       
    53                   stmt->u.set->dim);
       
    54                break;
       
    55             case A_PARAMETER:
       
    56                /* model parameter */
       
    57                xassert(stmt->u.par->array == NULL);
       
    58                switch (stmt->u.par->type)
       
    59                {  case A_NUMERIC:
       
    60                   case A_INTEGER:
       
    61                   case A_BINARY:
       
    62                      stmt->u.par->array = create_array(mpl, A_NUMERIC,
       
    63                         stmt->u.par->dim);
       
    64                      break;
       
    65                   case A_SYMBOLIC:
       
    66                      stmt->u.par->array = create_array(mpl, A_SYMBOLIC,
       
    67                         stmt->u.par->dim);
       
    68                      break;
       
    69                   default:
       
    70                      xassert(stmt != stmt);
       
    71                }
       
    72                break;
       
    73             case A_VARIABLE:
       
    74                /* model variable */
       
    75                xassert(stmt->u.var->array == NULL);
       
    76                stmt->u.var->array = create_array(mpl, A_ELEMVAR,
       
    77                   stmt->u.var->dim);
       
    78                break;
       
    79             case A_CONSTRAINT:
       
    80                /* model constraint/objective */
       
    81                xassert(stmt->u.con->array == NULL);
       
    82                stmt->u.con->array = create_array(mpl, A_ELEMCON,
       
    83                   stmt->u.con->dim);
       
    84                break;
       
    85 #if 1 /* 11/II-2008 */
       
    86             case A_TABLE:
       
    87 #endif
       
    88             case A_SOLVE:
       
    89             case A_CHECK:
       
    90             case A_DISPLAY:
       
    91             case A_PRINTF:
       
    92             case A_FOR:
       
    93                /* functional statements have no content array */
       
    94                break;
       
    95             default:
       
    96                xassert(stmt != stmt);
       
    97          }
       
    98       }
       
    99       return;
       
   100 }
       
   101 
       
   102 /*----------------------------------------------------------------------
       
   103 -- generate_model - generate model.
       
   104 --
       
   105 -- This routine executes the model statements which precede the solve
       
   106 -- statement. */
       
   107 
       
   108 void generate_model(MPL *mpl)
       
   109 {     STATEMENT *stmt;
       
   110       xassert(!mpl->flag_p);
       
   111       for (stmt = mpl->model; stmt != NULL; stmt = stmt->next)
       
   112       {  execute_statement(mpl, stmt);
       
   113          if (mpl->stmt->type == A_SOLVE) break;
       
   114       }
       
   115       mpl->stmt = stmt;
       
   116       return;
       
   117 }
       
   118 
       
   119 /*----------------------------------------------------------------------
       
   120 -- build_problem - build problem instance.
       
   121 --
       
   122 -- This routine builds lists of rows and columns for problem instance,
       
   123 -- which corresponds to the generated model. */
       
   124 
       
   125 void build_problem(MPL *mpl)
       
   126 {     STATEMENT *stmt;
       
   127       MEMBER *memb;
       
   128       VARIABLE *v;
       
   129       CONSTRAINT *c;
       
   130       FORMULA *t;
       
   131       int i, j;
       
   132       xassert(mpl->m == 0);
       
   133       xassert(mpl->n == 0);
       
   134       xassert(mpl->row == NULL);
       
   135       xassert(mpl->col == NULL);
       
   136       /* check that all elemental variables has zero column numbers */
       
   137       for (stmt = mpl->model; stmt != NULL; stmt = stmt->next)
       
   138       {  if (stmt->type == A_VARIABLE)
       
   139          {  v = stmt->u.var;
       
   140             for (memb = v->array->head; memb != NULL; memb = memb->next)
       
   141                xassert(memb->value.var->j == 0);
       
   142          }
       
   143       }
       
   144       /* assign row numbers to elemental constraints and objectives */
       
   145       for (stmt = mpl->model; stmt != NULL; stmt = stmt->next)
       
   146       {  if (stmt->type == A_CONSTRAINT)
       
   147          {  c = stmt->u.con;
       
   148             for (memb = c->array->head; memb != NULL; memb = memb->next)
       
   149             {  xassert(memb->value.con->i == 0);
       
   150                memb->value.con->i = ++mpl->m;
       
   151                /* walk through linear form and mark elemental variables,
       
   152                   which are referenced at least once */
       
   153                for (t = memb->value.con->form; t != NULL; t = t->next)
       
   154                {  xassert(t->var != NULL);
       
   155                   t->var->memb->value.var->j = -1;
       
   156                }
       
   157             }
       
   158          }
       
   159       }
       
   160       /* assign column numbers to marked elemental variables */
       
   161       for (stmt = mpl->model; stmt != NULL; stmt = stmt->next)
       
   162       {  if (stmt->type == A_VARIABLE)
       
   163          {  v = stmt->u.var;
       
   164             for (memb = v->array->head; memb != NULL; memb = memb->next)
       
   165                if (memb->value.var->j != 0) memb->value.var->j =
       
   166                   ++mpl->n;
       
   167          }
       
   168       }
       
   169       /* build list of rows */
       
   170       mpl->row = xcalloc(1+mpl->m, sizeof(ELEMCON *));
       
   171       for (i = 1; i <= mpl->m; i++) mpl->row[i] = NULL;
       
   172       for (stmt = mpl->model; stmt != NULL; stmt = stmt->next)
       
   173       {  if (stmt->type == A_CONSTRAINT)
       
   174          {  c = stmt->u.con;
       
   175             for (memb = c->array->head; memb != NULL; memb = memb->next)
       
   176             {  i = memb->value.con->i;
       
   177                xassert(1 <= i && i <= mpl->m);
       
   178                xassert(mpl->row[i] == NULL);
       
   179                mpl->row[i] = memb->value.con;
       
   180             }
       
   181          }
       
   182       }
       
   183       for (i = 1; i <= mpl->m; i++) xassert(mpl->row[i] != NULL);
       
   184       /* build list of columns */
       
   185       mpl->col = xcalloc(1+mpl->n, sizeof(ELEMVAR *));
       
   186       for (j = 1; j <= mpl->n; j++) mpl->col[j] = NULL;
       
   187       for (stmt = mpl->model; stmt != NULL; stmt = stmt->next)
       
   188       {  if (stmt->type == A_VARIABLE)
       
   189          {  v = stmt->u.var;
       
   190             for (memb = v->array->head; memb != NULL; memb = memb->next)
       
   191             {  j = memb->value.var->j;
       
   192                if (j == 0) continue;
       
   193                xassert(1 <= j && j <= mpl->n);
       
   194                xassert(mpl->col[j] == NULL);
       
   195                mpl->col[j] = memb->value.var;
       
   196             }
       
   197          }
       
   198       }
       
   199       for (j = 1; j <= mpl->n; j++) xassert(mpl->col[j] != NULL);
       
   200       return;
       
   201 }
       
   202 
       
   203 /*----------------------------------------------------------------------
       
   204 -- postsolve_model - postsolve model.
       
   205 --
       
   206 -- This routine executes the model statements which follow the solve
       
   207 -- statement. */
       
   208 
       
   209 void postsolve_model(MPL *mpl)
       
   210 {     STATEMENT *stmt;
       
   211       xassert(!mpl->flag_p);
       
   212       mpl->flag_p = 1;
       
   213       for (stmt = mpl->stmt; stmt != NULL; stmt = stmt->next)
       
   214          execute_statement(mpl, stmt);
       
   215       mpl->stmt = NULL;
       
   216       return;
       
   217 }
       
   218 
       
   219 /*----------------------------------------------------------------------
       
   220 -- clean_model - clean model content.
       
   221 --
       
   222 -- This routine cleans the model content that assumes deleting all stuff
       
   223 -- dynamically allocated on generating/postsolving phase.
       
   224 --
       
   225 -- Actually cleaning model content is not needed. This function is used
       
   226 -- mainly to be sure that there were no logical errors on using dynamic
       
   227 -- memory pools during the generation phase.
       
   228 --
       
   229 -- NOTE: This routine must not be called if any errors were detected on
       
   230 --       the generation phase. */
       
   231 
       
   232 void clean_model(MPL *mpl)
       
   233 {     STATEMENT *stmt;
       
   234       for (stmt = mpl->model; stmt != NULL; stmt = stmt->next)
       
   235          clean_statement(mpl, stmt);
       
   236       /* check that all atoms have been returned to their pools */
       
   237       if (dmp_in_use(mpl->strings).lo != 0)
       
   238          error(mpl, "internal logic error: %d string segment(s) were lo"
       
   239             "st", dmp_in_use(mpl->strings).lo);
       
   240       if (dmp_in_use(mpl->symbols).lo != 0)
       
   241          error(mpl, "internal logic error: %d symbol(s) were lost",
       
   242             dmp_in_use(mpl->symbols).lo);
       
   243       if (dmp_in_use(mpl->tuples).lo != 0)
       
   244          error(mpl, "internal logic error: %d n-tuple component(s) were"
       
   245             " lost", dmp_in_use(mpl->tuples).lo);
       
   246       if (dmp_in_use(mpl->arrays).lo != 0)
       
   247          error(mpl, "internal logic error: %d array(s) were lost",
       
   248             dmp_in_use(mpl->arrays).lo);
       
   249       if (dmp_in_use(mpl->members).lo != 0)
       
   250          error(mpl, "internal logic error: %d array member(s) were lost"
       
   251             , dmp_in_use(mpl->members).lo);
       
   252       if (dmp_in_use(mpl->elemvars).lo != 0)
       
   253          error(mpl, "internal logic error: %d elemental variable(s) wer"
       
   254             "e lost", dmp_in_use(mpl->elemvars).lo);
       
   255       if (dmp_in_use(mpl->formulae).lo != 0)
       
   256          error(mpl, "internal logic error: %d linear term(s) were lost",
       
   257             dmp_in_use(mpl->formulae).lo);
       
   258       if (dmp_in_use(mpl->elemcons).lo != 0)
       
   259          error(mpl, "internal logic error: %d elemental constraint(s) w"
       
   260             "ere lost", dmp_in_use(mpl->elemcons).lo);
       
   261       return;
       
   262 }
       
   263 
       
   264 /**********************************************************************/
       
   265 /* * *                        INPUT/OUTPUT                        * * */
       
   266 /**********************************************************************/
       
   267 
       
   268 /*----------------------------------------------------------------------
       
   269 -- open_input - open input text file.
       
   270 --
       
   271 -- This routine opens the input text file for scanning. */
       
   272 
       
   273 void open_input(MPL *mpl, char *file)
       
   274 {     mpl->line = 0;
       
   275       mpl->c = '\n';
       
   276       mpl->token = 0;
       
   277       mpl->imlen = 0;
       
   278       mpl->image[0] = '\0';
       
   279       mpl->value = 0.0;
       
   280       mpl->b_token = T_EOF;
       
   281       mpl->b_imlen = 0;
       
   282       mpl->b_image[0] = '\0';
       
   283       mpl->b_value = 0.0;
       
   284       mpl->f_dots = 0;
       
   285       mpl->f_scan = 0;
       
   286       mpl->f_token = 0;
       
   287       mpl->f_imlen = 0;
       
   288       mpl->f_image[0] = '\0';
       
   289       mpl->f_value = 0.0;
       
   290       memset(mpl->context, ' ', CONTEXT_SIZE);
       
   291       mpl->c_ptr = 0;
       
   292       xassert(mpl->in_fp == NULL);
       
   293       mpl->in_fp = xfopen(file, "r");
       
   294       if (mpl->in_fp == NULL)
       
   295          error(mpl, "unable to open %s - %s", file, xerrmsg());
       
   296       mpl->in_file = file;
       
   297       /* scan the very first character */
       
   298       get_char(mpl);
       
   299       /* scan the very first token */
       
   300       get_token(mpl);
       
   301       return;
       
   302 }
       
   303 
       
   304 /*----------------------------------------------------------------------
       
   305 -- read_char - read next character from input text file.
       
   306 --
       
   307 -- This routine returns a next ASCII character read from the input text
       
   308 -- file. If the end of file has been reached, EOF is returned. */
       
   309 
       
   310 int read_char(MPL *mpl)
       
   311 {     int c;
       
   312       xassert(mpl->in_fp != NULL);
       
   313       c = xfgetc(mpl->in_fp);
       
   314       if (c < 0)
       
   315       {  if (xferror(mpl->in_fp))
       
   316             error(mpl, "read error on %s - %s", mpl->in_file,
       
   317                xerrmsg());
       
   318          c = EOF;
       
   319       }
       
   320       return c;
       
   321 }
       
   322 
       
   323 /*----------------------------------------------------------------------
       
   324 -- close_input - close input text file.
       
   325 --
       
   326 -- This routine closes the input text file. */
       
   327 
       
   328 void close_input(MPL *mpl)
       
   329 {     xassert(mpl->in_fp != NULL);
       
   330       xfclose(mpl->in_fp);
       
   331       mpl->in_fp = NULL;
       
   332       mpl->in_file = NULL;
       
   333       return;
       
   334 }
       
   335 
       
   336 /*----------------------------------------------------------------------
       
   337 -- open_output - open output text file.
       
   338 --
       
   339 -- This routine opens the output text file for writing data produced by
       
   340 -- display and printf statements. */
       
   341 
       
   342 void open_output(MPL *mpl, char *file)
       
   343 {     xassert(mpl->out_fp == NULL);
       
   344       if (file == NULL)
       
   345       {  file = "<stdout>";
       
   346          mpl->out_fp = (void *)stdout;
       
   347       }
       
   348       else
       
   349       {  mpl->out_fp = xfopen(file, "w");
       
   350          if (mpl->out_fp == NULL)
       
   351             error(mpl, "unable to create %s - %s", file, xerrmsg());
       
   352       }
       
   353       mpl->out_file = xmalloc(strlen(file)+1);
       
   354       strcpy(mpl->out_file, file);
       
   355       return;
       
   356 }
       
   357 
       
   358 /*----------------------------------------------------------------------
       
   359 -- write_char - write next character to output text file.
       
   360 --
       
   361 -- This routine writes an ASCII character to the output text file. */
       
   362 
       
   363 void write_char(MPL *mpl, int c)
       
   364 {     xassert(mpl->out_fp != NULL);
       
   365       if (mpl->out_fp == (void *)stdout)
       
   366          xprintf("%c", c);
       
   367       else
       
   368          xfprintf(mpl->out_fp, "%c", c);
       
   369       return;
       
   370 }
       
   371 
       
   372 /*----------------------------------------------------------------------
       
   373 -- write_text - format and write text to output text file.
       
   374 --
       
   375 -- This routine formats a text using the format control string and then
       
   376 -- writes this text to the output text file. */
       
   377 
       
   378 void write_text(MPL *mpl, char *fmt, ...)
       
   379 {     va_list arg;
       
   380       char buf[OUTBUF_SIZE], *c;
       
   381       va_start(arg, fmt);
       
   382       vsprintf(buf, fmt, arg);
       
   383       xassert(strlen(buf) < sizeof(buf));
       
   384       va_end(arg);
       
   385       for (c = buf; *c != '\0'; c++) write_char(mpl, *c);
       
   386       return;
       
   387 }
       
   388 
       
   389 /*----------------------------------------------------------------------
       
   390 -- flush_output - finalize writing data to output text file.
       
   391 --
       
   392 -- This routine finalizes writing data to the output text file. */
       
   393 
       
   394 void flush_output(MPL *mpl)
       
   395 {     xassert(mpl->out_fp != NULL);
       
   396       if (mpl->out_fp != (void *)stdout)
       
   397       {  xfflush(mpl->out_fp);
       
   398          if (xferror(mpl->out_fp))
       
   399             error(mpl, "write error on %s - %s", mpl->out_file,
       
   400                xerrmsg());
       
   401       }
       
   402       return;
       
   403 }
       
   404 
       
   405 /**********************************************************************/
       
   406 /* * *                      SOLVER INTERFACE                      * * */
       
   407 /**********************************************************************/
       
   408 
       
   409 /*----------------------------------------------------------------------
       
   410 -- error - print error message and terminate model processing.
       
   411 --
       
   412 -- This routine formats and prints an error message and then terminates
       
   413 -- model processing. */
       
   414 
       
   415 void error(MPL *mpl, char *fmt, ...)
       
   416 {     va_list arg;
       
   417       char msg[4095+1];
       
   418       va_start(arg, fmt);
       
   419       vsprintf(msg, fmt, arg);
       
   420       xassert(strlen(msg) < sizeof(msg));
       
   421       va_end(arg);
       
   422       switch (mpl->phase)
       
   423       {  case 1:
       
   424          case 2:
       
   425             /* translation phase */
       
   426             xprintf("%s:%d: %s\n",
       
   427                mpl->in_file == NULL ? "(unknown)" : mpl->in_file,
       
   428                mpl->line, msg);
       
   429             print_context(mpl);
       
   430             break;
       
   431          case 3:
       
   432             /* generation/postsolve phase */
       
   433             xprintf("%s:%d: %s\n",
       
   434                mpl->mod_file == NULL ? "(unknown)" : mpl->mod_file,
       
   435                mpl->stmt == NULL ? 0 : mpl->stmt->line, msg);
       
   436             break;
       
   437          default:
       
   438             xassert(mpl != mpl);
       
   439       }
       
   440       mpl->phase = 4;
       
   441       longjmp(mpl->jump, 1);
       
   442       /* no return */
       
   443 }
       
   444 
       
   445 /*----------------------------------------------------------------------
       
   446 -- warning - print warning message and continue model processing.
       
   447 --
       
   448 -- This routine formats and prints a warning message and returns to the
       
   449 -- calling program. */
       
   450 
       
   451 void warning(MPL *mpl, char *fmt, ...)
       
   452 {     va_list arg;
       
   453       char msg[4095+1];
       
   454       va_start(arg, fmt);
       
   455       vsprintf(msg, fmt, arg);
       
   456       xassert(strlen(msg) < sizeof(msg));
       
   457       va_end(arg);
       
   458       switch (mpl->phase)
       
   459       {  case 1:
       
   460          case 2:
       
   461             /* translation phase */
       
   462             xprintf("%s:%d: warning: %s\n",
       
   463                mpl->in_file == NULL ? "(unknown)" : mpl->in_file,
       
   464                mpl->line, msg);
       
   465             break;
       
   466          case 3:
       
   467             /* generation/postsolve phase */
       
   468             xprintf("%s:%d: warning: %s\n",
       
   469                mpl->mod_file == NULL ? "(unknown)" : mpl->mod_file,
       
   470                mpl->stmt == NULL ? 0 : mpl->stmt->line, msg);
       
   471             break;
       
   472          default:
       
   473             xassert(mpl != mpl);
       
   474       }
       
   475       return;
       
   476 }
       
   477 
       
   478 /*----------------------------------------------------------------------
       
   479 -- mpl_initialize - create and initialize translator database.
       
   480 --
       
   481 -- *Synopsis*
       
   482 --
       
   483 -- #include "glpmpl.h"
       
   484 -- MPL *mpl_initialize(void);
       
   485 --
       
   486 -- *Description*
       
   487 --
       
   488 -- The routine mpl_initialize creates and initializes the database used
       
   489 -- by the GNU MathProg translator.
       
   490 --
       
   491 -- *Returns*
       
   492 --
       
   493 -- The routine returns a pointer to the database created. */
       
   494 
       
   495 MPL *mpl_initialize(void)
       
   496 {     MPL *mpl;
       
   497       mpl = xmalloc(sizeof(MPL));
       
   498       /* scanning segment */
       
   499       mpl->line = 0;
       
   500       mpl->c = 0;
       
   501       mpl->token = 0;
       
   502       mpl->imlen = 0;
       
   503       mpl->image = xcalloc(MAX_LENGTH+1, sizeof(char));
       
   504       mpl->image[0] = '\0';
       
   505       mpl->value = 0.0;
       
   506       mpl->b_token = 0;
       
   507       mpl->b_imlen = 0;
       
   508       mpl->b_image = xcalloc(MAX_LENGTH+1, sizeof(char));
       
   509       mpl->b_image[0] = '\0';
       
   510       mpl->b_value = 0.0;
       
   511       mpl->f_dots = 0;
       
   512       mpl->f_scan = 0;
       
   513       mpl->f_token = 0;
       
   514       mpl->f_imlen = 0;
       
   515       mpl->f_image = xcalloc(MAX_LENGTH+1, sizeof(char));
       
   516       mpl->f_image[0] = '\0';
       
   517       mpl->f_value = 0.0;
       
   518       mpl->context = xcalloc(CONTEXT_SIZE, sizeof(char));
       
   519       memset(mpl->context, ' ', CONTEXT_SIZE);
       
   520       mpl->c_ptr = 0;
       
   521       mpl->flag_d = 0;
       
   522       /* translating segment */
       
   523       mpl->pool = dmp_create_poolx(0);
       
   524       mpl->tree = avl_create_tree(avl_strcmp, NULL);
       
   525       mpl->model = NULL;
       
   526       mpl->flag_x = 0;
       
   527       mpl->as_within = 0;
       
   528       mpl->as_in = 0;
       
   529       mpl->as_binary = 0;
       
   530       mpl->flag_s = 0;
       
   531       /* common segment */
       
   532       mpl->strings = dmp_create_poolx(sizeof(STRING));
       
   533       mpl->symbols = dmp_create_poolx(sizeof(SYMBOL));
       
   534       mpl->tuples = dmp_create_poolx(sizeof(TUPLE));
       
   535       mpl->arrays = dmp_create_poolx(sizeof(ARRAY));
       
   536       mpl->members = dmp_create_poolx(sizeof(MEMBER));
       
   537       mpl->elemvars = dmp_create_poolx(sizeof(ELEMVAR));
       
   538       mpl->formulae = dmp_create_poolx(sizeof(FORMULA));
       
   539       mpl->elemcons = dmp_create_poolx(sizeof(ELEMCON));
       
   540       mpl->a_list = NULL;
       
   541       mpl->sym_buf = xcalloc(255+1, sizeof(char));
       
   542       mpl->sym_buf[0] = '\0';
       
   543       mpl->tup_buf = xcalloc(255+1, sizeof(char));
       
   544       mpl->tup_buf[0] = '\0';
       
   545       /* generating/postsolving segment */
       
   546       mpl->rand = rng_create_rand();
       
   547       mpl->flag_p = 0;
       
   548       mpl->stmt = NULL;
       
   549 #if 1 /* 11/II-2008 */
       
   550       mpl->dca = NULL;
       
   551 #endif
       
   552       mpl->m = 0;
       
   553       mpl->n = 0;
       
   554       mpl->row = NULL;
       
   555       mpl->col = NULL;
       
   556       /* input/output segment */
       
   557       mpl->in_fp = NULL;
       
   558       mpl->in_file = NULL;
       
   559       mpl->out_fp = NULL;
       
   560       mpl->out_file = NULL;
       
   561       mpl->prt_fp = NULL;
       
   562       mpl->prt_file = NULL;
       
   563       /* solver interface segment */
       
   564       if (setjmp(mpl->jump)) xassert(mpl != mpl);
       
   565       mpl->phase = 0;
       
   566       mpl->mod_file = NULL;
       
   567       mpl->mpl_buf = xcalloc(255+1, sizeof(char));
       
   568       mpl->mpl_buf[0] = '\0';
       
   569       return mpl;
       
   570 }
       
   571 
       
   572 /*----------------------------------------------------------------------
       
   573 -- mpl_read_model - read model section and optional data section.
       
   574 --
       
   575 -- *Synopsis*
       
   576 --
       
   577 -- #include "glpmpl.h"
       
   578 -- int mpl_read_model(MPL *mpl, char *file, int skip_data);
       
   579 --
       
   580 -- *Description*
       
   581 --
       
   582 -- The routine mpl_read_model reads model section and optionally data
       
   583 -- section, which may follow the model section, from the text file,
       
   584 -- whose name is the character string file, performs translating model
       
   585 -- statements and data blocks, and stores all the information in the
       
   586 -- translator database.
       
   587 --
       
   588 -- The parameter skip_data is a flag. If the input file contains the
       
   589 -- data section and this flag is set, the data section is not read as
       
   590 -- if there were no data section and a warning message is issued. This
       
   591 -- allows reading the data section from another input file.
       
   592 --
       
   593 -- This routine should be called once after the routine mpl_initialize
       
   594 -- and before other API routines.
       
   595 --
       
   596 -- *Returns*
       
   597 --
       
   598 -- The routine mpl_read_model returns one the following codes:
       
   599 --
       
   600 -- 1 - translation successful. The input text file contains only model
       
   601 --     section. In this case the calling program may call the routine
       
   602 --     mpl_read_data to read data section from another file.
       
   603 -- 2 - translation successful. The input text file contains both model
       
   604 --     and data section.
       
   605 -- 4 - processing failed due to some errors. In this case the calling
       
   606 --     program should call the routine mpl_terminate to terminate model
       
   607 --     processing. */
       
   608 
       
   609 int mpl_read_model(MPL *mpl, char *file, int skip_data)
       
   610 {     if (mpl->phase != 0)
       
   611          xfault("mpl_read_model: invalid call sequence\n");
       
   612       if (file == NULL)
       
   613          xfault("mpl_read_model: no input filename specified\n");
       
   614       /* set up error handler */
       
   615       if (setjmp(mpl->jump)) goto done;
       
   616       /* translate model section */
       
   617       mpl->phase = 1;
       
   618       xprintf("Reading model section from %s...\n", file);
       
   619       open_input(mpl, file);
       
   620       model_section(mpl);
       
   621       if (mpl->model == NULL)
       
   622          error(mpl, "empty model section not allowed");
       
   623       /* save name of the input text file containing model section for
       
   624          error diagnostics during the generation phase */
       
   625       mpl->mod_file = xcalloc(strlen(file)+1, sizeof(char));
       
   626       strcpy(mpl->mod_file, mpl->in_file);
       
   627       /* allocate content arrays for all model objects */
       
   628       alloc_content(mpl);
       
   629       /* optional data section may begin with the keyword 'data' */
       
   630       if (is_keyword(mpl, "data"))
       
   631       {  if (skip_data)
       
   632          {  warning(mpl, "data section ignored");
       
   633             goto skip;
       
   634          }
       
   635          mpl->flag_d = 1;
       
   636          get_token(mpl /* data */);
       
   637          if (mpl->token != T_SEMICOLON)
       
   638             error(mpl, "semicolon missing where expected");
       
   639          get_token(mpl /* ; */);
       
   640          /* translate data section */
       
   641          mpl->phase = 2;
       
   642          xprintf("Reading data section from %s...\n", file);
       
   643          data_section(mpl);
       
   644       }
       
   645       /* process end statement */
       
   646       end_statement(mpl);
       
   647 skip: xprintf("%d line%s were read\n",
       
   648          mpl->line, mpl->line == 1 ? "" : "s");
       
   649       close_input(mpl);
       
   650 done: /* return to the calling program */
       
   651       return mpl->phase;
       
   652 }
       
   653 
       
   654 /*----------------------------------------------------------------------
       
   655 -- mpl_read_data - read data section.
       
   656 --
       
   657 -- *Synopsis*
       
   658 --
       
   659 -- #include "glpmpl.h"
       
   660 -- int mpl_read_data(MPL *mpl, char *file);
       
   661 --
       
   662 -- *Description*
       
   663 --
       
   664 -- The routine mpl_read_data reads data section from the text file,
       
   665 -- whose name is the character string file, performs translating data
       
   666 -- blocks, and stores the data read in the translator database.
       
   667 --
       
   668 -- If this routine is used, it should be called once after the routine
       
   669 -- mpl_read_model and if the latter returned the code 1.
       
   670 --
       
   671 -- *Returns*
       
   672 --
       
   673 -- The routine mpl_read_data returns one of the following codes:
       
   674 --
       
   675 -- 2 - data section has been successfully processed.
       
   676 -- 4 - processing failed due to some errors. In this case the calling
       
   677 --     program should call the routine mpl_terminate to terminate model
       
   678 --     processing. */
       
   679 
       
   680 int mpl_read_data(MPL *mpl, char *file)
       
   681 #if 0 /* 02/X-2008 */
       
   682 {     if (mpl->phase != 1)
       
   683 #else
       
   684 {     if (!(mpl->phase == 1 || mpl->phase == 2))
       
   685 #endif
       
   686          xfault("mpl_read_data: invalid call sequence\n");
       
   687       if (file == NULL)
       
   688          xfault("mpl_read_data: no input filename specified\n");
       
   689       /* set up error handler */
       
   690       if (setjmp(mpl->jump)) goto done;
       
   691       /* process data section */
       
   692       mpl->phase = 2;
       
   693       xprintf("Reading data section from %s...\n", file);
       
   694       mpl->flag_d = 1;
       
   695       open_input(mpl, file);
       
   696       /* in this case the keyword 'data' is optional */
       
   697       if (is_literal(mpl, "data"))
       
   698       {  get_token(mpl /* data */);
       
   699          if (mpl->token != T_SEMICOLON)
       
   700             error(mpl, "semicolon missing where expected");
       
   701          get_token(mpl /* ; */);
       
   702       }
       
   703       data_section(mpl);
       
   704       /* process end statement */
       
   705       end_statement(mpl);
       
   706       xprintf("%d line%s were read\n",
       
   707          mpl->line, mpl->line == 1 ? "" : "s");
       
   708       close_input(mpl);
       
   709 done: /* return to the calling program */
       
   710       return mpl->phase;
       
   711 }
       
   712 
       
   713 /*----------------------------------------------------------------------
       
   714 -- mpl_generate - generate model.
       
   715 --
       
   716 -- *Synopsis*
       
   717 --
       
   718 -- #include "glpmpl.h"
       
   719 -- int mpl_generate(MPL *mpl, char *file);
       
   720 --
       
   721 -- *Description*
       
   722 --
       
   723 -- The routine mpl_generate generates the model using its description
       
   724 -- stored in the translator database. This phase means generating all
       
   725 -- variables, constraints, and objectives, executing check and display
       
   726 -- statements, which precede the solve statement (if it is presented),
       
   727 -- and building the problem instance.
       
   728 --
       
   729 -- The character string file specifies the name of output text file, to
       
   730 -- which output produced by display statements should be written. It is
       
   731 -- allowed to specify NULL, in which case the output goes to stdout via
       
   732 -- the routine print.
       
   733 --
       
   734 -- This routine should be called once after the routine mpl_read_model
       
   735 -- or mpl_read_data and if one of the latters returned the code 2.
       
   736 --
       
   737 -- *Returns*
       
   738 --
       
   739 -- The routine mpl_generate returns one of the following codes:
       
   740 --
       
   741 -- 3 - model has been successfully generated. In this case the calling
       
   742 --     program may call other api routines to obtain components of the
       
   743 --     problem instance from the translator database.
       
   744 -- 4 - processing failed due to some errors. In this case the calling
       
   745 --     program should call the routine mpl_terminate to terminate model
       
   746 --     processing. */
       
   747 
       
   748 int mpl_generate(MPL *mpl, char *file)
       
   749 {     if (!(mpl->phase == 1 || mpl->phase == 2))
       
   750          xfault("mpl_generate: invalid call sequence\n");
       
   751       /* set up error handler */
       
   752       if (setjmp(mpl->jump)) goto done;
       
   753       /* generate model */
       
   754       mpl->phase = 3;
       
   755       open_output(mpl, file);
       
   756       generate_model(mpl);
       
   757       flush_output(mpl);
       
   758       /* build problem instance */
       
   759       build_problem(mpl);
       
   760       /* generation phase has been finished */
       
   761       xprintf("Model has been successfully generated\n");
       
   762 done: /* return to the calling program */
       
   763       return mpl->phase;
       
   764 }
       
   765 
       
   766 /*----------------------------------------------------------------------
       
   767 -- mpl_get_prob_name - obtain problem (model) name.
       
   768 --
       
   769 -- *Synopsis*
       
   770 --
       
   771 -- #include "glpmpl.h"
       
   772 -- char *mpl_get_prob_name(MPL *mpl);
       
   773 --
       
   774 -- *Returns*
       
   775 --
       
   776 -- The routine mpl_get_prob_name returns a pointer to internal buffer,
       
   777 -- which contains symbolic name of the problem (model).
       
   778 --
       
   779 -- *Note*
       
   780 --
       
   781 -- Currently MathProg has no feature to assign a symbolic name to the
       
   782 -- model. Therefore the routine mpl_get_prob_name tries to construct
       
   783 -- such name using the name of input text file containing model section,
       
   784 -- although this is not a good idea (due to portability problems). */
       
   785 
       
   786 char *mpl_get_prob_name(MPL *mpl)
       
   787 {     char *name = mpl->mpl_buf;
       
   788       char *file = mpl->mod_file;
       
   789       int k;
       
   790       if (mpl->phase != 3)
       
   791          xfault("mpl_get_prob_name: invalid call sequence\n");
       
   792       for (;;)
       
   793       {  if (strchr(file, '/') != NULL)
       
   794             file = strchr(file, '/') + 1;
       
   795          else if (strchr(file, '\\') != NULL)
       
   796             file = strchr(file, '\\') + 1;
       
   797          else if (strchr(file, ':') != NULL)
       
   798             file = strchr(file, ':') + 1;
       
   799          else
       
   800             break;
       
   801       }
       
   802       for (k = 0; ; k++)
       
   803       {  if (k == 255) break;
       
   804          if (!(isalnum((unsigned char)*file) || *file == '_')) break;
       
   805          name[k] = *file++;
       
   806       }
       
   807       if (k == 0)
       
   808          strcpy(name, "Unknown");
       
   809       else
       
   810          name[k] = '\0';
       
   811       xassert(strlen(name) <= 255);
       
   812       return name;
       
   813 }
       
   814 
       
   815 /*----------------------------------------------------------------------
       
   816 -- mpl_get_num_rows - determine number of rows.
       
   817 --
       
   818 -- *Synopsis*
       
   819 --
       
   820 -- #include "glpmpl.h"
       
   821 -- int mpl_get_num_rows(MPL *mpl);
       
   822 --
       
   823 -- *Returns*
       
   824 --
       
   825 -- The routine mpl_get_num_rows returns total number of rows in the
       
   826 -- problem, where each row is an individual constraint or objective. */
       
   827 
       
   828 int mpl_get_num_rows(MPL *mpl)
       
   829 {     if (mpl->phase != 3)
       
   830          xfault("mpl_get_num_rows: invalid call sequence\n");
       
   831       return mpl->m;
       
   832 }
       
   833 
       
   834 /*----------------------------------------------------------------------
       
   835 -- mpl_get_num_cols - determine number of columns.
       
   836 --
       
   837 -- *Synopsis*
       
   838 --
       
   839 -- #include "glpmpl.h"
       
   840 -- int mpl_get_num_cols(MPL *mpl);
       
   841 --
       
   842 -- *Returns*
       
   843 --
       
   844 -- The routine mpl_get_num_cols returns total number of columns in the
       
   845 -- problem, where each column is an individual variable. */
       
   846 
       
   847 int mpl_get_num_cols(MPL *mpl)
       
   848 {     if (mpl->phase != 3)
       
   849          xfault("mpl_get_num_cols: invalid call sequence\n");
       
   850       return mpl->n;
       
   851 }
       
   852 
       
   853 /*----------------------------------------------------------------------
       
   854 -- mpl_get_row_name - obtain row name.
       
   855 --
       
   856 -- *Synopsis*
       
   857 --
       
   858 -- #include "glpmpl.h"
       
   859 -- char *mpl_get_row_name(MPL *mpl, int i);
       
   860 --
       
   861 -- *Returns*
       
   862 --
       
   863 -- The routine mpl_get_row_name returns a pointer to internal buffer,
       
   864 -- which contains symbolic name of i-th row of the problem. */
       
   865 
       
   866 char *mpl_get_row_name(MPL *mpl, int i)
       
   867 {     char *name = mpl->mpl_buf, *t;
       
   868       int len;
       
   869       if (mpl->phase != 3)
       
   870          xfault("mpl_get_row_name: invalid call sequence\n");
       
   871       if (!(1 <= i && i <= mpl->m))
       
   872          xfault("mpl_get_row_name: i = %d; row number out of range\n",
       
   873             i);
       
   874       strcpy(name, mpl->row[i]->con->name);
       
   875       len = strlen(name);
       
   876       xassert(len <= 255);
       
   877       t = format_tuple(mpl, '[', mpl->row[i]->memb->tuple);
       
   878       while (*t)
       
   879       {  if (len == 255) break;
       
   880          name[len++] = *t++;
       
   881       }
       
   882       name[len] = '\0';
       
   883       if (len == 255) strcpy(name+252, "...");
       
   884       xassert(strlen(name) <= 255);
       
   885       return name;
       
   886 }
       
   887 
       
   888 /*----------------------------------------------------------------------
       
   889 -- mpl_get_row_kind - determine row kind.
       
   890 --
       
   891 -- *Synopsis*
       
   892 --
       
   893 -- #include "glpmpl.h"
       
   894 -- int mpl_get_row_kind(MPL *mpl, int i);
       
   895 --
       
   896 -- *Returns*
       
   897 --
       
   898 -- The routine mpl_get_row_kind returns the kind of i-th row, which can
       
   899 -- be one of the following:
       
   900 --
       
   901 -- MPL_ST  - non-free (constraint) row;
       
   902 -- MPL_MIN - free (objective) row to be minimized;
       
   903 -- MPL_MAX - free (objective) row to be maximized. */
       
   904 
       
   905 int mpl_get_row_kind(MPL *mpl, int i)
       
   906 {     int kind;
       
   907       if (mpl->phase != 3)
       
   908          xfault("mpl_get_row_kind: invalid call sequence\n");
       
   909       if (!(1 <= i && i <= mpl->m))
       
   910          xfault("mpl_get_row_kind: i = %d; row number out of range\n",
       
   911             i);
       
   912       switch (mpl->row[i]->con->type)
       
   913       {  case A_CONSTRAINT:
       
   914             kind = MPL_ST; break;
       
   915          case A_MINIMIZE:
       
   916             kind = MPL_MIN; break;
       
   917          case A_MAXIMIZE:
       
   918             kind = MPL_MAX; break;
       
   919          default:
       
   920             xassert(mpl != mpl);
       
   921       }
       
   922       return kind;
       
   923 }
       
   924 
       
   925 /*----------------------------------------------------------------------
       
   926 -- mpl_get_row_bnds - obtain row bounds.
       
   927 --
       
   928 -- *Synopsis*
       
   929 --
       
   930 -- #include "glpmpl.h"
       
   931 -- int mpl_get_row_bnds(MPL *mpl, int i, double *lb, double *ub);
       
   932 --
       
   933 -- *Description*
       
   934 --
       
   935 -- The routine mpl_get_row_bnds stores lower and upper bounds of i-th
       
   936 -- row of the problem to the locations, which the parameters lb and ub
       
   937 -- point to, respectively. Besides the routine returns the type of the
       
   938 -- i-th row.
       
   939 --
       
   940 -- If some of the parameters lb and ub is NULL, the corresponding bound
       
   941 -- value is not stored.
       
   942 --
       
   943 -- Types and bounds have the following meaning:
       
   944 --
       
   945 --     Type           Bounds          Note
       
   946 --    -----------------------------------------------------------
       
   947 --    MPL_FR   -inf <  f(x) <  +inf   Free linear form
       
   948 --    MPL_LO     lb <= f(x) <  +inf   Inequality f(x) >= lb
       
   949 --    MPL_UP   -inf <  f(x) <=  ub    Inequality f(x) <= ub
       
   950 --    MPL_DB     lb <= f(x) <=  ub    Inequality lb <= f(x) <= ub
       
   951 --    MPL_FX           f(x)  =  lb    Equality f(x) = lb
       
   952 --
       
   953 -- where f(x) is the corresponding linear form of the i-th row.
       
   954 --
       
   955 -- If the row has no lower bound, *lb is set to zero; if the row has
       
   956 -- no upper bound, *ub is set to zero; and if the row is of fixed type,
       
   957 -- both *lb and *ub are set to the same value.
       
   958 --
       
   959 -- *Returns*
       
   960 --
       
   961 -- The routine returns the type of the i-th row as it is stated in the
       
   962 -- table above. */
       
   963 
       
   964 int mpl_get_row_bnds(MPL *mpl, int i, double *_lb, double *_ub)
       
   965 {     ELEMCON *con;
       
   966       int type;
       
   967       double lb, ub;
       
   968       if (mpl->phase != 3)
       
   969          xfault("mpl_get_row_bnds: invalid call sequence\n");
       
   970       if (!(1 <= i && i <= mpl->m))
       
   971          xfault("mpl_get_row_bnds: i = %d; row number out of range\n",
       
   972             i);
       
   973       con = mpl->row[i];
       
   974 #if 0 /* 21/VII-2006 */
       
   975       if (con->con->lbnd == NULL && con->con->ubnd == NULL)
       
   976          type = MPL_FR, lb = ub = 0.0;
       
   977       else if (con->con->ubnd == NULL)
       
   978          type = MPL_LO, lb = con->lbnd, ub = 0.0;
       
   979       else if (con->con->lbnd == NULL)
       
   980          type = MPL_UP, lb = 0.0, ub = con->ubnd;
       
   981       else if (con->con->lbnd != con->con->ubnd)
       
   982          type = MPL_DB, lb = con->lbnd, ub = con->ubnd;
       
   983       else
       
   984          type = MPL_FX, lb = ub = con->lbnd;
       
   985 #else
       
   986       lb = (con->con->lbnd == NULL ? -DBL_MAX : con->lbnd);
       
   987       ub = (con->con->ubnd == NULL ? +DBL_MAX : con->ubnd);
       
   988       if (lb == -DBL_MAX && ub == +DBL_MAX)
       
   989          type = MPL_FR, lb = ub = 0.0;
       
   990       else if (ub == +DBL_MAX)
       
   991          type = MPL_LO, ub = 0.0;
       
   992       else if (lb == -DBL_MAX)
       
   993          type = MPL_UP, lb = 0.0;
       
   994       else if (con->con->lbnd != con->con->ubnd)
       
   995          type = MPL_DB;
       
   996       else
       
   997          type = MPL_FX;
       
   998 #endif
       
   999       if (_lb != NULL) *_lb = lb;
       
  1000       if (_ub != NULL) *_ub = ub;
       
  1001       return type;
       
  1002 }
       
  1003 
       
  1004 /*----------------------------------------------------------------------
       
  1005 -- mpl_get_mat_row - obtain row of the constraint matrix.
       
  1006 --
       
  1007 -- *Synopsis*
       
  1008 --
       
  1009 -- #include "glpmpl.h"
       
  1010 -- int mpl_get_mat_row(MPL *mpl, int i, int ndx[], double val[]);
       
  1011 --
       
  1012 -- *Description*
       
  1013 --
       
  1014 -- The routine mpl_get_mat_row stores column indices and numeric values
       
  1015 -- of constraint coefficients for the i-th row to locations ndx[1], ...,
       
  1016 -- ndx[len] and val[1], ..., val[len], respectively, where 0 <= len <= n
       
  1017 -- is number of (structural) non-zero constraint coefficients, and n is
       
  1018 -- number of columns in the problem.
       
  1019 --
       
  1020 -- If the parameter ndx is NULL, column indices are not stored. If the
       
  1021 -- parameter val is NULL, numeric values are not stored.
       
  1022 --
       
  1023 -- Note that free rows may have constant terms, which are not part of
       
  1024 -- the constraint matrix and therefore not reported by this routine. The
       
  1025 -- constant term of a particular row can be obtained, if necessary, via
       
  1026 -- the routine mpl_get_row_c0.
       
  1027 --
       
  1028 -- *Returns*
       
  1029 --
       
  1030 -- The routine mpl_get_mat_row returns len, which is length of i-th row
       
  1031 -- of the constraint matrix (i.e. number of non-zero coefficients). */
       
  1032 
       
  1033 int mpl_get_mat_row(MPL *mpl, int i, int ndx[], double val[])
       
  1034 {     FORMULA *term;
       
  1035       int len = 0;
       
  1036       if (mpl->phase != 3)
       
  1037          xfault("mpl_get_mat_row: invalid call sequence\n");
       
  1038       if (!(1 <= i && i <= mpl->m))
       
  1039          xfault("mpl_get_mat_row: i = %d; row number out of range\n",
       
  1040             i);
       
  1041       for (term = mpl->row[i]->form; term != NULL; term = term->next)
       
  1042       {  xassert(term->var != NULL);
       
  1043          len++;
       
  1044          xassert(len <= mpl->n);
       
  1045          if (ndx != NULL) ndx[len] = term->var->j;
       
  1046          if (val != NULL) val[len] = term->coef;
       
  1047       }
       
  1048       return len;
       
  1049 }
       
  1050 
       
  1051 /*----------------------------------------------------------------------
       
  1052 -- mpl_get_row_c0 - obtain constant term of free row.
       
  1053 --
       
  1054 -- *Synopsis*
       
  1055 --
       
  1056 -- #include "glpmpl.h"
       
  1057 -- double mpl_get_row_c0(MPL *mpl, int i);
       
  1058 --
       
  1059 -- *Returns*
       
  1060 --
       
  1061 -- The routine mpl_get_row_c0 returns numeric value of constant term of
       
  1062 -- i-th row.
       
  1063 --
       
  1064 -- Note that only free rows may have non-zero constant terms. Therefore
       
  1065 -- if i-th row is not free, the routine returns zero. */
       
  1066 
       
  1067 double mpl_get_row_c0(MPL *mpl, int i)
       
  1068 {     ELEMCON *con;
       
  1069       double c0;
       
  1070       if (mpl->phase != 3)
       
  1071          xfault("mpl_get_row_c0: invalid call sequence\n");
       
  1072       if (!(1 <= i && i <= mpl->m))
       
  1073          xfault("mpl_get_row_c0: i = %d; row number out of range\n",
       
  1074             i);
       
  1075       con = mpl->row[i];
       
  1076       if (con->con->lbnd == NULL && con->con->ubnd == NULL)
       
  1077          c0 = - con->lbnd;
       
  1078       else
       
  1079          c0 = 0.0;
       
  1080       return c0;
       
  1081 }
       
  1082 
       
  1083 /*----------------------------------------------------------------------
       
  1084 -- mpl_get_col_name - obtain column name.
       
  1085 --
       
  1086 -- *Synopsis*
       
  1087 --
       
  1088 -- #include "glpmpl.h"
       
  1089 -- char *mpl_get_col_name(MPL *mpl, int j);
       
  1090 --
       
  1091 -- *Returns*
       
  1092 --
       
  1093 -- The routine mpl_get_col_name returns a pointer to internal buffer,
       
  1094 -- which contains symbolic name of j-th column of the problem. */
       
  1095 
       
  1096 char *mpl_get_col_name(MPL *mpl, int j)
       
  1097 {     char *name = mpl->mpl_buf, *t;
       
  1098       int len;
       
  1099       if (mpl->phase != 3)
       
  1100          xfault("mpl_get_col_name: invalid call sequence\n");
       
  1101       if (!(1 <= j && j <= mpl->n))
       
  1102          xfault("mpl_get_col_name: j = %d; column number out of range\n"
       
  1103             , j);
       
  1104       strcpy(name, mpl->col[j]->var->name);
       
  1105       len = strlen(name);
       
  1106       xassert(len <= 255);
       
  1107       t = format_tuple(mpl, '[', mpl->col[j]->memb->tuple);
       
  1108       while (*t)
       
  1109       {  if (len == 255) break;
       
  1110          name[len++] = *t++;
       
  1111       }
       
  1112       name[len] = '\0';
       
  1113       if (len == 255) strcpy(name+252, "...");
       
  1114       xassert(strlen(name) <= 255);
       
  1115       return name;
       
  1116 }
       
  1117 
       
  1118 /*----------------------------------------------------------------------
       
  1119 -- mpl_get_col_kind - determine column kind.
       
  1120 --
       
  1121 -- *Synopsis*
       
  1122 --
       
  1123 -- #include "glpmpl.h"
       
  1124 -- int mpl_get_col_kind(MPL *mpl, int j);
       
  1125 --
       
  1126 -- *Returns*
       
  1127 --
       
  1128 -- The routine mpl_get_col_kind returns the kind of j-th column, which
       
  1129 -- can be one of the following:
       
  1130 --
       
  1131 -- MPL_NUM - continuous variable;
       
  1132 -- MPL_INT - integer variable;
       
  1133 -- MPL_BIN - binary variable.
       
  1134 --
       
  1135 -- Note that column kinds are defined independently on type and bounds
       
  1136 -- (reported by the routine mpl_get_col_bnds) of corresponding columns.
       
  1137 -- This means, in particular, that bounds of an integer column may be
       
  1138 -- fractional, or a binary column may have lower and upper bounds that
       
  1139 -- are not 0 and 1 (or it may have no lower/upper bound at all). */
       
  1140 
       
  1141 int mpl_get_col_kind(MPL *mpl, int j)
       
  1142 {     int kind;
       
  1143       if (mpl->phase != 3)
       
  1144          xfault("mpl_get_col_kind: invalid call sequence\n");
       
  1145       if (!(1 <= j && j <= mpl->n))
       
  1146          xfault("mpl_get_col_kind: j = %d; column number out of range\n"
       
  1147             , j);
       
  1148       switch (mpl->col[j]->var->type)
       
  1149       {  case A_NUMERIC:
       
  1150             kind = MPL_NUM; break;
       
  1151          case A_INTEGER:
       
  1152             kind = MPL_INT; break;
       
  1153          case A_BINARY:
       
  1154             kind = MPL_BIN; break;
       
  1155          default:
       
  1156             xassert(mpl != mpl);
       
  1157       }
       
  1158       return kind;
       
  1159 }
       
  1160 
       
  1161 /*----------------------------------------------------------------------
       
  1162 -- mpl_get_col_bnds - obtain column bounds.
       
  1163 --
       
  1164 -- *Synopsis*
       
  1165 --
       
  1166 -- #include "glpmpl.h"
       
  1167 -- int mpl_get_col_bnds(MPL *mpl, int j, double *lb, double *ub);
       
  1168 --
       
  1169 -- *Description*
       
  1170 --
       
  1171 -- The routine mpl_get_col_bnds stores lower and upper bound of j-th
       
  1172 -- column of the problem to the locations, which the parameters lb and
       
  1173 -- ub point to, respectively. Besides the routine returns the type of
       
  1174 -- the j-th column.
       
  1175 --
       
  1176 -- If some of the parameters lb and ub is NULL, the corresponding bound
       
  1177 -- value is not stored.
       
  1178 --
       
  1179 -- Types and bounds have the following meaning:
       
  1180 --
       
  1181 --     Type         Bounds         Note
       
  1182 --    ------------------------------------------------------
       
  1183 --    MPL_FR   -inf <  x <  +inf   Free (unbounded) variable
       
  1184 --    MPL_LO     lb <= x <  +inf   Variable with lower bound
       
  1185 --    MPL_UP   -inf <  x <=  ub    Variable with upper bound
       
  1186 --    MPL_DB     lb <= x <=  ub    Double-bounded variable
       
  1187 --    MPL_FX           x  =  lb    Fixed variable
       
  1188 --
       
  1189 -- where x is individual variable corresponding to the j-th column.
       
  1190 --
       
  1191 -- If the column has no lower bound, *lb is set to zero; if the column
       
  1192 -- has no upper bound, *ub is set to zero; and if the column is of fixed
       
  1193 -- type, both *lb and *ub are set to the same value.
       
  1194 --
       
  1195 -- *Returns*
       
  1196 --
       
  1197 -- The routine returns the type of the j-th column as it is stated in
       
  1198 -- the table above. */
       
  1199 
       
  1200 int mpl_get_col_bnds(MPL *mpl, int j, double *_lb, double *_ub)
       
  1201 {     ELEMVAR *var;
       
  1202       int type;
       
  1203       double lb, ub;
       
  1204       if (mpl->phase != 3)
       
  1205          xfault("mpl_get_col_bnds: invalid call sequence\n");
       
  1206       if (!(1 <= j && j <= mpl->n))
       
  1207          xfault("mpl_get_col_bnds: j = %d; column number out of range\n"
       
  1208             , j);
       
  1209       var = mpl->col[j];
       
  1210 #if 0 /* 21/VII-2006 */
       
  1211       if (var->var->lbnd == NULL && var->var->ubnd == NULL)
       
  1212          type = MPL_FR, lb = ub = 0.0;
       
  1213       else if (var->var->ubnd == NULL)
       
  1214          type = MPL_LO, lb = var->lbnd, ub = 0.0;
       
  1215       else if (var->var->lbnd == NULL)
       
  1216          type = MPL_UP, lb = 0.0, ub = var->ubnd;
       
  1217       else if (var->var->lbnd != var->var->ubnd)
       
  1218          type = MPL_DB, lb = var->lbnd, ub = var->ubnd;
       
  1219       else
       
  1220          type = MPL_FX, lb = ub = var->lbnd;
       
  1221 #else
       
  1222       lb = (var->var->lbnd == NULL ? -DBL_MAX : var->lbnd);
       
  1223       ub = (var->var->ubnd == NULL ? +DBL_MAX : var->ubnd);
       
  1224       if (lb == -DBL_MAX && ub == +DBL_MAX)
       
  1225          type = MPL_FR, lb = ub = 0.0;
       
  1226       else if (ub == +DBL_MAX)
       
  1227          type = MPL_LO, ub = 0.0;
       
  1228       else if (lb == -DBL_MAX)
       
  1229          type = MPL_UP, lb = 0.0;
       
  1230       else if (var->var->lbnd != var->var->ubnd)
       
  1231          type = MPL_DB;
       
  1232       else
       
  1233          type = MPL_FX;
       
  1234 #endif
       
  1235       if (_lb != NULL) *_lb = lb;
       
  1236       if (_ub != NULL) *_ub = ub;
       
  1237       return type;
       
  1238 }
       
  1239 
       
  1240 /*----------------------------------------------------------------------
       
  1241 -- mpl_has_solve_stmt - check if model has solve statement.
       
  1242 --
       
  1243 -- *Synopsis*
       
  1244 --
       
  1245 -- #include "glpmpl.h"
       
  1246 -- int mpl_has_solve_stmt(MPL *mpl);
       
  1247 --
       
  1248 -- *Returns*
       
  1249 --
       
  1250 -- If the model has the solve statement, the routine returns non-zero,
       
  1251 -- otherwise zero is returned. */
       
  1252 
       
  1253 int mpl_has_solve_stmt(MPL *mpl)
       
  1254 {     if (mpl->phase != 3)
       
  1255          xfault("mpl_has_solve_stmt: invalid call sequence\n");
       
  1256       return mpl->flag_s;
       
  1257 }
       
  1258 
       
  1259 #if 1 /* 15/V-2010 */
       
  1260 void mpl_put_row_soln(MPL *mpl, int i, int stat, double prim,
       
  1261       double dual)
       
  1262 {     /* store row (constraint/objective) solution components */
       
  1263       xassert(mpl->phase == 3);
       
  1264       xassert(1 <= i && i <= mpl->m);
       
  1265       mpl->row[i]->stat = stat;
       
  1266       mpl->row[i]->prim = prim;
       
  1267       mpl->row[i]->dual = dual;
       
  1268       return;
       
  1269 }
       
  1270 #endif
       
  1271 
       
  1272 #if 1 /* 15/V-2010 */
       
  1273 void mpl_put_col_soln(MPL *mpl, int j, int stat, double prim,
       
  1274       double dual)
       
  1275 {     /* store column (variable) solution components */
       
  1276       xassert(mpl->phase == 3);
       
  1277       xassert(1 <= j && j <= mpl->n);
       
  1278       mpl->col[j]->stat = stat;
       
  1279       mpl->col[j]->prim = prim;
       
  1280       mpl->col[j]->dual = dual;
       
  1281       return;
       
  1282 }
       
  1283 #endif
       
  1284 
       
  1285 #if 0 /* 15/V-2010 */
       
  1286 /*----------------------------------------------------------------------
       
  1287 -- mpl_put_col_value - store column value.
       
  1288 --
       
  1289 -- *Synopsis*
       
  1290 --
       
  1291 -- #include "glpmpl.h"
       
  1292 -- void mpl_put_col_value(MPL *mpl, int j, double val);
       
  1293 --
       
  1294 -- *Description*
       
  1295 --
       
  1296 -- The routine mpl_put_col_value stores numeric value of j-th column
       
  1297 -- into the translator database. It is assumed that the column value is
       
  1298 -- provided by the solver. */
       
  1299 
       
  1300 void mpl_put_col_value(MPL *mpl, int j, double val)
       
  1301 {     if (mpl->phase != 3)
       
  1302          xfault("mpl_put_col_value: invalid call sequence\n");
       
  1303       if (!(1 <= j && j <= mpl->n))
       
  1304          xfault(
       
  1305          "mpl_put_col_value: j = %d; column number out of range\n", j);
       
  1306       mpl->col[j]->prim = val;
       
  1307       return;
       
  1308 }
       
  1309 #endif
       
  1310 
       
  1311 /*----------------------------------------------------------------------
       
  1312 -- mpl_postsolve - postsolve model.
       
  1313 --
       
  1314 -- *Synopsis*
       
  1315 --
       
  1316 -- #include "glpmpl.h"
       
  1317 -- int mpl_postsolve(MPL *mpl);
       
  1318 --
       
  1319 -- *Description*
       
  1320 --
       
  1321 -- The routine mpl_postsolve performs postsolving of the model using
       
  1322 -- its description stored in the translator database. This phase means
       
  1323 -- executing statements, which follow the solve statement.
       
  1324 --
       
  1325 -- If this routine is used, it should be called once after the routine
       
  1326 -- mpl_generate and if the latter returned the code 3.
       
  1327 --
       
  1328 -- *Returns*
       
  1329 --
       
  1330 -- The routine mpl_postsolve returns one of the following codes:
       
  1331 --
       
  1332 -- 3 - model has been successfully postsolved.
       
  1333 -- 4 - processing failed due to some errors. In this case the calling
       
  1334 --     program should call the routine mpl_terminate to terminate model
       
  1335 --     processing. */
       
  1336 
       
  1337 int mpl_postsolve(MPL *mpl)
       
  1338 {     if (!(mpl->phase == 3 && !mpl->flag_p))
       
  1339          xfault("mpl_postsolve: invalid call sequence\n");
       
  1340       /* set up error handler */
       
  1341       if (setjmp(mpl->jump)) goto done;
       
  1342       /* perform postsolving */
       
  1343       postsolve_model(mpl);
       
  1344       flush_output(mpl);
       
  1345       /* postsolving phase has been finished */
       
  1346       xprintf("Model has been successfully processed\n");
       
  1347 done: /* return to the calling program */
       
  1348       return mpl->phase;
       
  1349 }
       
  1350 
       
  1351 /*----------------------------------------------------------------------
       
  1352 -- mpl_terminate - free all resources used by translator.
       
  1353 --
       
  1354 -- *Synopsis*
       
  1355 --
       
  1356 -- #include "glpmpl.h"
       
  1357 -- void mpl_terminate(MPL *mpl);
       
  1358 --
       
  1359 -- *Description*
       
  1360 --
       
  1361 -- The routine mpl_terminate frees all the resources used by the GNU
       
  1362 -- MathProg translator. */
       
  1363 
       
  1364 void mpl_terminate(MPL *mpl)
       
  1365 {     if (setjmp(mpl->jump)) xassert(mpl != mpl);
       
  1366       switch (mpl->phase)
       
  1367       {  case 0:
       
  1368          case 1:
       
  1369          case 2:
       
  1370          case 3:
       
  1371             /* there were no errors; clean the model content */
       
  1372             clean_model(mpl);
       
  1373             xassert(mpl->a_list == NULL);
       
  1374 #if 1 /* 11/II-2008 */
       
  1375             xassert(mpl->dca == NULL);
       
  1376 #endif
       
  1377             break;
       
  1378          case 4:
       
  1379             /* model processing has been finished due to error; delete
       
  1380                search trees, which may be created for some arrays */
       
  1381             {  ARRAY *a;
       
  1382                for (a = mpl->a_list; a != NULL; a = a->next)
       
  1383                   if (a->tree != NULL) avl_delete_tree(a->tree);
       
  1384             }
       
  1385 #if 1 /* 11/II-2008 */
       
  1386             free_dca(mpl);
       
  1387 #endif
       
  1388             break;
       
  1389          default:
       
  1390             xassert(mpl != mpl);
       
  1391       }
       
  1392       /* delete the translator database */
       
  1393       xfree(mpl->image);
       
  1394       xfree(mpl->b_image);
       
  1395       xfree(mpl->f_image);
       
  1396       xfree(mpl->context);
       
  1397       dmp_delete_pool(mpl->pool);
       
  1398       avl_delete_tree(mpl->tree);
       
  1399       dmp_delete_pool(mpl->strings);
       
  1400       dmp_delete_pool(mpl->symbols);
       
  1401       dmp_delete_pool(mpl->tuples);
       
  1402       dmp_delete_pool(mpl->arrays);
       
  1403       dmp_delete_pool(mpl->members);
       
  1404       dmp_delete_pool(mpl->elemvars);
       
  1405       dmp_delete_pool(mpl->formulae);
       
  1406       dmp_delete_pool(mpl->elemcons);
       
  1407       xfree(mpl->sym_buf);
       
  1408       xfree(mpl->tup_buf);
       
  1409       rng_delete_rand(mpl->rand);
       
  1410       if (mpl->row != NULL) xfree(mpl->row);
       
  1411       if (mpl->col != NULL) xfree(mpl->col);
       
  1412       if (mpl->in_fp != NULL) xfclose(mpl->in_fp);
       
  1413       if (mpl->out_fp != NULL && mpl->out_fp != (void *)stdout)
       
  1414          xfclose(mpl->out_fp);
       
  1415       if (mpl->out_file != NULL) xfree(mpl->out_file);
       
  1416       if (mpl->prt_fp != NULL) xfclose(mpl->prt_fp);
       
  1417       if (mpl->prt_file != NULL) xfree(mpl->prt_file);
       
  1418       if (mpl->mod_file != NULL) xfree(mpl->mod_file);
       
  1419       xfree(mpl->mpl_buf);
       
  1420       xfree(mpl);
       
  1421       return;
       
  1422 }
       
  1423 
       
  1424 /* eof */