parser-top-down.c

     
   1  //! @file parser-top-down.c
   2  //! @author J. Marcel van der Veer
   3  
   4  //! @section Copyright
   5  //!
   6  //! This file is part of Algol68G - an Algol 68 compiler-interpreter.
   7  //! Copyright 2001-2025 J. Marcel van der Veer [algol68g@xs4all.nl].
   8  
   9  //! @section License
  10  //!
  11  //! This program is free software; you can redistribute it and/or modify it 
  12  //! under the terms of the GNU General Public License as published by the 
  13  //! Free Software Foundation; either version 3 of the License, or 
  14  //! (at your option) any later version.
  15  //!
  16  //! This program is distributed in the hope that it will be useful, but 
  17  //! WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY 
  18  //! or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for 
  19  //! more details. You should have received a copy of the GNU General Public 
  20  //! License along with this program. If not, see [http://www.gnu.org/licenses/].
  21  
  22  //! @section Synopsis 
  23  //!
  24  //! Top-down parser for control structure.
  25  
  26  #include "a68g.h"
  27  #include "a68g-parser.h"
  28  
  29  // Top-down parser, elaborates the control structure.
  30  
  31  //! @brief Substitute brackets.
  32  
  33  void substitute_brackets (NODE_T * p)
  34  {
  35    for (; p != NO_NODE; FORWARD (p)) {
  36      substitute_brackets (SUB (p));
  37      switch (ATTRIBUTE (p)) {
  38      case ACCO_SYMBOL: {
  39          ATTRIBUTE (p) = OPEN_SYMBOL;
  40          break;
  41        }
  42      case OCCA_SYMBOL: {
  43          ATTRIBUTE (p) = CLOSE_SYMBOL;
  44          break;
  45        }
  46      case SUB_SYMBOL: {
  47          ATTRIBUTE (p) = OPEN_SYMBOL;
  48          break;
  49        }
  50      case BUS_SYMBOL: {
  51          ATTRIBUTE (p) = CLOSE_SYMBOL;
  52          break;
  53        }
  54      }
  55    }
  56  }
  57  
  58  //! @brief Intelligible diagnostic from syntax tree branch.
  59  
  60  char *phrase_to_text (NODE_T * p, NODE_T ** w)
  61  {
  62  #define MAX_TERMINALS 8
  63    int count = 0, line = -1;
  64    static BUFFER buffer;
  65    for (buffer[0] = NULL_CHAR; p != NO_NODE && count < MAX_TERMINALS; FORWARD (p)) {
  66      if (LINE_NUMBER (p) > 0) {
  67        int gatt = get_good_attribute (p);
  68        char *z = non_terminal_string (A68 (input_line), gatt);
  69  // Where to put the error message? Bob Uzgalis noted that actual content of a 
  70  // diagnostic is not as important as accurately indicating *were* the problem is! 
  71        if (w != NO_VAR) {
  72          if (count == 0 || (*w) == NO_NODE) {
  73            *w = p;
  74          } else if (dont_mark_here (*w)) {
  75            *w = p;
  76          }
  77        }
  78  // Add initiation.
  79        if (count == 0) {
  80          if (w != NO_VAR) {
  81            a68_bufcat (buffer, "construct beginning with", BUFFER_SIZE);
  82          }
  83        } else if (count == 1) {
  84          a68_bufcat (buffer, " followed by", BUFFER_SIZE);
  85        } else if (count == 2) {
  86          a68_bufcat (buffer, " and then", BUFFER_SIZE);
  87        } else if (count >= 3) {
  88          a68_bufcat (buffer, " and", BUFFER_SIZE);
  89        }
  90  // Attribute or symbol.
  91        if (z != NO_TEXT && SUB (p) != NO_NODE) {
  92          if (gatt == IDENTIFIER || gatt == OPERATOR || gatt == DENOTATION) {
  93            ASSERT (a68_bufprt (A68 (edit_line), SNPRINTF_SIZE, " \"%s\"", NSYMBOL (p)) >= 0);
  94            a68_bufcat (buffer, A68 (edit_line), BUFFER_SIZE);
  95          } else {
  96            if (strchr ("aeio", z[0]) != NO_TEXT) {
  97              a68_bufcat (buffer, " an", BUFFER_SIZE);
  98            } else {
  99              a68_bufcat (buffer, " a", BUFFER_SIZE);
 100            }
 101            ASSERT (a68_bufprt (A68 (edit_line), SNPRINTF_SIZE, " %s", z) >= 0);
 102            a68_bufcat (buffer, A68 (edit_line), BUFFER_SIZE);
 103          }
 104        } else if (z != NO_TEXT && SUB (p) == NO_NODE) {
 105          ASSERT (a68_bufprt (A68 (edit_line), SNPRINTF_SIZE, " \"%s\"", NSYMBOL (p)) >= 0);
 106          a68_bufcat (buffer, A68 (edit_line), BUFFER_SIZE);
 107        } else if (NSYMBOL (p) != NO_TEXT) {
 108          ASSERT (a68_bufprt (A68 (edit_line), SNPRINTF_SIZE, " \"%s\"", NSYMBOL (p)) >= 0);
 109          a68_bufcat (buffer, A68 (edit_line), BUFFER_SIZE);
 110        }
 111  // Add "starting in line nn".
 112        if (z != NO_TEXT && line != LINE_NUMBER (p)) {
 113          line = LINE_NUMBER (p);
 114          if (gatt == SERIAL_CLAUSE || gatt == ENQUIRY_CLAUSE || gatt == INITIALISER_SERIES) {
 115            a68_bufcat (buffer, " starting", BUFFER_SIZE);
 116          }
 117          ASSERT (a68_bufprt (A68 (edit_line), SNPRINTF_SIZE, " in line %d", line) >= 0);
 118          a68_bufcat (buffer, A68 (edit_line), BUFFER_SIZE);
 119        }
 120        count++;
 121      }
 122    }
 123    if (p != NO_NODE && count == MAX_TERMINALS) {
 124      a68_bufcat (buffer, " etcetera", BUFFER_SIZE);
 125    }
 126    return buffer;
 127  }
 128  
 129  // Next is a top-down parser that branches out the basic blocks.
 130  // After this we can assign symbol tables to basic blocks.
 131  // This renders the two-level grammar LALR.
 132  
 133  //! @brief Give diagnose from top-down parser.
 134  
 135  void top_down_diagnose (NODE_T * start, NODE_T * p, int clause, int expected)
 136  {
 137    NODE_T *issue = (p != NO_NODE ? p : start);
 138    if (expected != 0) {
 139      diagnostic (A68_SYNTAX_ERROR, issue, ERROR_EXPECTED_NEAR, expected, clause, NSYMBOL (start), LINE (INFO (start)));
 140    } else {
 141      diagnostic (A68_SYNTAX_ERROR, issue, ERROR_UNBALANCED_KEYWORD, clause, NSYMBOL (start), LINE (INFO (start)));
 142    }
 143  }
 144  
 145  //! @brief Check for premature exhaustion of tokens.
 146  
 147  void tokens_exhausted (const NODE_T * p, NODE_T * q)
 148  {
 149    if (p == NO_NODE) {
 150      diagnostic (A68_SYNTAX_ERROR, q, ERROR_KEYWORD);
 151      longjmp (A68_PARSER (top_down_crash_exit), 1);
 152    }
 153  }
 154  
 155  // This part specifically branches out loop clauses.
 156  
 157  //! @brief Whether in cast or formula with loop clause.
 158  
 159  int is_loop_cast_formula (NODE_T * p)
 160  {
 161  // Accept declarers that can appear in such casts but not much more.
 162    if (IS (p, VOID_SYMBOL)) {
 163      return 1;
 164    } else if (IS (p, INT_SYMBOL)) {
 165      return 1;
 166    } else if (IS_REF (p)) {
 167      return 1;
 168    } else if (is_one_of (p, OPERATOR, BOLD_TAG, STOP)) {
 169      return 1;
 170    } else if (whether (p, UNION_SYMBOL, OPEN_SYMBOL, STOP)) {
 171      return 2;
 172    } else if (is_one_of (p, OPEN_SYMBOL, SUB_SYMBOL, STOP)) {
 173      int k = 0;
 174      for (; p != NO_NODE && (is_one_of (p, OPEN_SYMBOL, SUB_SYMBOL, STOP)); FORWARD (p), k++) {
 175        ;
 176      }
 177      return p != NO_NODE && (whether (p, UNION_SYMBOL, OPEN_SYMBOL, STOP) ? k : 0);
 178    }
 179    return 0;
 180  }
 181  
 182  //! @brief Skip a unit in a loop clause (FROM u BY u TO u).
 183  
 184  NODE_T *top_down_skip_loop_unit (NODE_T * p)
 185  {
 186  // Unit may start with, or consist of, a loop.
 187    if (is_loop_keyword (p)) {
 188      p = top_down_loop (p);
 189    }
 190  // Skip rest of unit.
 191    while (p != NO_NODE) {
 192      int k = is_loop_cast_formula (p);
 193      if (k != 0) {
 194  // operator-cast series ...
 195        while (p != NO_NODE && k != 0) {
 196          while (k != 0) {
 197            FORWARD (p);
 198            k--;
 199          }
 200          k = is_loop_cast_formula (p);
 201        }
 202  // ... may be followed by a loop clause.
 203        if (is_loop_keyword (p)) {
 204          p = top_down_loop (p);
 205        }
 206      } else if (is_loop_keyword (p) || IS (p, OD_SYMBOL)) {
 207  // new loop or end-of-loop.
 208        return p;
 209      } else if (IS (p, COLON_SYMBOL)) {
 210        FORWARD (p);
 211  // skip routine header: loop clause.
 212        if (p != NO_NODE && is_loop_keyword (p)) {
 213          p = top_down_loop (p);
 214        }
 215      } else if (is_one_of (p, SEMI_SYMBOL, COMMA_SYMBOL, STOP) || IS (p, EXIT_SYMBOL)) {
 216  // Statement separators.
 217        return p;
 218      } else {
 219        FORWARD (p);
 220      }
 221    }
 222    return NO_NODE;
 223  }
 224  
 225  //! @brief Skip a loop clause.
 226  
 227  NODE_T *top_down_skip_loop_series (NODE_T * p)
 228  {
 229    BOOL_T siga;
 230    do {
 231      p = top_down_skip_loop_unit (p);
 232      siga = (BOOL_T) (p != NO_NODE && (is_one_of (p, SEMI_SYMBOL, EXIT_SYMBOL, COMMA_SYMBOL, COLON_SYMBOL, STOP)));
 233      if (siga) {
 234        FORWARD (p);
 235      }
 236    } while (!(p == NO_NODE || !siga));
 237    return p;
 238  }
 239  
 240  //! @brief Make branch of loop parts.
 241  
 242  NODE_T *top_down_loop (NODE_T * p)
 243  {
 244    NODE_T *start = p, *q = p;
 245    if (IS (q, FOR_SYMBOL)) {
 246      tokens_exhausted (FORWARD (q), start);
 247      if (IS (q, IDENTIFIER)) {
 248        ATTRIBUTE (q) = DEFINING_IDENTIFIER;
 249      } else {
 250        top_down_diagnose (start, q, LOOP_CLAUSE, IDENTIFIER);
 251        longjmp (A68_PARSER (top_down_crash_exit), 1);
 252      }
 253      tokens_exhausted (FORWARD (q), start);
 254      if (is_one_of (q, FROM_SYMBOL, BY_SYMBOL, TO_SYMBOL, DOWNTO_SYMBOL, WHILE_SYMBOL, STOP)) {
 255        ;
 256      } else if (IS (q, DO_SYMBOL)) {
 257        ATTRIBUTE (q) = ALT_DO_SYMBOL;
 258      } else {
 259        top_down_diagnose (start, q, LOOP_CLAUSE, STOP);
 260        longjmp (A68_PARSER (top_down_crash_exit), 1);
 261      }
 262    }
 263    if (IS (q, FROM_SYMBOL)) {
 264      start = q;
 265      q = top_down_skip_loop_unit (NEXT (q));
 266      tokens_exhausted (q, start);
 267      if (is_one_of (q, BY_SYMBOL, TO_SYMBOL, DOWNTO_SYMBOL, WHILE_SYMBOL, STOP)) {
 268        ;
 269      } else if (IS (q, DO_SYMBOL)) {
 270        ATTRIBUTE (q) = ALT_DO_SYMBOL;
 271      } else {
 272        top_down_diagnose (start, q, LOOP_CLAUSE, STOP);
 273        longjmp (A68_PARSER (top_down_crash_exit), 1);
 274      }
 275      make_sub (start, PREVIOUS (q), FROM_SYMBOL);
 276    }
 277    if (IS (q, BY_SYMBOL)) {
 278      start = q;
 279      q = top_down_skip_loop_series (NEXT (q));
 280      tokens_exhausted (q, start);
 281      if (is_one_of (q, TO_SYMBOL, DOWNTO_SYMBOL, WHILE_SYMBOL, STOP)) {
 282        ;
 283      } else if (IS (q, DO_SYMBOL)) {
 284        ATTRIBUTE (q) = ALT_DO_SYMBOL;
 285      } else {
 286        top_down_diagnose (start, q, LOOP_CLAUSE, STOP);
 287        longjmp (A68_PARSER (top_down_crash_exit), 1);
 288      }
 289      make_sub (start, PREVIOUS (q), BY_SYMBOL);
 290    }
 291    if (is_one_of (q, TO_SYMBOL, DOWNTO_SYMBOL, STOP)) {
 292      start = q;
 293      q = top_down_skip_loop_series (NEXT (q));
 294      tokens_exhausted (q, start);
 295      if (IS (q, WHILE_SYMBOL)) {
 296        ;
 297      } else if (IS (q, DO_SYMBOL)) {
 298        ATTRIBUTE (q) = ALT_DO_SYMBOL;
 299      } else {
 300        top_down_diagnose (start, q, LOOP_CLAUSE, STOP);
 301        longjmp (A68_PARSER (top_down_crash_exit), 1);
 302      }
 303      make_sub (start, PREVIOUS (q), TO_SYMBOL);
 304    }
 305    if (IS (q, WHILE_SYMBOL)) {
 306      start = q;
 307      q = top_down_skip_loop_series (NEXT (q));
 308      tokens_exhausted (q, start);
 309      if (IS (q, DO_SYMBOL)) {
 310        ATTRIBUTE (q) = ALT_DO_SYMBOL;
 311      } else {
 312        top_down_diagnose (start, q, LOOP_CLAUSE, DO_SYMBOL);
 313        longjmp (A68_PARSER (top_down_crash_exit), 1);
 314      }
 315      make_sub (start, PREVIOUS (q), WHILE_SYMBOL);
 316    }
 317    if (is_one_of (q, DO_SYMBOL, ALT_DO_SYMBOL, STOP)) {
 318      int k = ATTRIBUTE (q);
 319      start = q;
 320      q = top_down_skip_loop_series (NEXT (q));
 321      tokens_exhausted (q, start);
 322      if (!IS (q, OD_SYMBOL)) {
 323        top_down_diagnose (start, q, LOOP_CLAUSE, OD_SYMBOL);
 324        longjmp (A68_PARSER (top_down_crash_exit), 1);
 325      }
 326      make_sub (start, q, k);
 327    }
 328    NODE_T *save = NEXT (start);
 329    make_sub (p, start, LOOP_CLAUSE);
 330    return save;
 331  }
 332  
 333  //! @brief Driver for making branches of loop parts.
 334  
 335  void top_down_loops (NODE_T * p)
 336  {
 337    NODE_T *q = p;
 338    for (; q != NO_NODE; FORWARD (q)) {
 339      if (SUB (q) != NO_NODE) {
 340        top_down_loops (SUB (q));
 341      }
 342    }
 343    q = p;
 344    while (q != NO_NODE) {
 345      if (is_loop_keyword (q) != STOP) {
 346        q = top_down_loop (q);
 347      } else {
 348        FORWARD (q);
 349      }
 350    }
 351  }
 352  
 353  //! @brief Driver for making branches of until parts.
 354  
 355  void top_down_untils (NODE_T * p)
 356  {
 357    NODE_T *q = p;
 358    for (; q != NO_NODE; FORWARD (q)) {
 359      if (SUB (q) != NO_NODE) {
 360        top_down_untils (SUB (q));
 361      }
 362    }
 363    q = p;
 364    while (q != NO_NODE) {
 365      if (IS (q, UNTIL_SYMBOL)) {
 366        NODE_T *u = q;
 367        while (NEXT (u) != NO_NODE) {
 368          FORWARD (u);
 369        }
 370        make_sub (q, PREVIOUS (u), UNTIL_SYMBOL);
 371        return;
 372      } else {
 373        FORWARD (q);
 374      }
 375    }
 376  }
 377  
 378  // Branch anything except parts of a loop.
 379  
 380  //! @brief Skip serial/enquiry clause (unit series).
 381  
 382  NODE_T *top_down_series (NODE_T * p)
 383  {
 384    BOOL_T siga = A68_TRUE;
 385    while (siga) {
 386      siga = A68_FALSE;
 387      p = top_down_skip_unit (p);
 388      if (p != NO_NODE) {
 389        if (is_one_of (p, SEMI_SYMBOL, EXIT_SYMBOL, COMMA_SYMBOL, STOP)) {
 390          siga = A68_TRUE;
 391          FORWARD (p);
 392        }
 393      }
 394    }
 395    return p;
 396  }
 397  
 398  //! @brief Make branch of BEGIN .. END.
 399  
 400  NODE_T *top_down_begin (NODE_T * begin_p)
 401  {
 402    NODE_T *end_p = top_down_series (NEXT (begin_p));
 403    if (end_p == NO_NODE || !IS (end_p, END_SYMBOL)) {
 404      top_down_diagnose (begin_p, end_p, ENCLOSED_CLAUSE, END_SYMBOL);
 405      longjmp (A68_PARSER (top_down_crash_exit), 1);
 406      return NO_NODE;
 407    } else {
 408      make_sub (begin_p, end_p, BEGIN_SYMBOL);
 409      return NEXT (begin_p);
 410    }
 411  }
 412  
 413  //! @brief Make branch of CODE .. EDOC.
 414  
 415  NODE_T *top_down_code (NODE_T * code_p)
 416  {
 417    NODE_T *edoc_p = top_down_series (NEXT (code_p));
 418    if (edoc_p == NO_NODE || !IS (edoc_p, EDOC_SYMBOL)) {
 419      diagnostic (A68_SYNTAX_ERROR, code_p, ERROR_KEYWORD);
 420      longjmp (A68_PARSER (top_down_crash_exit), 1);
 421      return NO_NODE;
 422    } else {
 423      make_sub (code_p, edoc_p, CODE_SYMBOL);
 424      return NEXT (code_p);
 425    }
 426  }
 427  
 428  //! @brief Make branch of ( .. ).
 429  
 430  NODE_T *top_down_open (NODE_T * open_p)
 431  {
 432    NODE_T *then_bar_p = top_down_series (NEXT (open_p)), *elif_bar_p;
 433    if (then_bar_p != NO_NODE && IS (then_bar_p, CLOSE_SYMBOL)) {
 434      make_sub (open_p, then_bar_p, OPEN_SYMBOL);
 435      return NEXT (open_p);
 436    }
 437    if (then_bar_p == NO_NODE || !IS (then_bar_p, THEN_BAR_SYMBOL)) {
 438      top_down_diagnose (open_p, then_bar_p, ENCLOSED_CLAUSE, STOP);
 439      longjmp (A68_PARSER (top_down_crash_exit), 1);
 440    }
 441    make_sub (open_p, PREVIOUS (then_bar_p), OPEN_SYMBOL);
 442    elif_bar_p = top_down_series (NEXT (then_bar_p));
 443    if (elif_bar_p != NO_NODE && IS (elif_bar_p, CLOSE_SYMBOL)) {
 444      make_sub (then_bar_p, PREVIOUS (elif_bar_p), THEN_BAR_SYMBOL);
 445      make_sub (open_p, elif_bar_p, OPEN_SYMBOL);
 446      return NEXT (open_p);
 447    }
 448    if (elif_bar_p != NO_NODE && IS (elif_bar_p, THEN_BAR_SYMBOL)) {
 449      NODE_T *close_p = top_down_series (NEXT (elif_bar_p));
 450      if (close_p == NO_NODE || !IS (close_p, CLOSE_SYMBOL)) {
 451        top_down_diagnose (open_p, elif_bar_p, ENCLOSED_CLAUSE, CLOSE_SYMBOL);
 452        longjmp (A68_PARSER (top_down_crash_exit), 1);
 453      }
 454      make_sub (then_bar_p, PREVIOUS (elif_bar_p), THEN_BAR_SYMBOL);
 455      make_sub (elif_bar_p, PREVIOUS (close_p), THEN_BAR_SYMBOL);
 456      make_sub (open_p, close_p, OPEN_SYMBOL);
 457      return NEXT (open_p);
 458    }
 459    if (elif_bar_p != NO_NODE && IS (elif_bar_p, ELSE_BAR_SYMBOL)) {
 460      NODE_T *close_p = top_down_open (elif_bar_p);
 461      make_sub (then_bar_p, PREVIOUS (elif_bar_p), THEN_BAR_SYMBOL);
 462      make_sub (open_p, elif_bar_p, OPEN_SYMBOL);
 463      return close_p;
 464    } else {
 465      top_down_diagnose (open_p, elif_bar_p, ENCLOSED_CLAUSE, CLOSE_SYMBOL);
 466      longjmp (A68_PARSER (top_down_crash_exit), 1);
 467      return NO_NODE;
 468    }
 469  }
 470  
 471  //! @brief Make branch of [ .. ].
 472  
 473  NODE_T *top_down_sub (NODE_T * sub_p)
 474  {
 475    NODE_T *bus_p = top_down_series (NEXT (sub_p));
 476    if (bus_p != NO_NODE && IS (bus_p, BUS_SYMBOL)) {
 477      make_sub (sub_p, bus_p, SUB_SYMBOL);
 478      return NEXT (sub_p);
 479    } else {
 480      top_down_diagnose (sub_p, bus_p, 0, BUS_SYMBOL);
 481      longjmp (A68_PARSER (top_down_crash_exit), 1);
 482      return NO_NODE;
 483    }
 484  }
 485  
 486  //! @brief Make branch of { .. }.
 487  
 488  NODE_T *top_down_acco (NODE_T * acco_p)
 489  {
 490    NODE_T *occa_p = top_down_series (NEXT (acco_p));
 491    if (occa_p != NO_NODE && IS (occa_p, OCCA_SYMBOL)) {
 492      make_sub (acco_p, occa_p, ACCO_SYMBOL);
 493      return NEXT (acco_p);
 494    } else {
 495      top_down_diagnose (acco_p, occa_p, ENCLOSED_CLAUSE, OCCA_SYMBOL);
 496      longjmp (A68_PARSER (top_down_crash_exit), 1);
 497      return NO_NODE;
 498    }
 499  }
 500  
 501  //! @brief Make branch of IF .. THEN .. ELSE .. FI.
 502  
 503  NODE_T *top_down_if (NODE_T * if_p)
 504  {
 505    NODE_T *then_p = top_down_series (NEXT (if_p)), *elif_p;
 506    if (then_p == NO_NODE || !IS (then_p, THEN_SYMBOL)) {
 507      top_down_diagnose (if_p, then_p, CONDITIONAL_CLAUSE, THEN_SYMBOL);
 508      longjmp (A68_PARSER (top_down_crash_exit), 1);
 509    }
 510    make_sub (if_p, PREVIOUS (then_p), IF_SYMBOL);
 511    elif_p = top_down_series (NEXT (then_p));
 512    if (elif_p != NO_NODE && IS (elif_p, FI_SYMBOL)) {
 513      make_sub (then_p, PREVIOUS (elif_p), THEN_SYMBOL);
 514      make_sub (if_p, elif_p, IF_SYMBOL);
 515      return NEXT (if_p);
 516    }
 517    if (elif_p != NO_NODE && IS (elif_p, ELSE_SYMBOL)) {
 518      NODE_T *fi_p = top_down_series (NEXT (elif_p));
 519      if (fi_p == NO_NODE || !IS (fi_p, FI_SYMBOL)) {
 520        top_down_diagnose (if_p, fi_p, CONDITIONAL_CLAUSE, FI_SYMBOL);
 521        longjmp (A68_PARSER (top_down_crash_exit), 1);
 522      } else {
 523        make_sub (then_p, PREVIOUS (elif_p), THEN_SYMBOL);
 524        make_sub (elif_p, PREVIOUS (fi_p), ELSE_SYMBOL);
 525        make_sub (if_p, fi_p, IF_SYMBOL);
 526        return NEXT (if_p);
 527      }
 528    }
 529    if (elif_p != NO_NODE && IS (elif_p, ELIF_SYMBOL)) {
 530      NODE_T *fi_p = top_down_if (elif_p);
 531      make_sub (then_p, PREVIOUS (elif_p), THEN_SYMBOL);
 532      make_sub (if_p, elif_p, IF_SYMBOL);
 533      return fi_p;
 534    } else {
 535      top_down_diagnose (if_p, elif_p, CONDITIONAL_CLAUSE, FI_SYMBOL);
 536      longjmp (A68_PARSER (top_down_crash_exit), 1);
 537      return NO_NODE;
 538    }
 539  }
 540  
 541  //! @brief Make branch of CASE .. IN .. OUT .. ESAC.
 542  
 543  NODE_T *top_down_case (NODE_T * case_p)
 544  {
 545    NODE_T *in_p = top_down_series (NEXT (case_p)), *ouse_p;
 546    if (in_p == NO_NODE || !IS (in_p, IN_SYMBOL)) {
 547      top_down_diagnose (case_p, in_p, ENCLOSED_CLAUSE, IN_SYMBOL);
 548      longjmp (A68_PARSER (top_down_crash_exit), 1);
 549    }
 550    make_sub (case_p, PREVIOUS (in_p), CASE_SYMBOL);
 551    ouse_p = top_down_series (NEXT (in_p));
 552    if (ouse_p != NO_NODE && IS (ouse_p, ESAC_SYMBOL)) {
 553      make_sub (in_p, PREVIOUS (ouse_p), IN_SYMBOL);
 554      make_sub (case_p, ouse_p, CASE_SYMBOL);
 555      return NEXT (case_p);
 556    }
 557    if (ouse_p != NO_NODE && IS (ouse_p, OUT_SYMBOL)) {
 558      NODE_T *esac_p = top_down_series (NEXT (ouse_p));
 559      if (esac_p == NO_NODE || !IS (esac_p, ESAC_SYMBOL)) {
 560        top_down_diagnose (case_p, esac_p, ENCLOSED_CLAUSE, ESAC_SYMBOL);
 561        longjmp (A68_PARSER (top_down_crash_exit), 1);
 562      } else {
 563        make_sub (in_p, PREVIOUS (ouse_p), IN_SYMBOL);
 564        make_sub (ouse_p, PREVIOUS (esac_p), OUT_SYMBOL);
 565        make_sub (case_p, esac_p, CASE_SYMBOL);
 566        return NEXT (case_p);
 567      }
 568    }
 569    if (ouse_p != NO_NODE && IS (ouse_p, OUSE_SYMBOL)) {
 570      NODE_T *esac_p = top_down_case (ouse_p);
 571      make_sub (in_p, PREVIOUS (ouse_p), IN_SYMBOL);
 572      make_sub (case_p, ouse_p, CASE_SYMBOL);
 573      return esac_p;
 574    } else {
 575      top_down_diagnose (case_p, ouse_p, ENCLOSED_CLAUSE, ESAC_SYMBOL);
 576      longjmp (A68_PARSER (top_down_crash_exit), 1);
 577      return NO_NODE;
 578    }
 579  }
 580  
 581  //! @brief Skip a unit.
 582  
 583  NODE_T *top_down_skip_unit (NODE_T * p)
 584  {
 585    while (p != NO_NODE && !is_unit_terminator (p)) {
 586      if (IS (p, BEGIN_SYMBOL)) {
 587        p = top_down_begin (p);
 588      } else if (IS (p, SUB_SYMBOL)) {
 589        p = top_down_sub (p);
 590      } else if (IS (p, OPEN_SYMBOL)) {
 591        p = top_down_open (p);
 592      } else if (IS (p, IF_SYMBOL)) {
 593        p = top_down_if (p);
 594      } else if (IS (p, CASE_SYMBOL)) {
 595        p = top_down_case (p);
 596      } else if (IS (p, CODE_SYMBOL)) {
 597        p = top_down_code (p);
 598      } else if (IS (p, ACCO_SYMBOL)) {
 599        p = top_down_acco (p);
 600      } else {
 601        FORWARD (p);
 602      }
 603    }
 604    return p;
 605  }
 606  
 607  NODE_T *top_down_skip_format (NODE_T *);
 608  
 609  //! @brief Make branch of ( .. ) in a format.
 610  
 611  NODE_T *top_down_format_open (NODE_T * open_p)
 612  {
 613    NODE_T *close_p = top_down_skip_format (NEXT (open_p));
 614    if (close_p != NO_NODE && IS (close_p, FORMAT_CLOSE_SYMBOL)) {
 615      make_sub (open_p, close_p, FORMAT_OPEN_SYMBOL);
 616      return NEXT (open_p);
 617    } else {
 618      top_down_diagnose (open_p, close_p, 0, FORMAT_CLOSE_SYMBOL);
 619      longjmp (A68_PARSER (top_down_crash_exit), 1);
 620      return NO_NODE;
 621    }
 622  }
 623  
 624  //! @brief Skip a format text.
 625  
 626  NODE_T *top_down_skip_format (NODE_T * p)
 627  {
 628    while (p != NO_NODE) {
 629      if (IS (p, FORMAT_OPEN_SYMBOL)) {
 630        p = top_down_format_open (p);
 631      } else if (is_one_of (p, FORMAT_CLOSE_SYMBOL, FORMAT_DELIMITER_SYMBOL, STOP)) {
 632        return p;
 633      } else {
 634        FORWARD (p);
 635      }
 636    }
 637    return NO_NODE;
 638  }
 639  
 640  //! @brief Make branch of $ .. $.
 641  
 642  void top_down_formats (NODE_T * p)
 643  {
 644    for (NODE_T *q = p; q != NO_NODE; FORWARD (q)) {
 645      if (SUB (q) != NO_NODE) {
 646        top_down_formats (SUB (q));
 647      }
 648    }
 649    for (NODE_T *q = p; q != NO_NODE; FORWARD (q)) {
 650      if (IS (q, FORMAT_DELIMITER_SYMBOL)) {
 651        NODE_T *f = NEXT (q);
 652        while (f != NO_NODE && !IS (f, FORMAT_DELIMITER_SYMBOL)) {
 653          if (IS (f, FORMAT_OPEN_SYMBOL)) {
 654            f = top_down_format_open (f);
 655          } else {
 656            f = NEXT (f);
 657          }
 658        }
 659        if (f == NO_NODE) {
 660          top_down_diagnose (p, f, FORMAT_TEXT, FORMAT_DELIMITER_SYMBOL);
 661          longjmp (A68_PARSER (top_down_crash_exit), 1);
 662        } else {
 663          make_sub (q, f, FORMAT_DELIMITER_SYMBOL);
 664        }
 665      }
 666    }
 667  }
 668  
 669  //! @brief Make branches of phrases for the bottom-up parser.
 670  
 671  void top_down_parser (NODE_T * p)
 672  {
 673    if (p != NO_NODE) {
 674      if (!setjmp (A68_PARSER (top_down_crash_exit))) {
 675        (void) top_down_series (p);
 676        top_down_loops (p);
 677        top_down_untils (p);
 678        top_down_formats (p);
 679      }
 680    }
 681  }
     


© 2002-2025 J.M. van der Veer (jmvdveer@xs4all.nl)