table.c

     
   1  //! @file table.c
   2  //! @author J. Marcel van der Veer
   3  //
   4  //! @section Copyright
   5  //
   6  // This file is part of VIF - vintage FORTRAN compiler.
   7  // Copyright 2020-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  //! Symbol table routines.
  25  
  26  #include <vif.h>
  27  
  28  int_4 ord (char ch)
  29  {
  30    return (ch - 'a');
  31  }
  32  
  33  char *edit_name (char *name)
  34  {
  35    static RECORD idf, idf2;
  36    char *buf = idf;
  37    strcpy (buf, name);
  38    int_4 len = strlen (buf);
  39    while (len > 1 && buf[len - 1] == '_') {
  40      buf[len - 1] = '\0';
  41      len = strlen (buf);
  42    }
  43    while (len > 1 && buf[0] == '_') {
  44      buf++;
  45      len = strlen (buf);
  46    }
  47    strcpy (idf2, buf);
  48    return idf2;
  49  }
  50  
  51  char *edit_f (char *name)
  52  {
  53    static RECORD buf;
  54    _srecordf (buf, "_%s", edit_name (name));
  55    return buf;
  56  }
  57  
  58  char *edit_v (char *name)
  59  {
  60    static RECORD buf;
  61    _srecordf (buf, "%s_", edit_name (name));
  62    return buf;
  63  }
  64  
  65  char *edit_vn (char *name, int_4 nest)
  66  {
  67    static RECORD buf;
  68    _srecordf (buf, "%s_%d_", edit_name (name), nest);
  69    return buf;
  70  }
  71  
  72  char *edit_tmp (int_4 nest)
  73  {
  74    static RECORD buf;
  75    _srecordf (buf, "$%d_", nest);
  76    return buf;
  77  }
  78  
  79  char *edit_fmt (int_4 nest)
  80  {
  81    static RECORD buf;
  82    _srecordf (buf, "$f_%d_", nest);
  83    return buf;
  84  }
  85  
  86  char *edit_unit (int_4 nest)
  87  {
  88    static RECORD buf;
  89    _srecordf (buf, "$u_%d_", nest);
  90    return buf;
  91  }
  92  
  93  char *encode (char *buf, char *name)
  94  {
  95    bufcpy (buf, name, RECLN);
  96    int_4 len = (int_4) strlen (buf);
  97    if (buf[0] == '_' || buf[len - 1] == '_') {
  98      return buf;
  99    }
 100    bufcpy (buf, edit_v (name), RECLN);
 101    return buf;
 102  }
 103  
 104  char *c_name (char *name)
 105  {
 106    RECORD safe;
 107    RECCLR (safe);
 108    (void) encode (safe, name);
 109    return f_stralloc (safe);
 110  }
 111  
 112  int_4 same_name (char *name, char * id)
 113  {
 114  if (name == NO_TEXT || id == NO_TEXT) return FALSE;
 115    RECORD u, v;
 116    RECCLR (u);
 117    RECCLR (v);
 118    (void) encode (u, name);
 119    (void) encode (v, id);
 120    return EQUAL (u, v);
 121  }
 122  
 123  LBL *find_label (char *lab)
 124  {
 125    int_4 k, num;
 126    sscanf (lab, "%d", &num);
 127    for (k = 0; k < nlabels; k++) {
 128      LBL *L = &labels[k];
 129      if (num == L->num) {
 130        return L;
 131      }
 132    }
 133    return NO_LABEL;
 134  }
 135  
 136  void impl_type (char *name, MODE * mode)
 137  {
 138    int_4 k = ord (tolower (name[0]));
 139    (*mode) = implic[k].mode;
 140  }
 141  
 142  int_4 add_block (char *name)
 143  {
 144    int_4 k;
 145    for (k = 0; k < ncommons; k++) {
 146      if (same_name (name, commons[k])) {
 147        return k;
 148      }
 149    }
 150    if (ncommons >= MAX_COMMONS) {
 151      FATAL (2901, "too many common blocks", NO_TEXT);
 152    }
 153    k = ncommons++;
 154    commons[k] = c_name (name);
 155    return k;
 156  }
 157  
 158  IDENT *find_local (char *name, MODE * mode)
 159  {
 160    int_4 k;
 161    if (!IS_VAR (name)) {
 162      ERROR (2902, "not a variable name", curlex);
 163      return NO_IDENT;
 164    }
 165  // Search backwards, do not change.
 166    for (k = nlocals - 1; k >= 0; k--) {
 167      IDENT *idf = &locals[k];
 168      if (EQUAL (name, C_NAME (idf)) || EQUAL (name, FTN_NAME (idf))) {
 169        if (macro_nest > 0 ? TRUE : idf->nest == 0) {
 170          if (mode != NO_MODE) {
 171            (*mode) = idf->mode;
 172          }
 173          idf->used = TRUE;
 174          return idf;
 175        }
 176      }
 177    }
 178    return NO_IDENT;
 179  }
 180  
 181  IDENT *void_decl (char *name, MODE * mode)
 182  {
 183    if (!IS_VAR (name)) {
 184      ERROR (2903, "not a variable name", curlex);
 185      return NO_IDENT;
 186    }
 187  // Apparently a normal local variable.
 188    IDENT *idf = find_local (name, mode);
 189    if (idf == NO_IDENT) {
 190      if (nlocals >= MAX_IDENTS) {
 191        FATAL (2904, "too many identifiers", NO_TEXT);
 192        return NO_IDENT;
 193      }
 194      idf = &locals[nlocals++];
 195      memset (idf, 0, sizeof (IDENT));
 196      idf->line = curlin;
 197      C_NAME (idf) = c_name (name);
 198      FTN_NAME (idf) = f_stralloc (name);
 199      idf->external = FALSE;
 200      idf->mode.type = NOTYPE;
 201      idf->mode.len = 0;
 202      if (mode != NO_MODE) {
 203        *mode = idf->mode;
 204      }
 205    }
 206    return idf;
 207  }
 208  
 209  IDENT *add_local (char *name, int_4 type, int_4 len, int_4 uniq, int_4 apatch, int_4 arg, int_4 blck, int_4 src)
 210  {
 211    MODE mode;
 212    if (!IS_VAR (name)) {
 213      ERROR (2905, "not a variable name", curlex);
 214      return NO_IDENT;
 215    }
 216    IDENT *idf = find_local (name, &mode);
 217    if (type == CHARACTER) {
 218      MODE m = {.type = type, .len = len};
 219      norm_mode (&m);
 220      len = m.len;
 221      if (idf == NO_IDENT) {
 222        if (len == 0 && arg == FALSE) {
 223          ERROR (2906, "variable length character must be argument", name);
 224        }
 225      } else {
 226        if (len == 0 && idf->arg == FALSE) {
 227          ERROR (2907, "variable length character must be argument", name);
 228        }
 229      }
 230    }
 231    if (idf != NO_IDENT) {
 232      if (uniq) {
 233        if (idf->mode.type == NOTYPE) {
 234          if (type != NOTYPE) {
 235            // Do not overwrite 'dim', respect previous dimension statement.
 236            idf->mode.type = type;
 237            idf->mode.len = len;
 238          } else {
 239            idf->patch2 = apatch;
 240          }
 241        } else if (idf->parm != NO_TEXT) {
 242          if (accept_mode (idf->mode.type, idf->mode.len, type, len)) {
 243            idf->mode = PLAIN_MODE (type, len);
 244          } else {
 245            MODE err = PLAIN_MODE (type, len);
 246            MODE_ERROR (2908, qtype (&err), qtype (&(idf->mode)));
 247          }
 248        } else if (NOT_LOCAL (idf)) {
 249          idf->mode = PLAIN_MODE (type, len);
 250        } else if (idf->external) {
 251          idf->mode = PLAIN_MODE (type, len);
 252        } else if (idf->mode.type == type && idf->mode.len == len) {
 253          /* Let it pass */
 254        } else {
 255          ERROR (2909, "multiple definition", name);
 256        }
 257      }
 258      return idf;
 259    }
 260    if (nlocals >= MAX_IDENTS) {
 261      FATAL (2910, "too many identifiers", NO_TEXT);
 262      return NO_IDENT;
 263    }
 264    idf = &locals[nlocals++];
 265    memset (idf, 0, sizeof (IDENT));
 266    idf->line = curlin;
 267    C_NAME (idf) = c_name (name);
 268    FTN_NAME (idf) = f_stralloc (name);
 269    idf->arg = arg;
 270    idf->common = blck;
 271    idf->external = FALSE;
 272    idf->mode = PLAIN_MODE (type, len);
 273    idf->patch1 = apatch;
 274    idf->source = src;
 275    return idf;
 276  }
 277  
 278  IDENT *add_nest (char *name, int_4 nest, MODE *mode)
 279  {
 280    if (!IS_VAR (name)) {
 281      ERROR (2911, "not a variable name", curlex);
 282      return NO_IDENT;
 283    }
 284    if (nlocals >= MAX_IDENTS) {
 285      FATAL (2912, "too many identifiers", NO_TEXT);
 286      return NO_IDENT;
 287    }
 288  // Already declared? Take this mode.
 289    IDENT *pre = find_local (name, mode);
 290  //
 291    IDENT *idf = &locals[nlocals++];
 292    memset (idf, 0, sizeof (IDENT));
 293    idf->line = curlin;
 294    C_NAME (idf) = c_name (name);
 295    FTN_NAME (idf) = f_stralloc (name);
 296    idf->arg = NOARG;
 297    idf->external = FALSE;
 298    idf->common = 0;
 299    idf->patch1 = NOPATCH;
 300    idf->source = TEMP;
 301    idf->nest = nest;
 302    if (pre == NO_IDENT) {
 303      impl_type (name, &idf->mode);
 304      *mode = idf->mode;
 305    } else {
 306      *mode = pre->mode;
 307      idf->mode = *mode;
 308    }
 309    return idf;
 310  }
     


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