view Xwnmo/romkan_m/rk_modread.c @ 7:6ab41ec6f895

fix dtoa crash when it encounters malformed entry.
author Yoshiki Yazawa <yaz@cc.rim.or.jp>
date Tue, 18 Dec 2007 23:25:17 +0900
parents bbc77ca4def5
children
line wrap: on
line source

/*
 * $Id: rk_modread.c,v 1.2 2001/06/14 18:16:10 ura Exp $
 */

/*
 * FreeWnn is a network-extensible Kana-to-Kanji conversion system.
 * This file is part of FreeWnn.
 * 
 * Copyright Kyoto University Research Institute for Mathematical Sciences
 *                 1987, 1988, 1989, 1990, 1991, 1992
 * Copyright OMRON Corporation. 1987, 1988, 1989, 1990, 1991, 1992, 1999
 * Copyright ASTEC, Inc. 1987, 1988, 1989, 1990, 1991, 1992
 *
 * Author: OMRON SOFTWARE Co., Ltd. <freewnn@rd.kyoto.omronsoft.co.jp>
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2, or (at your option)
 * any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with GNU Emacs; see the file COPYING.  If not, write to the
 * Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 *
 * Commentary:
 *
 * Change log:
 *
 * Last modified date: 8,Feb.1999
 *
 * Code:
 *
 */
/***********************************************************************
                        rk_modread.c
                                                87.11.17  改 正

        モード定義表の読み込みを担当するプログラム。
***********************************************************************/
/*      Version 3.1     88/06/15        H.HASHIMOTO
 */
#ifndef OMRON_LIB
#include "rk_header.h"
#include "rk_extvars.h"
#ifdef  MULTI
#include "rk_multi.h"
#endif /*MULTI*/
#include <sys/types.h>
#include <sys/stat.h>
#include "rext.h"
#ifdef WNNDEFAULT
#       include "config.h"
  /* マクロLIBDIRの定義(のためだけ)。コンパイル時は、ヘッダファイルの
     サーチパスに、Wnnのインクルードファイルのありかを設定しておくこと。 */
#endif
#include <pwd.h>
#endif
#define xyint(X, Y) (((X) << 24) | (Y))
#define Terminator 0            /* intの列(naibu[])の終止コード */
#ifdef OMRON_SPEC
#define mystrcmp(s1, s2)        strcmp(s1, s2)
#define mystrcpy(s1, s2)        strcpy(s1, s2)
#endif
static void cond_evl (), rd_bcksla (), rd_ctrl (), look_choose (), hyouse (), mystrcpy (), ERMOPN (), ERRMOD ();
static int mystrcmp (), mod_evl (), fnmsrc_tourk (), pathsrc_tourk (),
dspnamsrc_tourk (), modsrc_tourk (), chkchar_getc (), read1tm (), get_hmdir (), rd_string (), ctov (), scan1tm (), modnamchk (), look_cond (), evlcond ();
static char codeeval (), *ename ();

extern char *chrcat (), *strend ();

#ifndef MULTI
/* 88/05/31 V3.1 */
static FILE *modefile;          /* モード定義表の fp */

/* 88/06/03 V3.1 */
/* エラー処理のためのもの */
static char *mcurdir;           /* モード表のサーチパス */
static char *mcurfnm;           /* モード表のファイル名 */
static char *mcurread;          /* モード表の現在行bufへのポインタ */
#endif /*!MULTI */

/* キーワードとその内部表現の対応を与える構造体。内部表現を
        持たないものに対しては0が与えられている。*/
struct kwdpair
{
  char *name;
  int code;
};
static                          /* V3.1 */
struct kwdpair modfn[] = {
  "defmode", 0,
  "if", xyint (2, 0),
  "when", xyint (2, 1),
  "path", 0,
  "search", 0,
  "on_dispmode", xyint (5, 0),
  "off_dispmode", xyint (5, 1),
  "on_unchg", xyint (6, 0),
  "off_unchg", xyint (6, 1),
  NULL
};                              /* 下を見よ キーワード定義表はまだあと二つあるのだ */

static                          /* V3.1 */
struct kwdpair modcond[] = {
  "not", xyint (3, 0),
  "and", xyint (3, 1),
  "or", xyint (3, 2),
  "true", xyint (3, 3),
  "false", xyint (3, 4),
  "=", xyint (3, 5),
  "!=", xyint (3, 6),
  "<", xyint (3, 7),
  ">", xyint (3, 8),
  NULL
};
static                          /* V3.1 */
int condarg[] = { 1, 2, 2, 0, 0, 2, 2, 2, 2 };  /* 条件判断関数の引数の個数 */

static                          /* V3.1 */
struct kwdpair swstat[] = {
  "on", 0,
  "off", 0,
  NULL
};

 /** キーワード(if, andなど)が正当なものかチェックし、その番号を返す */
static                          /* V3.1 */
  int
kwdsrc (hyou, wd)
     struct kwdpair *hyou;      /* どのキーワード表を使うか */
     char *wd;                  /* チェックされるキーワード */
{
  fast int i;                   /* V3.1 */

  for (i = 0; hyou[i].name != NULL; i++)
    if (!mystrcmp (hyou[i].name, wd))
      return (i);
  ERRMOD (9);
   /*NOTREACHED*/ return (-1);
}

 /** 与えられたファイルがディレクトリなら非0を返す */
static                          /* V3.1 */
  int
isdir (fnm)
     char *fnm;
{
  struct stat statbuf;

  /* Bug!!! 88/07/20 H.HASHIMOTO */
  /* return(stat(fnm, &statbuf) == 0 && (statbuf.st_mode & S_IFDIR)); */
  return (stat (fnm, &statbuf) == 0 && ((statbuf.st_mode & S_IFMT) == S_IFDIR));
}

/* 88/06/10 V3.1 */
 /** モード表の読み込み */
#ifdef OMRON_LIB
static
#endif
  int
readmode (filename)
     char *filename;            /* ユーザが指定したモード表の名 */
{
#ifndef MULTI
  char modefilename[REALFN + 5];        /* V3.1 */
  char pathname[REALFN];        /* V3.1 */
#endif /*!MULTI */

  struct stat statbuf;          /* V3.1 */
  char *buf, *bufp;             /* V3.1 */

#ifdef RKMODPATH
  char *genv, *pathenv, *pathp;
  extern char *getenv ();       /* V3.1 */
#endif

  /* 88/06/14 V3.1 */
  /* モード定義表のパス名の長さチェックを安全性のために、
     付け加えておきます。 */
  cur_rk_table->mcurdir = "";   /* エラー処理用 *//* V3.1 */
  if (strlen (cur_rk_table->mcurfnm = filename) > (REALFN - 1)) /* V3.1 */
    ERMOPN (3);                 /* V3.1 */
  strcpy (cur_rk_table->modefilename, filename);
  if (*(cur_rk_table->modefilename) == '\0' || *(strend (cur_rk_table->modefilename)) == KUGIRI)
    {
      strcat (cur_rk_table->modefilename, "mode");
    }
  else if (isdir (cur_rk_table->modefilename))
    {
      chrcat (cur_rk_table->modefilename, KUGIRI);
      strcat (cur_rk_table->modefilename, "mode");
    }
  /* 88/06/14 V3.1 */
  if (strlen (cur_rk_table->mcurfnm = cur_rk_table->modefilename) > (REALFN - 1))       /* V3.1 */
    ERMOPN (3);                 /* V3.1 */

#ifdef RKMODPATH
  /* 88/06/10 V3.1 */
  if (!fixednamep (cur_rk_table->modefilename) && (pathenv = genv = getenv (RKMODPATH)) != NULL && *genv != '\0')
    {
      /* PATHに少なくとも一つのサーチパスがある場合 */
      for (;;)
        {
          for (pathp = cur_rk_table->pathname; *genv != ':' && *genv; genv++)
            *pathp++ = *genv;
          *pathp = '\0';

          if (*(strend (cur_rk_table->pathname)) != KUGIRI)
            *pathp++ = KUGIRI;
          /* pathの区切りはDG(MV)であっても'/' */

          strcpy (pathp, cur_rk_table->modefilename);
          if ((stat (cur_rk_table->pathname, &statbuf) == 0) && ((cur_rk_table->modefile = fopen (cur_rk_table->pathname, "r")) != NULL))
            {
              /* Now Mode-hyo found */
              /*
                 if (cur_rk->flags & RK_VERBOS) {
                 fprintf(stderr,
                 "romkan: using Mode-hyo %s ...\r\n",
                 pathname);
                 }
               */
              cur_rk_table->mcurdir = cur_rk_table->pathname;
              /* この時点ではファイル名込みだが、
                 あとでパス名だけになる */
              cur_rk_table->mcurfnm = ename (cur_rk_table->modefilename);       /* V3.1 */
              break;
            }

          if (*genv != ':')
            {
              /* Mode-hyo not found */
              /*
                 if (flags & RK_VERBOS){
                 fprintf(stderr,
                 "no %s in ",cur_rk->modefilename);
                 for(genv = pathenv; *genv; genv++){
                 fputc((*genv == ':' ?
                 ' ' : *genv), stderr);
                 }
                 fprintf(stderr, ".\n");
                 }
               */
              ERMOPN (1);
            }
          /* coutinues searching Mode-hyo */
          genv++;
        }
    }
  else
#endif
    {
      /* 88/06/10 V3.1 */
      if (stat (cur_rk_table->modefilename, &statbuf) != 0)
        ERMOPN (1);
      if ((cur_rk_table->modefile = fopen (cur_rk_table->modefilename, "r")) == NULL)
        ERMOPN (2);
      /*
         if (cur_rk->flags & RK_VERBOS) {
         fprintf(stderr, "romkan: using Mode-hyo %s ...\r\n",
         cur_rk->modefilename);
         }
       */
      strcpy (cur_rk_table->pathname, cur_rk_table->modefilename);
    }

  /* サーチパスの先頭に、モード表のあるディレクトリを設定している。 */
  *(ename (cur_rk_table->pathname)) = '\0';     /* V3.1 */
  pathsrc_tourk (cur_rk_table->pathname);       /* V3.1 */

  /* 88/06/14 V3.1 */
  malloc_for_modenaibu (&(cur_rk_table->rk_modenaibu), statbuf.st_size + 1);
  *(cur_rk_table->rk_modenaibu.next) = Terminator;

  malloc_for_modebuf (&(cur_rk_table->rk_modebuf), statbuf.st_size + 1);
  cur_rk_table->mcurread = buf = cur_rk_table->rk_modebuf.org;  /* エラー処理用 *//* V3.1 */

  /* モード定義表の読み込み */
  while (bufp = buf, read1tm (&bufp, 0))        /* V3.1 */
    mod_evl (buf);              /* V3.1 */
  fclose (cur_rk_table->modefile);

  /* 88/06/14 V3.1 */
  free_for_modebuf (&(cur_rk_table->rk_modebuf));
  realloc_for_modenaibu (&(cur_rk_table->rk_modenaibu));

  return (cur_rk_table->rk_taiouhyo.count);     /* V3.1 */
}

 /**    モード表の一かたまり(リスト、ファイル名、モード表示文字列)を
        解釈する。返す値は、defmode,search及びpathの時0、それ以外なら1。*/
static                          /* V3.1 */
  int
mod_evl (s)
     char *s;                   /* モード表の内部表現の列へのポインタ */
{
  char md1[MDT1LN];             /* V3.1 */
  fast char *bgn, *end;         /* V3.1 */
  fast int num, retval = 1;     /* V3.1 */

  if (*s != '(')
    {
      if (*s != '"')
        {
          num = fnmsrc_tourk (s);
          *(cur_rk_table->rk_modenaibu.next++) = xyint (4, num);
        }
      else
        {
          s++;
          if (*(end = strend (s)) != '"')
            ERRMOD (10);
          *end = '\0';
          num = dspnamsrc_tourk (s);
          *(cur_rk_table->rk_modenaibu.next++) = xyint (5, 0);
          *(cur_rk_table->rk_modenaibu.next++) = num;
        }
    }
  else
    {
      s++;
      scan1tm (&s, md1, 1);
      switch (num = kwdsrc (modfn, md1))
        {
        case 0:         /* defmode */
          retval = 0;
          scan1tm (&s, md1, 1); /* modename */
          num = modsrc_tourk (md1, 0);
          if (scan1tm (&s, md1, 0))
            {                   /* 初期on-off */
              switch (kwdsrc (swstat, md1))
                {
                case 0:
                  cur_rk_table->rk_modesw.point[num] = 1;
                  break;
                case 1:
                  cur_rk_table->rk_modesw.point[num] = 0;
                  break;
                }
              scan1tm (&s, md1, 2);     /* あればerr */
            }
          else
            cur_rk_table->rk_modesw.point[num] = 0;     /* defaultはoff */
          break;
        case 1:         /* if */
        case 2:         /* when */
          *(cur_rk_table->rk_modenaibu.next++) = modfn[num].code;
          scan1tm (&s, md1, 1); /* condition */
          cond_evl (md1);
          while (scan1tm (&s, md1, 0))
            {
              if (mod_evl (md1) == 0)
                ERRMOD (8);
            }
          *(cur_rk_table->rk_modenaibu.next++) = Terminator;
          break;
        case 3:         /* path */
          /* 88/06/09 V3.1 */
          cur_rk_table->rk_path.count = 1;
          cur_rk_table->rk_path.next = cur_rk_table->rk_path.point[1];
        case 4:         /* search */
          retval = 0;
          /* 88/06/29 V3.1 */
          if (cur_rk_table->rk_taiouhyo.point[0] != NULL)
            ERRMOD (11);
          /* サーチパスの指定はファイル名の出現より
             先行しなければならないとしておく。 */

          while (scan1tm (&s, md1, 0))
            {                   /* find pathname */
              pathsrc_tourk (md1);
            }
          break;
        case 5:         /* on_dispmode */
        case 6:         /* off_dispmode */
          *(cur_rk_table->rk_modenaibu.next++) = modfn[num].code;
          scan1tm (&s, md1, 1); /* dispmode string */

          if (*(bgn = md1) != '"')
            ERRMOD (12);
          bgn++;
          if (*(end = strend (bgn)) != '"')
            ERRMOD (10);
          *end = '\0';
          *(cur_rk_table->rk_modenaibu.next++) = dspnamsrc_tourk (bgn);
          scan1tm (&s, md1, 2); /* あればerr */
          break;
        case 7:         /* on-unchg */
        case 8:         /* off-unchg */
          *(cur_rk_table->rk_modenaibu.next++) = modfn[num].code;
          scan1tm (&s, md1, 2); /* あればerr */
          break;
        }

    }
  *(cur_rk_table->rk_modenaibu.next) = Terminator;
  return (retval);
}

 /** 条件式(モード名 又はnot,andなどの式)一つを解釈 */
static                          /* V3.1 */
  void
cond_evl (cod)
     char *cod;                 /* 条件式の内部表現の列へのポインタ */
{
  char md1[MDT1LN];
  fast int num, i;              /* V3.1 */

  if (*cod != '(')
    {
      num = modsrc_tourk (cod, 1);
      *(cur_rk_table->rk_modenaibu.next++) = xyint (1, num);
    }
  else
    {
      cod++;
      scan1tm (&cod, md1, 1);   /* not;and;or */
      num = kwdsrc (modcond, md1);
      *(cur_rk_table->rk_modenaibu.next++) = xyint (3, num);
      for (i = condarg[num]; i; i--)
        {
          scan1tm (&cod, md1, 0);
          cond_evl (md1);
        }
      scan1tm (&cod, md1, 2);
    }
  *(cur_rk_table->rk_modenaibu.next) = Terminator;
}

/* 88/06/09 V3.1 */
static int
serach_in_modetable_struct (ptr, s, np)
     fast modetable *ptr;
     fast char *s;
     fast int *np;
{
  fast int n;

  for (n = 0; n < ptr->count; n++)
    {
      if (!mystrcmp (ptr->point[n], s))
        {
          *np = n;
          return (1);           /* found */
        }
    }
  *np = n;
  return (0);                   /* not found */
}

/* 88/06/13 V3.1 */
static void
entry_to_modetable_struct (ptr, n, s)
     fast modetable *ptr;
     int n;
     char *s;
{
  ptr->point[n] = ptr->next;
  mystrcpy (ptr->next, s);
  strtail (ptr->next);
  ++(ptr->next);
}


/* 88/06/07 V3.1 */
 /**    sで指定されたファイル名が既登録か探し、なければ登録。但し、既登録か
        どうかのチェックは厳密ではないが(例えば、同じファイルでも、
        パス名付きと無しとでは、同じと見ない)、ファイル名が既登録かどうか
        チェックするのは、メモリ節約のために同じ表を読み込むのを防ぐため
        だけなので、それ以外には別に困る点はない。*/
static                          /* V3.1 */
  int
fnmsrc_tourk (s)
     char *s;
{
  int n;

  if (serach_in_modetable_struct (&(cur_rk_table->rk_taiouhyo), s, &n))
    return (n);

  check_and_realloc_for_modetable_struct (&(cur_rk_table->rk_taiouhyo), n, s, RK_TAIOUHYO_MAX_LOT, RK_TAIOUHYO_MEM_LOT);

  /* チェックのみ */
  if (!(filnamchk (s)))
    ERRMOD (3);

  entry_to_modetable_struct (&(cur_rk_table->rk_taiouhyo), n, s);
  return (n);
}

/* 88/06/07 V3.1 */
 /**    sで指定されたサーチパス名が既登録か探し、なければ登録。但し、fnmsrc_
        tourk()同様、既登録かどうかのチェックは厳密ではないが問題ない。*/
/* 88/06/07 V3.1 */
static                          /* V3.1 */
  int
pathsrc_tourk (s)
     char *s;
{
  int n;
  char fnm_addsla[MDT1LN];

  mystrcpy (fnm_addsla, s);
  if (!(*fnm_addsla == '\0' || *(strend (fnm_addsla)) == KUGIRI))
    chrcat (fnm_addsla, KUGIRI);
  s = fnm_addsla;
  /* パス名が'/'で終わってなければ、それを付加する。 */

  if (serach_in_modetable_struct (&(cur_rk_table->rk_path), s, &n))
    return (n);

  check_and_realloc_for_modetable_struct (&(cur_rk_table->rk_path), n, s, RK_PATH_MAX_LOT, RK_PATH_MEM_LOT);

  entry_to_modetable_struct (&(cur_rk_table->rk_path), n, s);
  return (n);
}

/* 88/06/07 V3.1 */
 /** sで指定されたモード表示文字列が既登録か探し、なければ登録 */
static                          /* V3.1 */
  int
dspnamsrc_tourk (s)
     char *s;
{
  int n;

  if (serach_in_modetable_struct (&(cur_rk_table->rk_dspmode), s, &n))
    return (n);

  check_and_realloc_for_modetable_struct (&(cur_rk_table->rk_dspmode), n, s, RK_DSPMODE_MAX_LOT, RK_DSPMODE_MEM_LOT);

  entry_to_modetable_struct (&(cur_rk_table->rk_dspmode), n, s);
  return (n);
}

/* 88/06/09 V3.1 */
 /**    sで指定されたモード名を探し、なければ登録。但し、flgが非0なら、
        見つからなければエラー */
static                          /* V3.1 */
  int
modsrc_tourk (s, flg)
     char *s;
     int flg;
{
  int n;

  if (!modnamchk (s))
    ERRMOD (4);                 /* V3.1 */

  if (serach_in_modetable_struct (&(cur_rk_table->rk_defmode), s, &n))
    return (n);

  if (flg)
    ERRMOD (5);

  check_and_realloc_for_modetable_struct (&(cur_rk_table->rk_defmode), n, s, RK_DEFMODE_MAX_LOT, RK_DEFMODE_MEM_LOT);

  check_and_realloc_for_modesw (&(cur_rk_table->rk_modesw), n, RK_DEFMODE_MAX_LOT, RK_DEFMODE_MEM_LOT);

  entry_to_modetable_struct (&(cur_rk_table->rk_defmode), n, s);
  return (n);
}

 /** ファイルから一文字読む(空白文字は飛ばす)。読んだ文字がEOFなら0を返す */
static                          /* V3.1 */
  char
fspcpass ()
{
  fast int c;                   /* V3.1 */

  while (EOF != (c = chkchar_getc (cur_rk_table->modefile)) && is_nulsp (c));
  return (c == EOF ? '\0' : c);
}

 /**    モード表には空白文字以外のコントロール文字は生では混じらないものと
        する。混じっていた場合はチェックしつつ、getcを行う。*/
static                          /* V3.1 */
  int
chkchar_getc (f)
     fast FILE *f;
{
  fast int c;                   /* V3.1 */

  c = getc (f);
  if (is_cntrl (c) && !isspace (c))
    {
      sprintf (cur_rk_table->mcurread, "\\%03o", c);
      ERRMOD (16);
    }
  return (c);
}

 /**    モード表の一かたまり(リスト、ファイル名、モード表示文字列)を
        切り出す。その際、特殊な表記('^','\'による)は、'\(8進);' の
        形に直す。flgが非0なら、EOFでエラーを起こし、')'で0を返す。*/
static                          /* V3.1 */
  int
read1tm (sptr, flg)
     char **sptr;               /* モード表の内部表現の列へのポインタへのポインタ。
                                   rd_bcksla()、rd_ctrl()、codeeval()でも同様 */
     int flg;
{
  fast int c;                   /* V3.1 */
  fast char *head;              /* V3.1 */
  int retval = 1;               /* V3.1 */
  char *s;                      /* V3.1 */

  s = *sptr;

  while ((c = fspcpass ()) == ';')
    {
      /* 注釈文を検出したら、行末までとばして再試行。 */
      while ((c = chkchar_getc (cur_rk_table->modefile)) != '\n' && c != EOF);
    }

  switch (c)
    {
    case '\0':                  /* EOFを表す */
      if (flg)
        ERRMOD (0);
      else
        retval = 0;
      break;
    case ')':
      if (flg)
        retval = 0;
      else
        ERRMOD (1);
      break;
    case '(':
      *s++ = c;
      *s++ = ' ';
      while (read1tm (&s, 1))
        *s++ = ' ';
      *s++ = ')';
      break;
    case '"':
      *s++ = c;
      while ((c = chkchar_getc (cur_rk_table->modefile)) != '"')
        {
          switch (c)
            {
            case EOF:
              ERRMOD (0);
#ifndef OMRON_SPEC
            case '\\':
              rd_bcksla (&s);
              break;
            case '^':
              rd_ctrl (&s);
              break;
#endif
            default:
              *s++ = c;
            }
        }
      *s++ = '"';
      break;
    default:
      /* 先頭が @ 又は ~ の時は、特殊処理。 */
      if (c == '@')
        {                       /* @HOME, @MODEDIR, @LIBDIR */
          *s++ = c;
          head = s;
          rd_string (&s, 1);

          if (mystrcmp ("HOME", head) == 0)
            {
              s = --head;
              if (get_hmdir (&s, (char *) NULL) != 0)
                {
                  cur_rk_table->mcurread = head;
                  /* エラー表示の調整。抜けずにすぐにエラー表示
                     ルーチンに行くのでheadの中身は失われない。 */
                  ERRMOD (13);
                }
            }
          else if (mystrcmp ("MODEDIR", head) == 0)
            {
              /* 88/06/09 V3.1 */
              strcpy (s = --head, cur_rk_table->rk_path.point[0]);
              if (KUGIRI == *(s = strend (s)))
                *s = '\0';
            }
          else
#ifdef WNNDEFAULT
          if (mystrcmp ("LIBDIR", head) == 0)
            {
              strcpy (s = --head, LIBDIR);
              strtail (s);
            }
          else
#endif
            {
              cur_rk_table->mcurread = --head;
              ERRMOD (14);
            }
        }
      else if (c == '~')
        {                       /* ~user */
          *s++ = c;
          head = s;
          rd_string (&s, 1);

          mystrcpy (head, head);
          s = head - 1;
          if ((get_hmdir (&s, (*head ? head : NULL))) != 0)
            {
              cur_rk_table->mcurread = --head;
              /* エラー表示の調整 */
              ERRMOD (15);
            }
        }
      else
        {
          ungetc (c, cur_rk_table->modefile);
        }
      /* 先頭の特殊処理 終わり */

      if ((c = rd_string (&s, 0)) == EOF && flg)
        ERRMOD (0);
      if (c == ')' && !flg)
        ERRMOD (1);
    }

  *s = '\0';
  *sptr = s;
  return (retval);
}

/* 88/06/03 V3.1 */
 /**    socの名のユーザのログイン・ディレクトリ名をdestに入れ、*destにその末尾を
        指させる。但しsocが空列または、NULLなら自分のログイン・ディレクトリ名、
        いずれの場合も、不成功時は、何もしない。返値は、不成功時-1。*/
static                          /* V3.1 */
  int
get_hmdir (dest, soc)
     char **dest, *soc;
{
  fast struct passwd *usr;      /* V3.1 */
  extern struct passwd *getpwnam (), *getpwuid ();      /* V3.1 */

  /* 88/05/26 V3.1 */
  if (soc == NULL || *soc == '\0')
    {
      if (NULL == (usr = getpwuid (getuid ())))
        return (-1);
    }
  else
    {
      if (NULL == (usr = getpwnam (soc)))
        return (-1);
    }
  strcpy (*dest, usr->pw_dir);
  strtail (*dest);
  return (0);
}



 /**    モード表・対応表中の、ファイル名の部分の読み込み。先頭が @ 又は ~ の
        時は、特殊処理を行う。引数は、一字読み込み・一字戻し・文字列取り出しの
        関数と、結果を入れるエリアの番地へのポインタ、次に読まれる文字を入れる
        ポインタ。返値は、正常終了時0、@HOMEでホーム・ディレクトリが取れない時
        1、@のあとに変なものが来たら2、~で自分のホーム・ディレクトリが取れない
        時3、~のあとに存在しないユーザ名が来たら4。*/
int
readfnm (readchar_func, unreadc_func, readstr_func, areap, lastcptr)
     register int (*readchar_func) (), (*unreadc_func) (), (*readstr_func) ();
     char **areap;
     int *lastcptr;
{
  char *head;
  register int c;

  c = (*readchar_func) ();
  if (c == '@')
    {                           /* @HOME, @MODEDIR, @LIBDIR */
      *(*areap)++ = c;
      head = *areap;
      (*readstr_func) (areap, 1);

      if (mystrcmp ("HOME", head) == 0)
        {
          *areap = --head;
          if (get_hmdir (areap, (char *) NULL) != 0)
            {
              *areap = head;
              return (1);
            }
        }
      else if (mystrcmp ("MODEDIR", head) == 0)
        {
          strcpy (*areap = --head, cur_rk_table->pathname);
          if (KUGIRI == *(*areap = strend (*areap)))
            **areap = '\0';
        }
      else
#ifdef WNNDEFAULT
      if (mystrcmp ("LIBDIR", head) == 0)
        {
          strcpy (*areap = --head, LIBDIR);
          strtail (*areap);
        }
      else
#endif
        {
          *areap = --head;
          return (2);
        }

    }
  else if (c == '~')
    {                           /* ~user */
      int err;

      *(*areap)++ = c;
      head = *areap;
      (*readstr_func) (areap, 1);

      mystrcpy (head, head);
      *areap = head - 1;
      if ((err = get_hmdir (areap, (*head ? head : NULL))) != 0)
        {
          *areap = --head;
          return (err == -2 ? 3 : 4);
        }

    }
  else
    {
      (*unreadc_func) (c);
    }

  *lastcptr = (*readstr_func) (areap, 0);
  return (0);
}

 /**    モード表から一文字分取り出す作業を、空白・EOF・括弧のどれかが来るまで
        続ける。flgが非0なら、'/'が来ても終わる。返値は、次に読まれる文字。*/
static                          /* V3.1 */
  int
rd_string (sptr, flg)
     char **sptr;
     int flg;
{
  fast int c;                   /* V3.1 */

  while (c = chkchar_getc (cur_rk_table->modefile), !(is_nulsp (c) || c == EOF || c == '(' || c == ')' || (flg == 1 && c == KUGIRI)))
    {
      switch (c)
        {
#ifndef OMRON_SPEC
        case '\\':
          rd_bcksla (sptr);
          break;
        case '^':
          rd_ctrl (sptr);
          break;
#endif
        default:
          *(*sptr)++ = c;
        }
    }
  **sptr = '\0';
  return (ungetc (c, cur_rk_table->modefile));
}

/* 88/06/02 V3.1 */
#ifndef OMRON_SPEC
 /**    モード表からバックスラッシュ形式の一文字分を取り出し、'\(8進);' の
        形に直す。但し、先頭の'\\'は既に読まれたあと。*/
static                          /* V3.1 */
  void
rd_bcksla (sptr)
     char **sptr;
{
  fast int c, code = 0, digflg = 0;     /* V3.1 */

  switch (c = chkchar_getc (cur_rk_table->modefile))
    {
    case 'n':
      code = '\n';
      digflg = 1;
      break;
    case 't':
      code = '\t';
      digflg = 1;
      break;
    case 'b':
      code = '\b';
      digflg = 1;
      break;
    case 'r':
      code = '\r';
      digflg = 1;
      break;
    case 'f':
      code = '\f';
      digflg = 1;
      break;
    case 'e':
    case 'E':
      code = ESCCHR;
      digflg = 1;
      break;
    case 'o':
      while (c = chkchar_getc (cur_rk_table->modefile), is_octal (c))
        {
          code <<= 3;
          code += ctov (c);
          digflg = 1;
        }
      if (c != ';')
        ungetc (c, cur_rk_table->modefile);
      break;
    case 'd':
      while (c = chkchar_getc (cur_rk_table->modefile), is_digit (c))
        {
          code *= 10;
          code += ctov (c);
          digflg = 1;
        }
      if (c != ';')
        ungetc (c, cur_rk_table->modefile);
      break;
    case 'x':
      while (c = chkchar_getc (cur_rk_table->modefile), is_xdigit (c))
        {
          code <<= 4;
          code += ctov (c);
          digflg = 1;
        }
      if (c != ';')
        ungetc (c, cur_rk_table->modefile);
      break;
    default:
      if (is_octal (c))
        {
          digflg = 1;
          code = ctov (c);
          while (c = chkchar_getc (cur_rk_table->modefile), is_octal (c))
            {
              code <<= 3;
              code += ctov (c);
            }
          if (c != ';')
            ungetc (c, cur_rk_table->modefile);
        }
      else
        {
          code = c;
          digflg = 1;
        }
    }

  if (digflg == 0)
    ERRMOD (7);
  sprintf (*sptr, "\\%o;", code);
  strtail (*sptr);
}


 /**    モード表からコントロールコード形式の一文字分を取り出し、'\(8進);' の
        形に直す。但し、先頭の'^'は既に読まれたあと。*/
static                          /* V3.1 */
  void
rd_ctrl (sptr)
     char **sptr;
{
  fast int c;                   /* V3.1 */

  if (!(' ' <= (c = chkchar_getc (cur_rk_table->modefile)) && c < '\177'))
    ERRMOD (7);
  if (c == '?')
    c = '\177';
  else
    c &= 0x1f;

  sprintf (*sptr, "\\%o;", c);
  strtail (*sptr);
}
#endif

 /**    8・10・16進コード用のキャラクタを実際のコードに直す。入力のチェックは
        しない。*/
static int
ctov (c)
     fast char c;
{
  if (is_upper (c))
    return (c - 'A' + 10);
  if (is_lower (c))
    return (c - 'a' + 10);
  return (c - '0');
}

 /**    リストの中身のscanに専用。')'で0を返す。EOLは来ないはず。
        flg == 1 のとき、取り出しに失敗したらエラー。
        flg == 2 のとき、取り出しに成功したらエラー。
        特殊なコード表記は既に全て '\(8進);' の形に直っている筈。*/
static                          /* V3.1 */
  int
scan1tm (socp, dest, flg)
     char **socp, *dest;
         /* socpの指しているポインタが指している所から取り出してdestに入れる。
            その後、socpが指しているポインタを進める。 */
     int flg;
{
  fast char c;                  /* V3.1 */
  fast int retval = 1;          /* V3.1 */

  while (c = *(*socp)++, is_nulsp (c))
    if (c == '\0')
      ERRMOD (6);
  switch (c)
    {
    case ')':
      retval = 0;
      break;
    case '(':
      *dest++ = c;
      *dest++ = ' ';
      while (scan1tm (socp, dest, 0))
        {
          strtail (dest);
          *dest++ = ' ';
        }
      *dest++ = ')';
      break;
    case '"':
      *dest++ = c;
      while ((c = *dest++ = *(*socp)++) != '"')
        {
          if (c == '\\')
            {                   /* '\(8進);'の解釈 */
              while (c = *dest++ = *(*socp)++, is_octal (c));
            }
        }
      break;
    default:
      *dest++ = c;
      while (!is_nulsp (**socp))
        *dest++ = *(*socp)++;
    }

  *dest = '\0';
  if (flg == 1 && retval == 0 || flg == 2 && retval == 1)
    ERRMOD (6);
  return (retval);
}

 /** モード名として正当かチェック。英数字からなっていればいい */
static                          /* V3.1 */
  int
modnamchk (s)
     fast char *s;              /* V3.1 */
{
  if (is_digit (*s))
    return (0);
  for (; *s; s++)
    {
      if (!is_alnum (*s))
        {
          if (*s != '_')
            {
              return (0);
            }
        }
    }
  return (1);
/*
        if(is_digit(*s)) return(0);
        for(; *s; s++) if(!is_alnum(*s) && *s != '_') return(0);
        return(1);
*/
}

 /** 変換対応表の選択を行う */
#ifdef OMRON_LIB
static
#endif
  void
choosehyo ()
{
  fast int i;                   /* V3.1 */
  int *naibup;                  /* V3.1 */

  if (cur_rk->rk_usehyo.usehyo != NULL)
    cur_rk->rk_usehyo.usemaehyo[0] = cur_rk->rk_usehyo.usehyo[0] = cur_rk->rk_usehyo.useatohyo[0] = -1;
  for (i = 0; i < 2; i++)
    {
      cur_rk->dspmod[1][i] = cur_rk->dspmod[0][i];
      cur_rk->dspmod[0][i] = NULL;
    }

  if ((naibup = cur_rk_table->rk_modenaibu.org) != NULL)
    look_choose (&naibup, 1);   /* V3.1 */
}

 /**    モード表の内部形式を順次見ていき、使用表の選択及びモード表示文字列の
        選択を行っていく。但しflgが0ならスキップするだけ */
static                          /* V3.1 */
  void
look_choose (naibupp, flg)
     int **naibupp;             /* モード表の内部表現の列へのポインタへのポインタ。
                                   look_cond()、evlcond()でも同様 */
     int flg;
{
  fast int naibu1, naibu2, branch, lcrsl;       /* V3.1 */
  int *naibup;                  /* V3.1 */

  naibup = *naibupp;

  while ((naibu1 = *naibup++) != Terminator)
    {
      switch (SHUBET (naibu1))
        {
        case 4:         /* 表名 */
          if (flg)
            hyouse (LWRMSK (naibu1));
          break;
        case 2:         /* 条件式 */
          branch = LWRMSK (naibu1);     /* if;when */
          lcrsl = look_cond (&naibup, flg);
          if (branch == 0 && lcrsl)
            flg = 0;
          break;
        case 5:         /* romkanがon・off時それぞれの
                                   モード表示文字列 */
          naibu2 = *naibup++;
          if (flg)
            cur_rk->dspmod[0][LWRMSK (naibu1)] = cur_rk_table->rk_dspmode.point[naibu2];
          break;
        case 6:         /* romkanがそれぞれon・off時のモード表示
                                   文字列を前のままに */
          if (flg)
            cur_rk->dspmod[0][LWRMSK (naibu1)] = cur_rk->dspmod[1][LWRMSK (naibu1)];
          break;
        default:
          BUGreport (6);
        }
    }

  *naibupp = naibup;
}

 /**    *naibupp が、内部表現の列で条件式を表すところを指している筈なので、
        それを評価し、真ならその続きを解釈しにいく。偽なら読み飛ばす。
        返値は、最初に評価した条件式の真偽値。*/
static                          /* V3.1 */
  int
look_cond (naibupp, flg)
     int **naibupp, flg;
{
  fast int condrsl;             /* V3.1 */
  int *naibup;

  naibup = *naibupp;

  /* 88/06/06 V3.1 */
  condrsl = evlcond (&naibup);  /* 必ず評価しないといけないため */
  look_choose (&naibup, flg &= condrsl);
  *naibupp = naibup;
  return (flg);
}

 /** 条件式の真偽値の評価 */
static                          /* V3.1 */
  int
evlcond (naibupp)
     int **naibupp;
{
  int *naibup, tmpval[ARGMAX];  /* V3.1 */
  fast int naibu1, retval = 0, imax;    /* V3.1 */
  fast int i;                   /* V3.1 */

  naibup = *naibupp;

  naibu1 = *naibup++;
  switch (SHUBET (naibu1))
    {
    case 7:                     /* 数値 */
      retval = *naibup++;
      break;
    case 1:                     /* モード名 */
      retval = cur_rk->rk_modesw.point[LWRMSK (naibu1)];
      break;
    case 3:                     /* if;when */
      imax = condarg[LWRMSK (naibu1)];
      for (i = 0; i < imax; i++)
        tmpval[i] = evlcond (&naibup);
      switch (LWRMSK (naibu1))
        {
          /* 上から順にtrue,false,not,and,or */
        case 0:
          retval = !tmpval[0];
          break;
        case 1:
          retval = tmpval[0] && tmpval[1];
          break;
        case 2:
          retval = tmpval[0] || tmpval[1];
          break;
        case 3:
          retval = 1;
          break;
        case 4:
          retval = 0;
          break;
        case 5:
          retval = (tmpval[0] == tmpval[1]);
          break;
        case 6:
          retval = (tmpval[0] != tmpval[1]);
          break;
        case 7:
          retval = ((unsigned int) tmpval[0] < (unsigned int) tmpval[1]);
          break;
        case 8:
          retval = ((unsigned int) tmpval[0] > (unsigned int) tmpval[1]);
          break;

        }
    }

  *naibupp = naibup;
  return (retval);
}

 /** num番目の表を、使用するものとして登録する。前・本・後処理の区別もする */
static                          /* V3.1 */
  void
hyouse (num)
     int num;
{
  fast int *ptr = NULL;         /* V3.1 */

  switch (cur_rk_table->rk_hyo.point[num].hyoshu)
    {
    case 1:
      ptr = cur_rk->rk_usehyo.usemaehyo;
      break;
    case 2:
      ptr = cur_rk->rk_usehyo.usehyo;
      break;
    case 3:
      ptr = cur_rk->rk_usehyo.useatohyo;
      break;
    default:
      BUGreport (11);
    }
  if (ptr == NULL)
    return;
  for (; *ptr != -1; ptr++)
    if (*ptr == num)
      return;
  *ptr = num;
  *++ptr = -1;
}

/* 88/06/02 V3.1 */
#ifndef OMRON_SPEC
 /** strcmpと同等  但し、'\(8進);'も解釈する。*/
static                          /* V3.1 */
  int
mystrcmp (s1, s2)
     char *s1, *s2;
{
  fast char c1, c2;             /* V3.1 */

  while ((c1 = codeeval (&s1)) == (c2 = codeeval (&s2)))
    if (c1 == '\0')
      return (0);
  return (c1 > c2 ? 1 : -1);
}

 /** strcpyと同等 但し'\(8進);'も解釈する。s1 <= s2なら正常動作するはず */
static                          /* V3.1 */
  void
mystrcpy (s1, s2)
     fast char *s1;
     char *s2;
{
  while (*s1++ = codeeval (&s2));
}

 /**    一文字の解釈を行う。普通の文字はそのまま、'\(8進);'は実際のコードに
        直す。その後、文字列へのポインタを一文字分進めておく(少なくとも
        1バイト分進むことが保証されるはず)。*/
static                          /* V3.1 */
  char
codeeval (sptr)
     fast char **sptr;          /* V3.1 */
{
  fast char c;                  /* V3.1 */
  fast char code = 0;           /* V3.1 */

  if ((c = *(*sptr)++) != '\\')
    return (c);
  while (c = *(*sptr)++, is_octal (c))
    {
      code <<= 3;
      code += ctov (c);
    }
  if (c != ';')
    BUGreport (12);
  return (code);
}
#endif

 /** romkanがon時のモード表示文字列を返す関数。無指定であってRK_DSPNILフラグが
     立っている時は空文字列を返す。*/
char *
romkan_dispmode ()
{
  return (cur_rk->dspmod[0][0] == NULL && (cur_rk_table->flags & RK_DSPNIL) ? "" : cur_rk->dspmod[0][0]);       /* V3.1 */
}

 /** romkanがoff時のモード表示文字列を返す関数。無指定であってRK_DSPNILフラグ
     が立っている時は空文字列を返す。*/
char *
romkan_offmode ()
{
  return (cur_rk->dspmod[0][1] == NULL && (cur_rk_table->flags & RK_DSPNIL) ? "" : cur_rk->dspmod[0][1]);       /* V3.1 */
}

/* 88/06/02 V3.1 */
 /** ファイル名からパス名を除いた部分の先頭を返す。*/
static char *
ename (s)
     char *s;
{
  fast char *p;                 /* V3.1 */

  p = rindex (s, KUGIRI);
  return (p == NULL ? s : p + 1);
}

 /**    ファイル名のチェック。先頭(パス名は除く)が'1'〜'3'でないといけない。
        正当なものなら1〜3(前・本・後処理表の区別を表す)を返し、そうでないと
        0を返す。*/
#ifdef OMRON_LIB
static
#endif
  int
filnamchk (s)
     char *s;
{
  fast int c;                   /* V3.1 */

  c = *(ename (s)) - '0';
  return ((1 <= c && c <= 3) ? c : 0);
}

/* 88/06/02 V3.1 */
static void
ERMOPN (n)                      /* モード定義表がオープンできない */
     fast unsigned int n;       /* V3.1 */
{
  static char *ermes[] = {
    /*  0 */ "Unprintable error",
    "Mode-hyo does't exist",
    "Can't open Mode-hyo",
    "Too many long Mode-hyo filenames",
  };

  if (n >= numberof (ermes))
    n = 0;

  fprintf (stderr, "\r\nromkan: Mode-hyo %s ---\r\n", cur_rk_table->mcurfnm);
  fprintf (stderr, "\tError No.%d: %s.\r\n", n, ermes[n]);
  longjmp (cur_rk_table->env0, 1);
}

/* 88/06/03 V3.1 */
static void
ERRMOD (n)                      /* モード定義表のエラー */
     fast unsigned int n;       /* V3.1 */
{
  static char *ermes[] = {
    /*  0 */ "Table incomplete",
    "')' mismatch",
    "Unprintable error",
    "Illegal filename",
    "Illegal modename",
    /*  5 */ "Undefined mode",
    "Illegal content(s) of list",
    "Illegal ^,\\o,\\x or \\d expression",
    "Illegal defmode",
    "Unrecognized keyword",
    /* 10 */ "Incomplete string",
    "Search path specified after filename",
    "Argument must be a string",
    "Can't get home directory",
    "Illegal @-keyword",
    /* 15 */ "User doesn't exist",
    "Illegal character",
  };

  if (n >= numberof (ermes))
    n = 2;

  fprintf (stderr, "\r\nromkan: Mode-hyo %s%s ---\r\n%s\r\n", cur_rk_table->mcurdir, cur_rk_table->mcurfnm, cur_rk_table->mcurread);
  fprintf (stderr, "\tError No.%d: %s.\r\n", n, ermes[n]);
  fclose (cur_rk_table->modefile);
  longjmp (cur_rk_table->env0, 1);
}