src/glpmpl01.c
author Alpar Juttner <alpar@cs.elte.hu>
Sun, 05 Dec 2010 17:35:23 +0100
changeset 2 4c8956a7bdf4
permissions -rw-r--r--
Set up CMAKE build environment
     1 /* glpmpl01.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_STDIO
    26 #include "glpmpl.h"
    27 #define dmp_get_atomv dmp_get_atom
    28 
    29 /**********************************************************************/
    30 /* * *                  PROCESSING MODEL SECTION                  * * */
    31 /**********************************************************************/
    32 
    33 /*----------------------------------------------------------------------
    34 -- enter_context - enter current token into context queue.
    35 --
    36 -- This routine enters the current token into the context queue. */
    37 
    38 void enter_context(MPL *mpl)
    39 {     char *image, *s;
    40       if (mpl->token == T_EOF)
    41          image = "_|_";
    42       else if (mpl->token == T_STRING)
    43          image = "'...'";
    44       else
    45          image = mpl->image;
    46       xassert(0 <= mpl->c_ptr && mpl->c_ptr < CONTEXT_SIZE);
    47       mpl->context[mpl->c_ptr++] = ' ';
    48       if (mpl->c_ptr == CONTEXT_SIZE) mpl->c_ptr = 0;
    49       for (s = image; *s != '\0'; s++)
    50       {  mpl->context[mpl->c_ptr++] = *s;
    51          if (mpl->c_ptr == CONTEXT_SIZE) mpl->c_ptr = 0;
    52       }
    53       return;
    54 }
    55 
    56 /*----------------------------------------------------------------------
    57 -- print_context - print current content of context queue.
    58 --
    59 -- This routine prints current content of the context queue. */
    60 
    61 void print_context(MPL *mpl)
    62 {     int c;
    63       while (mpl->c_ptr > 0)
    64       {  mpl->c_ptr--;
    65          c = mpl->context[0];
    66          memmove(mpl->context, mpl->context+1, CONTEXT_SIZE-1);
    67          mpl->context[CONTEXT_SIZE-1] = (char)c;
    68       }
    69       xprintf("Context: %s%.*s\n", mpl->context[0] == ' ' ? "" : "...",
    70          CONTEXT_SIZE, mpl->context);
    71       return;
    72 }
    73 
    74 /*----------------------------------------------------------------------
    75 -- get_char - scan next character from input text file.
    76 --
    77 -- This routine scans a next ASCII character from the input text file.
    78 -- In case of end-of-file, the character is assigned EOF. */
    79 
    80 void get_char(MPL *mpl)
    81 {     int c;
    82       if (mpl->c == EOF) goto done;
    83       if (mpl->c == '\n') mpl->line++;
    84       c = read_char(mpl);
    85       if (c == EOF)
    86       {  if (mpl->c == '\n')
    87             mpl->line--;
    88          else
    89             warning(mpl, "final NL missing before end of file");
    90       }
    91       else if (c == '\n')
    92          ;
    93       else if (isspace(c))
    94          c = ' ';
    95       else if (iscntrl(c))
    96       {  enter_context(mpl);
    97          error(mpl, "control character 0x%02X not allowed", c);
    98       }
    99       mpl->c = c;
   100 done: return;
   101 }
   102 
   103 /*----------------------------------------------------------------------
   104 -- append_char - append character to current token.
   105 --
   106 -- This routine appends the current character to the current token and
   107 -- then scans a next character. */
   108 
   109 void append_char(MPL *mpl)
   110 {     xassert(0 <= mpl->imlen && mpl->imlen <= MAX_LENGTH);
   111       if (mpl->imlen == MAX_LENGTH)
   112       {  switch (mpl->token)
   113          {  case T_NAME:
   114                enter_context(mpl);
   115                error(mpl, "symbolic name %s... too long", mpl->image);
   116             case T_SYMBOL:
   117                enter_context(mpl);
   118                error(mpl, "symbol %s... too long", mpl->image);
   119             case T_NUMBER:
   120                enter_context(mpl);
   121                error(mpl, "numeric literal %s... too long", mpl->image);
   122             case T_STRING:
   123                enter_context(mpl);
   124                error(mpl, "string literal too long");
   125             default:
   126                xassert(mpl != mpl);
   127          }
   128       }
   129       mpl->image[mpl->imlen++] = (char)mpl->c;
   130       mpl->image[mpl->imlen] = '\0';
   131       get_char(mpl);
   132       return;
   133 }
   134 
   135 /*----------------------------------------------------------------------
   136 -- get_token - scan next token from input text file.
   137 --
   138 -- This routine scans a next token from the input text file using the
   139 -- standard finite automation technique. */
   140 
   141 void get_token(MPL *mpl)
   142 {     /* save the current token */
   143       mpl->b_token = mpl->token;
   144       mpl->b_imlen = mpl->imlen;
   145       strcpy(mpl->b_image, mpl->image);
   146       mpl->b_value = mpl->value;
   147       /* if the next token is already scanned, make it current */
   148       if (mpl->f_scan)
   149       {  mpl->f_scan = 0;
   150          mpl->token = mpl->f_token;
   151          mpl->imlen = mpl->f_imlen;
   152          strcpy(mpl->image, mpl->f_image);
   153          mpl->value = mpl->f_value;
   154          goto done;
   155       }
   156 loop: /* nothing has been scanned so far */
   157       mpl->token = 0;
   158       mpl->imlen = 0;
   159       mpl->image[0] = '\0';
   160       mpl->value = 0.0;
   161       /* skip any uninteresting characters */
   162       while (mpl->c == ' ' || mpl->c == '\n') get_char(mpl);
   163       /* recognize and construct the token */
   164       if (mpl->c == EOF)
   165       {  /* end-of-file reached */
   166          mpl->token = T_EOF;
   167       }
   168       else if (mpl->c == '#')
   169       {  /* comment; skip anything until end-of-line */
   170          while (mpl->c != '\n' && mpl->c != EOF) get_char(mpl);
   171          goto loop;
   172       }
   173       else if (!mpl->flag_d && (isalpha(mpl->c) || mpl->c == '_'))
   174       {  /* symbolic name or reserved keyword */
   175          mpl->token = T_NAME;
   176          while (isalnum(mpl->c) || mpl->c == '_') append_char(mpl);
   177          if (strcmp(mpl->image, "and") == 0)
   178             mpl->token = T_AND;
   179          else if (strcmp(mpl->image, "by") == 0)
   180             mpl->token = T_BY;
   181          else if (strcmp(mpl->image, "cross") == 0)
   182             mpl->token = T_CROSS;
   183          else if (strcmp(mpl->image, "diff") == 0)
   184             mpl->token = T_DIFF;
   185          else if (strcmp(mpl->image, "div") == 0)
   186             mpl->token = T_DIV;
   187          else if (strcmp(mpl->image, "else") == 0)
   188             mpl->token = T_ELSE;
   189          else if (strcmp(mpl->image, "if") == 0)
   190             mpl->token = T_IF;
   191          else if (strcmp(mpl->image, "in") == 0)
   192             mpl->token = T_IN;
   193 #if 1 /* 21/VII-2006 */
   194          else if (strcmp(mpl->image, "Infinity") == 0)
   195             mpl->token = T_INFINITY;
   196 #endif
   197          else if (strcmp(mpl->image, "inter") == 0)
   198             mpl->token = T_INTER;
   199          else if (strcmp(mpl->image, "less") == 0)
   200             mpl->token = T_LESS;
   201          else if (strcmp(mpl->image, "mod") == 0)
   202             mpl->token = T_MOD;
   203          else if (strcmp(mpl->image, "not") == 0)
   204             mpl->token = T_NOT;
   205          else if (strcmp(mpl->image, "or") == 0)
   206             mpl->token = T_OR;
   207          else if (strcmp(mpl->image, "s") == 0 && mpl->c == '.')
   208          {  mpl->token = T_SPTP;
   209             append_char(mpl);
   210             if (mpl->c != 't')
   211 sptp:       {  enter_context(mpl);
   212                error(mpl, "keyword s.t. incomplete");
   213             }
   214             append_char(mpl);
   215             if (mpl->c != '.') goto sptp;
   216             append_char(mpl);
   217          }
   218          else if (strcmp(mpl->image, "symdiff") == 0)
   219             mpl->token = T_SYMDIFF;
   220          else if (strcmp(mpl->image, "then") == 0)
   221             mpl->token = T_THEN;
   222          else if (strcmp(mpl->image, "union") == 0)
   223             mpl->token = T_UNION;
   224          else if (strcmp(mpl->image, "within") == 0)
   225             mpl->token = T_WITHIN;
   226       }
   227       else if (!mpl->flag_d && isdigit(mpl->c))
   228       {  /* numeric literal */
   229          mpl->token = T_NUMBER;
   230          /* scan integer part */
   231          while (isdigit(mpl->c)) append_char(mpl);
   232          /* scan optional fractional part */
   233          if (mpl->c == '.')
   234          {  append_char(mpl);
   235             if (mpl->c == '.')
   236             {  /* hmm, it is not the fractional part, it is dots that
   237                   follow the integer part */
   238                mpl->imlen--;
   239                mpl->image[mpl->imlen] = '\0';
   240                mpl->f_dots = 1;
   241                goto conv;
   242             }
   243 frac:       while (isdigit(mpl->c)) append_char(mpl);
   244          }
   245          /* scan optional decimal exponent */
   246          if (mpl->c == 'e' || mpl->c == 'E')
   247          {  append_char(mpl);
   248             if (mpl->c == '+' || mpl->c == '-') append_char(mpl);
   249             if (!isdigit(mpl->c))
   250             {  enter_context(mpl);
   251                error(mpl, "numeric literal %s incomplete", mpl->image);
   252             }
   253             while (isdigit(mpl->c)) append_char(mpl);
   254          }
   255          /* there must be no letter following the numeric literal */
   256          if (isalpha(mpl->c) || mpl->c == '_')
   257          {  enter_context(mpl);
   258             error(mpl, "symbol %s%c... should be enclosed in quotes",
   259                mpl->image, mpl->c);
   260          }
   261 conv:    /* convert numeric literal to floating-point */
   262          if (str2num(mpl->image, &mpl->value))
   263 err:     {  enter_context(mpl);
   264             error(mpl, "cannot convert numeric literal %s to floating-p"
   265                "oint number", mpl->image);
   266          }
   267       }
   268       else if (mpl->c == '\'' || mpl->c == '"')
   269       {  /* character string */
   270          int quote = mpl->c;
   271          mpl->token = T_STRING;
   272          get_char(mpl);
   273          for (;;)
   274          {  if (mpl->c == '\n' || mpl->c == EOF)
   275             {  enter_context(mpl);
   276                error(mpl, "unexpected end of line; string literal incom"
   277                   "plete");
   278             }
   279             if (mpl->c == quote)
   280             {  get_char(mpl);
   281                if (mpl->c != quote) break;
   282             }
   283             append_char(mpl);
   284          }
   285       }
   286       else if (!mpl->flag_d && mpl->c == '+')
   287          mpl->token = T_PLUS, append_char(mpl);
   288       else if (!mpl->flag_d && mpl->c == '-')
   289          mpl->token = T_MINUS, append_char(mpl);
   290       else if (mpl->c == '*')
   291       {  mpl->token = T_ASTERISK, append_char(mpl);
   292          if (mpl->c == '*')
   293             mpl->token = T_POWER, append_char(mpl);
   294       }
   295       else if (mpl->c == '/')
   296       {  mpl->token = T_SLASH, append_char(mpl);
   297          if (mpl->c == '*')
   298          {  /* comment sequence */
   299             get_char(mpl);
   300             for (;;)
   301             {  if (mpl->c == EOF)
   302                {  /* do not call enter_context at this point */
   303                   error(mpl, "unexpected end of file; comment sequence "
   304                      "incomplete");
   305                }
   306                else if (mpl->c == '*')
   307                {  get_char(mpl);
   308                   if (mpl->c == '/') break;
   309                }
   310                else
   311                   get_char(mpl);
   312             }
   313             get_char(mpl);
   314             goto loop;
   315          }
   316       }
   317       else if (mpl->c == '^')
   318          mpl->token = T_POWER, append_char(mpl);
   319       else if (mpl->c == '<')
   320       {  mpl->token = T_LT, append_char(mpl);
   321          if (mpl->c == '=')
   322             mpl->token = T_LE, append_char(mpl);
   323          else if (mpl->c == '>')
   324             mpl->token = T_NE, append_char(mpl);
   325 #if 1 /* 11/II-2008 */
   326          else if (mpl->c == '-')
   327             mpl->token = T_INPUT, append_char(mpl);
   328 #endif
   329       }
   330       else if (mpl->c == '=')
   331       {  mpl->token = T_EQ, append_char(mpl);
   332          if (mpl->c == '=') append_char(mpl);
   333       }
   334       else if (mpl->c == '>')
   335       {  mpl->token = T_GT, append_char(mpl);
   336          if (mpl->c == '=')
   337             mpl->token = T_GE, append_char(mpl);
   338 #if 1 /* 14/VII-2006 */
   339          else if (mpl->c == '>')
   340             mpl->token = T_APPEND, append_char(mpl);
   341 #endif
   342       }
   343       else if (mpl->c == '!')
   344       {  mpl->token = T_NOT, append_char(mpl);
   345          if (mpl->c == '=')
   346             mpl->token = T_NE, append_char(mpl);
   347       }
   348       else if (mpl->c == '&')
   349       {  mpl->token = T_CONCAT, append_char(mpl);
   350          if (mpl->c == '&')
   351             mpl->token = T_AND, append_char(mpl);
   352       }
   353       else if (mpl->c == '|')
   354       {  mpl->token = T_BAR, append_char(mpl);
   355          if (mpl->c == '|')
   356             mpl->token = T_OR, append_char(mpl);
   357       }
   358       else if (!mpl->flag_d && mpl->c == '.')
   359       {  mpl->token = T_POINT, append_char(mpl);
   360          if (mpl->f_dots)
   361          {  /* dots; the first dot was read on the previous call to the
   362                scanner, so the current character is the second dot */
   363             mpl->token = T_DOTS;
   364             mpl->imlen = 2;
   365             strcpy(mpl->image, "..");
   366             mpl->f_dots = 0;
   367          }
   368          else if (mpl->c == '.')
   369             mpl->token = T_DOTS, append_char(mpl);
   370          else if (isdigit(mpl->c))
   371          {  /* numeric literal that begins with the decimal point */
   372             mpl->token = T_NUMBER, append_char(mpl);
   373             goto frac;
   374          }
   375       }
   376       else if (mpl->c == ',')
   377          mpl->token = T_COMMA, append_char(mpl);
   378       else if (mpl->c == ':')
   379       {  mpl->token = T_COLON, append_char(mpl);
   380          if (mpl->c == '=')
   381             mpl->token = T_ASSIGN, append_char(mpl);
   382       }
   383       else if (mpl->c == ';')
   384          mpl->token = T_SEMICOLON, append_char(mpl);
   385       else if (mpl->c == '(')
   386          mpl->token = T_LEFT, append_char(mpl);
   387       else if (mpl->c == ')')
   388          mpl->token = T_RIGHT, append_char(mpl);
   389       else if (mpl->c == '[')
   390          mpl->token = T_LBRACKET, append_char(mpl);
   391       else if (mpl->c == ']')
   392          mpl->token = T_RBRACKET, append_char(mpl);
   393       else if (mpl->c == '{')
   394          mpl->token = T_LBRACE, append_char(mpl);
   395       else if (mpl->c == '}')
   396          mpl->token = T_RBRACE, append_char(mpl);
   397 #if 1 /* 11/II-2008 */
   398       else if (mpl->c == '~')
   399          mpl->token = T_TILDE, append_char(mpl);
   400 #endif
   401       else if (isalnum(mpl->c) || strchr("+-._", mpl->c) != NULL)
   402       {  /* symbol */
   403          xassert(mpl->flag_d);
   404          mpl->token = T_SYMBOL;
   405          while (isalnum(mpl->c) || strchr("+-._", mpl->c) != NULL)
   406             append_char(mpl);
   407          switch (str2num(mpl->image, &mpl->value))
   408          {  case 0:
   409                mpl->token = T_NUMBER;
   410                break;
   411             case 1:
   412                goto err;
   413             case 2:
   414                break;
   415             default:
   416                xassert(mpl != mpl);
   417          }
   418       }
   419       else
   420       {  enter_context(mpl);
   421          error(mpl, "character %c not allowed", mpl->c);
   422       }
   423       /* enter the current token into the context queue */
   424       enter_context(mpl);
   425       /* reset the flag, which may be set by indexing_expression() and
   426          is used by expression_list() */
   427       mpl->flag_x = 0;
   428 done: return;
   429 }
   430 
   431 /*----------------------------------------------------------------------
   432 -- unget_token - return current token back to input stream.
   433 --
   434 -- This routine returns the current token back to the input stream, so
   435 -- the previously scanned token becomes the current one. */
   436 
   437 void unget_token(MPL *mpl)
   438 {     /* save the current token, which becomes the next one */
   439       xassert(!mpl->f_scan);
   440       mpl->f_scan = 1;
   441       mpl->f_token = mpl->token;
   442       mpl->f_imlen = mpl->imlen;
   443       strcpy(mpl->f_image, mpl->image);
   444       mpl->f_value = mpl->value;
   445       /* restore the previous token, which becomes the current one */
   446       mpl->token = mpl->b_token;
   447       mpl->imlen = mpl->b_imlen;
   448       strcpy(mpl->image, mpl->b_image);
   449       mpl->value = mpl->b_value;
   450       return;
   451 }
   452 
   453 /*----------------------------------------------------------------------
   454 -- is_keyword - check if current token is given non-reserved keyword.
   455 --
   456 -- If the current token is given (non-reserved) keyword, this routine
   457 -- returns non-zero. Otherwise zero is returned. */
   458 
   459 int is_keyword(MPL *mpl, char *keyword)
   460 {     return
   461          mpl->token == T_NAME && strcmp(mpl->image, keyword) == 0;
   462 }
   463 
   464 /*----------------------------------------------------------------------
   465 -- is_reserved - check if current token is reserved keyword.
   466 --
   467 -- If the current token is a reserved keyword, this routine returns
   468 -- non-zero. Otherwise zero is returned. */
   469 
   470 int is_reserved(MPL *mpl)
   471 {     return
   472          mpl->token == T_AND && mpl->image[0] == 'a' ||
   473          mpl->token == T_BY ||
   474          mpl->token == T_CROSS ||
   475          mpl->token == T_DIFF ||
   476          mpl->token == T_DIV ||
   477          mpl->token == T_ELSE ||
   478          mpl->token == T_IF ||
   479          mpl->token == T_IN ||
   480          mpl->token == T_INTER ||
   481          mpl->token == T_LESS ||
   482          mpl->token == T_MOD ||
   483          mpl->token == T_NOT && mpl->image[0] == 'n' ||
   484          mpl->token == T_OR && mpl->image[0] == 'o' ||
   485          mpl->token == T_SYMDIFF ||
   486          mpl->token == T_THEN ||
   487          mpl->token == T_UNION ||
   488          mpl->token == T_WITHIN;
   489 }
   490 
   491 /*----------------------------------------------------------------------
   492 -- make_code - generate pseudo-code (basic routine).
   493 --
   494 -- This routine generates specified pseudo-code. It is assumed that all
   495 -- other translator routines use this basic routine. */
   496 
   497 CODE *make_code(MPL *mpl, int op, OPERANDS *arg, int type, int dim)
   498 {     CODE *code;
   499       DOMAIN *domain;
   500       DOMAIN_BLOCK *block;
   501       ARG_LIST *e;
   502       /* generate pseudo-code */
   503       code = alloc(CODE);
   504       code->op = op;
   505       code->vflag = 0; /* is inherited from operand(s) */
   506       /* copy operands and also make them referring to the pseudo-code
   507          being generated, because the latter becomes the parent for all
   508          its operands */
   509       memset(&code->arg, '?', sizeof(OPERANDS));
   510       switch (op)
   511       {  case O_NUMBER:
   512             code->arg.num = arg->num;
   513             break;
   514          case O_STRING:
   515             code->arg.str = arg->str;
   516             break;
   517          case O_INDEX:
   518             code->arg.index.slot = arg->index.slot;
   519             code->arg.index.next = arg->index.next;
   520             break;
   521          case O_MEMNUM:
   522          case O_MEMSYM:
   523             for (e = arg->par.list; e != NULL; e = e->next)
   524             {  xassert(e->x != NULL);
   525                xassert(e->x->up == NULL);
   526                e->x->up = code;
   527                code->vflag |= e->x->vflag;
   528             }
   529             code->arg.par.par = arg->par.par;
   530             code->arg.par.list = arg->par.list;
   531             break;
   532          case O_MEMSET:
   533             for (e = arg->set.list; e != NULL; e = e->next)
   534             {  xassert(e->x != NULL);
   535                xassert(e->x->up == NULL);
   536                e->x->up = code;
   537                code->vflag |= e->x->vflag;
   538             }
   539             code->arg.set.set = arg->set.set;
   540             code->arg.set.list = arg->set.list;
   541             break;
   542          case O_MEMVAR:
   543             for (e = arg->var.list; e != NULL; e = e->next)
   544             {  xassert(e->x != NULL);
   545                xassert(e->x->up == NULL);
   546                e->x->up = code;
   547                code->vflag |= e->x->vflag;
   548             }
   549             code->arg.var.var = arg->var.var;
   550             code->arg.var.list = arg->var.list;
   551 #if 1 /* 15/V-2010 */
   552             code->arg.var.suff = arg->var.suff;
   553 #endif
   554             break;
   555 #if 1 /* 15/V-2010 */
   556          case O_MEMCON:
   557             for (e = arg->con.list; e != NULL; e = e->next)
   558             {  xassert(e->x != NULL);
   559                xassert(e->x->up == NULL);
   560                e->x->up = code;
   561                code->vflag |= e->x->vflag;
   562             }
   563             code->arg.con.con = arg->con.con;
   564             code->arg.con.list = arg->con.list;
   565             code->arg.con.suff = arg->con.suff;
   566             break;
   567 #endif
   568          case O_TUPLE:
   569          case O_MAKE:
   570             for (e = arg->list; e != NULL; e = e->next)
   571             {  xassert(e->x != NULL);
   572                xassert(e->x->up == NULL);
   573                e->x->up = code;
   574                code->vflag |= e->x->vflag;
   575             }
   576             code->arg.list = arg->list;
   577             break;
   578          case O_SLICE:
   579             xassert(arg->slice != NULL);
   580             code->arg.slice = arg->slice;
   581             break;
   582          case O_IRAND224:
   583          case O_UNIFORM01:
   584          case O_NORMAL01:
   585          case O_GMTIME:
   586             code->vflag = 1;
   587             break;
   588          case O_CVTNUM:
   589          case O_CVTSYM:
   590          case O_CVTLOG:
   591          case O_CVTTUP:
   592          case O_CVTLFM:
   593          case O_PLUS:
   594          case O_MINUS:
   595          case O_NOT:
   596          case O_ABS:
   597          case O_CEIL:
   598          case O_FLOOR:
   599          case O_EXP:
   600          case O_LOG:
   601          case O_LOG10:
   602          case O_SQRT:
   603          case O_SIN:
   604          case O_COS:
   605          case O_ATAN:
   606          case O_ROUND:
   607          case O_TRUNC:
   608          case O_CARD:
   609          case O_LENGTH:
   610             /* unary operation */
   611             xassert(arg->arg.x != NULL);
   612             xassert(arg->arg.x->up == NULL);
   613             arg->arg.x->up = code;
   614             code->vflag |= arg->arg.x->vflag;
   615             code->arg.arg.x = arg->arg.x;
   616             break;
   617          case O_ADD:
   618          case O_SUB:
   619          case O_LESS:
   620          case O_MUL:
   621          case O_DIV:
   622          case O_IDIV:
   623          case O_MOD:
   624          case O_POWER:
   625          case O_ATAN2:
   626          case O_ROUND2:
   627          case O_TRUNC2:
   628          case O_UNIFORM:
   629             if (op == O_UNIFORM) code->vflag = 1;
   630          case O_NORMAL:
   631             if (op == O_NORMAL) code->vflag = 1;
   632          case O_CONCAT:
   633          case O_LT:
   634          case O_LE:
   635          case O_EQ:
   636          case O_GE:
   637          case O_GT:
   638          case O_NE:
   639          case O_AND:
   640          case O_OR:
   641          case O_UNION:
   642          case O_DIFF:
   643          case O_SYMDIFF:
   644          case O_INTER:
   645          case O_CROSS:
   646          case O_IN:
   647          case O_NOTIN:
   648          case O_WITHIN:
   649          case O_NOTWITHIN:
   650          case O_SUBSTR:
   651          case O_STR2TIME:
   652          case O_TIME2STR:
   653             /* binary operation */
   654             xassert(arg->arg.x != NULL);
   655             xassert(arg->arg.x->up == NULL);
   656             arg->arg.x->up = code;
   657             code->vflag |= arg->arg.x->vflag;
   658             xassert(arg->arg.y != NULL);
   659             xassert(arg->arg.y->up == NULL);
   660             arg->arg.y->up = code;
   661             code->vflag |= arg->arg.y->vflag;
   662             code->arg.arg.x = arg->arg.x;
   663             code->arg.arg.y = arg->arg.y;
   664             break;
   665          case O_DOTS:
   666          case O_FORK:
   667          case O_SUBSTR3:
   668             /* ternary operation */
   669             xassert(arg->arg.x != NULL);
   670             xassert(arg->arg.x->up == NULL);
   671             arg->arg.x->up = code;
   672             code->vflag |= arg->arg.x->vflag;
   673             xassert(arg->arg.y != NULL);
   674             xassert(arg->arg.y->up == NULL);
   675             arg->arg.y->up = code;
   676             code->vflag |= arg->arg.y->vflag;
   677             if (arg->arg.z != NULL)
   678             {  xassert(arg->arg.z->up == NULL);
   679                arg->arg.z->up = code;
   680                code->vflag |= arg->arg.z->vflag;
   681             }
   682             code->arg.arg.x = arg->arg.x;
   683             code->arg.arg.y = arg->arg.y;
   684             code->arg.arg.z = arg->arg.z;
   685             break;
   686          case O_MIN:
   687          case O_MAX:
   688             /* n-ary operation */
   689             for (e = arg->list; e != NULL; e = e->next)
   690             {  xassert(e->x != NULL);
   691                xassert(e->x->up == NULL);
   692                e->x->up = code;
   693                code->vflag |= e->x->vflag;
   694             }
   695             code->arg.list = arg->list;
   696             break;
   697          case O_SUM:
   698          case O_PROD:
   699          case O_MINIMUM:
   700          case O_MAXIMUM:
   701          case O_FORALL:
   702          case O_EXISTS:
   703          case O_SETOF:
   704          case O_BUILD:
   705             /* iterated operation */
   706             domain = arg->loop.domain;
   707             xassert(domain != NULL);
   708             if (domain->code != NULL)
   709             {  xassert(domain->code->up == NULL);
   710                domain->code->up = code;
   711                code->vflag |= domain->code->vflag;
   712             }
   713             for (block = domain->list; block != NULL; block =
   714                block->next)
   715             {  xassert(block->code != NULL);
   716                xassert(block->code->up == NULL);
   717                block->code->up = code;
   718                code->vflag |= block->code->vflag;
   719             }
   720             if (arg->loop.x != NULL)
   721             {  xassert(arg->loop.x->up == NULL);
   722                arg->loop.x->up = code;
   723                code->vflag |= arg->loop.x->vflag;
   724             }
   725             code->arg.loop.domain = arg->loop.domain;
   726             code->arg.loop.x = arg->loop.x;
   727             break;
   728          default:
   729             xassert(op != op);
   730       }
   731       /* set other attributes of the pseudo-code */
   732       code->type = type;
   733       code->dim = dim;
   734       code->up = NULL;
   735       code->valid = 0;
   736       memset(&code->value, '?', sizeof(VALUE));
   737       return code;
   738 }
   739 
   740 /*----------------------------------------------------------------------
   741 -- make_unary - generate pseudo-code for unary operation.
   742 --
   743 -- This routine generates pseudo-code for unary operation. */
   744 
   745 CODE *make_unary(MPL *mpl, int op, CODE *x, int type, int dim)
   746 {     CODE *code;
   747       OPERANDS arg;
   748       xassert(x != NULL);
   749       arg.arg.x = x;
   750       code = make_code(mpl, op, &arg, type, dim);
   751       return code;
   752 }
   753 
   754 /*----------------------------------------------------------------------
   755 -- make_binary - generate pseudo-code for binary operation.
   756 --
   757 -- This routine generates pseudo-code for binary operation. */
   758 
   759 CODE *make_binary(MPL *mpl, int op, CODE *x, CODE *y, int type,
   760       int dim)
   761 {     CODE *code;
   762       OPERANDS arg;
   763       xassert(x != NULL);
   764       xassert(y != NULL);
   765       arg.arg.x = x;
   766       arg.arg.y = y;
   767       code = make_code(mpl, op, &arg, type, dim);
   768       return code;
   769 }
   770 
   771 /*----------------------------------------------------------------------
   772 -- make_ternary - generate pseudo-code for ternary operation.
   773 --
   774 -- This routine generates pseudo-code for ternary operation. */
   775 
   776 CODE *make_ternary(MPL *mpl, int op, CODE *x, CODE *y, CODE *z,
   777       int type, int dim)
   778 {     CODE *code;
   779       OPERANDS arg;
   780       xassert(x != NULL);
   781       xassert(y != NULL);
   782       /* third operand can be NULL */
   783       arg.arg.x = x;
   784       arg.arg.y = y;
   785       arg.arg.z = z;
   786       code = make_code(mpl, op, &arg, type, dim);
   787       return code;
   788 }
   789 
   790 /*----------------------------------------------------------------------
   791 -- numeric_literal - parse reference to numeric literal.
   792 --
   793 -- This routine parses primary expression using the syntax:
   794 --
   795 -- <primary expression> ::= <numeric literal> */
   796 
   797 CODE *numeric_literal(MPL *mpl)
   798 {     CODE *code;
   799       OPERANDS arg;
   800       xassert(mpl->token == T_NUMBER);
   801       arg.num = mpl->value;
   802       code = make_code(mpl, O_NUMBER, &arg, A_NUMERIC, 0);
   803       get_token(mpl /* <numeric literal> */);
   804       return code;
   805 }
   806 
   807 /*----------------------------------------------------------------------
   808 -- string_literal - parse reference to string literal.
   809 --
   810 -- This routine parses primary expression using the syntax:
   811 --
   812 -- <primary expression> ::= <string literal> */
   813 
   814 CODE *string_literal(MPL *mpl)
   815 {     CODE *code;
   816       OPERANDS arg;
   817       xassert(mpl->token == T_STRING);
   818       arg.str = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
   819       strcpy(arg.str, mpl->image);
   820       code = make_code(mpl, O_STRING, &arg, A_SYMBOLIC, 0);
   821       get_token(mpl /* <string literal> */);
   822       return code;
   823 }
   824 
   825 /*----------------------------------------------------------------------
   826 -- create_arg_list - create empty operands list.
   827 --
   828 -- This routine creates operands list, which is initially empty. */
   829 
   830 ARG_LIST *create_arg_list(MPL *mpl)
   831 {     ARG_LIST *list;
   832       xassert(mpl == mpl);
   833       list = NULL;
   834       return list;
   835 }
   836 
   837 /*----------------------------------------------------------------------
   838 -- expand_arg_list - append operand to operands list.
   839 --
   840 -- This routine appends new operand to specified operands list. */
   841 
   842 ARG_LIST *expand_arg_list(MPL *mpl, ARG_LIST *list, CODE *x)
   843 {     ARG_LIST *tail, *temp;
   844       xassert(x != NULL);
   845       /* create new operands list entry */
   846       tail = alloc(ARG_LIST);
   847       tail->x = x;
   848       tail->next = NULL;
   849       /* and append it to the operands list */
   850       if (list == NULL)
   851          list = tail;
   852       else
   853       {  for (temp = list; temp->next != NULL; temp = temp->next);
   854          temp->next = tail;
   855       }
   856       return list;
   857 }
   858 
   859 /*----------------------------------------------------------------------
   860 -- arg_list_len - determine length of operands list.
   861 --
   862 -- This routine returns the number of operands in operands list. */
   863 
   864 int arg_list_len(MPL *mpl, ARG_LIST *list)
   865 {     ARG_LIST *temp;
   866       int len;
   867       xassert(mpl == mpl);
   868       len = 0;
   869       for (temp = list; temp != NULL; temp = temp->next) len++;
   870       return len;
   871 }
   872 
   873 /*----------------------------------------------------------------------
   874 -- subscript_list - parse subscript list.
   875 --
   876 -- This routine parses subscript list using the syntax:
   877 --
   878 -- <subscript list> ::= <subscript>
   879 -- <subscript list> ::= <subscript list> , <subscript>
   880 -- <subscript> ::= <expression 5> */
   881 
   882 ARG_LIST *subscript_list(MPL *mpl)
   883 {     ARG_LIST *list;
   884       CODE *x;
   885       list = create_arg_list(mpl);
   886       for (;;)
   887       {  /* parse subscript expression */
   888          x = expression_5(mpl);
   889          /* convert it to symbolic type, if necessary */
   890          if (x->type == A_NUMERIC)
   891             x = make_unary(mpl, O_CVTSYM, x, A_SYMBOLIC, 0);
   892          /* check that now the expression is of symbolic type */
   893          if (x->type != A_SYMBOLIC)
   894             error(mpl, "subscript expression has invalid type");
   895          xassert(x->dim == 0);
   896          /* and append it to the subscript list */
   897          list = expand_arg_list(mpl, list, x);
   898          /* check a token that follows the subscript expression */
   899          if (mpl->token == T_COMMA)
   900             get_token(mpl /* , */);
   901          else if (mpl->token == T_RBRACKET)
   902             break;
   903          else
   904             error(mpl, "syntax error in subscript list");
   905       }
   906       return list;
   907 }
   908 
   909 #if 1 /* 15/V-2010 */
   910 /*----------------------------------------------------------------------
   911 -- object_reference - parse reference to named object.
   912 --
   913 -- This routine parses primary expression using the syntax:
   914 --
   915 -- <primary expression> ::= <dummy index>
   916 -- <primary expression> ::= <set name>
   917 -- <primary expression> ::= <set name> [ <subscript list> ]
   918 -- <primary expression> ::= <parameter name>
   919 -- <primary expression> ::= <parameter name> [ <subscript list> ]
   920 -- <primary expression> ::= <variable name> <suffix>
   921 -- <primary expression> ::= <variable name> [ <subscript list> ]
   922 --                          <suffix>
   923 -- <primary expression> ::= <constraint name> <suffix>
   924 -- <primary expression> ::= <constraint name> [ <subscript list> ]
   925 --                          <suffix>
   926 -- <dummy index> ::= <symbolic name>
   927 -- <set name> ::= <symbolic name>
   928 -- <parameter name> ::= <symbolic name>
   929 -- <variable name> ::= <symbolic name>
   930 -- <constraint name> ::= <symbolic name>
   931 -- <suffix> ::= <empty> | .lb | .ub | .status | .val | .dual */
   932 
   933 CODE *object_reference(MPL *mpl)
   934 {     AVLNODE *node;
   935       DOMAIN_SLOT *slot;
   936       SET *set;
   937       PARAMETER *par;
   938       VARIABLE *var;
   939       CONSTRAINT *con;
   940       ARG_LIST *list;
   941       OPERANDS arg;
   942       CODE *code;
   943       char *name;
   944       int dim, suff;
   945       /* find the object in the symbolic name table */
   946       xassert(mpl->token == T_NAME);
   947       node = avl_find_node(mpl->tree, mpl->image);
   948       if (node == NULL)
   949          error(mpl, "%s not defined", mpl->image);
   950       /* check the object type and obtain its dimension */
   951       switch (avl_get_node_type(node))
   952       {  case A_INDEX:
   953             /* dummy index */
   954             slot = (DOMAIN_SLOT *)avl_get_node_link(node);
   955             name = slot->name;
   956             dim = 0;
   957             break;
   958          case A_SET:
   959             /* model set */
   960             set = (SET *)avl_get_node_link(node);
   961             name = set->name;
   962             dim = set->dim;
   963             /* if a set object is referenced in its own declaration and
   964                the dimen attribute is not specified yet, use dimen 1 by
   965                default */
   966             if (set->dimen == 0) set->dimen = 1;
   967             break;
   968          case A_PARAMETER:
   969             /* model parameter */
   970             par = (PARAMETER *)avl_get_node_link(node);
   971             name = par->name;
   972             dim = par->dim;
   973             break;
   974          case A_VARIABLE:
   975             /* model variable */
   976             var = (VARIABLE *)avl_get_node_link(node);
   977             name = var->name;
   978             dim = var->dim;
   979             break;
   980          case A_CONSTRAINT:
   981             /* model constraint or objective */
   982             con = (CONSTRAINT *)avl_get_node_link(node);
   983             name = con->name;
   984             dim = con->dim;
   985             break;
   986          default:
   987             xassert(node != node);
   988       }
   989       get_token(mpl /* <symbolic name> */);
   990       /* parse optional subscript list */
   991       if (mpl->token == T_LBRACKET)
   992       {  /* subscript list is specified */
   993          if (dim == 0)
   994             error(mpl, "%s cannot be subscripted", name);
   995          get_token(mpl /* [ */);
   996          list = subscript_list(mpl);
   997          if (dim != arg_list_len(mpl, list))
   998             error(mpl, "%s must have %d subscript%s rather than %d",
   999                name, dim, dim == 1 ? "" : "s", arg_list_len(mpl, list));
  1000          xassert(mpl->token == T_RBRACKET);
  1001          get_token(mpl /* ] */);
  1002       }
  1003       else
  1004       {  /* subscript list is not specified */
  1005          if (dim != 0)
  1006             error(mpl, "%s must be subscripted", name);
  1007          list = create_arg_list(mpl);
  1008       }
  1009       /* parse optional suffix */
  1010       if (!mpl->flag_s && avl_get_node_type(node) == A_VARIABLE)
  1011          suff = DOT_NONE;
  1012       else
  1013          suff = DOT_VAL;
  1014       if (mpl->token == T_POINT)
  1015       {  get_token(mpl /* . */);
  1016          if (mpl->token != T_NAME)
  1017             error(mpl, "invalid use of period");
  1018          if (!(avl_get_node_type(node) == A_VARIABLE ||
  1019                avl_get_node_type(node) == A_CONSTRAINT))
  1020             error(mpl, "%s cannot have a suffix", name);
  1021          if (strcmp(mpl->image, "lb") == 0)
  1022             suff = DOT_LB;
  1023          else if (strcmp(mpl->image, "ub") == 0)
  1024             suff = DOT_UB;
  1025          else if (strcmp(mpl->image, "status") == 0)
  1026             suff = DOT_STATUS;
  1027          else if (strcmp(mpl->image, "val") == 0)
  1028             suff = DOT_VAL;
  1029          else if (strcmp(mpl->image, "dual") == 0)
  1030             suff = DOT_DUAL;
  1031          else
  1032             error(mpl, "suffix .%s invalid", mpl->image);
  1033          get_token(mpl /* suffix */);
  1034       }
  1035       /* generate pseudo-code to take value of the object */
  1036       switch (avl_get_node_type(node))
  1037       {  case A_INDEX:
  1038             arg.index.slot = slot;
  1039             arg.index.next = slot->list;
  1040             code = make_code(mpl, O_INDEX, &arg, A_SYMBOLIC, 0);
  1041             slot->list = code;
  1042             break;
  1043          case A_SET:
  1044             arg.set.set = set;
  1045             arg.set.list = list;
  1046             code = make_code(mpl, O_MEMSET, &arg, A_ELEMSET,
  1047                set->dimen);
  1048             break;
  1049          case A_PARAMETER:
  1050             arg.par.par = par;
  1051             arg.par.list = list;
  1052             if (par->type == A_SYMBOLIC)
  1053                code = make_code(mpl, O_MEMSYM, &arg, A_SYMBOLIC, 0);
  1054             else
  1055                code = make_code(mpl, O_MEMNUM, &arg, A_NUMERIC, 0);
  1056             break;
  1057          case A_VARIABLE:
  1058             if (!mpl->flag_s && (suff == DOT_STATUS || suff == DOT_VAL
  1059                || suff == DOT_DUAL))
  1060                error(mpl, "invalid reference to status, primal value, o"
  1061                   "r dual value of variable %s above solve statement",
  1062                   var->name);
  1063             arg.var.var = var;
  1064             arg.var.list = list;
  1065             arg.var.suff = suff;
  1066             code = make_code(mpl, O_MEMVAR, &arg, suff == DOT_NONE ?
  1067                A_FORMULA : A_NUMERIC, 0);
  1068             break;
  1069          case A_CONSTRAINT:
  1070             if (!mpl->flag_s && (suff == DOT_STATUS || suff == DOT_VAL
  1071                || suff == DOT_DUAL))
  1072                error(mpl, "invalid reference to status, primal value, o"
  1073                   "r dual value of %s %s above solve statement",
  1074                   con->type == A_CONSTRAINT ? "constraint" : "objective"
  1075                   , con->name);
  1076             arg.con.con = con;
  1077             arg.con.list = list;
  1078             arg.con.suff = suff;
  1079             code = make_code(mpl, O_MEMCON, &arg, A_NUMERIC, 0);
  1080             break;
  1081          default:
  1082             xassert(node != node);
  1083       }
  1084       return code;
  1085 }
  1086 #endif
  1087 
  1088 /*----------------------------------------------------------------------
  1089 -- numeric_argument - parse argument passed to built-in function.
  1090 --
  1091 -- This routine parses an argument passed to numeric built-in function
  1092 -- using the syntax:
  1093 --
  1094 -- <arg> ::= <expression 5> */
  1095 
  1096 CODE *numeric_argument(MPL *mpl, char *func)
  1097 {     CODE *x;
  1098       x = expression_5(mpl);
  1099       /* convert the argument to numeric type, if necessary */
  1100       if (x->type == A_SYMBOLIC)
  1101          x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  1102       /* check that now the argument is of numeric type */
  1103       if (x->type != A_NUMERIC)
  1104          error(mpl, "argument for %s has invalid type", func);
  1105       xassert(x->dim == 0);
  1106       return x;
  1107 }
  1108 
  1109 #if 1 /* 15/VII-2006 */
  1110 CODE *symbolic_argument(MPL *mpl, char *func)
  1111 {     CODE *x;
  1112       x = expression_5(mpl);
  1113       /* convert the argument to symbolic type, if necessary */
  1114       if (x->type == A_NUMERIC)
  1115          x = make_unary(mpl, O_CVTSYM, x, A_SYMBOLIC, 0);
  1116       /* check that now the argument is of symbolic type */
  1117       if (x->type != A_SYMBOLIC)
  1118          error(mpl, "argument for %s has invalid type", func);
  1119       xassert(x->dim == 0);
  1120       return x;
  1121 }
  1122 #endif
  1123 
  1124 #if 1 /* 15/VII-2006 */
  1125 CODE *elemset_argument(MPL *mpl, char *func)
  1126 {     CODE *x;
  1127       x = expression_9(mpl);
  1128       if (x->type != A_ELEMSET)
  1129          error(mpl, "argument for %s has invalid type", func);
  1130       xassert(x->dim > 0);
  1131       return x;
  1132 }
  1133 #endif
  1134 
  1135 /*----------------------------------------------------------------------
  1136 -- function_reference - parse reference to built-in function.
  1137 --
  1138 -- This routine parses primary expression using the syntax:
  1139 --
  1140 -- <primary expression> ::= abs ( <arg> )
  1141 -- <primary expression> ::= ceil ( <arg> )
  1142 -- <primary expression> ::= floor ( <arg> )
  1143 -- <primary expression> ::= exp ( <arg> )
  1144 -- <primary expression> ::= log ( <arg> )
  1145 -- <primary expression> ::= log10 ( <arg> )
  1146 -- <primary expression> ::= max ( <arg list> )
  1147 -- <primary expression> ::= min ( <arg list> )
  1148 -- <primary expression> ::= sqrt ( <arg> )
  1149 -- <primary expression> ::= sin ( <arg> )
  1150 -- <primary expression> ::= cos ( <arg> )
  1151 -- <primary expression> ::= atan ( <arg> )
  1152 -- <primary expression> ::= atan2 ( <arg> , <arg> )
  1153 -- <primary expression> ::= round ( <arg> )
  1154 -- <primary expression> ::= round ( <arg> , <arg> )
  1155 -- <primary expression> ::= trunc ( <arg> )
  1156 -- <primary expression> ::= trunc ( <arg> , <arg> )
  1157 -- <primary expression> ::= Irand224 ( )
  1158 -- <primary expression> ::= Uniform01 ( )
  1159 -- <primary expression> ::= Uniform ( <arg> , <arg> )
  1160 -- <primary expression> ::= Normal01 ( )
  1161 -- <primary expression> ::= Normal ( <arg> , <arg> )
  1162 -- <primary expression> ::= card ( <arg> )
  1163 -- <primary expression> ::= length ( <arg> )
  1164 -- <primary expression> ::= substr ( <arg> , <arg> )
  1165 -- <primary expression> ::= substr ( <arg> , <arg> , <arg> )
  1166 -- <primary expression> ::= str2time ( <arg> , <arg> )
  1167 -- <primary expression> ::= time2str ( <arg> , <arg> )
  1168 -- <primary expression> ::= gmtime ( )
  1169 -- <arg list> ::= <arg>
  1170 -- <arg list> ::= <arg list> , <arg> */
  1171 
  1172 CODE *function_reference(MPL *mpl)
  1173 {     CODE *code;
  1174       OPERANDS arg;
  1175       int op;
  1176       char func[15+1];
  1177       /* determine operation code */
  1178       xassert(mpl->token == T_NAME);
  1179       if (strcmp(mpl->image, "abs") == 0)
  1180          op = O_ABS;
  1181       else if (strcmp(mpl->image, "ceil") == 0)
  1182          op = O_CEIL;
  1183       else if (strcmp(mpl->image, "floor") == 0)
  1184          op = O_FLOOR;
  1185       else if (strcmp(mpl->image, "exp") == 0)
  1186          op = O_EXP;
  1187       else if (strcmp(mpl->image, "log") == 0)
  1188          op = O_LOG;
  1189       else if (strcmp(mpl->image, "log10") == 0)
  1190          op = O_LOG10;
  1191       else if (strcmp(mpl->image, "sqrt") == 0)
  1192          op = O_SQRT;
  1193       else if (strcmp(mpl->image, "sin") == 0)
  1194          op = O_SIN;
  1195       else if (strcmp(mpl->image, "cos") == 0)
  1196          op = O_COS;
  1197       else if (strcmp(mpl->image, "atan") == 0)
  1198          op = O_ATAN;
  1199       else if (strcmp(mpl->image, "min") == 0)
  1200          op = O_MIN;
  1201       else if (strcmp(mpl->image, "max") == 0)
  1202          op = O_MAX;
  1203       else if (strcmp(mpl->image, "round") == 0)
  1204          op = O_ROUND;
  1205       else if (strcmp(mpl->image, "trunc") == 0)
  1206          op = O_TRUNC;
  1207       else if (strcmp(mpl->image, "Irand224") == 0)
  1208          op = O_IRAND224;
  1209       else if (strcmp(mpl->image, "Uniform01") == 0)
  1210          op = O_UNIFORM01;
  1211       else if (strcmp(mpl->image, "Uniform") == 0)
  1212          op = O_UNIFORM;
  1213       else if (strcmp(mpl->image, "Normal01") == 0)
  1214          op = O_NORMAL01;
  1215       else if (strcmp(mpl->image, "Normal") == 0)
  1216          op = O_NORMAL;
  1217       else if (strcmp(mpl->image, "card") == 0)
  1218          op = O_CARD;
  1219       else if (strcmp(mpl->image, "length") == 0)
  1220          op = O_LENGTH;
  1221       else if (strcmp(mpl->image, "substr") == 0)
  1222          op = O_SUBSTR;
  1223       else if (strcmp(mpl->image, "str2time") == 0)
  1224          op = O_STR2TIME;
  1225       else if (strcmp(mpl->image, "time2str") == 0)
  1226          op = O_TIME2STR;
  1227       else if (strcmp(mpl->image, "gmtime") == 0)
  1228          op = O_GMTIME;
  1229       else
  1230          error(mpl, "function %s unknown", mpl->image);
  1231       /* save symbolic name of the function */
  1232       strcpy(func, mpl->image);
  1233       xassert(strlen(func) < sizeof(func));
  1234       get_token(mpl /* <symbolic name> */);
  1235       /* check the left parenthesis that follows the function name */
  1236       xassert(mpl->token == T_LEFT);
  1237       get_token(mpl /* ( */);
  1238       /* parse argument list */
  1239       if (op == O_MIN || op == O_MAX)
  1240       {  /* min and max allow arbitrary number of arguments */
  1241          arg.list = create_arg_list(mpl);
  1242          /* parse argument list */
  1243          for (;;)
  1244          {  /* parse argument and append it to the operands list */
  1245             arg.list = expand_arg_list(mpl, arg.list,
  1246                numeric_argument(mpl, func));
  1247             /* check a token that follows the argument */
  1248             if (mpl->token == T_COMMA)
  1249                get_token(mpl /* , */);
  1250             else if (mpl->token == T_RIGHT)
  1251                break;
  1252             else
  1253                error(mpl, "syntax error in argument list for %s", func);
  1254          }
  1255       }
  1256       else if (op == O_IRAND224 || op == O_UNIFORM01 || op ==
  1257          O_NORMAL01 || op == O_GMTIME)
  1258       {  /* Irand224, Uniform01, Normal01, gmtime need no arguments */
  1259          if (mpl->token != T_RIGHT)
  1260             error(mpl, "%s needs no arguments", func);
  1261       }
  1262       else if (op == O_UNIFORM || op == O_NORMAL)
  1263       {  /* Uniform and Normal need two arguments */
  1264          /* parse the first argument */
  1265          arg.arg.x = numeric_argument(mpl, func);
  1266          /* check a token that follows the first argument */
  1267          if (mpl->token == T_COMMA)
  1268             ;
  1269          else if (mpl->token == T_RIGHT)
  1270             error(mpl, "%s needs two arguments", func);
  1271          else
  1272             error(mpl, "syntax error in argument for %s", func);
  1273          get_token(mpl /* , */);
  1274          /* parse the second argument */
  1275          arg.arg.y = numeric_argument(mpl, func);
  1276          /* check a token that follows the second argument */
  1277          if (mpl->token == T_COMMA)
  1278             error(mpl, "%s needs two argument", func);
  1279          else if (mpl->token == T_RIGHT)
  1280             ;
  1281          else
  1282             error(mpl, "syntax error in argument for %s", func);
  1283       }
  1284       else if (op == O_ATAN || op == O_ROUND || op == O_TRUNC)
  1285       {  /* atan, round, and trunc need one or two arguments */
  1286          /* parse the first argument */
  1287          arg.arg.x = numeric_argument(mpl, func);
  1288          /* parse the second argument, if specified */
  1289          if (mpl->token == T_COMMA)
  1290          {  switch (op)
  1291             {  case O_ATAN:  op = O_ATAN2;  break;
  1292                case O_ROUND: op = O_ROUND2; break;
  1293                case O_TRUNC: op = O_TRUNC2; break;
  1294                default: xassert(op != op);
  1295             }
  1296             get_token(mpl /* , */);
  1297             arg.arg.y = numeric_argument(mpl, func);
  1298          }
  1299          /* check a token that follows the last argument */
  1300          if (mpl->token == T_COMMA)
  1301             error(mpl, "%s needs one or two arguments", func);
  1302          else if (mpl->token == T_RIGHT)
  1303             ;
  1304          else
  1305             error(mpl, "syntax error in argument for %s", func);
  1306       }
  1307       else if (op == O_SUBSTR)
  1308       {  /* substr needs two or three arguments */
  1309          /* parse the first argument */
  1310          arg.arg.x = symbolic_argument(mpl, func);
  1311          /* check a token that follows the first argument */
  1312          if (mpl->token == T_COMMA)
  1313             ;
  1314          else if (mpl->token == T_RIGHT)
  1315             error(mpl, "%s needs two or three arguments", func);
  1316          else
  1317             error(mpl, "syntax error in argument for %s", func);
  1318          get_token(mpl /* , */);
  1319          /* parse the second argument */
  1320          arg.arg.y = numeric_argument(mpl, func);
  1321          /* parse the third argument, if specified */
  1322          if (mpl->token == T_COMMA)
  1323          {  op = O_SUBSTR3;
  1324             get_token(mpl /* , */);
  1325             arg.arg.z = numeric_argument(mpl, func);
  1326          }
  1327          /* check a token that follows the last argument */
  1328          if (mpl->token == T_COMMA)
  1329             error(mpl, "%s needs two or three arguments", func);
  1330          else if (mpl->token == T_RIGHT)
  1331             ;
  1332          else
  1333             error(mpl, "syntax error in argument for %s", func);
  1334       }
  1335       else if (op == O_STR2TIME)
  1336       {  /* str2time needs two arguments, both symbolic */
  1337          /* parse the first argument */
  1338          arg.arg.x = symbolic_argument(mpl, func);
  1339          /* check a token that follows the first argument */
  1340          if (mpl->token == T_COMMA)
  1341             ;
  1342          else if (mpl->token == T_RIGHT)
  1343             error(mpl, "%s needs two arguments", func);
  1344          else
  1345             error(mpl, "syntax error in argument for %s", func);
  1346          get_token(mpl /* , */);
  1347          /* parse the second argument */
  1348          arg.arg.y = symbolic_argument(mpl, func);
  1349          /* check a token that follows the second argument */
  1350          if (mpl->token == T_COMMA)
  1351             error(mpl, "%s needs two argument", func);
  1352          else if (mpl->token == T_RIGHT)
  1353             ;
  1354          else
  1355             error(mpl, "syntax error in argument for %s", func);
  1356       }
  1357       else if (op == O_TIME2STR)
  1358       {  /* time2str needs two arguments, numeric and symbolic */
  1359          /* parse the first argument */
  1360          arg.arg.x = numeric_argument(mpl, func);
  1361          /* check a token that follows the first argument */
  1362          if (mpl->token == T_COMMA)
  1363             ;
  1364          else if (mpl->token == T_RIGHT)
  1365             error(mpl, "%s needs two arguments", func);
  1366          else
  1367             error(mpl, "syntax error in argument for %s", func);
  1368          get_token(mpl /* , */);
  1369          /* parse the second argument */
  1370          arg.arg.y = symbolic_argument(mpl, func);
  1371          /* check a token that follows the second argument */
  1372          if (mpl->token == T_COMMA)
  1373             error(mpl, "%s needs two argument", func);
  1374          else if (mpl->token == T_RIGHT)
  1375             ;
  1376          else
  1377             error(mpl, "syntax error in argument for %s", func);
  1378       }
  1379       else
  1380       {  /* other functions need one argument */
  1381          if (op == O_CARD)
  1382             arg.arg.x = elemset_argument(mpl, func);
  1383          else if (op == O_LENGTH)
  1384             arg.arg.x = symbolic_argument(mpl, func);
  1385          else
  1386             arg.arg.x = numeric_argument(mpl, func);
  1387          /* check a token that follows the argument */
  1388          if (mpl->token == T_COMMA)
  1389             error(mpl, "%s needs one argument", func);
  1390          else if (mpl->token == T_RIGHT)
  1391             ;
  1392          else
  1393             error(mpl, "syntax error in argument for %s", func);
  1394       }
  1395       /* make pseudo-code to call the built-in function */
  1396       if (op == O_SUBSTR || op == O_SUBSTR3 || op == O_TIME2STR)
  1397          code = make_code(mpl, op, &arg, A_SYMBOLIC, 0);
  1398       else
  1399          code = make_code(mpl, op, &arg, A_NUMERIC, 0);
  1400       /* the reference ends with the right parenthesis */
  1401       xassert(mpl->token == T_RIGHT);
  1402       get_token(mpl /* ) */);
  1403       return code;
  1404 }
  1405 
  1406 /*----------------------------------------------------------------------
  1407 -- create_domain - create empty domain.
  1408 --
  1409 -- This routine creates empty domain, which is initially empty, i.e.
  1410 -- has no domain blocks. */
  1411 
  1412 DOMAIN *create_domain(MPL *mpl)
  1413 {     DOMAIN *domain;
  1414       domain = alloc(DOMAIN);
  1415       domain->list = NULL;
  1416       domain->code = NULL;
  1417       return domain;
  1418 }
  1419 
  1420 /*----------------------------------------------------------------------
  1421 -- create_block - create empty domain block.
  1422 --
  1423 -- This routine creates empty domain block, which is initially empty,
  1424 -- i.e. has no domain slots. */
  1425 
  1426 DOMAIN_BLOCK *create_block(MPL *mpl)
  1427 {     DOMAIN_BLOCK *block;
  1428       block = alloc(DOMAIN_BLOCK);
  1429       block->list = NULL;
  1430       block->code = NULL;
  1431       block->backup = NULL;
  1432       block->next = NULL;
  1433       return block;
  1434 }
  1435 
  1436 /*----------------------------------------------------------------------
  1437 -- append_block - append domain block to specified domain.
  1438 --
  1439 -- This routine adds given domain block to the end of the block list of
  1440 -- specified domain. */
  1441 
  1442 void append_block(MPL *mpl, DOMAIN *domain, DOMAIN_BLOCK *block)
  1443 {     DOMAIN_BLOCK *temp;
  1444       xassert(mpl == mpl);
  1445       xassert(domain != NULL);
  1446       xassert(block != NULL);
  1447       xassert(block->next == NULL);
  1448       if (domain->list == NULL)
  1449          domain->list = block;
  1450       else
  1451       {  for (temp = domain->list; temp->next != NULL; temp =
  1452             temp->next);
  1453          temp->next = block;
  1454       }
  1455       return;
  1456 }
  1457 
  1458 /*----------------------------------------------------------------------
  1459 -- append_slot - create and append new slot to domain block.
  1460 --
  1461 -- This routine creates new domain slot and adds it to the end of slot
  1462 -- list of specified domain block.
  1463 --
  1464 -- The parameter name is symbolic name of the dummy index associated
  1465 -- with the slot (the character string must be allocated). NULL means
  1466 -- the dummy index is not explicitly specified.
  1467 --
  1468 -- The parameter code is pseudo-code for computing symbolic value, at
  1469 -- which the dummy index is bounded. NULL means the dummy index is free
  1470 -- in the domain scope. */
  1471 
  1472 DOMAIN_SLOT *append_slot(MPL *mpl, DOMAIN_BLOCK *block, char *name,
  1473       CODE *code)
  1474 {     DOMAIN_SLOT *slot, *temp;
  1475       xassert(block != NULL);
  1476       slot = alloc(DOMAIN_SLOT);
  1477       slot->name = name;
  1478       slot->code = code;
  1479       slot->value = NULL;
  1480       slot->list = NULL;
  1481       slot->next = NULL;
  1482       if (block->list == NULL)
  1483          block->list = slot;
  1484       else
  1485       {  for (temp = block->list; temp->next != NULL; temp =
  1486             temp->next);
  1487          temp->next = slot;
  1488       }
  1489       return slot;
  1490 }
  1491 
  1492 /*----------------------------------------------------------------------
  1493 -- expression_list - parse expression list.
  1494 --
  1495 -- This routine parses a list of one or more expressions enclosed into
  1496 -- the parentheses using the syntax:
  1497 --
  1498 -- <primary expression> ::= ( <expression list> )
  1499 -- <expression list> ::= <expression 13>
  1500 -- <expression list> ::= <expression 13> , <expression list>
  1501 --
  1502 -- Note that this construction may have three different meanings:
  1503 --
  1504 -- 1. If <expression list> consists of only one expression, <primary
  1505 --    expression> is a parenthesized expression, which may be of any
  1506 --    valid type (not necessarily 1-tuple).
  1507 --
  1508 -- 2. If <expression list> consists of several expressions separated by
  1509 --    commae, where no expression is undeclared symbolic name, <primary
  1510 --    expression> is a n-tuple.
  1511 --
  1512 -- 3. If <expression list> consists of several expressions separated by
  1513 --    commae, where at least one expression is undeclared symbolic name
  1514 --    (that denotes a dummy index), <primary expression> is a slice and
  1515 --    can be only used as constituent of indexing expression. */
  1516 
  1517 #define max_dim 20
  1518 /* maximal number of components allowed within parentheses */
  1519 
  1520 CODE *expression_list(MPL *mpl)
  1521 {     CODE *code;
  1522       OPERANDS arg;
  1523       struct { char *name; CODE *code; } list[1+max_dim];
  1524       int flag_x, next_token, dim, j, slice = 0;
  1525       xassert(mpl->token == T_LEFT);
  1526       /* the flag, which allows recognizing undeclared symbolic names
  1527          as dummy indices, will be automatically reset by get_token(),
  1528          so save it before scanning the next token */
  1529       flag_x = mpl->flag_x;
  1530       get_token(mpl /* ( */);
  1531       /* parse <expression list> */
  1532       for (dim = 1; ; dim++)
  1533       {  if (dim > max_dim)
  1534             error(mpl, "too many components within parentheses");
  1535          /* current component of <expression list> can be either dummy
  1536             index or expression */
  1537          if (mpl->token == T_NAME)
  1538          {  /* symbolic name is recognized as dummy index only if:
  1539                the flag, which allows that, is set, and
  1540                the name is followed by comma or right parenthesis, and
  1541                the name is undeclared */
  1542             get_token(mpl /* <symbolic name> */);
  1543             next_token = mpl->token;
  1544             unget_token(mpl);
  1545             if (!(flag_x &&
  1546                   (next_token == T_COMMA || next_token == T_RIGHT) &&
  1547                   avl_find_node(mpl->tree, mpl->image) == NULL))
  1548             {  /* this is not dummy index */
  1549                goto expr;
  1550             }
  1551             /* all dummy indices within the same slice must have unique
  1552                symbolic names */
  1553             for (j = 1; j < dim; j++)
  1554             {  if (list[j].name != NULL && strcmp(list[j].name,
  1555                   mpl->image) == 0)
  1556                   error(mpl, "duplicate dummy index %s not allowed",
  1557                      mpl->image);
  1558             }
  1559             /* current component of <expression list> is dummy index */
  1560             list[dim].name
  1561                = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  1562             strcpy(list[dim].name, mpl->image);
  1563             list[dim].code = NULL;
  1564             get_token(mpl /* <symbolic name> */);
  1565             /* <expression list> is a slice, because at least one dummy
  1566                index has appeared */
  1567             slice = 1;
  1568             /* note that the context ( <dummy index> ) is not allowed,
  1569                i.e. in this case <primary expression> is considered as
  1570                a parenthesized expression */
  1571             if (dim == 1 && mpl->token == T_RIGHT)
  1572                error(mpl, "%s not defined", list[dim].name);
  1573          }
  1574          else
  1575 expr:    {  /* current component of <expression list> is expression */
  1576             code = expression_13(mpl);
  1577             /* if the current expression is followed by comma or it is
  1578                not the very first expression, entire <expression list>
  1579                is n-tuple or slice, in which case the current expression
  1580                should be converted to symbolic type, if necessary */
  1581             if (mpl->token == T_COMMA || dim > 1)
  1582             {  if (code->type == A_NUMERIC)
  1583                   code = make_unary(mpl, O_CVTSYM, code, A_SYMBOLIC, 0);
  1584                /* now the expression must be of symbolic type */
  1585                if (code->type != A_SYMBOLIC)
  1586                   error(mpl, "component expression has invalid type");
  1587                xassert(code->dim == 0);
  1588             }
  1589             list[dim].name = NULL;
  1590             list[dim].code = code;
  1591          }
  1592          /* check a token that follows the current component */
  1593          if (mpl->token == T_COMMA)
  1594             get_token(mpl /* , */);
  1595          else if (mpl->token == T_RIGHT)
  1596             break;
  1597          else
  1598             error(mpl, "right parenthesis missing where expected");
  1599       }
  1600       /* generate pseudo-code for <primary expression> */
  1601       if (dim == 1 && !slice)
  1602       {  /* <primary expression> is a parenthesized expression */
  1603          code = list[1].code;
  1604       }
  1605       else if (!slice)
  1606       {  /* <primary expression> is a n-tuple */
  1607          arg.list = create_arg_list(mpl);
  1608          for (j = 1; j <= dim; j++)
  1609             arg.list = expand_arg_list(mpl, arg.list, list[j].code);
  1610          code = make_code(mpl, O_TUPLE, &arg, A_TUPLE, dim);
  1611       }
  1612       else
  1613       {  /* <primary expression> is a slice */
  1614          arg.slice = create_block(mpl);
  1615          for (j = 1; j <= dim; j++)
  1616             append_slot(mpl, arg.slice, list[j].name, list[j].code);
  1617          /* note that actually pseudo-codes with op = O_SLICE are never
  1618             evaluated */
  1619          code = make_code(mpl, O_SLICE, &arg, A_TUPLE, dim);
  1620       }
  1621       get_token(mpl /* ) */);
  1622       /* if <primary expression> is a slice, there must be the keyword
  1623          'in', which follows the right parenthesis */
  1624       if (slice && mpl->token != T_IN)
  1625          error(mpl, "keyword in missing where expected");
  1626       /* if the slice flag is set and there is the keyword 'in', which
  1627          follows <primary expression>, the latter must be a slice */
  1628       if (flag_x && mpl->token == T_IN && !slice)
  1629       {  if (dim == 1)
  1630             error(mpl, "syntax error in indexing expression");
  1631          else
  1632             error(mpl, "0-ary slice not allowed");
  1633       }
  1634       return code;
  1635 }
  1636 
  1637 /*----------------------------------------------------------------------
  1638 -- literal set - parse literal set.
  1639 --
  1640 -- This routine parses literal set using the syntax:
  1641 --
  1642 -- <literal set> ::= { <member list> }
  1643 -- <member list> ::= <member expression>
  1644 -- <member list> ::= <member list> , <member expression>
  1645 -- <member expression> ::= <expression 5>
  1646 --
  1647 -- It is assumed that the left curly brace and the very first member
  1648 -- expression that follows it are already parsed. The right curly brace
  1649 -- remains unscanned on exit. */
  1650 
  1651 CODE *literal_set(MPL *mpl, CODE *code)
  1652 {     OPERANDS arg;
  1653       int j;
  1654       xassert(code != NULL);
  1655       arg.list = create_arg_list(mpl);
  1656       /* parse <member list> */
  1657       for (j = 1; ; j++)
  1658       {  /* all member expressions must be n-tuples; so, if the current
  1659             expression is not n-tuple, convert it to 1-tuple */
  1660          if (code->type == A_NUMERIC)
  1661             code = make_unary(mpl, O_CVTSYM, code, A_SYMBOLIC, 0);
  1662          if (code->type == A_SYMBOLIC)
  1663             code = make_unary(mpl, O_CVTTUP, code, A_TUPLE, 1);
  1664          /* now the expression must be n-tuple */
  1665          if (code->type != A_TUPLE)
  1666             error(mpl, "member expression has invalid type");
  1667          /* all member expressions must have identical dimension */
  1668          if (arg.list != NULL && arg.list->x->dim != code->dim)
  1669             error(mpl, "member %d has %d component%s while member %d ha"
  1670                "s %d component%s",
  1671                j-1, arg.list->x->dim, arg.list->x->dim == 1 ? "" : "s",
  1672                j, code->dim, code->dim == 1 ? "" : "s");
  1673          /* append the current expression to the member list */
  1674          arg.list = expand_arg_list(mpl, arg.list, code);
  1675          /* check a token that follows the current expression */
  1676          if (mpl->token == T_COMMA)
  1677             get_token(mpl /* , */);
  1678          else if (mpl->token == T_RBRACE)
  1679             break;
  1680          else
  1681             error(mpl, "syntax error in literal set");
  1682          /* parse the next expression that follows the comma */
  1683          code = expression_5(mpl);
  1684       }
  1685       /* generate pseudo-code for <literal set> */
  1686       code = make_code(mpl, O_MAKE, &arg, A_ELEMSET, arg.list->x->dim);
  1687       return code;
  1688 }
  1689 
  1690 /*----------------------------------------------------------------------
  1691 -- indexing_expression - parse indexing expression.
  1692 --
  1693 -- This routine parses indexing expression using the syntax:
  1694 --
  1695 -- <indexing expression> ::= <literal set>
  1696 -- <indexing expression> ::= { <indexing list> }
  1697 -- <indexing expression> ::= { <indexing list> : <logical expression> }
  1698 -- <indexing list> ::= <indexing element>
  1699 -- <indexing list> ::= <indexing list> , <indexing element>
  1700 -- <indexing element> ::= <basic expression>
  1701 -- <indexing element> ::= <dummy index> in <basic expression>
  1702 -- <indexing element> ::= <slice> in <basic expression>
  1703 -- <dummy index> ::= <symbolic name>
  1704 -- <slice> ::= ( <expression list> )
  1705 -- <basic expression> ::= <expression 9>
  1706 -- <logical expression> ::= <expression 13>
  1707 --
  1708 -- This routine creates domain for <indexing expression>, where each
  1709 -- domain block corresponds to <indexing element>, and each domain slot
  1710 -- corresponds to individual indexing position. */
  1711 
  1712 DOMAIN *indexing_expression(MPL *mpl)
  1713 {     DOMAIN *domain;
  1714       DOMAIN_BLOCK *block;
  1715       DOMAIN_SLOT *slot;
  1716       CODE *code;
  1717       xassert(mpl->token == T_LBRACE);
  1718       get_token(mpl /* { */);
  1719       if (mpl->token == T_RBRACE)
  1720          error(mpl, "empty indexing expression not allowed");
  1721       /* create domain to be constructed */
  1722       domain = create_domain(mpl);
  1723       /* parse either <member list> or <indexing list> that follows the
  1724          left brace */
  1725       for (;;)
  1726       {  /* domain block for <indexing element> is not created yet */
  1727          block = NULL;
  1728          /* pseudo-code for <basic expression> is not generated yet */
  1729          code = NULL;
  1730          /* check a token, which <indexing element> begins with */
  1731          if (mpl->token == T_NAME)
  1732          {  /* it is a symbolic name */
  1733             int next_token;
  1734             char *name;
  1735             /* symbolic name is recognized as dummy index only if it is
  1736                followed by the keyword 'in' and not declared */
  1737             get_token(mpl /* <symbolic name> */);
  1738             next_token = mpl->token;
  1739             unget_token(mpl);
  1740             if (!(next_token == T_IN &&
  1741                   avl_find_node(mpl->tree, mpl->image) == NULL))
  1742             {  /* this is not dummy index; the symbolic name begins an
  1743                   expression, which is either <basic expression> or the
  1744                   very first <member expression> in <literal set> */
  1745                goto expr;
  1746             }
  1747             /* create domain block with one slot, which is assigned the
  1748                dummy index */
  1749             block = create_block(mpl);
  1750             name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  1751             strcpy(name, mpl->image);
  1752             append_slot(mpl, block, name, NULL);
  1753             get_token(mpl /* <symbolic name> */);
  1754             /* the keyword 'in' is already checked above */
  1755             xassert(mpl->token == T_IN);
  1756             get_token(mpl /* in */);
  1757             /* <basic expression> that follows the keyword 'in' will be
  1758                parsed below */
  1759          }
  1760          else if (mpl->token == T_LEFT)
  1761          {  /* it is the left parenthesis; parse expression that begins
  1762                with this parenthesis (the flag is set in order to allow
  1763                recognizing slices; see the routine expression_list) */
  1764             mpl->flag_x = 1;
  1765             code = expression_9(mpl);
  1766             if (code->op != O_SLICE)
  1767             {  /* this is either <basic expression> or the very first
  1768                   <member expression> in <literal set> */
  1769                goto expr;
  1770             }
  1771             /* this is a slice; besides the corresponding domain block
  1772                is already created by expression_list() */
  1773             block = code->arg.slice;
  1774             code = NULL; /* <basic expression> is not parsed yet */
  1775             /* the keyword 'in' following the slice is already checked
  1776                by expression_list() */
  1777             xassert(mpl->token == T_IN);
  1778             get_token(mpl /* in */);
  1779             /* <basic expression> that follows the keyword 'in' will be
  1780                parsed below */
  1781          }
  1782 expr:    /* parse expression that follows either the keyword 'in' (in
  1783             which case it can be <basic expression) or the left brace
  1784             (in which case it can be <basic expression> as well as the
  1785             very first <member expression> in <literal set>); note that
  1786             this expression can be already parsed above */
  1787          if (code == NULL) code = expression_9(mpl);
  1788          /* check the type of the expression just parsed */
  1789          if (code->type != A_ELEMSET)
  1790          {  /* it is not <basic expression> and therefore it can only
  1791                be the very first <member expression> in <literal set>;
  1792                however, then there must be no dummy index neither slice
  1793                between the left brace and this expression */
  1794             if (block != NULL)
  1795                error(mpl, "domain expression has invalid type");
  1796             /* parse the rest part of <literal set> and make this set
  1797                be <basic expression>, i.e. the construction {a, b, c}
  1798                is parsed as it were written as {A}, where A = {a, b, c}
  1799                is a temporary elemental set */
  1800             code = literal_set(mpl, code);
  1801          }
  1802          /* now pseudo-code for <basic set> has been built */
  1803          xassert(code != NULL);
  1804          xassert(code->type == A_ELEMSET);
  1805          xassert(code->dim > 0);
  1806          /* if domain block for the current <indexing element> is still
  1807             not created, create it for fake slice of the same dimension
  1808             as <basic set> */
  1809          if (block == NULL)
  1810          {  int j;
  1811             block = create_block(mpl);
  1812             for (j = 1; j <= code->dim; j++)
  1813                append_slot(mpl, block, NULL, NULL);
  1814          }
  1815          /* number of indexing positions in <indexing element> must be
  1816             the same as dimension of n-tuples in basic set */
  1817          {  int dim = 0;
  1818             for (slot = block->list; slot != NULL; slot = slot->next)
  1819                dim++;
  1820             if (dim != code->dim)
  1821                error(mpl,"%d %s specified for set of dimension %d",
  1822                   dim, dim == 1 ? "index" : "indices", code->dim);
  1823          }
  1824          /* store pseudo-code for <basic set> in the domain block */
  1825          xassert(block->code == NULL);
  1826          block->code = code;
  1827          /* and append the domain block to the domain */
  1828          append_block(mpl, domain, block);
  1829          /* the current <indexing element> has been completely parsed;
  1830             include all its dummy indices into the symbolic name table
  1831             to make them available for referencing from expressions;
  1832             implicit declarations of dummy indices remain valid while
  1833             the corresponding domain scope is valid */
  1834          for (slot = block->list; slot != NULL; slot = slot->next)
  1835          if (slot->name != NULL)
  1836          {  AVLNODE *node;
  1837             xassert(avl_find_node(mpl->tree, slot->name) == NULL);
  1838             node = avl_insert_node(mpl->tree, slot->name);
  1839             avl_set_node_type(node, A_INDEX);
  1840             avl_set_node_link(node, (void *)slot);
  1841          }
  1842          /* check a token that follows <indexing element> */
  1843          if (mpl->token == T_COMMA)
  1844             get_token(mpl /* , */);
  1845          else if (mpl->token == T_COLON || mpl->token == T_RBRACE)
  1846             break;
  1847          else
  1848             error(mpl, "syntax error in indexing expression");
  1849       }
  1850       /* parse <logical expression> that follows the colon */
  1851       if (mpl->token == T_COLON)
  1852       {  get_token(mpl /* : */);
  1853          code = expression_13(mpl);
  1854          /* convert the expression to logical type, if necessary */
  1855          if (code->type == A_SYMBOLIC)
  1856             code = make_unary(mpl, O_CVTNUM, code, A_NUMERIC, 0);
  1857          if (code->type == A_NUMERIC)
  1858             code = make_unary(mpl, O_CVTLOG, code, A_LOGICAL, 0);
  1859          /* now the expression must be of logical type */
  1860          if (code->type != A_LOGICAL)
  1861             error(mpl, "expression following colon has invalid type");
  1862          xassert(code->dim == 0);
  1863          domain->code = code;
  1864          /* the right brace must follow the logical expression */
  1865          if (mpl->token != T_RBRACE)
  1866             error(mpl, "syntax error in indexing expression");
  1867       }
  1868       get_token(mpl /* } */);
  1869       return domain;
  1870 }
  1871 
  1872 /*----------------------------------------------------------------------
  1873 -- close_scope - close scope of indexing expression.
  1874 --
  1875 -- The routine closes the scope of indexing expression specified by its
  1876 -- domain and thereby makes all dummy indices introduced in the indexing
  1877 -- expression no longer available for referencing. */
  1878 
  1879 void close_scope(MPL *mpl, DOMAIN *domain)
  1880 {     DOMAIN_BLOCK *block;
  1881       DOMAIN_SLOT *slot;
  1882       AVLNODE *node;
  1883       xassert(domain != NULL);
  1884       /* remove all dummy indices from the symbolic names table */
  1885       for (block = domain->list; block != NULL; block = block->next)
  1886       {  for (slot = block->list; slot != NULL; slot = slot->next)
  1887          {  if (slot->name != NULL)
  1888             {  node = avl_find_node(mpl->tree, slot->name);
  1889                xassert(node != NULL);
  1890                xassert(avl_get_node_type(node) == A_INDEX);
  1891                avl_delete_node(mpl->tree, node);
  1892             }
  1893          }
  1894       }
  1895       return;
  1896 }
  1897 
  1898 /*----------------------------------------------------------------------
  1899 -- iterated_expression - parse iterated expression.
  1900 --
  1901 -- This routine parses primary expression using the syntax:
  1902 --
  1903 -- <primary expression> ::= <iterated expression>
  1904 -- <iterated expression> ::= sum <indexing expression> <expression 3>
  1905 -- <iterated expression> ::= prod <indexing expression> <expression 3>
  1906 -- <iterated expression> ::= min <indexing expression> <expression 3>
  1907 -- <iterated expression> ::= max <indexing expression> <expression 3>
  1908 -- <iterated expression> ::= exists <indexing expression>
  1909 --                           <expression 12>
  1910 -- <iterated expression> ::= forall <indexing expression>
  1911 --                           <expression 12>
  1912 -- <iterated expression> ::= setof <indexing expression> <expression 5>
  1913 --
  1914 -- Note that parsing "integrand" depends on the iterated operator. */
  1915 
  1916 #if 1 /* 07/IX-2008 */
  1917 static void link_up(CODE *code)
  1918 {     /* if we have something like sum{(i+1,j,k-1) in E} x[i,j,k],
  1919          where i and k are dummy indices defined out of the iterated
  1920          expression, we should link up pseudo-code for computing i+1
  1921          and k-1 to pseudo-code for computing the iterated expression;
  1922          this is needed to invalidate current value of the iterated
  1923          expression once i or k have been changed */
  1924       DOMAIN_BLOCK *block;
  1925       DOMAIN_SLOT *slot;
  1926       for (block = code->arg.loop.domain->list; block != NULL;
  1927          block = block->next)
  1928       {  for (slot = block->list; slot != NULL; slot = slot->next)
  1929          {  if (slot->code != NULL)
  1930             {  xassert(slot->code->up == NULL);
  1931                slot->code->up = code;
  1932             }
  1933          }
  1934       }
  1935       return;
  1936 }
  1937 #endif
  1938 
  1939 CODE *iterated_expression(MPL *mpl)
  1940 {     CODE *code;
  1941       OPERANDS arg;
  1942       int op;
  1943       char opstr[8];
  1944       /* determine operation code */
  1945       xassert(mpl->token == T_NAME);
  1946       if (strcmp(mpl->image, "sum") == 0)
  1947          op = O_SUM;
  1948       else if (strcmp(mpl->image, "prod") == 0)
  1949          op = O_PROD;
  1950       else if (strcmp(mpl->image, "min") == 0)
  1951          op = O_MINIMUM;
  1952       else if (strcmp(mpl->image, "max") == 0)
  1953          op = O_MAXIMUM;
  1954       else if (strcmp(mpl->image, "forall") == 0)
  1955          op = O_FORALL;
  1956       else if (strcmp(mpl->image, "exists") == 0)
  1957          op = O_EXISTS;
  1958       else if (strcmp(mpl->image, "setof") == 0)
  1959          op = O_SETOF;
  1960       else
  1961          error(mpl, "operator %s unknown", mpl->image);
  1962       strcpy(opstr, mpl->image);
  1963       xassert(strlen(opstr) < sizeof(opstr));
  1964       get_token(mpl /* <symbolic name> */);
  1965       /* check the left brace that follows the operator name */
  1966       xassert(mpl->token == T_LBRACE);
  1967       /* parse indexing expression that controls iterating */
  1968       arg.loop.domain = indexing_expression(mpl);
  1969       /* parse "integrand" expression and generate pseudo-code */
  1970       switch (op)
  1971       {  case O_SUM:
  1972          case O_PROD:
  1973          case O_MINIMUM:
  1974          case O_MAXIMUM:
  1975             arg.loop.x = expression_3(mpl);
  1976             /* convert the integrand to numeric type, if necessary */
  1977             if (arg.loop.x->type == A_SYMBOLIC)
  1978                arg.loop.x = make_unary(mpl, O_CVTNUM, arg.loop.x,
  1979                   A_NUMERIC, 0);
  1980             /* now the integrand must be of numeric type or linear form
  1981                (the latter is only allowed for the sum operator) */
  1982             if (!(arg.loop.x->type == A_NUMERIC ||
  1983                   op == O_SUM && arg.loop.x->type == A_FORMULA))
  1984 err:           error(mpl, "integrand following %s{...} has invalid type"
  1985                   , opstr);
  1986             xassert(arg.loop.x->dim == 0);
  1987             /* generate pseudo-code */
  1988             code = make_code(mpl, op, &arg, arg.loop.x->type, 0);
  1989             break;
  1990          case O_FORALL:
  1991          case O_EXISTS:
  1992             arg.loop.x = expression_12(mpl);
  1993             /* convert the integrand to logical type, if necessary */
  1994             if (arg.loop.x->type == A_SYMBOLIC)
  1995                arg.loop.x = make_unary(mpl, O_CVTNUM, arg.loop.x,
  1996                   A_NUMERIC, 0);
  1997             if (arg.loop.x->type == A_NUMERIC)
  1998                arg.loop.x = make_unary(mpl, O_CVTLOG, arg.loop.x,
  1999                   A_LOGICAL, 0);
  2000             /* now the integrand must be of logical type */
  2001             if (arg.loop.x->type != A_LOGICAL) goto err;
  2002             xassert(arg.loop.x->dim == 0);
  2003             /* generate pseudo-code */
  2004             code = make_code(mpl, op, &arg, A_LOGICAL, 0);
  2005             break;
  2006          case O_SETOF:
  2007             arg.loop.x = expression_5(mpl);
  2008             /* convert the integrand to 1-tuple, if necessary */
  2009             if (arg.loop.x->type == A_NUMERIC)
  2010                arg.loop.x = make_unary(mpl, O_CVTSYM, arg.loop.x,
  2011                   A_SYMBOLIC, 0);
  2012             if (arg.loop.x->type == A_SYMBOLIC)
  2013                arg.loop.x = make_unary(mpl, O_CVTTUP, arg.loop.x,
  2014                   A_TUPLE, 1);
  2015             /* now the integrand must be n-tuple */
  2016             if (arg.loop.x->type != A_TUPLE) goto err;
  2017             xassert(arg.loop.x->dim > 0);
  2018             /* generate pseudo-code */
  2019             code = make_code(mpl, op, &arg, A_ELEMSET, arg.loop.x->dim);
  2020             break;
  2021          default:
  2022             xassert(op != op);
  2023       }
  2024       /* close the scope of the indexing expression */
  2025       close_scope(mpl, arg.loop.domain);
  2026 #if 1 /* 07/IX-2008 */
  2027       link_up(code);
  2028 #endif
  2029       return code;
  2030 }
  2031 
  2032 /*----------------------------------------------------------------------
  2033 -- domain_arity - determine arity of domain.
  2034 --
  2035 -- This routine returns arity of specified domain, which is number of
  2036 -- its free dummy indices. */
  2037 
  2038 int domain_arity(MPL *mpl, DOMAIN *domain)
  2039 {     DOMAIN_BLOCK *block;
  2040       DOMAIN_SLOT *slot;
  2041       int arity;
  2042       xassert(mpl == mpl);
  2043       arity = 0;
  2044       for (block = domain->list; block != NULL; block = block->next)
  2045          for (slot = block->list; slot != NULL; slot = slot->next)
  2046             if (slot->code == NULL) arity++;
  2047       return arity;
  2048 }
  2049 
  2050 /*----------------------------------------------------------------------
  2051 -- set_expression - parse set expression.
  2052 --
  2053 -- This routine parses primary expression using the syntax:
  2054 --
  2055 -- <primary expression> ::= { }
  2056 -- <primary expression> ::= <indexing expression> */
  2057 
  2058 CODE *set_expression(MPL *mpl)
  2059 {     CODE *code;
  2060       OPERANDS arg;
  2061       xassert(mpl->token == T_LBRACE);
  2062       get_token(mpl /* { */);
  2063       /* check a token that follows the left brace */
  2064       if (mpl->token == T_RBRACE)
  2065       {  /* it is the right brace, so the resultant is an empty set of
  2066             dimension 1 */
  2067          arg.list = NULL;
  2068          /* generate pseudo-code to build the resultant set */
  2069          code = make_code(mpl, O_MAKE, &arg, A_ELEMSET, 1);
  2070          get_token(mpl /* } */);
  2071       }
  2072       else
  2073       {  /* the next token begins an indexing expression */
  2074          unget_token(mpl);
  2075          arg.loop.domain = indexing_expression(mpl);
  2076          arg.loop.x = NULL; /* integrand is not used */
  2077          /* close the scope of the indexing expression */
  2078          close_scope(mpl, arg.loop.domain);
  2079          /* generate pseudo-code to build the resultant set */
  2080          code = make_code(mpl, O_BUILD, &arg, A_ELEMSET,
  2081             domain_arity(mpl, arg.loop.domain));
  2082 #if 1 /* 07/IX-2008 */
  2083          link_up(code);
  2084 #endif
  2085       }
  2086       return code;
  2087 }
  2088 
  2089 /*----------------------------------------------------------------------
  2090 -- branched_expression - parse conditional expression.
  2091 --
  2092 -- This routine parses primary expression using the syntax:
  2093 --
  2094 -- <primary expression> ::= <branched expression>
  2095 -- <branched expression> ::= if <logical expression> then <expression 9>
  2096 -- <branched expression> ::= if <logical expression> then <expression 9>
  2097 --                           else <expression 9>
  2098 -- <logical expression> ::= <expression 13> */
  2099 
  2100 CODE *branched_expression(MPL *mpl)
  2101 {     CODE *code, *x, *y, *z;
  2102       xassert(mpl->token == T_IF);
  2103       get_token(mpl /* if */);
  2104       /* parse <logical expression> that follows 'if' */
  2105       x = expression_13(mpl);
  2106       /* convert the expression to logical type, if necessary */
  2107       if (x->type == A_SYMBOLIC)
  2108          x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2109       if (x->type == A_NUMERIC)
  2110          x = make_unary(mpl, O_CVTLOG, x, A_LOGICAL, 0);
  2111       /* now the expression must be of logical type */
  2112       if (x->type != A_LOGICAL)
  2113          error(mpl, "expression following if has invalid type");
  2114       xassert(x->dim == 0);
  2115       /* the keyword 'then' must follow the logical expression */
  2116       if (mpl->token != T_THEN)
  2117          error(mpl, "keyword then missing where expected");
  2118       get_token(mpl /* then */);
  2119       /* parse <expression> that follows 'then' and check its type */
  2120       y = expression_9(mpl);
  2121       if (!(y->type == A_NUMERIC || y->type == A_SYMBOLIC ||
  2122             y->type == A_ELEMSET || y->type == A_FORMULA))
  2123          error(mpl, "expression following then has invalid type");
  2124       /* if the expression that follows the keyword 'then' is elemental
  2125          set, the keyword 'else' cannot be omitted; otherwise else-part
  2126          is optional */
  2127       if (mpl->token != T_ELSE)
  2128       {  if (y->type == A_ELEMSET)
  2129             error(mpl, "keyword else missing where expected");
  2130          z = NULL;
  2131          goto skip;
  2132       }
  2133       get_token(mpl /* else */);
  2134       /* parse <expression> that follow 'else' and check its type */
  2135       z = expression_9(mpl);
  2136       if (!(z->type == A_NUMERIC || z->type == A_SYMBOLIC ||
  2137             z->type == A_ELEMSET || z->type == A_FORMULA))
  2138          error(mpl, "expression following else has invalid type");
  2139       /* convert to identical types, if necessary */
  2140       if (y->type == A_FORMULA || z->type == A_FORMULA)
  2141       {  if (y->type == A_SYMBOLIC)
  2142             y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0);
  2143          if (y->type == A_NUMERIC)
  2144             y = make_unary(mpl, O_CVTLFM, y, A_FORMULA, 0);
  2145          if (z->type == A_SYMBOLIC)
  2146             z = make_unary(mpl, O_CVTNUM, z, A_NUMERIC, 0);
  2147          if (z->type == A_NUMERIC)
  2148             z = make_unary(mpl, O_CVTLFM, z, A_FORMULA, 0);
  2149       }
  2150       if (y->type == A_SYMBOLIC || z->type == A_SYMBOLIC)
  2151       {  if (y->type == A_NUMERIC)
  2152             y = make_unary(mpl, O_CVTSYM, y, A_SYMBOLIC, 0);
  2153          if (z->type == A_NUMERIC)
  2154             z = make_unary(mpl, O_CVTSYM, z, A_SYMBOLIC, 0);
  2155       }
  2156       /* now both expressions must have identical types */
  2157       if (y->type != z->type)
  2158          error(mpl, "expressions following then and else have incompati"
  2159             "ble types");
  2160       /* and identical dimensions */
  2161       if (y->dim != z->dim)
  2162          error(mpl, "expressions following then and else have different"
  2163             " dimensions %d and %d, respectively", y->dim, z->dim);
  2164 skip: /* generate pseudo-code to perform branching */
  2165       code = make_ternary(mpl, O_FORK, x, y, z, y->type, y->dim);
  2166       return code;
  2167 }
  2168 
  2169 /*----------------------------------------------------------------------
  2170 -- primary_expression - parse primary expression.
  2171 --
  2172 -- This routine parses primary expression using the syntax:
  2173 --
  2174 -- <primary expression> ::= <numeric literal>
  2175 -- <primary expression> ::= Infinity
  2176 -- <primary expression> ::= <string literal>
  2177 -- <primary expression> ::= <dummy index>
  2178 -- <primary expression> ::= <set name>
  2179 -- <primary expression> ::= <set name> [ <subscript list> ]
  2180 -- <primary expression> ::= <parameter name>
  2181 -- <primary expression> ::= <parameter name> [ <subscript list> ]
  2182 -- <primary expression> ::= <variable name>
  2183 -- <primary expression> ::= <variable name> [ <subscript list> ]
  2184 -- <primary expression> ::= <built-in function> ( <argument list> )
  2185 -- <primary expression> ::= ( <expression list> )
  2186 -- <primary expression> ::= <iterated expression>
  2187 -- <primary expression> ::= { }
  2188 -- <primary expression> ::= <indexing expression>
  2189 -- <primary expression> ::= <branched expression>
  2190 --
  2191 -- For complete list of syntactic rules for <primary expression> see
  2192 -- comments to the corresponding parsing routines. */
  2193 
  2194 CODE *primary_expression(MPL *mpl)
  2195 {     CODE *code;
  2196       if (mpl->token == T_NUMBER)
  2197       {  /* parse numeric literal */
  2198          code = numeric_literal(mpl);
  2199       }
  2200 #if 1 /* 21/VII-2006 */
  2201       else if (mpl->token == T_INFINITY)
  2202       {  /* parse "infinity" */
  2203          OPERANDS arg;
  2204          arg.num = DBL_MAX;
  2205          code = make_code(mpl, O_NUMBER, &arg, A_NUMERIC, 0);
  2206          get_token(mpl /* Infinity */);
  2207       }
  2208 #endif
  2209       else if (mpl->token == T_STRING)
  2210       {  /* parse string literal */
  2211          code = string_literal(mpl);
  2212       }
  2213       else if (mpl->token == T_NAME)
  2214       {  int next_token;
  2215          get_token(mpl /* <symbolic name> */);
  2216          next_token = mpl->token;
  2217          unget_token(mpl);
  2218          /* check a token that follows <symbolic name> */
  2219          switch (next_token)
  2220          {  case T_LBRACKET:
  2221                /* parse reference to subscripted object */
  2222                code = object_reference(mpl);
  2223                break;
  2224             case T_LEFT:
  2225                /* parse reference to built-in function */
  2226                code = function_reference(mpl);
  2227                break;
  2228             case T_LBRACE:
  2229                /* parse iterated expression */
  2230                code = iterated_expression(mpl);
  2231                break;
  2232             default:
  2233                /* parse reference to unsubscripted object */
  2234                code = object_reference(mpl);
  2235                break;
  2236          }
  2237       }
  2238       else if (mpl->token == T_LEFT)
  2239       {  /* parse parenthesized expression */
  2240          code = expression_list(mpl);
  2241       }
  2242       else if (mpl->token == T_LBRACE)
  2243       {  /* parse set expression */
  2244          code = set_expression(mpl);
  2245       }
  2246       else if (mpl->token == T_IF)
  2247       {  /* parse conditional expression */
  2248          code = branched_expression(mpl);
  2249       }
  2250       else if (is_reserved(mpl))
  2251       {  /* other reserved keywords cannot be used here */
  2252          error(mpl, "invalid use of reserved keyword %s", mpl->image);
  2253       }
  2254       else
  2255          error(mpl, "syntax error in expression");
  2256       return code;
  2257 }
  2258 
  2259 /*----------------------------------------------------------------------
  2260 -- error_preceding - raise error if preceding operand has wrong type.
  2261 --
  2262 -- This routine is called to raise error if operand that precedes some
  2263 -- infix operator has invalid type. */
  2264 
  2265 void error_preceding(MPL *mpl, char *opstr)
  2266 {     error(mpl, "operand preceding %s has invalid type", opstr);
  2267       /* no return */
  2268 }
  2269 
  2270 /*----------------------------------------------------------------------
  2271 -- error_following - raise error if following operand has wrong type.
  2272 --
  2273 -- This routine is called to raise error if operand that follows some
  2274 -- infix operator has invalid type. */
  2275 
  2276 void error_following(MPL *mpl, char *opstr)
  2277 {     error(mpl, "operand following %s has invalid type", opstr);
  2278       /* no return */
  2279 }
  2280 
  2281 /*----------------------------------------------------------------------
  2282 -- error_dimension - raise error if operands have different dimension.
  2283 --
  2284 -- This routine is called to raise error if two operands of some infix
  2285 -- operator have different dimension. */
  2286 
  2287 void error_dimension(MPL *mpl, char *opstr, int dim1, int dim2)
  2288 {     error(mpl, "operands preceding and following %s have different di"
  2289          "mensions %d and %d, respectively", opstr, dim1, dim2);
  2290       /* no return */
  2291 }
  2292 
  2293 /*----------------------------------------------------------------------
  2294 -- expression_0 - parse expression of level 0.
  2295 --
  2296 -- This routine parses expression of level 0 using the syntax:
  2297 --
  2298 -- <expression 0> ::= <primary expression> */
  2299 
  2300 CODE *expression_0(MPL *mpl)
  2301 {     CODE *code;
  2302       code = primary_expression(mpl);
  2303       return code;
  2304 }
  2305 
  2306 /*----------------------------------------------------------------------
  2307 -- expression_1 - parse expression of level 1.
  2308 --
  2309 -- This routine parses expression of level 1 using the syntax:
  2310 --
  2311 -- <expression 1> ::= <expression 0>
  2312 -- <expression 1> ::= <expression 0> <power> <expression 1>
  2313 -- <expression 1> ::= <expression 0> <power> <expression 2>
  2314 -- <power> ::= ^ | ** */
  2315 
  2316 CODE *expression_1(MPL *mpl)
  2317 {     CODE *x, *y;
  2318       char opstr[8];
  2319       x = expression_0(mpl);
  2320       if (mpl->token == T_POWER)
  2321       {  strcpy(opstr, mpl->image);
  2322          xassert(strlen(opstr) < sizeof(opstr));
  2323          if (x->type == A_SYMBOLIC)
  2324             x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2325          if (x->type != A_NUMERIC)
  2326             error_preceding(mpl, opstr);
  2327          get_token(mpl /* ^ | ** */);
  2328          if (mpl->token == T_PLUS || mpl->token == T_MINUS)
  2329             y = expression_2(mpl);
  2330          else
  2331             y = expression_1(mpl);
  2332          if (y->type == A_SYMBOLIC)
  2333             y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0);
  2334          if (y->type != A_NUMERIC)
  2335             error_following(mpl, opstr);
  2336          x = make_binary(mpl, O_POWER, x, y, A_NUMERIC, 0);
  2337       }
  2338       return x;
  2339 }
  2340 
  2341 /*----------------------------------------------------------------------
  2342 -- expression_2 - parse expression of level 2.
  2343 --
  2344 -- This routine parses expression of level 2 using the syntax:
  2345 --
  2346 -- <expression 2> ::= <expression 1>
  2347 -- <expression 2> ::= + <expression 1>
  2348 -- <expression 2> ::= - <expression 1> */
  2349 
  2350 CODE *expression_2(MPL *mpl)
  2351 {     CODE *x;
  2352       if (mpl->token == T_PLUS)
  2353       {  get_token(mpl /* + */);
  2354          x = expression_1(mpl);
  2355          if (x->type == A_SYMBOLIC)
  2356             x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2357          if (!(x->type == A_NUMERIC || x->type == A_FORMULA))
  2358             error_following(mpl, "+");
  2359          x = make_unary(mpl, O_PLUS, x, x->type, 0);
  2360       }
  2361       else if (mpl->token == T_MINUS)
  2362       {  get_token(mpl /* - */);
  2363          x = expression_1(mpl);
  2364          if (x->type == A_SYMBOLIC)
  2365             x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2366          if (!(x->type == A_NUMERIC || x->type == A_FORMULA))
  2367             error_following(mpl, "-");
  2368          x = make_unary(mpl, O_MINUS, x, x->type, 0);
  2369       }
  2370       else
  2371          x = expression_1(mpl);
  2372       return x;
  2373 }
  2374 
  2375 /*----------------------------------------------------------------------
  2376 -- expression_3 - parse expression of level 3.
  2377 --
  2378 -- This routine parses expression of level 3 using the syntax:
  2379 --
  2380 -- <expression 3> ::= <expression 2>
  2381 -- <expression 3> ::= <expression 3> * <expression 2>
  2382 -- <expression 3> ::= <expression 3> / <expression 2>
  2383 -- <expression 3> ::= <expression 3> div <expression 2>
  2384 -- <expression 3> ::= <expression 3> mod <expression 2> */
  2385 
  2386 CODE *expression_3(MPL *mpl)
  2387 {     CODE *x, *y;
  2388       x = expression_2(mpl);
  2389       for (;;)
  2390       {  if (mpl->token == T_ASTERISK)
  2391          {  if (x->type == A_SYMBOLIC)
  2392                x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2393             if (!(x->type == A_NUMERIC || x->type == A_FORMULA))
  2394                error_preceding(mpl, "*");
  2395             get_token(mpl /* * */);
  2396             y = expression_2(mpl);
  2397             if (y->type == A_SYMBOLIC)
  2398                y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0);
  2399             if (!(y->type == A_NUMERIC || y->type == A_FORMULA))
  2400                error_following(mpl, "*");
  2401             if (x->type == A_FORMULA && y->type == A_FORMULA)
  2402                error(mpl, "multiplication of linear forms not allowed");
  2403             if (x->type == A_NUMERIC && y->type == A_NUMERIC)
  2404                x = make_binary(mpl, O_MUL, x, y, A_NUMERIC, 0);
  2405             else
  2406                x = make_binary(mpl, O_MUL, x, y, A_FORMULA, 0);
  2407          }
  2408          else if (mpl->token == T_SLASH)
  2409          {  if (x->type == A_SYMBOLIC)
  2410                x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2411             if (!(x->type == A_NUMERIC || x->type == A_FORMULA))
  2412                error_preceding(mpl, "/");
  2413             get_token(mpl /* / */);
  2414             y = expression_2(mpl);
  2415             if (y->type == A_SYMBOLIC)
  2416                y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0);
  2417             if (y->type != A_NUMERIC)
  2418                error_following(mpl, "/");
  2419             if (x->type == A_NUMERIC)
  2420                x = make_binary(mpl, O_DIV, x, y, A_NUMERIC, 0);
  2421             else
  2422                x = make_binary(mpl, O_DIV, x, y, A_FORMULA, 0);
  2423          }
  2424          else if (mpl->token == T_DIV)
  2425          {  if (x->type == A_SYMBOLIC)
  2426                x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2427             if (x->type != A_NUMERIC)
  2428                error_preceding(mpl, "div");
  2429             get_token(mpl /* div */);
  2430             y = expression_2(mpl);
  2431             if (y->type == A_SYMBOLIC)
  2432                y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0);
  2433             if (y->type != A_NUMERIC)
  2434                error_following(mpl, "div");
  2435             x = make_binary(mpl, O_IDIV, x, y, A_NUMERIC, 0);
  2436          }
  2437          else if (mpl->token == T_MOD)
  2438          {  if (x->type == A_SYMBOLIC)
  2439                x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2440             if (x->type != A_NUMERIC)
  2441                error_preceding(mpl, "mod");
  2442             get_token(mpl /* mod */);
  2443             y = expression_2(mpl);
  2444             if (y->type == A_SYMBOLIC)
  2445                y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0);
  2446             if (y->type != A_NUMERIC)
  2447                error_following(mpl, "mod");
  2448             x = make_binary(mpl, O_MOD, x, y, A_NUMERIC, 0);
  2449          }
  2450          else
  2451             break;
  2452       }
  2453       return x;
  2454 }
  2455 
  2456 /*----------------------------------------------------------------------
  2457 -- expression_4 - parse expression of level 4.
  2458 --
  2459 -- This routine parses expression of level 4 using the syntax:
  2460 --
  2461 -- <expression 4> ::= <expression 3>
  2462 -- <expression 4> ::= <expression 4> + <expression 3>
  2463 -- <expression 4> ::= <expression 4> - <expression 3>
  2464 -- <expression 4> ::= <expression 4> less <expression 3> */
  2465 
  2466 CODE *expression_4(MPL *mpl)
  2467 {     CODE *x, *y;
  2468       x = expression_3(mpl);
  2469       for (;;)
  2470       {  if (mpl->token == T_PLUS)
  2471          {  if (x->type == A_SYMBOLIC)
  2472                x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2473             if (!(x->type == A_NUMERIC || x->type == A_FORMULA))
  2474                error_preceding(mpl, "+");
  2475             get_token(mpl /* + */);
  2476             y = expression_3(mpl);
  2477             if (y->type == A_SYMBOLIC)
  2478                y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0);
  2479             if (!(y->type == A_NUMERIC || y->type == A_FORMULA))
  2480                error_following(mpl, "+");
  2481             if (x->type == A_NUMERIC && y->type == A_FORMULA)
  2482                x = make_unary(mpl, O_CVTLFM, x, A_FORMULA, 0);
  2483             if (x->type == A_FORMULA && y->type == A_NUMERIC)
  2484                y = make_unary(mpl, O_CVTLFM, y, A_FORMULA, 0);
  2485             x = make_binary(mpl, O_ADD, x, y, x->type, 0);
  2486          }
  2487          else if (mpl->token == T_MINUS)
  2488          {  if (x->type == A_SYMBOLIC)
  2489                x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2490             if (!(x->type == A_NUMERIC || x->type == A_FORMULA))
  2491                error_preceding(mpl, "-");
  2492             get_token(mpl /* - */);
  2493             y = expression_3(mpl);
  2494             if (y->type == A_SYMBOLIC)
  2495                y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0);
  2496             if (!(y->type == A_NUMERIC || y->type == A_FORMULA))
  2497                error_following(mpl, "-");
  2498             if (x->type == A_NUMERIC && y->type == A_FORMULA)
  2499                x = make_unary(mpl, O_CVTLFM, x, A_FORMULA, 0);
  2500             if (x->type == A_FORMULA && y->type == A_NUMERIC)
  2501                y = make_unary(mpl, O_CVTLFM, y, A_FORMULA, 0);
  2502             x = make_binary(mpl, O_SUB, x, y, x->type, 0);
  2503          }
  2504          else if (mpl->token == T_LESS)
  2505          {  if (x->type == A_SYMBOLIC)
  2506                x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2507             if (x->type != A_NUMERIC)
  2508                error_preceding(mpl, "less");
  2509             get_token(mpl /* less */);
  2510             y = expression_3(mpl);
  2511             if (y->type == A_SYMBOLIC)
  2512                y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0);
  2513             if (y->type != A_NUMERIC)
  2514                error_following(mpl, "less");
  2515             x = make_binary(mpl, O_LESS, x, y, A_NUMERIC, 0);
  2516          }
  2517          else
  2518             break;
  2519       }
  2520       return x;
  2521 }
  2522 
  2523 /*----------------------------------------------------------------------
  2524 -- expression_5 - parse expression of level 5.
  2525 --
  2526 -- This routine parses expression of level 5 using the syntax:
  2527 --
  2528 -- <expression 5> ::= <expression 4>
  2529 -- <expression 5> ::= <expression 5> & <expression 4> */
  2530 
  2531 CODE *expression_5(MPL *mpl)
  2532 {     CODE *x, *y;
  2533       x = expression_4(mpl);
  2534       for (;;)
  2535       {  if (mpl->token == T_CONCAT)
  2536          {  if (x->type == A_NUMERIC)
  2537                x = make_unary(mpl, O_CVTSYM, x, A_SYMBOLIC, 0);
  2538             if (x->type != A_SYMBOLIC)
  2539                error_preceding(mpl, "&");
  2540             get_token(mpl /* & */);
  2541             y = expression_4(mpl);
  2542             if (y->type == A_NUMERIC)
  2543                y = make_unary(mpl, O_CVTSYM, y, A_SYMBOLIC, 0);
  2544             if (y->type != A_SYMBOLIC)
  2545                error_following(mpl, "&");
  2546             x = make_binary(mpl, O_CONCAT, x, y, A_SYMBOLIC, 0);
  2547          }
  2548          else
  2549             break;
  2550       }
  2551       return x;
  2552 }
  2553 
  2554 /*----------------------------------------------------------------------
  2555 -- expression_6 - parse expression of level 6.
  2556 --
  2557 -- This routine parses expression of level 6 using the syntax:
  2558 --
  2559 -- <expression 6> ::= <expression 5>
  2560 -- <expression 6> ::= <expression 5> .. <expression 5>
  2561 -- <expression 6> ::= <expression 5> .. <expression 5> by
  2562 --                    <expression 5> */
  2563 
  2564 CODE *expression_6(MPL *mpl)
  2565 {     CODE *x, *y, *z;
  2566       x = expression_5(mpl);
  2567       if (mpl->token == T_DOTS)
  2568       {  if (x->type == A_SYMBOLIC)
  2569             x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2570          if (x->type != A_NUMERIC)
  2571             error_preceding(mpl, "..");
  2572          get_token(mpl /* .. */);
  2573          y = expression_5(mpl);
  2574          if (y->type == A_SYMBOLIC)
  2575             y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0);
  2576          if (y->type != A_NUMERIC)
  2577             error_following(mpl, "..");
  2578          if (mpl->token == T_BY)
  2579          {  get_token(mpl /* by */);
  2580             z = expression_5(mpl);
  2581             if (z->type == A_SYMBOLIC)
  2582                z = make_unary(mpl, O_CVTNUM, z, A_NUMERIC, 0);
  2583             if (z->type != A_NUMERIC)
  2584                error_following(mpl, "by");
  2585          }
  2586          else
  2587             z = NULL;
  2588          x = make_ternary(mpl, O_DOTS, x, y, z, A_ELEMSET, 1);
  2589       }
  2590       return x;
  2591 }
  2592 
  2593 /*----------------------------------------------------------------------
  2594 -- expression_7 - parse expression of level 7.
  2595 --
  2596 -- This routine parses expression of level 7 using the syntax:
  2597 --
  2598 -- <expression 7> ::= <expression 6>
  2599 -- <expression 7> ::= <expression 7> cross <expression 6> */
  2600 
  2601 CODE *expression_7(MPL *mpl)
  2602 {     CODE *x, *y;
  2603       x = expression_6(mpl);
  2604       for (;;)
  2605       {  if (mpl->token == T_CROSS)
  2606          {  if (x->type != A_ELEMSET)
  2607                error_preceding(mpl, "cross");
  2608             get_token(mpl /* cross */);
  2609             y = expression_6(mpl);
  2610             if (y->type != A_ELEMSET)
  2611                error_following(mpl, "cross");
  2612             x = make_binary(mpl, O_CROSS, x, y, A_ELEMSET,
  2613                x->dim + y->dim);
  2614          }
  2615          else
  2616             break;
  2617       }
  2618       return x;
  2619 }
  2620 
  2621 /*----------------------------------------------------------------------
  2622 -- expression_8 - parse expression of level 8.
  2623 --
  2624 -- This routine parses expression of level 8 using the syntax:
  2625 --
  2626 -- <expression 8> ::= <expression 7>
  2627 -- <expression 8> ::= <expression 8> inter <expression 7> */
  2628 
  2629 CODE *expression_8(MPL *mpl)
  2630 {     CODE *x, *y;
  2631       x = expression_7(mpl);
  2632       for (;;)
  2633       {  if (mpl->token == T_INTER)
  2634          {  if (x->type != A_ELEMSET)
  2635                error_preceding(mpl, "inter");
  2636             get_token(mpl /* inter */);
  2637             y = expression_7(mpl);
  2638             if (y->type != A_ELEMSET)
  2639                error_following(mpl, "inter");
  2640             if (x->dim != y->dim)
  2641                error_dimension(mpl, "inter", x->dim, y->dim);
  2642             x = make_binary(mpl, O_INTER, x, y, A_ELEMSET, x->dim);
  2643          }
  2644          else
  2645             break;
  2646       }
  2647       return x;
  2648 }
  2649 
  2650 /*----------------------------------------------------------------------
  2651 -- expression_9 - parse expression of level 9.
  2652 --
  2653 -- This routine parses expression of level 9 using the syntax:
  2654 --
  2655 -- <expression 9> ::= <expression 8>
  2656 -- <expression 9> ::= <expression 9> union <expression 8>
  2657 -- <expression 9> ::= <expression 9> diff <expression 8>
  2658 -- <expression 9> ::= <expression 9> symdiff <expression 8> */
  2659 
  2660 CODE *expression_9(MPL *mpl)
  2661 {     CODE *x, *y;
  2662       x = expression_8(mpl);
  2663       for (;;)
  2664       {  if (mpl->token == T_UNION)
  2665          {  if (x->type != A_ELEMSET)
  2666                error_preceding(mpl, "union");
  2667             get_token(mpl /* union */);
  2668             y = expression_8(mpl);
  2669             if (y->type != A_ELEMSET)
  2670                error_following(mpl, "union");
  2671             if (x->dim != y->dim)
  2672                error_dimension(mpl, "union", x->dim, y->dim);
  2673             x = make_binary(mpl, O_UNION, x, y, A_ELEMSET, x->dim);
  2674          }
  2675          else if (mpl->token == T_DIFF)
  2676          {  if (x->type != A_ELEMSET)
  2677                error_preceding(mpl, "diff");
  2678             get_token(mpl /* diff */);
  2679             y = expression_8(mpl);
  2680             if (y->type != A_ELEMSET)
  2681                error_following(mpl, "diff");
  2682             if (x->dim != y->dim)
  2683                error_dimension(mpl, "diff", x->dim, y->dim);
  2684             x = make_binary(mpl, O_DIFF, x, y, A_ELEMSET, x->dim);
  2685          }
  2686          else if (mpl->token == T_SYMDIFF)
  2687          {  if (x->type != A_ELEMSET)
  2688                error_preceding(mpl, "symdiff");
  2689             get_token(mpl /* symdiff */);
  2690             y = expression_8(mpl);
  2691             if (y->type != A_ELEMSET)
  2692                error_following(mpl, "symdiff");
  2693             if (x->dim != y->dim)
  2694                error_dimension(mpl, "symdiff", x->dim, y->dim);
  2695             x = make_binary(mpl, O_SYMDIFF, x, y, A_ELEMSET, x->dim);
  2696          }
  2697          else
  2698             break;
  2699       }
  2700       return x;
  2701 }
  2702 
  2703 /*----------------------------------------------------------------------
  2704 -- expression_10 - parse expression of level 10.
  2705 --
  2706 -- This routine parses expression of level 10 using the syntax:
  2707 --
  2708 -- <expression 10> ::= <expression 9>
  2709 -- <expression 10> ::= <expression 9> <rho> <expression 9>
  2710 -- <rho> ::= < | <= | = | == | >= | > | <> | != | in | not in | ! in |
  2711 --           within | not within | ! within */
  2712 
  2713 CODE *expression_10(MPL *mpl)
  2714 {     CODE *x, *y;
  2715       int op = -1;
  2716       char opstr[16];
  2717       x = expression_9(mpl);
  2718       strcpy(opstr, "");
  2719       switch (mpl->token)
  2720       {  case T_LT:
  2721             op = O_LT; break;
  2722          case T_LE:
  2723             op = O_LE; break;
  2724          case T_EQ:
  2725             op = O_EQ; break;
  2726          case T_GE:
  2727             op = O_GE; break;
  2728          case T_GT:
  2729             op = O_GT; break;
  2730          case T_NE:
  2731             op = O_NE; break;
  2732          case T_IN:
  2733             op = O_IN; break;
  2734          case T_WITHIN:
  2735             op = O_WITHIN; break;
  2736          case T_NOT:
  2737             strcpy(opstr, mpl->image);
  2738             get_token(mpl /* not | ! */);
  2739             if (mpl->token == T_IN)
  2740                op = O_NOTIN;
  2741             else if (mpl->token == T_WITHIN)
  2742                op = O_NOTWITHIN;
  2743             else
  2744                error(mpl, "invalid use of %s", opstr);
  2745             strcat(opstr, " ");
  2746             break;
  2747          default:
  2748             goto done;
  2749       }
  2750       strcat(opstr, mpl->image);
  2751       xassert(strlen(opstr) < sizeof(opstr));
  2752       switch (op)
  2753       {  case O_EQ:
  2754          case O_NE:
  2755 #if 1 /* 02/VIII-2008 */
  2756          case O_LT:
  2757          case O_LE:
  2758          case O_GT:
  2759          case O_GE:
  2760 #endif
  2761             if (!(x->type == A_NUMERIC || x->type == A_SYMBOLIC))
  2762                error_preceding(mpl, opstr);
  2763             get_token(mpl /* <rho> */);
  2764             y = expression_9(mpl);
  2765             if (!(y->type == A_NUMERIC || y->type == A_SYMBOLIC))
  2766                error_following(mpl, opstr);
  2767             if (x->type == A_NUMERIC && y->type == A_SYMBOLIC)
  2768                x = make_unary(mpl, O_CVTSYM, x, A_SYMBOLIC, 0);
  2769             if (x->type == A_SYMBOLIC && y->type == A_NUMERIC)
  2770                y = make_unary(mpl, O_CVTSYM, y, A_SYMBOLIC, 0);
  2771             x = make_binary(mpl, op, x, y, A_LOGICAL, 0);
  2772             break;
  2773 #if 0 /* 02/VIII-2008 */
  2774          case O_LT:
  2775          case O_LE:
  2776          case O_GT:
  2777          case O_GE:
  2778             if (x->type == A_SYMBOLIC)
  2779                x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2780             if (x->type != A_NUMERIC)
  2781                error_preceding(mpl, opstr);
  2782             get_token(mpl /* <rho> */);
  2783             y = expression_9(mpl);
  2784             if (y->type == A_SYMBOLIC)
  2785                y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0);
  2786             if (y->type != A_NUMERIC)
  2787                error_following(mpl, opstr);
  2788             x = make_binary(mpl, op, x, y, A_LOGICAL, 0);
  2789             break;
  2790 #endif
  2791          case O_IN:
  2792          case O_NOTIN:
  2793             if (x->type == A_NUMERIC)
  2794                x = make_unary(mpl, O_CVTSYM, x, A_SYMBOLIC, 0);
  2795             if (x->type == A_SYMBOLIC)
  2796                x = make_unary(mpl, O_CVTTUP, x, A_TUPLE, 1);
  2797             if (x->type != A_TUPLE)
  2798                error_preceding(mpl, opstr);
  2799             get_token(mpl /* <rho> */);
  2800             y = expression_9(mpl);
  2801             if (y->type != A_ELEMSET)
  2802                error_following(mpl, opstr);
  2803             if (x->dim != y->dim)
  2804                error_dimension(mpl, opstr, x->dim, y->dim);
  2805             x = make_binary(mpl, op, x, y, A_LOGICAL, 0);
  2806             break;
  2807          case O_WITHIN:
  2808          case O_NOTWITHIN:
  2809             if (x->type != A_ELEMSET)
  2810                error_preceding(mpl, opstr);
  2811             get_token(mpl /* <rho> */);
  2812             y = expression_9(mpl);
  2813             if (y->type != A_ELEMSET)
  2814                error_following(mpl, opstr);
  2815             if (x->dim != y->dim)
  2816                error_dimension(mpl, opstr, x->dim, y->dim);
  2817             x = make_binary(mpl, op, x, y, A_LOGICAL, 0);
  2818             break;
  2819          default:
  2820             xassert(op != op);
  2821       }
  2822 done: return x;
  2823 }
  2824 
  2825 /*----------------------------------------------------------------------
  2826 -- expression_11 - parse expression of level 11.
  2827 --
  2828 -- This routine parses expression of level 11 using the syntax:
  2829 --
  2830 -- <expression 11> ::= <expression 10>
  2831 -- <expression 11> ::= not <expression 10>
  2832 -- <expression 11> ::= ! <expression 10> */
  2833 
  2834 CODE *expression_11(MPL *mpl)
  2835 {     CODE *x;
  2836       char opstr[8];
  2837       if (mpl->token == T_NOT)
  2838       {  strcpy(opstr, mpl->image);
  2839          xassert(strlen(opstr) < sizeof(opstr));
  2840          get_token(mpl /* not | ! */);
  2841          x = expression_10(mpl);
  2842          if (x->type == A_SYMBOLIC)
  2843             x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2844          if (x->type == A_NUMERIC)
  2845             x = make_unary(mpl, O_CVTLOG, x, A_LOGICAL, 0);
  2846          if (x->type != A_LOGICAL)
  2847             error_following(mpl, opstr);
  2848          x = make_unary(mpl, O_NOT, x, A_LOGICAL, 0);
  2849       }
  2850       else
  2851          x = expression_10(mpl);
  2852       return x;
  2853 }
  2854 
  2855 /*----------------------------------------------------------------------
  2856 -- expression_12 - parse expression of level 12.
  2857 --
  2858 -- This routine parses expression of level 12 using the syntax:
  2859 --
  2860 -- <expression 12> ::= <expression 11>
  2861 -- <expression 12> ::= <expression 12> and <expression 11>
  2862 -- <expression 12> ::= <expression 12> && <expression 11> */
  2863 
  2864 CODE *expression_12(MPL *mpl)
  2865 {     CODE *x, *y;
  2866       char opstr[8];
  2867       x = expression_11(mpl);
  2868       for (;;)
  2869       {  if (mpl->token == T_AND)
  2870          {  strcpy(opstr, mpl->image);
  2871             xassert(strlen(opstr) < sizeof(opstr));
  2872             if (x->type == A_SYMBOLIC)
  2873                x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2874             if (x->type == A_NUMERIC)
  2875                x = make_unary(mpl, O_CVTLOG, x, A_LOGICAL, 0);
  2876             if (x->type != A_LOGICAL)
  2877                error_preceding(mpl, opstr);
  2878             get_token(mpl /* and | && */);
  2879             y = expression_11(mpl);
  2880             if (y->type == A_SYMBOLIC)
  2881                y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0);
  2882             if (y->type == A_NUMERIC)
  2883                y = make_unary(mpl, O_CVTLOG, y, A_LOGICAL, 0);
  2884             if (y->type != A_LOGICAL)
  2885                error_following(mpl, opstr);
  2886             x = make_binary(mpl, O_AND, x, y, A_LOGICAL, 0);
  2887          }
  2888          else
  2889             break;
  2890       }
  2891       return x;
  2892 }
  2893 
  2894 /*----------------------------------------------------------------------
  2895 -- expression_13 - parse expression of level 13.
  2896 --
  2897 -- This routine parses expression of level 13 using the syntax:
  2898 --
  2899 -- <expression 13> ::= <expression 12>
  2900 -- <expression 13> ::= <expression 13> or <expression 12>
  2901 -- <expression 13> ::= <expression 13> || <expression 12> */
  2902 
  2903 CODE *expression_13(MPL *mpl)
  2904 {     CODE *x, *y;
  2905       char opstr[8];
  2906       x = expression_12(mpl);
  2907       for (;;)
  2908       {  if (mpl->token == T_OR)
  2909          {  strcpy(opstr, mpl->image);
  2910             xassert(strlen(opstr) < sizeof(opstr));
  2911             if (x->type == A_SYMBOLIC)
  2912                x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0);
  2913             if (x->type == A_NUMERIC)
  2914                x = make_unary(mpl, O_CVTLOG, x, A_LOGICAL, 0);
  2915             if (x->type != A_LOGICAL)
  2916                error_preceding(mpl, opstr);
  2917             get_token(mpl /* or | || */);
  2918             y = expression_12(mpl);
  2919             if (y->type == A_SYMBOLIC)
  2920                y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0);
  2921             if (y->type == A_NUMERIC)
  2922                y = make_unary(mpl, O_CVTLOG, y, A_LOGICAL, 0);
  2923             if (y->type != A_LOGICAL)
  2924                error_following(mpl, opstr);
  2925             x = make_binary(mpl, O_OR, x, y, A_LOGICAL, 0);
  2926          }
  2927          else
  2928             break;
  2929       }
  2930       return x;
  2931 }
  2932 
  2933 /*----------------------------------------------------------------------
  2934 -- set_statement - parse set statement.
  2935 --
  2936 -- This routine parses set statement using the syntax:
  2937 --
  2938 -- <set statement> ::= set <symbolic name> <alias> <domain>
  2939 --                     <attributes> ;
  2940 -- <alias> ::= <empty>
  2941 -- <alias> ::= <string literal>
  2942 -- <domain> ::= <empty>
  2943 -- <domain> ::= <indexing expression>
  2944 -- <attributes> ::= <empty>
  2945 -- <attributes> ::= <attributes> , dimen <numeric literal>
  2946 -- <attributes> ::= <attributes> , within <expression 9>
  2947 -- <attributes> ::= <attributes> , := <expression 9>
  2948 -- <attributes> ::= <attributes> , default <expression 9>
  2949 --
  2950 -- Commae in <attributes> are optional and may be omitted anywhere. */
  2951 
  2952 SET *set_statement(MPL *mpl)
  2953 {     SET *set;
  2954       int dimen_used = 0;
  2955       xassert(is_keyword(mpl, "set"));
  2956       get_token(mpl /* set */);
  2957       /* symbolic name must follow the keyword 'set' */
  2958       if (mpl->token == T_NAME)
  2959          ;
  2960       else if (is_reserved(mpl))
  2961          error(mpl, "invalid use of reserved keyword %s", mpl->image);
  2962       else
  2963          error(mpl, "symbolic name missing where expected");
  2964       /* there must be no other object with the same name */
  2965       if (avl_find_node(mpl->tree, mpl->image) != NULL)
  2966          error(mpl, "%s multiply declared", mpl->image);
  2967       /* create model set */
  2968       set = alloc(SET);
  2969       set->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  2970       strcpy(set->name, mpl->image);
  2971       set->alias = NULL;
  2972       set->dim = 0;
  2973       set->domain = NULL;
  2974       set->dimen = 0;
  2975       set->within = NULL;
  2976       set->assign = NULL;
  2977       set->option = NULL;
  2978       set->gadget = NULL;
  2979       set->data = 0;
  2980       set->array = NULL;
  2981       get_token(mpl /* <symbolic name> */);
  2982       /* parse optional alias */
  2983       if (mpl->token == T_STRING)
  2984       {  set->alias = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  2985          strcpy(set->alias, mpl->image);
  2986          get_token(mpl /* <string literal> */);
  2987       }
  2988       /* parse optional indexing expression */
  2989       if (mpl->token == T_LBRACE)
  2990       {  set->domain = indexing_expression(mpl);
  2991          set->dim = domain_arity(mpl, set->domain);
  2992       }
  2993       /* include the set name in the symbolic names table */
  2994       {  AVLNODE *node;
  2995          node = avl_insert_node(mpl->tree, set->name);
  2996          avl_set_node_type(node, A_SET);
  2997          avl_set_node_link(node, (void *)set);
  2998       }
  2999       /* parse the list of optional attributes */
  3000       for (;;)
  3001       {  if (mpl->token == T_COMMA)
  3002             get_token(mpl /* , */);
  3003          else if (mpl->token == T_SEMICOLON)
  3004             break;
  3005          if (is_keyword(mpl, "dimen"))
  3006          {  /* dimension of set members */
  3007             int dimen;
  3008             get_token(mpl /* dimen */);
  3009             if (!(mpl->token == T_NUMBER &&
  3010                   1.0 <= mpl->value && mpl->value <= 20.0 &&
  3011                   floor(mpl->value) == mpl->value))
  3012                error(mpl, "dimension must be integer between 1 and 20");
  3013             dimen = (int)(mpl->value + 0.5);
  3014             if (dimen_used)
  3015                error(mpl, "at most one dimension attribute allowed");
  3016             if (set->dimen > 0)
  3017                error(mpl, "dimension %d conflicts with dimension %d alr"
  3018                   "eady determined", dimen, set->dimen);
  3019             set->dimen = dimen;
  3020             dimen_used = 1;
  3021             get_token(mpl /* <numeric literal> */);
  3022          }
  3023          else if (mpl->token == T_WITHIN || mpl->token == T_IN)
  3024          {  /* restricting superset */
  3025             WITHIN *within, *temp;
  3026             if (mpl->token == T_IN && !mpl->as_within)
  3027             {  warning(mpl, "keyword in understood as within");
  3028                mpl->as_within = 1;
  3029             }
  3030             get_token(mpl /* within */);
  3031             /* create new restricting superset list entry and append it
  3032                to the within-list */
  3033             within = alloc(WITHIN);
  3034             within->code = NULL;
  3035             within->next = NULL;
  3036             if (set->within == NULL)
  3037                set->within = within;
  3038             else
  3039             {  for (temp = set->within; temp->next != NULL; temp =
  3040                   temp->next);
  3041                temp->next = within;
  3042             }
  3043             /* parse an expression that follows 'within' */
  3044             within->code = expression_9(mpl);
  3045             if (within->code->type != A_ELEMSET)
  3046                error(mpl, "expression following within has invalid type"
  3047                   );
  3048             xassert(within->code->dim > 0);
  3049             /* check/set dimension of set members */
  3050             if (set->dimen == 0) set->dimen = within->code->dim;
  3051             if (set->dimen != within->code->dim)
  3052                error(mpl, "set expression following within must have di"
  3053                   "mension %d rather than %d",
  3054                   set->dimen, within->code->dim);
  3055          }
  3056          else if (mpl->token == T_ASSIGN)
  3057          {  /* assignment expression */
  3058             if (!(set->assign == NULL && set->option == NULL &&
  3059                   set->gadget == NULL))
  3060 err:           error(mpl, "at most one := or default/data allowed");
  3061             get_token(mpl /* := */);
  3062             /* parse an expression that follows ':=' */
  3063             set->assign = expression_9(mpl);
  3064             if (set->assign->type != A_ELEMSET)
  3065                error(mpl, "expression following := has invalid type");
  3066             xassert(set->assign->dim > 0);
  3067             /* check/set dimension of set members */
  3068             if (set->dimen == 0) set->dimen = set->assign->dim;
  3069             if (set->dimen != set->assign->dim)
  3070                error(mpl, "set expression following := must have dimens"
  3071                   "ion %d rather than %d",
  3072                   set->dimen, set->assign->dim);
  3073          }
  3074          else if (is_keyword(mpl, "default"))
  3075          {  /* expression for default value */
  3076             if (!(set->assign == NULL && set->option == NULL)) goto err;
  3077             get_token(mpl /* := */);
  3078             /* parse an expression that follows 'default' */
  3079             set->option = expression_9(mpl);
  3080             if (set->option->type != A_ELEMSET)
  3081                error(mpl, "expression following default has invalid typ"
  3082                   "e");
  3083             xassert(set->option->dim > 0);
  3084             /* check/set dimension of set members */
  3085             if (set->dimen == 0) set->dimen = set->option->dim;
  3086             if (set->dimen != set->option->dim)
  3087                error(mpl, "set expression following default must have d"
  3088                   "imension %d rather than %d",
  3089                   set->dimen, set->option->dim);
  3090          }
  3091 #if 1 /* 12/XII-2008 */
  3092          else if (is_keyword(mpl, "data"))
  3093          {  /* gadget to initialize the set by data from plain set */
  3094             GADGET *gadget;
  3095             AVLNODE *node;
  3096             int i, k, fff[20];
  3097             if (!(set->assign == NULL && set->gadget == NULL)) goto err;
  3098             get_token(mpl /* data */);
  3099             set->gadget = gadget = alloc(GADGET);
  3100             /* set name must follow the keyword 'data' */
  3101             if (mpl->token == T_NAME)
  3102                ;
  3103             else if (is_reserved(mpl))
  3104                error(mpl, "invalid use of reserved keyword %s",
  3105                   mpl->image);
  3106             else
  3107                error(mpl, "set name missing where expected");
  3108             /* find the set in the symbolic name table */
  3109             node = avl_find_node(mpl->tree, mpl->image);
  3110             if (node == NULL)
  3111                error(mpl, "%s not defined", mpl->image);
  3112             if (avl_get_node_type(node) != A_SET)
  3113 err1:          error(mpl, "%s not a plain set", mpl->image);
  3114             gadget->set = avl_get_node_link(node);
  3115             if (gadget->set->dim != 0) goto err1;
  3116             if (gadget->set == set)
  3117                error(mpl, "set cannot be initialized by itself");
  3118             /* check and set dimensions */
  3119             if (set->dim >= gadget->set->dimen)
  3120 err2:          error(mpl, "dimension of %s too small", mpl->image);
  3121             if (set->dimen == 0)
  3122                set->dimen = gadget->set->dimen - set->dim;
  3123             if (set->dim + set->dimen > gadget->set->dimen)
  3124                goto err2;
  3125             else if (set->dim + set->dimen < gadget->set->dimen)
  3126                error(mpl, "dimension of %s too big", mpl->image);
  3127             get_token(mpl /* set name */);
  3128             /* left parenthesis must follow the set name */
  3129             if (mpl->token == T_LEFT)
  3130                get_token(mpl /* ( */);
  3131             else
  3132                error(mpl, "left parenthesis missing where expected");
  3133             /* parse permutation of component numbers */
  3134             for (k = 0; k < gadget->set->dimen; k++) fff[k] = 0;
  3135             k = 0;
  3136             for (;;)
  3137             {  if (mpl->token != T_NUMBER)
  3138                   error(mpl, "component number missing where expected");
  3139                if (str2int(mpl->image, &i) != 0)
  3140 err3:             error(mpl, "component number must be integer between "
  3141                      "1 and %d", gadget->set->dimen);
  3142                if (!(1 <= i && i <= gadget->set->dimen)) goto err3;
  3143                if (fff[i-1] != 0)
  3144                   error(mpl, "component %d multiply specified", i);
  3145                gadget->ind[k++] = i, fff[i-1] = 1;
  3146                xassert(k <= gadget->set->dimen);
  3147                get_token(mpl /* number */);
  3148                if (mpl->token == T_COMMA)
  3149                   get_token(mpl /* , */);
  3150                else if (mpl->token == T_RIGHT)
  3151                   break;
  3152                else
  3153                   error(mpl, "syntax error in data attribute");
  3154             }
  3155             if (k < gadget->set->dimen)
  3156                error(mpl, "there are must be %d components rather than "
  3157                   "%d", gadget->set->dimen, k);
  3158             get_token(mpl /* ) */);
  3159          }
  3160 #endif
  3161          else
  3162             error(mpl, "syntax error in set statement");
  3163       }
  3164       /* close the domain scope */
  3165       if (set->domain != NULL) close_scope(mpl, set->domain);
  3166       /* if dimension of set members is still unknown, set it to 1 */
  3167       if (set->dimen == 0) set->dimen = 1;
  3168       /* the set statement has been completely parsed */
  3169       xassert(mpl->token == T_SEMICOLON);
  3170       get_token(mpl /* ; */);
  3171       return set;
  3172 }
  3173 
  3174 /*----------------------------------------------------------------------
  3175 -- parameter_statement - parse parameter statement.
  3176 --
  3177 -- This routine parses parameter statement using the syntax:
  3178 --
  3179 -- <parameter statement> ::= param <symbolic name> <alias> <domain>
  3180 --                           <attributes> ;
  3181 -- <alias> ::= <empty>
  3182 -- <alias> ::= <string literal>
  3183 -- <domain> ::= <empty>
  3184 -- <domain> ::= <indexing expression>
  3185 -- <attributes> ::= <empty>
  3186 -- <attributes> ::= <attributes> , integer
  3187 -- <attributes> ::= <attributes> , binary
  3188 -- <attributes> ::= <attributes> , symbolic
  3189 -- <attributes> ::= <attributes> , <rho> <expression 5>
  3190 -- <attributes> ::= <attributes> , in <expression 9>
  3191 -- <attributes> ::= <attributes> , := <expression 5>
  3192 -- <attributes> ::= <attributes> , default <expression 5>
  3193 -- <rho> ::= < | <= | = | == | >= | > | <> | !=
  3194 --
  3195 -- Commae in <attributes> are optional and may be omitted anywhere. */
  3196 
  3197 PARAMETER *parameter_statement(MPL *mpl)
  3198 {     PARAMETER *par;
  3199       int integer_used = 0, binary_used = 0, symbolic_used = 0;
  3200       xassert(is_keyword(mpl, "param"));
  3201       get_token(mpl /* param */);
  3202       /* symbolic name must follow the keyword 'param' */
  3203       if (mpl->token == T_NAME)
  3204          ;
  3205       else if (is_reserved(mpl))
  3206          error(mpl, "invalid use of reserved keyword %s", mpl->image);
  3207       else
  3208          error(mpl, "symbolic name missing where expected");
  3209       /* there must be no other object with the same name */
  3210       if (avl_find_node(mpl->tree, mpl->image) != NULL)
  3211          error(mpl, "%s multiply declared", mpl->image);
  3212       /* create model parameter */
  3213       par = alloc(PARAMETER);
  3214       par->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  3215       strcpy(par->name, mpl->image);
  3216       par->alias = NULL;
  3217       par->dim = 0;
  3218       par->domain = NULL;
  3219       par->type = A_NUMERIC;
  3220       par->cond = NULL;
  3221       par->in = NULL;
  3222       par->assign = NULL;
  3223       par->option = NULL;
  3224       par->data = 0;
  3225       par->defval = NULL;
  3226       par->array = NULL;
  3227       get_token(mpl /* <symbolic name> */);
  3228       /* parse optional alias */
  3229       if (mpl->token == T_STRING)
  3230       {  par->alias = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  3231          strcpy(par->alias, mpl->image);
  3232          get_token(mpl /* <string literal> */);
  3233       }
  3234       /* parse optional indexing expression */
  3235       if (mpl->token == T_LBRACE)
  3236       {  par->domain = indexing_expression(mpl);
  3237          par->dim = domain_arity(mpl, par->domain);
  3238       }
  3239       /* include the parameter name in the symbolic names table */
  3240       {  AVLNODE *node;
  3241          node = avl_insert_node(mpl->tree, par->name);
  3242          avl_set_node_type(node, A_PARAMETER);
  3243          avl_set_node_link(node, (void *)par);
  3244       }
  3245       /* parse the list of optional attributes */
  3246       for (;;)
  3247       {  if (mpl->token == T_COMMA)
  3248             get_token(mpl /* , */);
  3249          else if (mpl->token == T_SEMICOLON)
  3250             break;
  3251          if (is_keyword(mpl, "integer"))
  3252          {  if (integer_used)
  3253                error(mpl, "at most one integer allowed");
  3254             if (par->type == A_SYMBOLIC)
  3255                error(mpl, "symbolic parameter cannot be integer");
  3256             if (par->type != A_BINARY) par->type = A_INTEGER;
  3257             integer_used = 1;
  3258             get_token(mpl /* integer */);
  3259          }
  3260          else if (is_keyword(mpl, "binary"))
  3261 bin:     {  if (binary_used)
  3262                error(mpl, "at most one binary allowed");
  3263             if (par->type == A_SYMBOLIC)
  3264                error(mpl, "symbolic parameter cannot be binary");
  3265             par->type = A_BINARY;
  3266             binary_used = 1;
  3267             get_token(mpl /* binary */);
  3268          }
  3269          else if (is_keyword(mpl, "logical"))
  3270          {  if (!mpl->as_binary)
  3271             {  warning(mpl, "keyword logical understood as binary");
  3272                mpl->as_binary = 1;
  3273             }
  3274             goto bin;
  3275          }
  3276          else if (is_keyword(mpl, "symbolic"))
  3277          {  if (symbolic_used)
  3278                error(mpl, "at most one symbolic allowed");
  3279             if (par->type != A_NUMERIC)
  3280                error(mpl, "integer or binary parameter cannot be symbol"
  3281                   "ic");
  3282             /* the parameter may be referenced from expressions given
  3283                in the same parameter declaration, so its type must be
  3284                completed before parsing that expressions */
  3285             if (!(par->cond == NULL && par->in == NULL &&
  3286                   par->assign == NULL && par->option == NULL))
  3287                error(mpl, "keyword symbolic must precede any other para"
  3288                   "meter attributes");
  3289             par->type = A_SYMBOLIC;
  3290             symbolic_used = 1;
  3291             get_token(mpl /* symbolic */);
  3292          }
  3293          else if (mpl->token == T_LT || mpl->token == T_LE ||
  3294                   mpl->token == T_EQ || mpl->token == T_GE ||
  3295                   mpl->token == T_GT || mpl->token == T_NE)
  3296          {  /* restricting condition */
  3297             CONDITION *cond, *temp;
  3298             char opstr[8];
  3299             /* create new restricting condition list entry and append
  3300                it to the conditions list */
  3301             cond = alloc(CONDITION);
  3302             switch (mpl->token)
  3303             {  case T_LT:
  3304                   cond->rho = O_LT, strcpy(opstr, mpl->image); break;
  3305                case T_LE:
  3306                   cond->rho = O_LE, strcpy(opstr, mpl->image); break;
  3307                case T_EQ:
  3308                   cond->rho = O_EQ, strcpy(opstr, mpl->image); break;
  3309                case T_GE:
  3310                   cond->rho = O_GE, strcpy(opstr, mpl->image); break;
  3311                case T_GT:
  3312                   cond->rho = O_GT, strcpy(opstr, mpl->image); break;
  3313                case T_NE:
  3314                   cond->rho = O_NE, strcpy(opstr, mpl->image); break;
  3315                default:
  3316                   xassert(mpl->token != mpl->token);
  3317             }
  3318             xassert(strlen(opstr) < sizeof(opstr));
  3319             cond->code = NULL;
  3320             cond->next = NULL;
  3321             if (par->cond == NULL)
  3322                par->cond = cond;
  3323             else
  3324             {  for (temp = par->cond; temp->next != NULL; temp =
  3325                   temp->next);
  3326                temp->next = cond;
  3327             }
  3328 #if 0 /* 13/VIII-2008 */
  3329             if (par->type == A_SYMBOLIC &&
  3330                !(cond->rho == O_EQ || cond->rho == O_NE))
  3331                error(mpl, "inequality restriction not allowed");
  3332 #endif
  3333             get_token(mpl /* rho */);
  3334             /* parse an expression that follows relational operator */
  3335             cond->code = expression_5(mpl);
  3336             if (!(cond->code->type == A_NUMERIC ||
  3337                   cond->code->type == A_SYMBOLIC))
  3338                error(mpl, "expression following %s has invalid type",
  3339                   opstr);
  3340             xassert(cond->code->dim == 0);
  3341             /* convert to the parameter type, if necessary */
  3342             if (par->type != A_SYMBOLIC && cond->code->type ==
  3343                A_SYMBOLIC)
  3344                cond->code = make_unary(mpl, O_CVTNUM, cond->code,
  3345                   A_NUMERIC, 0);
  3346             if (par->type == A_SYMBOLIC && cond->code->type !=
  3347                A_SYMBOLIC)
  3348                cond->code = make_unary(mpl, O_CVTSYM, cond->code,
  3349                   A_SYMBOLIC, 0);
  3350          }
  3351          else if (mpl->token == T_IN || mpl->token == T_WITHIN)
  3352          {  /* restricting superset */
  3353             WITHIN *in, *temp;
  3354             if (mpl->token == T_WITHIN && !mpl->as_in)
  3355             {  warning(mpl, "keyword within understood as in");
  3356                mpl->as_in = 1;
  3357             }
  3358             get_token(mpl /* in */);
  3359             /* create new restricting superset list entry and append it
  3360                to the in-list */
  3361             in = alloc(WITHIN);
  3362             in->code = NULL;
  3363             in->next = NULL;
  3364             if (par->in == NULL)
  3365                par->in = in;
  3366             else
  3367             {  for (temp = par->in; temp->next != NULL; temp =
  3368                   temp->next);
  3369                temp->next = in;
  3370             }
  3371             /* parse an expression that follows 'in' */
  3372             in->code = expression_9(mpl);
  3373             if (in->code->type != A_ELEMSET)
  3374                error(mpl, "expression following in has invalid type");
  3375             xassert(in->code->dim > 0);
  3376             if (in->code->dim != 1)
  3377                error(mpl, "set expression following in must have dimens"
  3378                   "ion 1 rather than %d", in->code->dim);
  3379          }
  3380          else if (mpl->token == T_ASSIGN)
  3381          {  /* assignment expression */
  3382             if (!(par->assign == NULL && par->option == NULL))
  3383 err:           error(mpl, "at most one := or default allowed");
  3384             get_token(mpl /* := */);
  3385             /* parse an expression that follows ':=' */
  3386             par->assign = expression_5(mpl);
  3387             /* the expression must be of numeric/symbolic type */
  3388             if (!(par->assign->type == A_NUMERIC ||
  3389                   par->assign->type == A_SYMBOLIC))
  3390                error(mpl, "expression following := has invalid type");
  3391             xassert(par->assign->dim == 0);
  3392             /* convert to the parameter type, if necessary */
  3393             if (par->type != A_SYMBOLIC && par->assign->type ==
  3394                A_SYMBOLIC)
  3395                par->assign = make_unary(mpl, O_CVTNUM, par->assign,
  3396                   A_NUMERIC, 0);
  3397             if (par->type == A_SYMBOLIC && par->assign->type !=
  3398                A_SYMBOLIC)
  3399                par->assign = make_unary(mpl, O_CVTSYM, par->assign,
  3400                   A_SYMBOLIC, 0);
  3401          }
  3402          else if (is_keyword(mpl, "default"))
  3403          {  /* expression for default value */
  3404             if (!(par->assign == NULL && par->option == NULL)) goto err;
  3405             get_token(mpl /* default */);
  3406             /* parse an expression that follows 'default' */
  3407             par->option = expression_5(mpl);
  3408             if (!(par->option->type == A_NUMERIC ||
  3409                   par->option->type == A_SYMBOLIC))
  3410                error(mpl, "expression following default has invalid typ"
  3411                   "e");
  3412             xassert(par->option->dim == 0);
  3413             /* convert to the parameter type, if necessary */
  3414             if (par->type != A_SYMBOLIC && par->option->type ==
  3415                A_SYMBOLIC)
  3416                par->option = make_unary(mpl, O_CVTNUM, par->option,
  3417                   A_NUMERIC, 0);
  3418             if (par->type == A_SYMBOLIC && par->option->type !=
  3419                A_SYMBOLIC)
  3420                par->option = make_unary(mpl, O_CVTSYM, par->option,
  3421                   A_SYMBOLIC, 0);
  3422          }
  3423          else
  3424             error(mpl, "syntax error in parameter statement");
  3425       }
  3426       /* close the domain scope */
  3427       if (par->domain != NULL) close_scope(mpl, par->domain);
  3428       /* the parameter statement has been completely parsed */
  3429       xassert(mpl->token == T_SEMICOLON);
  3430       get_token(mpl /* ; */);
  3431       return par;
  3432 }
  3433 
  3434 /*----------------------------------------------------------------------
  3435 -- variable_statement - parse variable statement.
  3436 --
  3437 -- This routine parses variable statement using the syntax:
  3438 --
  3439 -- <variable statement> ::= var <symbolic name> <alias> <domain>
  3440 --                          <attributes> ;
  3441 -- <alias> ::= <empty>
  3442 -- <alias> ::= <string literal>
  3443 -- <domain> ::= <empty>
  3444 -- <domain> ::= <indexing expression>
  3445 -- <attributes> ::= <empty>
  3446 -- <attributes> ::= <attributes> , integer
  3447 -- <attributes> ::= <attributes> , binary
  3448 -- <attributes> ::= <attributes> , <rho> <expression 5>
  3449 -- <rho> ::= >= | <= | = | ==
  3450 --
  3451 -- Commae in <attributes> are optional and may be omitted anywhere. */
  3452 
  3453 VARIABLE *variable_statement(MPL *mpl)
  3454 {     VARIABLE *var;
  3455       int integer_used = 0, binary_used = 0;
  3456       xassert(is_keyword(mpl, "var"));
  3457       if (mpl->flag_s)
  3458          error(mpl, "variable statement must precede solve statement");
  3459       get_token(mpl /* var */);
  3460       /* symbolic name must follow the keyword 'var' */
  3461       if (mpl->token == T_NAME)
  3462          ;
  3463       else if (is_reserved(mpl))
  3464          error(mpl, "invalid use of reserved keyword %s", mpl->image);
  3465       else
  3466          error(mpl, "symbolic name missing where expected");
  3467       /* there must be no other object with the same name */
  3468       if (avl_find_node(mpl->tree, mpl->image) != NULL)
  3469          error(mpl, "%s multiply declared", mpl->image);
  3470       /* create model variable */
  3471       var = alloc(VARIABLE);
  3472       var->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  3473       strcpy(var->name, mpl->image);
  3474       var->alias = NULL;
  3475       var->dim = 0;
  3476       var->domain = NULL;
  3477       var->type = A_NUMERIC;
  3478       var->lbnd = NULL;
  3479       var->ubnd = NULL;
  3480       var->array = NULL;
  3481       get_token(mpl /* <symbolic name> */);
  3482       /* parse optional alias */
  3483       if (mpl->token == T_STRING)
  3484       {  var->alias = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  3485          strcpy(var->alias, mpl->image);
  3486          get_token(mpl /* <string literal> */);
  3487       }
  3488       /* parse optional indexing expression */
  3489       if (mpl->token == T_LBRACE)
  3490       {  var->domain = indexing_expression(mpl);
  3491          var->dim = domain_arity(mpl, var->domain);
  3492       }
  3493       /* include the variable name in the symbolic names table */
  3494       {  AVLNODE *node;
  3495          node = avl_insert_node(mpl->tree, var->name);
  3496          avl_set_node_type(node, A_VARIABLE);
  3497          avl_set_node_link(node, (void *)var);
  3498       }
  3499       /* parse the list of optional attributes */
  3500       for (;;)
  3501       {  if (mpl->token == T_COMMA)
  3502             get_token(mpl /* , */);
  3503          else if (mpl->token == T_SEMICOLON)
  3504             break;
  3505          if (is_keyword(mpl, "integer"))
  3506          {  if (integer_used)
  3507                error(mpl, "at most one integer allowed");
  3508             if (var->type != A_BINARY) var->type = A_INTEGER;
  3509             integer_used = 1;
  3510             get_token(mpl /* integer */);
  3511          }
  3512          else if (is_keyword(mpl, "binary"))
  3513 bin:     {  if (binary_used)
  3514                error(mpl, "at most one binary allowed");
  3515             var->type = A_BINARY;
  3516             binary_used = 1;
  3517             get_token(mpl /* binary */);
  3518          }
  3519          else if (is_keyword(mpl, "logical"))
  3520          {  if (!mpl->as_binary)
  3521             {  warning(mpl, "keyword logical understood as binary");
  3522                mpl->as_binary = 1;
  3523             }
  3524             goto bin;
  3525          }
  3526          else if (is_keyword(mpl, "symbolic"))
  3527             error(mpl, "variable cannot be symbolic");
  3528          else if (mpl->token == T_GE)
  3529          {  /* lower bound */
  3530             if (var->lbnd != NULL)
  3531             {  if (var->lbnd == var->ubnd)
  3532                   error(mpl, "both fixed value and lower bound not allo"
  3533                      "wed");
  3534                else
  3535                   error(mpl, "at most one lower bound allowed");
  3536             }
  3537             get_token(mpl /* >= */);
  3538             /* parse an expression that specifies the lower bound */
  3539             var->lbnd = expression_5(mpl);
  3540             if (var->lbnd->type == A_SYMBOLIC)
  3541                var->lbnd = make_unary(mpl, O_CVTNUM, var->lbnd,
  3542                   A_NUMERIC, 0);
  3543             if (var->lbnd->type != A_NUMERIC)
  3544                error(mpl, "expression following >= has invalid type");
  3545             xassert(var->lbnd->dim == 0);
  3546          }
  3547          else if (mpl->token == T_LE)
  3548          {  /* upper bound */
  3549             if (var->ubnd != NULL)
  3550             {  if (var->ubnd == var->lbnd)
  3551                   error(mpl, "both fixed value and upper bound not allo"
  3552                      "wed");
  3553                else
  3554                   error(mpl, "at most one upper bound allowed");
  3555             }
  3556             get_token(mpl /* <= */);
  3557             /* parse an expression that specifies the upper bound */
  3558             var->ubnd = expression_5(mpl);
  3559             if (var->ubnd->type == A_SYMBOLIC)
  3560                var->ubnd = make_unary(mpl, O_CVTNUM, var->ubnd,
  3561                   A_NUMERIC, 0);
  3562             if (var->ubnd->type != A_NUMERIC)
  3563                error(mpl, "expression following <= has invalid type");
  3564             xassert(var->ubnd->dim == 0);
  3565          }
  3566          else if (mpl->token == T_EQ)
  3567          {  /* fixed value */
  3568             char opstr[8];
  3569             if (!(var->lbnd == NULL && var->ubnd == NULL))
  3570             {  if (var->lbnd == var->ubnd)
  3571                   error(mpl, "at most one fixed value allowed");
  3572                else if (var->lbnd != NULL)
  3573                   error(mpl, "both lower bound and fixed value not allo"
  3574                      "wed");
  3575                else
  3576                   error(mpl, "both upper bound and fixed value not allo"
  3577                      "wed");
  3578             }
  3579             strcpy(opstr, mpl->image);
  3580             xassert(strlen(opstr) < sizeof(opstr));
  3581             get_token(mpl /* = | == */);
  3582             /* parse an expression that specifies the fixed value */
  3583             var->lbnd = expression_5(mpl);
  3584             if (var->lbnd->type == A_SYMBOLIC)
  3585                var->lbnd = make_unary(mpl, O_CVTNUM, var->lbnd,
  3586                   A_NUMERIC, 0);
  3587             if (var->lbnd->type != A_NUMERIC)
  3588                error(mpl, "expression following %s has invalid type",
  3589                   opstr);
  3590             xassert(var->lbnd->dim == 0);
  3591             /* indicate that the variable is fixed, not bounded */
  3592             var->ubnd = var->lbnd;
  3593          }
  3594          else if (mpl->token == T_LT || mpl->token == T_GT ||
  3595                   mpl->token == T_NE)
  3596             error(mpl, "strict bound not allowed");
  3597          else
  3598             error(mpl, "syntax error in variable statement");
  3599       }
  3600       /* close the domain scope */
  3601       if (var->domain != NULL) close_scope(mpl, var->domain);
  3602       /* the variable statement has been completely parsed */
  3603       xassert(mpl->token == T_SEMICOLON);
  3604       get_token(mpl /* ; */);
  3605       return var;
  3606 }
  3607 
  3608 /*----------------------------------------------------------------------
  3609 -- constraint_statement - parse constraint statement.
  3610 --
  3611 -- This routine parses constraint statement using the syntax:
  3612 --
  3613 -- <constraint statement> ::= <subject to> <symbolic name> <alias>
  3614 --                            <domain> : <constraint> ;
  3615 -- <subject to> ::= <empty>
  3616 -- <subject to> ::= subject to
  3617 -- <subject to> ::= subj to
  3618 -- <subject to> ::= s.t.
  3619 -- <alias> ::= <empty>
  3620 -- <alias> ::= <string literal>
  3621 -- <domain> ::= <empty>
  3622 -- <domain> ::= <indexing expression>
  3623 -- <constraint> ::= <formula> , >= <formula>
  3624 -- <constraint> ::= <formula> , <= <formula>
  3625 -- <constraint> ::= <formula> , = <formula>
  3626 -- <constraint> ::= <formula> , <= <formula> , <= <formula>
  3627 -- <constraint> ::= <formula> , >= <formula> , >= <formula>
  3628 -- <formula> ::= <expression 5>
  3629 --
  3630 -- Commae in <constraint> are optional and may be omitted anywhere. */
  3631 
  3632 CONSTRAINT *constraint_statement(MPL *mpl)
  3633 {     CONSTRAINT *con;
  3634       CODE *first, *second, *third;
  3635       int rho;
  3636       char opstr[8];
  3637       if (mpl->flag_s)
  3638          error(mpl, "constraint statement must precede solve statement")
  3639             ;
  3640       if (is_keyword(mpl, "subject"))
  3641       {  get_token(mpl /* subject */);
  3642          if (!is_keyword(mpl, "to"))
  3643             error(mpl, "keyword subject to incomplete");
  3644          get_token(mpl /* to */);
  3645       }
  3646       else if (is_keyword(mpl, "subj"))
  3647       {  get_token(mpl /* subj */);
  3648          if (!is_keyword(mpl, "to"))
  3649             error(mpl, "keyword subj to incomplete");
  3650          get_token(mpl /* to */);
  3651       }
  3652       else if (mpl->token == T_SPTP)
  3653          get_token(mpl /* s.t. */);
  3654       /* the current token must be symbolic name of constraint */
  3655       if (mpl->token == T_NAME)
  3656          ;
  3657       else if (is_reserved(mpl))
  3658          error(mpl, "invalid use of reserved keyword %s", mpl->image);
  3659       else
  3660          error(mpl, "symbolic name missing where expected");
  3661       /* there must be no other object with the same name */
  3662       if (avl_find_node(mpl->tree, mpl->image) != NULL)
  3663          error(mpl, "%s multiply declared", mpl->image);
  3664       /* create model constraint */
  3665       con = alloc(CONSTRAINT);
  3666       con->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  3667       strcpy(con->name, mpl->image);
  3668       con->alias = NULL;
  3669       con->dim = 0;
  3670       con->domain = NULL;
  3671       con->type = A_CONSTRAINT;
  3672       con->code = NULL;
  3673       con->lbnd = NULL;
  3674       con->ubnd = NULL;
  3675       con->array = NULL;
  3676       get_token(mpl /* <symbolic name> */);
  3677       /* parse optional alias */
  3678       if (mpl->token == T_STRING)
  3679       {  con->alias = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  3680          strcpy(con->alias, mpl->image);
  3681          get_token(mpl /* <string literal> */);
  3682       }
  3683       /* parse optional indexing expression */
  3684       if (mpl->token == T_LBRACE)
  3685       {  con->domain = indexing_expression(mpl);
  3686          con->dim = domain_arity(mpl, con->domain);
  3687       }
  3688       /* include the constraint name in the symbolic names table */
  3689       {  AVLNODE *node;
  3690          node = avl_insert_node(mpl->tree, con->name);
  3691          avl_set_node_type(node, A_CONSTRAINT);
  3692          avl_set_node_link(node, (void *)con);
  3693       }
  3694       /* the colon must precede the first expression */
  3695       if (mpl->token != T_COLON)
  3696          error(mpl, "colon missing where expected");
  3697       get_token(mpl /* : */);
  3698       /* parse the first expression */
  3699       first = expression_5(mpl);
  3700       if (first->type == A_SYMBOLIC)
  3701          first = make_unary(mpl, O_CVTNUM, first, A_NUMERIC, 0);
  3702       if (!(first->type == A_NUMERIC || first->type == A_FORMULA))
  3703          error(mpl, "expression following colon has invalid type");
  3704       xassert(first->dim == 0);
  3705       /* relational operator must follow the first expression */
  3706       if (mpl->token == T_COMMA) get_token(mpl /* , */);
  3707       switch (mpl->token)
  3708       {  case T_LE:
  3709          case T_GE:
  3710          case T_EQ:
  3711             break;
  3712          case T_LT:
  3713          case T_GT:
  3714          case T_NE:
  3715             error(mpl, "strict inequality not allowed");
  3716          case T_SEMICOLON:
  3717             error(mpl, "constraint must be equality or inequality");
  3718          default:
  3719             goto err;
  3720       }
  3721       rho = mpl->token;
  3722       strcpy(opstr, mpl->image);
  3723       xassert(strlen(opstr) < sizeof(opstr));
  3724       get_token(mpl /* rho */);
  3725       /* parse the second expression */
  3726       second = expression_5(mpl);
  3727       if (second->type == A_SYMBOLIC)
  3728          second = make_unary(mpl, O_CVTNUM, second, A_NUMERIC, 0);
  3729       if (!(second->type == A_NUMERIC || second->type == A_FORMULA))
  3730          error(mpl, "expression following %s has invalid type", opstr);
  3731       xassert(second->dim == 0);
  3732       /* check a token that follow the second expression */
  3733       if (mpl->token == T_COMMA)
  3734       {  get_token(mpl /* , */);
  3735          if (mpl->token == T_SEMICOLON) goto err;
  3736       }
  3737       if (mpl->token == T_LT || mpl->token == T_LE ||
  3738           mpl->token == T_EQ || mpl->token == T_GE ||
  3739           mpl->token == T_GT || mpl->token == T_NE)
  3740       {  /* it is another relational operator, therefore the constraint
  3741             is double inequality */
  3742          if (rho == T_EQ || mpl->token != rho)
  3743             error(mpl, "double inequality must be ... <= ... <= ... or "
  3744                "... >= ... >= ...");
  3745          /* the first expression cannot be linear form */
  3746          if (first->type == A_FORMULA)
  3747             error(mpl, "leftmost expression in double inequality cannot"
  3748                " be linear form");
  3749          get_token(mpl /* rho */);
  3750          /* parse the third expression */
  3751          third = expression_5(mpl);
  3752          if (third->type == A_SYMBOLIC)
  3753             third = make_unary(mpl, O_CVTNUM, second, A_NUMERIC, 0);
  3754          if (!(third->type == A_NUMERIC || third->type == A_FORMULA))
  3755             error(mpl, "rightmost expression in double inequality const"
  3756                "raint has invalid type");
  3757          xassert(third->dim == 0);
  3758          /* the third expression also cannot be linear form */
  3759          if (third->type == A_FORMULA)
  3760             error(mpl, "rightmost expression in double inequality canno"
  3761                "t be linear form");
  3762       }
  3763       else
  3764       {  /* the constraint is equality or single inequality */
  3765          third = NULL;
  3766       }
  3767       /* close the domain scope */
  3768       if (con->domain != NULL) close_scope(mpl, con->domain);
  3769       /* convert all expressions to linear form, if necessary */
  3770       if (first->type != A_FORMULA)
  3771          first = make_unary(mpl, O_CVTLFM, first, A_FORMULA, 0);
  3772       if (second->type != A_FORMULA)
  3773          second = make_unary(mpl, O_CVTLFM, second, A_FORMULA, 0);
  3774       if (third != NULL)
  3775          third = make_unary(mpl, O_CVTLFM, third, A_FORMULA, 0);
  3776       /* arrange expressions in the constraint */
  3777       if (third == NULL)
  3778       {  /* the constraint is equality or single inequality */
  3779          switch (rho)
  3780          {  case T_LE:
  3781                /* first <= second */
  3782                con->code = first;
  3783                con->lbnd = NULL;
  3784                con->ubnd = second;
  3785                break;
  3786             case T_GE:
  3787                /* first >= second */
  3788                con->code = first;
  3789                con->lbnd = second;
  3790                con->ubnd = NULL;
  3791                break;
  3792             case T_EQ:
  3793                /* first = second */
  3794                con->code = first;
  3795                con->lbnd = second;
  3796                con->ubnd = second;
  3797                break;
  3798             default:
  3799                xassert(rho != rho);
  3800          }
  3801       }
  3802       else
  3803       {  /* the constraint is double inequality */
  3804          switch (rho)
  3805          {  case T_LE:
  3806                /* first <= second <= third */
  3807                con->code = second;
  3808                con->lbnd = first;
  3809                con->ubnd = third;
  3810                break;
  3811             case T_GE:
  3812                /* first >= second >= third */
  3813                con->code = second;
  3814                con->lbnd = third;
  3815                con->ubnd = first;
  3816                break;
  3817             default:
  3818                xassert(rho != rho);
  3819          }
  3820       }
  3821       /* the constraint statement has been completely parsed */
  3822       if (mpl->token != T_SEMICOLON)
  3823 err:     error(mpl, "syntax error in constraint statement");
  3824       get_token(mpl /* ; */);
  3825       return con;
  3826 }
  3827 
  3828 /*----------------------------------------------------------------------
  3829 -- objective_statement - parse objective statement.
  3830 --
  3831 -- This routine parses objective statement using the syntax:
  3832 --
  3833 -- <objective statement> ::= <verb> <symbolic name> <alias> <domain> :
  3834 --                           <formula> ;
  3835 -- <verb> ::= minimize
  3836 -- <verb> ::= maximize
  3837 -- <alias> ::= <empty>
  3838 -- <alias> ::= <string literal>
  3839 -- <domain> ::= <empty>
  3840 -- <domain> ::= <indexing expression>
  3841 -- <formula> ::= <expression 5> */
  3842 
  3843 CONSTRAINT *objective_statement(MPL *mpl)
  3844 {     CONSTRAINT *obj;
  3845       int type;
  3846       if (is_keyword(mpl, "minimize"))
  3847          type = A_MINIMIZE;
  3848       else if (is_keyword(mpl, "maximize"))
  3849          type = A_MAXIMIZE;
  3850       else
  3851          xassert(mpl != mpl);
  3852       if (mpl->flag_s)
  3853          error(mpl, "objective statement must precede solve statement");
  3854       get_token(mpl /* minimize | maximize */);
  3855       /* symbolic name must follow the verb 'minimize' or 'maximize' */
  3856       if (mpl->token == T_NAME)
  3857          ;
  3858       else if (is_reserved(mpl))
  3859          error(mpl, "invalid use of reserved keyword %s", mpl->image);
  3860       else
  3861          error(mpl, "symbolic name missing where expected");
  3862       /* there must be no other object with the same name */
  3863       if (avl_find_node(mpl->tree, mpl->image) != NULL)
  3864          error(mpl, "%s multiply declared", mpl->image);
  3865       /* create model objective */
  3866       obj = alloc(CONSTRAINT);
  3867       obj->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  3868       strcpy(obj->name, mpl->image);
  3869       obj->alias = NULL;
  3870       obj->dim = 0;
  3871       obj->domain = NULL;
  3872       obj->type = type;
  3873       obj->code = NULL;
  3874       obj->lbnd = NULL;
  3875       obj->ubnd = NULL;
  3876       obj->array = NULL;
  3877       get_token(mpl /* <symbolic name> */);
  3878       /* parse optional alias */
  3879       if (mpl->token == T_STRING)
  3880       {  obj->alias = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  3881          strcpy(obj->alias, mpl->image);
  3882          get_token(mpl /* <string literal> */);
  3883       }
  3884       /* parse optional indexing expression */
  3885       if (mpl->token == T_LBRACE)
  3886       {  obj->domain = indexing_expression(mpl);
  3887          obj->dim = domain_arity(mpl, obj->domain);
  3888       }
  3889       /* include the constraint name in the symbolic names table */
  3890       {  AVLNODE *node;
  3891          node = avl_insert_node(mpl->tree, obj->name);
  3892          avl_set_node_type(node, A_CONSTRAINT);
  3893          avl_set_node_link(node, (void *)obj);
  3894       }
  3895       /* the colon must precede the objective expression */
  3896       if (mpl->token != T_COLON)
  3897          error(mpl, "colon missing where expected");
  3898       get_token(mpl /* : */);
  3899       /* parse the objective expression */
  3900       obj->code = expression_5(mpl);
  3901       if (obj->code->type == A_SYMBOLIC)
  3902          obj->code = make_unary(mpl, O_CVTNUM, obj->code, A_NUMERIC, 0);
  3903       if (obj->code->type == A_NUMERIC)
  3904          obj->code = make_unary(mpl, O_CVTLFM, obj->code, A_FORMULA, 0);
  3905       if (obj->code->type != A_FORMULA)
  3906          error(mpl, "expression following colon has invalid type");
  3907       xassert(obj->code->dim == 0);
  3908       /* close the domain scope */
  3909       if (obj->domain != NULL) close_scope(mpl, obj->domain);
  3910       /* the objective statement has been completely parsed */
  3911       if (mpl->token != T_SEMICOLON)
  3912          error(mpl, "syntax error in objective statement");
  3913       get_token(mpl /* ; */);
  3914       return obj;
  3915 }
  3916 
  3917 #if 1 /* 11/II-2008 */
  3918 /***********************************************************************
  3919 *  table_statement - parse table statement
  3920 *
  3921 *  This routine parses table statement using the syntax:
  3922 *
  3923 *  <table statement> ::= <input table statement>
  3924 *  <table statement> ::= <output table statement>
  3925 *
  3926 *  <input table statement> ::=
  3927 *        table <table name> <alias> IN <argument list> :
  3928 *        <input set> [ <field list> ] , <input list> ;
  3929 *  <alias> ::= <empty>
  3930 *  <alias> ::= <string literal>
  3931 *  <argument list> ::= <expression 5>
  3932 *  <argument list> ::= <argument list> <expression 5>
  3933 *  <argument list> ::= <argument list> , <expression 5>
  3934 *  <input set> ::= <empty>
  3935 *  <input set> ::= <set name> <-
  3936 *  <field list> ::= <field name>
  3937 *  <field list> ::= <field list> , <field name>
  3938 *  <input list> ::= <input item>
  3939 *  <input list> ::= <input list> , <input item>
  3940 *  <input item> ::= <parameter name>
  3941 *  <input item> ::= <parameter name> ~ <field name>
  3942 *
  3943 *  <output table statement> ::=
  3944 *        table <table name> <alias> <domain> OUT <argument list> :
  3945 *        <output list> ;
  3946 *  <domain> ::= <indexing expression>
  3947 *  <output list> ::= <output item>
  3948 *  <output list> ::= <output list> , <output item>
  3949 *  <output item> ::= <expression 5>
  3950 *  <output item> ::= <expression 5> ~ <field name> */
  3951 
  3952 TABLE *table_statement(MPL *mpl)
  3953 {     TABLE *tab;
  3954       TABARG *last_arg, *arg;
  3955       TABFLD *last_fld, *fld;
  3956       TABIN *last_in, *in;
  3957       TABOUT *last_out, *out;
  3958       AVLNODE *node;
  3959       int nflds;
  3960       char name[MAX_LENGTH+1];
  3961       xassert(is_keyword(mpl, "table"));
  3962       get_token(mpl /* solve */);
  3963       /* symbolic name must follow the keyword table */
  3964       if (mpl->token == T_NAME)
  3965          ;
  3966       else if (is_reserved(mpl))
  3967          error(mpl, "invalid use of reserved keyword %s", mpl->image);
  3968       else
  3969          error(mpl, "symbolic name missing where expected");
  3970       /* there must be no other object with the same name */
  3971       if (avl_find_node(mpl->tree, mpl->image) != NULL)
  3972          error(mpl, "%s multiply declared", mpl->image);
  3973       /* create data table */
  3974       tab = alloc(TABLE);
  3975       tab->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  3976       strcpy(tab->name, mpl->image);
  3977       get_token(mpl /* <symbolic name> */);
  3978       /* parse optional alias */
  3979       if (mpl->token == T_STRING)
  3980       {  tab->alias = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  3981          strcpy(tab->alias, mpl->image);
  3982          get_token(mpl /* <string literal> */);
  3983       }
  3984       else
  3985          tab->alias = NULL;
  3986       /* parse optional indexing expression */
  3987       if (mpl->token == T_LBRACE)
  3988       {  /* this is output table */
  3989          tab->type = A_OUTPUT;
  3990          tab->u.out.domain = indexing_expression(mpl);
  3991          if (!is_keyword(mpl, "OUT"))
  3992             error(mpl, "keyword OUT missing where expected");
  3993          get_token(mpl /* OUT */);
  3994       }
  3995       else
  3996       {  /* this is input table */
  3997          tab->type = A_INPUT;
  3998          if (!is_keyword(mpl, "IN"))
  3999             error(mpl, "keyword IN missing where expected");
  4000          get_token(mpl /* IN */);
  4001       }
  4002       /* parse argument list */
  4003       tab->arg = last_arg = NULL;
  4004       for (;;)
  4005       {  /* create argument list entry */
  4006          arg = alloc(TABARG);
  4007          /* parse argument expression */
  4008          if (mpl->token == T_COMMA || mpl->token == T_COLON ||
  4009              mpl->token == T_SEMICOLON)
  4010             error(mpl, "argument expression missing where expected");
  4011          arg->code = expression_5(mpl);
  4012          /* convert the result to symbolic type, if necessary */
  4013          if (arg->code->type == A_NUMERIC)
  4014             arg->code =
  4015                make_unary(mpl, O_CVTSYM, arg->code, A_SYMBOLIC, 0);
  4016          /* check that now the result is of symbolic type */
  4017          if (arg->code->type != A_SYMBOLIC)
  4018             error(mpl, "argument expression has invalid type");
  4019          /* add the entry to the end of the list */
  4020          arg->next = NULL;
  4021          if (last_arg == NULL)
  4022             tab->arg = arg;
  4023          else
  4024             last_arg->next = arg;
  4025          last_arg = arg;
  4026          /* argument expression has been parsed */
  4027          if (mpl->token == T_COMMA)
  4028             get_token(mpl /* , */);
  4029          else if (mpl->token == T_COLON || mpl->token == T_SEMICOLON)
  4030             break;
  4031       }
  4032       xassert(tab->arg != NULL);
  4033       /* argument list must end with colon */
  4034       if (mpl->token == T_COLON)
  4035          get_token(mpl /* : */);
  4036       else
  4037          error(mpl, "colon missing where expected");
  4038       /* parse specific part of the table statement */
  4039       switch (tab->type)
  4040       {  case A_INPUT:  goto input_table;
  4041          case A_OUTPUT: goto output_table;
  4042          default:       xassert(tab != tab);
  4043       }
  4044 input_table:
  4045       /* parse optional set name */
  4046       if (mpl->token == T_NAME)
  4047       {  node = avl_find_node(mpl->tree, mpl->image);
  4048          if (node == NULL)
  4049             error(mpl, "%s not defined", mpl->image);
  4050          if (avl_get_node_type(node) != A_SET)
  4051             error(mpl, "%s not a set", mpl->image);
  4052          tab->u.in.set = (SET *)avl_get_node_link(node);
  4053          if (tab->u.in.set->assign != NULL)
  4054             error(mpl, "%s needs no data", mpl->image);
  4055          if (tab->u.in.set->dim != 0)
  4056             error(mpl, "%s must be a simple set", mpl->image);
  4057          get_token(mpl /* <symbolic name> */);
  4058          if (mpl->token == T_INPUT)
  4059             get_token(mpl /* <- */);
  4060          else
  4061             error(mpl, "delimiter <- missing where expected");
  4062       }
  4063       else if (is_reserved(mpl))
  4064          error(mpl, "invalid use of reserved keyword %s", mpl->image);
  4065       else
  4066          tab->u.in.set = NULL;
  4067       /* parse field list */
  4068       tab->u.in.fld = last_fld = NULL;
  4069       nflds = 0;
  4070       if (mpl->token == T_LBRACKET)
  4071          get_token(mpl /* [ */);
  4072       else
  4073          error(mpl, "field list missing where expected");
  4074       for (;;)
  4075       {  /* create field list entry */
  4076          fld = alloc(TABFLD);
  4077          /* parse field name */
  4078          if (mpl->token == T_NAME)
  4079             ;
  4080          else if (is_reserved(mpl))
  4081             error(mpl,
  4082                "invalid use of reserved keyword %s", mpl->image);
  4083          else
  4084             error(mpl, "field name missing where expected");
  4085          fld->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1);
  4086          strcpy(fld->name, mpl->image);
  4087          get_token(mpl /* <symbolic name> */);
  4088          /* add the entry to the end of the list */
  4089          fld->next = NULL;
  4090          if (last_fld == NULL)
  4091             tab->u.in.fld = fld;
  4092          else
  4093             last_fld->next = fld;
  4094          last_fld = fld;
  4095          nflds++;
  4096          /* field name has been parsed */
  4097          if (mpl->token == T_COMMA)
  4098             get_token(mpl /* , */);
  4099          else if (mpl->token == T_RBRACKET)
  4100             break;
  4101          else
  4102             error(mpl, "syntax error in field list");
  4103       }
  4104       /* check that the set dimen is equal to the number of fields */
  4105       if (tab->u.in.set != NULL && tab->u.in.set->dimen != nflds)
  4106          error(mpl, "there must be %d field%s rather than %d",
  4107             tab->u.in.set->dimen, tab->u.in.set->dimen == 1 ? "" : "s",
  4108             nflds);
  4109       get_token(mpl /* ] */);
  4110       /* parse optional input list */
  4111       tab->u.in.list = last_in = NULL;
  4112       while (mpl->token == T_COMMA)
  4113       {  get_token(mpl /* , */);
  4114          /* create input list entry */
  4115          in = alloc(TABIN);
  4116          /* parse parameter name */
  4117          if (mpl->token == T_NAME)
  4118             ;
  4119          else if (is_reserved(mpl))
  4120             error(mpl,
  4121                "invalid use of reserved keyword %s", mpl->image);
  4122          else
  4123             error(mpl, "parameter name missing where expected");
  4124          node = avl_find_node(mpl->tree, mpl->image);
  4125          if (node == NULL)
  4126             error(mpl, "%s not defined", mpl->image);
  4127          if (avl_get_node_type(node) != A_PARAMETER)
  4128             error(mpl, "%s not a parameter", mpl->image);
  4129          in->par = (PARAMETER *)avl_get_node_link(node);
  4130          if (in->par->dim != nflds)
  4131             error(mpl, "%s must have %d subscript%s rather than %d",
  4132                mpl->image, nflds, nflds == 1 ? "" : "s", in->par->dim);
  4133          if (in->par->assign != NULL)
  4134             error(mpl, "%s needs no data", mpl->image);
  4135          get_token(mpl /* <symbolic name> */);
  4136          /* parse optional field name */
  4137          if (mpl->token == T_TILDE)
  4138          {  get_token(mpl /* ~ */);
  4139             /* parse field name */
  4140             if (mpl->token == T_NAME)
  4141                ;
  4142             else if (is_reserved(mpl))
  4143                error(mpl,
  4144                   "invalid use of reserved keyword %s", mpl->image);
  4145             else
  4146                error(mpl, "field name missing where expected");
  4147             xassert(strlen(mpl->image) < sizeof(name));
  4148             strcpy(name, mpl->image);
  4149             get_token(mpl /* <symbolic name> */);
  4150          }
  4151          else
  4152          {  /* field name is the same as the parameter name */
  4153             xassert(strlen(in->par->name) < sizeof(name));
  4154             strcpy(name, in->par->name);
  4155          }
  4156          /* assign field name */
  4157          in->name = dmp_get_atomv(mpl->pool, strlen(name)+1);
  4158          strcpy(in->name, name);
  4159          /* add the entry to the end of the list */
  4160          in->next = NULL;
  4161          if (last_in == NULL)
  4162             tab->u.in.list = in;
  4163          else
  4164             last_in->next = in;
  4165          last_in = in;
  4166       }
  4167       goto end_of_table;
  4168 output_table:
  4169       /* parse output list */
  4170       tab->u.out.list = last_out = NULL;
  4171       for (;;)
  4172       {  /* create output list entry */
  4173          out = alloc(TABOUT);
  4174          /* parse expression */
  4175          if (mpl->token == T_COMMA || mpl->token == T_SEMICOLON)
  4176             error(mpl, "expression missing where expected");
  4177          if (mpl->token == T_NAME)
  4178          {  xassert(strlen(mpl->image) < sizeof(name));
  4179             strcpy(name, mpl->image);
  4180          }
  4181          else
  4182             name[0] = '\0';
  4183          out->code = expression_5(mpl);
  4184          /* parse optional field name */
  4185          if (mpl->token == T_TILDE)
  4186          {  get_token(mpl /* ~ */);
  4187             /* parse field name */
  4188             if (mpl->token == T_NAME)
  4189                ;
  4190             else if (is_reserved(mpl))
  4191                error(mpl,
  4192                   "invalid use of reserved keyword %s", mpl->image);
  4193             else
  4194                error(mpl, "field name missing where expected");
  4195             xassert(strlen(mpl->image) < sizeof(name));
  4196             strcpy(name, mpl->image);
  4197             get_token(mpl /* <symbolic name> */);
  4198          }
  4199          /* assign field name */
  4200          if (name[0] == '\0')
  4201             error(mpl, "field name required");
  4202          out->name = dmp_get_atomv(mpl->pool, strlen(name)+1);
  4203          strcpy(out->name, name);
  4204          /* add the entry to the end of the list */
  4205          out->next = NULL;
  4206          if (last_out == NULL)
  4207             tab->u.out.list = out;
  4208          else
  4209             last_out->next = out;
  4210          last_out = out;
  4211          /* output item has been parsed */
  4212          if (mpl->token == T_COMMA)
  4213             get_token(mpl /* , */);
  4214          else if (mpl->token == T_SEMICOLON)
  4215             break;
  4216          else
  4217             error(mpl, "syntax error in output list");
  4218       }
  4219       /* close the domain scope */
  4220       close_scope(mpl,tab->u.out.domain);
  4221 end_of_table:
  4222       /* the table statement must end with semicolon */
  4223       if (mpl->token != T_SEMICOLON)
  4224          error(mpl, "syntax error in table statement");
  4225       get_token(mpl /* ; */);
  4226       return tab;
  4227 }
  4228 #endif
  4229 
  4230 /*----------------------------------------------------------------------
  4231 -- solve_statement - parse solve statement.
  4232 --
  4233 -- This routine parses solve statement using the syntax:
  4234 --
  4235 -- <solve statement> ::= solve ;
  4236 --
  4237 -- The solve statement can be used at most once. */
  4238 
  4239 void *solve_statement(MPL *mpl)
  4240 {     xassert(is_keyword(mpl, "solve"));
  4241       if (mpl->flag_s)
  4242          error(mpl, "at most one solve statement allowed");
  4243       mpl->flag_s = 1;
  4244       get_token(mpl /* solve */);
  4245       /* semicolon must follow solve statement */
  4246       if (mpl->token != T_SEMICOLON)
  4247          error(mpl, "syntax error in solve statement");
  4248       get_token(mpl /* ; */);
  4249       return NULL;
  4250 }
  4251 
  4252 /*----------------------------------------------------------------------
  4253 -- check_statement - parse check statement.
  4254 --
  4255 -- This routine parses check statement using the syntax:
  4256 --
  4257 -- <check statement> ::= check <domain> : <expression 13> ;
  4258 -- <domain> ::= <empty>
  4259 -- <domain> ::= <indexing expression>
  4260 --
  4261 -- If <domain> is omitted, colon following it may also be omitted. */
  4262 
  4263 CHECK *check_statement(MPL *mpl)
  4264 {     CHECK *chk;
  4265       xassert(is_keyword(mpl, "check"));
  4266       /* create check descriptor */
  4267       chk = alloc(CHECK);
  4268       chk->domain = NULL;
  4269       chk->code = NULL;
  4270       get_token(mpl /* check */);
  4271       /* parse optional indexing expression */
  4272       if (mpl->token == T_LBRACE)
  4273       {  chk->domain = indexing_expression(mpl);
  4274 #if 0
  4275          if (mpl->token != T_COLON)
  4276             error(mpl, "colon missing where expected");
  4277 #endif
  4278       }
  4279       /* skip optional colon */
  4280       if (mpl->token == T_COLON) get_token(mpl /* : */);
  4281       /* parse logical expression */
  4282       chk->code = expression_13(mpl);
  4283       if (chk->code->type != A_LOGICAL)
  4284          error(mpl, "expression has invalid type");
  4285       xassert(chk->code->dim == 0);
  4286       /* close the domain scope */
  4287       if (chk->domain != NULL) close_scope(mpl, chk->domain);
  4288       /* the check statement has been completely parsed */
  4289       if (mpl->token != T_SEMICOLON)
  4290          error(mpl, "syntax error in check statement");
  4291       get_token(mpl /* ; */);
  4292       return chk;
  4293 }
  4294 
  4295 #if 1 /* 15/V-2010 */
  4296 /*----------------------------------------------------------------------
  4297 -- display_statement - parse display statement.
  4298 --
  4299 -- This routine parses display statement using the syntax:
  4300 --
  4301 -- <display statement> ::= display <domain> : <display list> ;
  4302 -- <display statement> ::= display <domain> <display list> ;
  4303 -- <domain> ::= <empty>
  4304 -- <domain> ::= <indexing expression>
  4305 -- <display list> ::= <display entry>
  4306 -- <display list> ::= <display list> , <display entry>
  4307 -- <display entry> ::= <dummy index>
  4308 -- <display entry> ::= <set name>
  4309 -- <display entry> ::= <set name> [ <subscript list> ]
  4310 -- <display entry> ::= <parameter name>
  4311 -- <display entry> ::= <parameter name> [ <subscript list> ]
  4312 -- <display entry> ::= <variable name>
  4313 -- <display entry> ::= <variable name> [ <subscript list> ]
  4314 -- <display entry> ::= <constraint name>
  4315 -- <display entry> ::= <constraint name> [ <subscript list> ]
  4316 -- <display entry> ::= <expression 13> */
  4317 
  4318 DISPLAY *display_statement(MPL *mpl)
  4319 {     DISPLAY *dpy;
  4320       DISPLAY1 *entry, *last_entry;
  4321       xassert(is_keyword(mpl, "display"));
  4322       /* create display descriptor */
  4323       dpy = alloc(DISPLAY);
  4324       dpy->domain = NULL;
  4325       dpy->list = last_entry = NULL;
  4326       get_token(mpl /* display */);
  4327       /* parse optional indexing expression */
  4328       if (mpl->token == T_LBRACE)
  4329          dpy->domain = indexing_expression(mpl);
  4330       /* skip optional colon */
  4331       if (mpl->token == T_COLON) get_token(mpl /* : */);
  4332       /* parse display list */
  4333       for (;;)
  4334       {  /* create new display entry */
  4335          entry = alloc(DISPLAY1);
  4336          entry->type = 0;
  4337          entry->next = NULL;
  4338          /* and append it to the display list */
  4339          if (dpy->list == NULL)
  4340             dpy->list = entry;
  4341          else
  4342             last_entry->next = entry;
  4343          last_entry = entry;
  4344          /* parse display entry */
  4345          if (mpl->token == T_NAME)
  4346          {  AVLNODE *node;
  4347             int next_token;
  4348             get_token(mpl /* <symbolic name> */);
  4349             next_token = mpl->token;
  4350             unget_token(mpl);
  4351             if (!(next_token == T_COMMA || next_token == T_SEMICOLON))
  4352             {  /* symbolic name begins expression */
  4353                goto expr;
  4354             }
  4355             /* display entry is dummy index or model object */
  4356             node = avl_find_node(mpl->tree, mpl->image);
  4357             if (node == NULL)
  4358                error(mpl, "%s not defined", mpl->image);
  4359             entry->type = avl_get_node_type(node);
  4360             switch (avl_get_node_type(node))
  4361             {  case A_INDEX:
  4362                   entry->u.slot =
  4363                      (DOMAIN_SLOT *)avl_get_node_link(node);
  4364                   break;
  4365                case A_SET:
  4366                   entry->u.set = (SET *)avl_get_node_link(node);
  4367                   break;
  4368                case A_PARAMETER:
  4369                   entry->u.par = (PARAMETER *)avl_get_node_link(node);
  4370                   break;
  4371                case A_VARIABLE:
  4372                   entry->u.var = (VARIABLE *)avl_get_node_link(node);
  4373                   if (!mpl->flag_s)
  4374                      error(mpl, "invalid reference to variable %s above"
  4375                         " solve statement", entry->u.var->name);
  4376                   break;
  4377                case A_CONSTRAINT:
  4378                   entry->u.con = (CONSTRAINT *)avl_get_node_link(node);
  4379                   if (!mpl->flag_s)
  4380                      error(mpl, "invalid reference to %s %s above solve"
  4381                         " statement",
  4382                         entry->u.con->type == A_CONSTRAINT ?
  4383                         "constraint" : "objective", entry->u.con->name);
  4384                   break;
  4385                default:
  4386                   xassert(node != node);
  4387             }
  4388             get_token(mpl /* <symbolic name> */);
  4389          }
  4390          else
  4391 expr:    {  /* display entry is expression */
  4392             entry->type = A_EXPRESSION;
  4393             entry->u.code = expression_13(mpl);
  4394          }
  4395          /* check a token that follows the entry parsed */
  4396          if (mpl->token == T_COMMA)
  4397             get_token(mpl /* , */);
  4398          else
  4399             break;
  4400       }
  4401       /* close the domain scope */
  4402       if (dpy->domain != NULL) close_scope(mpl, dpy->domain);
  4403       /* the display statement has been completely parsed */
  4404       if (mpl->token != T_SEMICOLON)
  4405          error(mpl, "syntax error in display statement");
  4406       get_token(mpl /* ; */);
  4407       return dpy;
  4408 }
  4409 #endif
  4410 
  4411 /*----------------------------------------------------------------------
  4412 -- printf_statement - parse printf statement.
  4413 --
  4414 -- This routine parses print statement using the syntax:
  4415 --
  4416 -- <printf statement> ::= <printf clause> ;
  4417 -- <printf statement> ::= <printf clause> > <file name> ;
  4418 -- <printf statement> ::= <printf clause> >> <file name> ;
  4419 -- <printf clause> ::= printf <domain> : <format> <printf list>
  4420 -- <printf clause> ::= printf <domain> <format> <printf list>
  4421 -- <domain> ::= <empty>
  4422 -- <domain> ::= <indexing expression>
  4423 -- <format> ::= <expression 5>
  4424 -- <printf list> ::= <empty>
  4425 -- <printf list> ::= <printf list> , <printf entry>
  4426 -- <printf entry> ::= <expression 9>
  4427 -- <file name> ::= <expression 5> */
  4428 
  4429 PRINTF *printf_statement(MPL *mpl)
  4430 {     PRINTF *prt;
  4431       PRINTF1 *entry, *last_entry;
  4432       xassert(is_keyword(mpl, "printf"));
  4433       /* create printf descriptor */
  4434       prt = alloc(PRINTF);
  4435       prt->domain = NULL;
  4436       prt->fmt = NULL;
  4437       prt->list = last_entry = NULL;
  4438       get_token(mpl /* printf */);
  4439       /* parse optional indexing expression */
  4440       if (mpl->token == T_LBRACE)
  4441       {  prt->domain = indexing_expression(mpl);
  4442 #if 0
  4443          if (mpl->token != T_COLON)
  4444             error(mpl, "colon missing where expected");
  4445 #endif
  4446       }
  4447       /* skip optional colon */
  4448       if (mpl->token == T_COLON) get_token(mpl /* : */);
  4449       /* parse expression for format string */
  4450       prt->fmt = expression_5(mpl);
  4451       /* convert it to symbolic type, if necessary */
  4452       if (prt->fmt->type == A_NUMERIC)
  4453          prt->fmt = make_unary(mpl, O_CVTSYM, prt->fmt, A_SYMBOLIC, 0);
  4454       /* check that now the expression is of symbolic type */
  4455       if (prt->fmt->type != A_SYMBOLIC)
  4456          error(mpl, "format expression has invalid type");
  4457       /* parse printf list */
  4458       while (mpl->token == T_COMMA)
  4459       {  get_token(mpl /* , */);
  4460          /* create new printf entry */
  4461          entry = alloc(PRINTF1);
  4462          entry->code = NULL;
  4463          entry->next = NULL;
  4464          /* and append it to the printf list */
  4465          if (prt->list == NULL)
  4466             prt->list = entry;
  4467          else
  4468             last_entry->next = entry;
  4469          last_entry = entry;
  4470          /* parse printf entry */
  4471          entry->code = expression_9(mpl);
  4472          if (!(entry->code->type == A_NUMERIC ||
  4473                entry->code->type == A_SYMBOLIC ||
  4474                entry->code->type == A_LOGICAL))
  4475             error(mpl, "only numeric, symbolic, or logical expression a"
  4476                "llowed");
  4477       }
  4478       /* close the domain scope */
  4479       if (prt->domain != NULL) close_scope(mpl, prt->domain);
  4480 #if 1 /* 14/VII-2006 */
  4481       /* parse optional redirection */
  4482       prt->fname = NULL, prt->app = 0;
  4483       if (mpl->token == T_GT || mpl->token == T_APPEND)
  4484       {  prt->app = (mpl->token == T_APPEND);
  4485          get_token(mpl /* > or >> */);
  4486          /* parse expression for file name string */
  4487          prt->fname = expression_5(mpl);
  4488          /* convert it to symbolic type, if necessary */
  4489          if (prt->fname->type == A_NUMERIC)
  4490             prt->fname = make_unary(mpl, O_CVTSYM, prt->fname,
  4491                A_SYMBOLIC, 0);
  4492          /* check that now the expression is of symbolic type */
  4493          if (prt->fname->type != A_SYMBOLIC)
  4494             error(mpl, "file name expression has invalid type");
  4495       }
  4496 #endif
  4497       /* the printf statement has been completely parsed */
  4498       if (mpl->token != T_SEMICOLON)
  4499          error(mpl, "syntax error in printf statement");
  4500       get_token(mpl /* ; */);
  4501       return prt;
  4502 }
  4503 
  4504 /*----------------------------------------------------------------------
  4505 -- for_statement - parse for statement.
  4506 --
  4507 -- This routine parses for statement using the syntax:
  4508 --
  4509 -- <for statement> ::= for <domain> <statement>
  4510 -- <for statement> ::= for <domain> { <statement list> }
  4511 -- <domain> ::= <indexing expression>
  4512 -- <statement list> ::= <empty>
  4513 -- <statement list> ::= <statement list> <statement>
  4514 -- <statement> ::= <check statement>
  4515 -- <statement> ::= <display statement>
  4516 -- <statement> ::= <printf statement>
  4517 -- <statement> ::= <for statement> */
  4518 
  4519 FOR *for_statement(MPL *mpl)
  4520 {     FOR *fur;
  4521       STATEMENT *stmt, *last_stmt;
  4522       xassert(is_keyword(mpl, "for"));
  4523       /* create for descriptor */
  4524       fur = alloc(FOR);
  4525       fur->domain = NULL;
  4526       fur->list = last_stmt = NULL;
  4527       get_token(mpl /* for */);
  4528       /* parse indexing expression */
  4529       if (mpl->token != T_LBRACE)
  4530          error(mpl, "indexing expression missing where expected");
  4531       fur->domain = indexing_expression(mpl);
  4532       /* skip optional colon */
  4533       if (mpl->token == T_COLON) get_token(mpl /* : */);
  4534       /* parse for statement body */
  4535       if (mpl->token != T_LBRACE)
  4536       {  /* parse simple statement */
  4537          fur->list = simple_statement(mpl, 1);
  4538       }
  4539       else
  4540       {  /* parse compound statement */
  4541          get_token(mpl /* { */);
  4542          while (mpl->token != T_RBRACE)
  4543          {  /* parse statement */
  4544             stmt = simple_statement(mpl, 1);
  4545             /* and append it to the end of the statement list */
  4546             if (last_stmt == NULL)
  4547                fur->list = stmt;
  4548             else
  4549                last_stmt->next = stmt;
  4550             last_stmt = stmt;
  4551          }
  4552          get_token(mpl /* } */);
  4553       }
  4554       /* close the domain scope */
  4555       xassert(fur->domain != NULL);
  4556       close_scope(mpl, fur->domain);
  4557       /* the for statement has been completely parsed */
  4558       return fur;
  4559 }
  4560 
  4561 /*----------------------------------------------------------------------
  4562 -- end_statement - parse end statement.
  4563 --
  4564 -- This routine parses end statement using the syntax:
  4565 --
  4566 -- <end statement> ::= end ; <eof> */
  4567 
  4568 void end_statement(MPL *mpl)
  4569 {     if (!mpl->flag_d && is_keyword(mpl, "end") ||
  4570            mpl->flag_d && is_literal(mpl, "end"))
  4571       {  get_token(mpl /* end */);
  4572          if (mpl->token == T_SEMICOLON)
  4573             get_token(mpl /* ; */);
  4574          else
  4575             warning(mpl, "no semicolon following end statement; missing"
  4576                " semicolon inserted");
  4577       }
  4578       else
  4579          warning(mpl, "unexpected end of file; missing end statement in"
  4580             "serted");
  4581       if (mpl->token != T_EOF)
  4582          warning(mpl, "some text detected beyond end statement; text ig"
  4583             "nored");
  4584       return;
  4585 }
  4586 
  4587 /*----------------------------------------------------------------------
  4588 -- simple_statement - parse simple statement.
  4589 --
  4590 -- This routine parses simple statement using the syntax:
  4591 --
  4592 -- <statement> ::= <set statement>
  4593 -- <statement> ::= <parameter statement>
  4594 -- <statement> ::= <variable statement>
  4595 -- <statement> ::= <constraint statement>
  4596 -- <statement> ::= <objective statement>
  4597 -- <statement> ::= <solve statement>
  4598 -- <statement> ::= <check statement>
  4599 -- <statement> ::= <display statement>
  4600 -- <statement> ::= <printf statement>
  4601 -- <statement> ::= <for statement>
  4602 --
  4603 -- If the flag spec is set, some statements cannot be used. */
  4604 
  4605 STATEMENT *simple_statement(MPL *mpl, int spec)
  4606 {     STATEMENT *stmt;
  4607       stmt = alloc(STATEMENT);
  4608       stmt->line = mpl->line;
  4609       stmt->next = NULL;
  4610       if (is_keyword(mpl, "set"))
  4611       {  if (spec)
  4612             error(mpl, "set statement not allowed here");
  4613          stmt->type = A_SET;
  4614          stmt->u.set = set_statement(mpl);
  4615       }
  4616       else if (is_keyword(mpl, "param"))
  4617       {  if (spec)
  4618             error(mpl, "parameter statement not allowed here");
  4619          stmt->type = A_PARAMETER;
  4620          stmt->u.par = parameter_statement(mpl);
  4621       }
  4622       else if (is_keyword(mpl, "var"))
  4623       {  if (spec)
  4624             error(mpl, "variable statement not allowed here");
  4625          stmt->type = A_VARIABLE;
  4626          stmt->u.var = variable_statement(mpl);
  4627       }
  4628       else if (is_keyword(mpl, "subject") ||
  4629                is_keyword(mpl, "subj") ||
  4630                mpl->token == T_SPTP)
  4631       {  if (spec)
  4632             error(mpl, "constraint statement not allowed here");
  4633          stmt->type = A_CONSTRAINT;
  4634          stmt->u.con = constraint_statement(mpl);
  4635       }
  4636       else if (is_keyword(mpl, "minimize") ||
  4637                is_keyword(mpl, "maximize"))
  4638       {  if (spec)
  4639             error(mpl, "objective statement not allowed here");
  4640          stmt->type = A_CONSTRAINT;
  4641          stmt->u.con = objective_statement(mpl);
  4642       }
  4643 #if 1 /* 11/II-2008 */
  4644       else if (is_keyword(mpl, "table"))
  4645       {  if (spec)
  4646             error(mpl, "table statement not allowed here");
  4647          stmt->type = A_TABLE;
  4648          stmt->u.tab = table_statement(mpl);
  4649       }
  4650 #endif
  4651       else if (is_keyword(mpl, "solve"))
  4652       {  if (spec)
  4653             error(mpl, "solve statement not allowed here");
  4654          stmt->type = A_SOLVE;
  4655          stmt->u.slv = solve_statement(mpl);
  4656       }
  4657       else if (is_keyword(mpl, "check"))
  4658       {  stmt->type = A_CHECK;
  4659          stmt->u.chk = check_statement(mpl);
  4660       }
  4661       else if (is_keyword(mpl, "display"))
  4662       {  stmt->type = A_DISPLAY;
  4663          stmt->u.dpy = display_statement(mpl);
  4664       }
  4665       else if (is_keyword(mpl, "printf"))
  4666       {  stmt->type = A_PRINTF;
  4667          stmt->u.prt = printf_statement(mpl);
  4668       }
  4669       else if (is_keyword(mpl, "for"))
  4670       {  stmt->type = A_FOR;
  4671          stmt->u.fur = for_statement(mpl);
  4672       }
  4673       else if (mpl->token == T_NAME)
  4674       {  if (spec)
  4675             error(mpl, "constraint statement not allowed here");
  4676          stmt->type = A_CONSTRAINT;
  4677          stmt->u.con = constraint_statement(mpl);
  4678       }
  4679       else if (is_reserved(mpl))
  4680          error(mpl, "invalid use of reserved keyword %s", mpl->image);
  4681       else
  4682          error(mpl, "syntax error in model section");
  4683       return stmt;
  4684 }
  4685 
  4686 /*----------------------------------------------------------------------
  4687 -- model_section - parse model section.
  4688 --
  4689 -- This routine parses model section using the syntax:
  4690 --
  4691 -- <model section> ::= <empty>
  4692 -- <model section> ::= <model section> <statement>
  4693 --
  4694 -- Parsing model section is terminated by either the keyword 'data', or
  4695 -- the keyword 'end', or the end of file. */
  4696 
  4697 void model_section(MPL *mpl)
  4698 {     STATEMENT *stmt, *last_stmt;
  4699       xassert(mpl->model == NULL);
  4700       last_stmt = NULL;
  4701       while (!(mpl->token == T_EOF || is_keyword(mpl, "data") ||
  4702                is_keyword(mpl, "end")))
  4703       {  /* parse statement */
  4704          stmt = simple_statement(mpl, 0);
  4705          /* and append it to the end of the statement list */
  4706          if (last_stmt == NULL)
  4707             mpl->model = stmt;
  4708          else
  4709             last_stmt->next = stmt;
  4710          last_stmt = stmt;
  4711       }
  4712       return;
  4713 }
  4714 
  4715 /* eof */