view Wnn/romkan/rk_main.c @ 21:22b754fbd8fe

added 2ch_face.u
author Yoshiki Yazawa <yaz@cc.rim.or.jp>
date Tue, 15 Apr 2008 19:32:47 +0900
parents bbc77ca4def5
children a7ccf412ba02
line wrap: on
line source

/*
 *  $Id: rk_main.c,v 1.4 2005/04/10 15:26:38 aonoto 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
 * Copyright FreeWnn Project 1999, 2000
 *
 * Maintainer:  FreeWnn Project   <freewnn@tomo.gr.jp>
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) any later version.
 *
 * This library 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
 * Boston, MA 02111-1307, USA.
 */

/***********************************************************************
                        rk_main.c
                                                88. 6.10  改 正

        ローマ字かな変換・その他、入力コードを変換するプログラム。
        ASCIIコード使用を、前提としている。
        このファイルは、変換のメインルーチン。
***********************************************************************/

#include "rk_header.h"
#include "rk_extvars.h"
#include "rk_fundecl.h"

struct matchpair                /*    現在マッチされている変数の番号と、マッチした文字の
                                   組を保持。 */
{
  int hennum;
  letter ltrmch;
};

letter memory[SIZALL];          /* 変換対応表や、変数の変域などを入れる領域 */
struct dat data[LINALL];        /* 対応表の一行ごとのデータへのポインタ */
letter *hensudefhyo[VARTOT];    /* 変数ごとの変域データへのポインタ */
struct matchpair henmatch[VARTOT];      /* 変数のマッチ状況をためておく */

#ifdef KDSP
#  ifdef MVUX
letter displine[DSPLIN];        /* デバッグ用  表示の一行分を記憶 */
#  endif
#endif

letter *curdis;
int codein_len;
letter *codeout, *remainkbf;
letter ungetc_buf = EOLTTR;     /* romkan_ungetcが一文字退避しておくバッファ */
letter unnext_buf = EOLTTR;     /* romkan_unnextが            〃             */

letter (*keyin_method) ();      /* キーイン関数のアドレス */
int (*bytcnt_method) ();        /* バイトカウント関数のアドレス */
int (*kbytcnt_method) ();       /* キー入力解釈用バイトカウント関数のアドレス */
char prv_modfnm[REALFN] = "\0"; /* モード表名(又はそのパス名)を保存 */
            /* 初期設定がいい加減! */

#define DISOUT ((flags & RK_NONISE) ? rk_output : disout)
 /* フラグの値はromkan_init3()の中で設定され、そのRK_NONISEのビットの値の
    標準は0。従ってDISOUTの標準値はdisout。これをrk_outputに変えると、偽コード
    (除:LTREOF)を一切出さない(但し、キー入力があれば必ず何かを返すように
    してある時に限り、EOLTTRをも返す)。 */

char eofflg;                    /* romkan_next()からLTREOFが来たときに、romkan_henkan()内で1
                                   にする。これの値が非0なら変換対応表の(error)を無視して、
                                   本処理バッファ内の未確定コードも、強制的に変換させる。更に
                                   LTREOFが来た時に特別にある出力を出すように指定されていれば
                                   その処理も行う(但し、その処理は急ごしらえで不完全)。その
                                   処理中はeofflgは2。 */

letter evalbuf[2][2][OUTSIZ];
char ebf_sw = 0;
letter saishu_out[OUTSIZ];
letter delchr, delchr2 = EOLTTR, nisedl;        /* DELのキャラクタは二個まで持てる */

modetyp chgmod (), incmod (), decmod ();


/* letterを返す関数の定義及び宣言(一部にはcharのものも混じっている) */

static letter mchedsrc ();
letter romkan_next (), romkan_unnext ();
letter romkan_getc (), romkan_ungetc (), *romkan_henkan ();
letter to_zenalpha (), to_zenhira (), to_zenkata ();
 /* to_zenhira;kata は濁点を持つ文字を一まとめにしない。 */

void BUGreport ();
static void maeato_henkan (), add_at_eof (), rk_delete (), set_rubout (), mchevl (), codeout_chg (), ltrevlcpy ();
int match ();
static int p_eq (), hen_ikisrc (), henkan_ok (), rk_rst (), head_bytecount ();
extern void readdata (), romkan_reset (), hank_setup (), to_hankata (), to_digit (), dakuadd (), handakuadd (), allchgmod ();
extern int ltov ();

static letter *
ltrcpy (lp1, lp2)
     letter *lp1, *lp2;
{
  letter *org;

  org = lp1;
  while ((*lp1++ = *lp2++) != EOLTTR);
  return (org);
}

#ifdef notuse
letter *
ltrncpy (lp1, lp2, n)           /* 末尾に'EOLTTR'をセット */
     letter *lp1, *lp2;
     int n;
{
  letter *org;

  org = lp1;
  for (; n; --n)
    if (EOLTTR == (*lp1++ = *lp2++))
      return (org);
  *lp1 = EOLTTR;
  return (org);
}
#endif

 /** letterの列の末尾に一文字つなげる */
static letter *
ltr1cat (lp, l)
     letter *lp, l;
{
  letter *org;

  org = lp;
  totail (lp);
  *lp++ = l;
  *lp = EOLTTR;
  return (org);
}

 /** 文字列の末尾に一文字つなげる。ltr1catのchar版 */
char *
chrcat (s, c)
     char *s, c;
{
  char *org;

  org = s;
  strtail (s);
  *s++ = c;
  *s = '\0';
  return (org);
}

static letter *
ltrcat (lp1, lp2)
     letter *lp1, *lp2;
{
  letter *org;

  org = lp1;
  totail (lp1);
  ltrcpy (lp1, lp2);
  return (org);
}

 /** letterの列lp2の各要素の最上位ビットを立ててから、lp1の後ろにつなげる。*/
static letter *
bitup_ltrcat (lp1, lp2)
     letter *lp1, *lp2;
{
  letter *org;

  org = lp1;
  totail (lp1);

  while ((*lp1 = *lp2++) != EOLTTR)
    *lp1++ |= HG1BIT;
         /** lp2 の要素全ての最上位ビットを立てる。*lp2がNISEBP(rk_spclval.h
             でdefine)などのときは最初から最上位が立ってるからよい */

  return (org);
}

 /** ltrcatしたあと、結果の文字列の末尾にポインタをもっていく。*/
letter *
ltrgrow (lp1, lp2)
     letter *lp1, *lp2;
{
  totail (lp1);
  while ((*lp1 = *lp2++) != EOLTTR)
    lp1++;
  return (lp1);
}

static int
ltrlen (lp)
     letter *lp;
{
  letter *org;

  for (org = lp; *lp != EOLTTR; lp++);
  return (lp - org);
}

 /**    letterの列の最後の文字へのポインタを返す。但し空文字列に対しては
        そのまま返す。*/
letter *
ltrend (lp)
     letter *lp;
{
  return ((*lp != EOLTTR) ? (lp + ltrlen (lp) - 1) : lp);
}

 /**    文字列の最後の文字へのポインタを返す。ltrend()のchar版。ltrend()と
        同様、空文字列のときは特例がある。*/
char *
strend (s)
     char *s;
{
  return (*s ? (s + strlen (s) - 1) : s);
}

int
ltrcmp (lp1, lp2)
     letter *lp1, *lp2;
{
  for (; *lp1 == *lp2; lp1++, lp2++)
    if (*lp1 == EOLTTR)
      return (0);
  return (*lp1 > *lp2 ? 1 : -1);
}

#ifdef notuse
int
ltrncmp (lp1, lp2, n)
     letter *lp1, *lp2;
     int n;
{
  for (; n && *lp1 == *lp2; n--, lp1++, lp2++)
    if (*lp1 == EOLTTR)
      return (0);
  return (n == 0 ? 0 : (*lp1 > *lp2 ? 1 : -1));
}

letter *
ltr_index (lp, l)
     letter l, *lp;
{
  for (; *lp != EOLTTR; lp++)
    if (*lp == l)
      return (lp);
  return (NULL);
}
#endif

static letter *
ltr_rindex (lp, l)
     letter l, *lp;
{
  letter *lp2;

  for (lp += ltrlen (lp2 = lp); lp2 != lp;)
    if (*--lp == l)
      return (lp);
  return (NULL);
}

 /** 文字列の末尾一文字をカットする。*/
letter *
ltr1cut (lp)
     letter *lp;
{
  int l;

  if (0 == (l = ltrlen (lp)))
    BUGreport (0);              /* Illegal POP */
  *(lp + --l) = EOLTTR;
  return (lp);
}

 /** letterの一文字を長さ1の文字列に変換する */
static letter *
ltr_to_ltrseq (lp, l)
     letter *lp, l;
{
  *lp++ = l;
  *lp-- = EOLTTR;
  return (lp);
}

 /** charの列からletterを一つ取り出す。但し'\0'はEOLTTRに変える */
/* *INDENT-OFF* */
letter
letterpick (lbfpptr)
     uns_chr **lbfpptr;
/* *INDENT-ON* */
{
  letter l = 0;
  register int i;

  for (i = (*bytcnt_method) (*lbfpptr); i; i--)
    l = (l << 8) + *(*lbfpptr)++;
  return (l == 0 ? EOLTTR : l);
}

 /**    入力されたコードを文字単位にまとめる。但しEOFはLTREOF(rk_spclval.h
        にて定義)に変える。*/
/* *INDENT-OFF* */
letter
romkan_next ()
/* *INDENT-ON* */
{
  letter in;
  int i, n;
  uns_chr c;

  /* unnextされている場合は、それを取り出す。 */
  if (unnext_buf != EOLTTR)
    return (in = unnext_buf, unnext_buf = EOLTTR, in);

  if ((letter) EOF == (in = (*keyin_method) ()))
    return (LTREOF);
  c = (in &= 0xff);
  n = (*kbytcnt_method) (&c);
  for (i = 1; i < n; i++)
    in = (in << 8) + ((*keyin_method) () & 0xff);
  return (in);
}

 /*     letterの文字 l とcharの文字 c の比較。エンドマーク同士(letter列の
    エンドマークはEOLTTR、char列のは'\0')も一致とみなす。 */
#define ltrchreq(l, c) ((l) == ((c) != 0 ? (c) : EOLTTR))

 /** letterの文字列とcharの文字列の比較 但し大小比較は符号なしとしてやる */
int
ltrstrcmp (l, s)
     register letter *l;
     register char *s;
{
  for (; ltrchreq (*l, *s); l++, s++)
    if (*s == 0)
      return (0);
  return ((*l < (uns_chr) * s || *l == EOLTTR) ? -1 : 1);
}

 /** 最後にマッチした文字 つまりurabufの最後。urabufが空だとEOLTTRを返す */
static letter
lastmch ()
{
  return (*(ltrend (urabuf)));
}

/* end of 'letter' functions */


#ifdef  OMAKE
 /** お  ま  け
        TVアニメのご案内(一部特撮を含む)。引数が非負整数の時はどれか一つ、
        -1なら全てを出力し、それ以外なら何も出力しない。いずれの場合も、データ
        の個数を返す。*/
takeoka (i)
     int i;
{
#define OPIMAX (numberof(opinion))

  static char *opinion[] = {
    "「魔法のアイドル パステルユーミ」は良い番組だ。by竹岡",
    "「機動戦士ガンダムΖΖ」の地球編は良い番組だ。by服部",
    "「魔法のスター マジカルエミ」は良い番組だ。by竹岡",
    "(参考意見)「マジカルエミ〔蝉時雨〕」は良いアニメだ。by森島",
    "「魔法の天使 クリィミーマミ」は良い番組だ。by竹岡",
    "「サザエさん」は良い番組だ。by鈴木",
    "「ムーミン」は良い番組だ。by鈴木",
    "「マンガ大作戦」のロードランナーシリーズは良い番組だ。by新出",
    "「スプーンおばさん」は良い番組だ。by鴨",
    "「はじめ人間ギャートルズ」は良い番組だ。by富永",
    "「宇宙の騎士テッカマン」は良い番組だ。by駒嵐",
    "「妖怪人間ベム」は良い番組だ。by鴨",
    "「アルプスの少女ハイジ」は良い番組だ。by富永",
    "「ときめきトゥナイト」は良い番組だ。by駒嵐",
    "「あしたのジョー」は良い番組だったね。by立木(ついき)",
    "「きまぐれオレンジ・ロード」は良い番組だ。by三好・服部",
    "「ゲッターロボ」(「ゲッターロボG」を含む)は良い番組だ。by新出",
    "「テッちゃん」(「じゃりン子チエ」のこと)は良い番組だ。by松田",
    "「チキチキマシン猛レース」は良い番組だ。by服部",
    "「小公女セーラ」は良い番組だ。by森島",
    "「秘密戦隊ゴレンジャー」は良い番組だ。by新出",
    "(白黒時代の)「鉄人28号」は良い番組だ。by桜川"
  };

  if (i >= 0)
    printf ("%s\n", opinion[i % OPIMAX]);
  else if (i == -1)
    for (i = 0; i < OPIMAX; i++)
      printf ("%s\n", opinion[i]);
  return (OPIMAX);
}
#endif /* OMAKE */


 /** 変換バッファのクリア */
void
romkan_clear ()
{
#ifdef KDSP
#  ifdef MVUX
  *displine = EOLTTR;
#  endif
#endif
  ungetc_buf = *keybuf = *urabuf = *disout = *rk_output = EOLTTR;
  curdis = DISOUT;
  lastoutlen = lastkbflen = 0;
}

 /**    表を読み込んで変換の初期設定をする。(part 1)*/
int
romkan_init (modhyo, delchr_, chmoutf, keyinfn, bytcntfn)
     char *modhyo, chmoutf;
     letter delchr_;
letter (*keyinfn) ();
     int (*bytcntfn) ();        /* 引数の詳細はromkan_init3を参照 */
{
  return (romkan_init2 (modhyo, delchr_, chmoutf, keyinfn, bytcntfn, 0, 0, 0));
}

#define ifflg(a, b) ((a) ? (b) : 0)

 /**    表を読み込んで変換の初期設定をする。(part 2: キー入力に対し必ず何か
        返すようにするか、キーバッファをクリアするかどうかなども指定可)*/
int
romkan_init2 (modhyo, delchr_, chmoutf, keyinfn, bytcntfn, keyackf, restartf, nonisecodf)
     char *modhyo, chmoutf, keyackf, restartf, nonisecodf;
     letter delchr_;
letter (*keyinfn) ();
     int (*bytcntfn) ();        /* 引数の詳細はromkan_init3を参照 */
{
  return (romkan_init3 (modhyo, delchr_, toNISE (delchr_), EOLTTR,
                        keyinfn, bytcntfn, (int (*)()) NULL, restartf, ifflg (chmoutf, RK_CHMOUT) | ifflg (keyackf, RK_KEYACK) | ifflg (nonisecodf, RK_NONISE) | RK_DSPNIL));
}

 /**    表を読み込んで変換の初期設定をする。(part 3)*/
int
romkan_init3 (modhyo, delchr_, nisedl_, delchr2_, keyinfn, bytcntfn, kbytcntfn, restartf, flags_)
     char *modhyo;              /* モード定義表の名又はそのパス名 */
     letter delchr_;            /* DELとして使うコード */
     letter nisedl_;            /* 偽DELとして使うコード */
     letter delchr2_;           /* DELとして使うコードが二つある場合そのもう一方 */
letter (*keyinfn) ();           /* キーイン関数 */
     int (*bytcntfn) ();        /* バイトカウント関数 */
     int (*kbytcntfn) ();       /* キーインの解釈用バイトカウント関数。NULL指定の時は
                                   bytcntfnと同じものが使われる */
     char restartf;
 /*     rk_rst()内から呼ばれた時に立つフラグ。これが立つと、バッファをクリア
    せずに済ます。この時はエラー検出時のメッセージも少し異なる */
     int flags_;
 /* 以下のフラグが利用可。これらはrk_spclval.hにて定義。
    RK_CHMOUT:モードチェンジを知らせるコードを返すか?
    RK_KEYACK:キーインに対し必ず何かを返すか
    RK_DSPNIL:romkan_disp(off)modeのデフォルトを空文字列にするか
    RK_NONISE:偽コードを出さないようにするか
    RK_REDRAW:Wnn用特殊フラグ(redraw用のフラグを出すかどうか)
    RK_SIMPLD:deleteの動作を単純にするか
    RK_VERBOS:verboseモードで起こすか
  */
{
  int errcod;                   /* 今の所1のみ */

  /* 偽コードを出すかの設定はromkan_clearで使うので、フラグの設定を
     先にやっておかないといけない。 */
  if ((flags = flags_) & RK_VERBOS)
    {
      fprintf (stderr, "romkan_init invoked.\r\n");
      fflush (stderr);
    }

#ifdef  OMAKE
  if (modhyo == NULL)
    takeoka (-1);               /* おまけ;通常は実行されない */
#endif /* OMAKE */

  keyin_method = keyinfn;
  bytcnt_method = (bytcntfn == NULL ? head_bytecount : bytcntfn);
  kbytcnt_method = (kbytcntfn == NULL ? bytcnt_method : kbytcntfn);
  strcpy (prv_modfnm, modhyo);

  delchr = delchr_;
  nisedl = nisedl_;
  delchr2 = delchr2_;
 /**    実はDELのキャラクタを二個まで持てる。二個目:delchr2は、未設定の
        とき(値がEOLTTR)は無視される。それ以外は、delchr2が入力されると
        delchrが入力されたのと同じ振る舞いをする。*/

  if ((errcod = setjmp (env0)) != 0)
    {
      fprintf (stderr, "romkan_init failed.");
      if (!restartf)
        {
          romkan_reset ();
          /*     fprintf(stderr,
             "No conversion is currently performed."); */
          fprintf (stderr, "\r\n");
        }
      else
        {
          fprintf (stderr, "Conversion method was unchanged.\r\n");
        }
    }
  else
    {
      readdata (memory, data, hensudefhyo, modhyo);
      if (!restartf)
        {
          romkan_clear ();
          hank_setup ();
        }
    }

  if (!errcod && (flags & RK_VERBOS))
    {
      fprintf (stderr, "romkan_init finished.\r\n");
      fflush (stderr);
    }
  return (errcod);              /* 正常終了なら0 */
}

 /**    バイトカウント関数のデフォルト。sの指している所に入っているのが
        何バイトコードかを返す。
        romkan_nextに注意。そちらでは、バイトカウント関数の値は
        引数の一文字目にのみ依存すると仮定している。*/
static int
head_bytecount (s)
     uns_chr *s;
{
#ifdef IKIS
  return ((*s <= 0xa0 || *s == 0xff) ? 1 : 2);
#else
  return (((*s <= 0xa0 && *s != HNKAK1) || *s == 0xff) ? 1 : 2);
#endif
}

 /** 変換された文字を順次返す */
/* *INDENT-OFF* */
letter
romkan_getc ()
/* *INDENT-ON* */
{
  letter l;

  /* ungetcされている場合は、それを取り出す。 */
  if (ungetc_buf != EOLTTR)
    return (l = ungetc_buf, ungetc_buf = EOLTTR, l);

  while (*curdis == EOLTTR)
    {
      /* romkan_next()の値がLTREOFの時も、そのまま送れば良い。 */
      curdis = romkan_henkan (romkan_next ());

      /* keyackflgが非0なら、キーインがあれば必ず何か返す。その
         ため、返すべきものがない時はEOLTTRを返すことにする。 */
      if (flags & RK_KEYACK)
        break;
    }

  if (EOLTTR != (l = *curdis))
    curdis++;
  return (l);
  /* 偽物の文字なら、HG1BITが立っている。 */
  /* 特別なコード(例えば偽のBEEPとしてNISEBP)を返すときがある。それらは
     rk_spclval.hに定義されている。 */
}

 /**    romkan_getcの下位関数として、入力を一文字受け取って
        変換結果の文字列を出力する。*/
letter *
romkan_henkan (mae_in)
     letter mae_in;
{
  letter mae_out[2], *p;

  /* if(*curdis != EOLTTR){p = curdis; curdis = nil; return(p);} */
  /* ↑これではmae_inが無視される */
  curdis = nil;

  /* 前・後処理への入力は常にただ一文字 */
  mae_out[0] = mae_out[1] = EOLTTR;

  eofflg = rk_errstat = 0;
  *rk_output = *disout = EOLTTR;

  maeato_henkan (mae_in, mae_out, usemaehyo);
  rk_input = *mae_out;

  if (rk_input == LTREOF)
    {
      /* LTREOFが来た場合、(error)を無視し、本処理バッファの末尾迄
         強制変換する。そのためにeofflgを1にする。その後、結果の
         末尾に、LTREOFが来た時の特別コード(指定されていれば)と、
         LTREOFをつなぐ。 */
      eofflg = 1;
      match ();

      eofflg = 2;
      add_at_eof ();            /* codeoutに、LTREOFが来た時出すコードが入る。 */
      ltr1cat (codeout, LTREOF);

      ltrcat (disout, codeout);
      ltrcat (rk_output, codeout);
    }
  else if (rk_input == EOLTTR)
    {
      /* EOLTTRが来た場合も、上と同様の処理を行うが、LTREOFは
         つながない。なお、これはromkan_getc()を呼んでいる時は
         起こらない(romkan_next()がEOLTTRを返さないから)。 */
      eofflg = 1;
      match ();
    }
  else if (rk_input == delchr || rk_input == delchr2)
    {
      /* delchr2が未設定ならその値はEOLTTRなのでrk_inputと等しくない。 */
      rk_delete (rk_input);
    }
  else
    {
      ltr1cat (keybuf, rk_input);
      ltr1cat (disout, toNISE (rk_input));
      match ();
    }

  if (!(flags & RK_CHMOUT))
    {
      /* chmoutflgが0の時は、CHMSIGを出さない。 */
      for (p = DISOUT;; p++)
        {
          while (*p == CHMSIG)
            ltrcpy (p, p + 1);
          if (*p == EOLTTR)
            break;
        }
    }

  if ((flags & RK_REDRAW) && NULL != (p = ltr_rindex (disout, nisedl)))
    {
      for (p++; *p != EOLTTR || (ltr1cat (disout, REDRAW), 0); p++)
        if (!isSPCL (*p))
          break;
    }
  /* wnnのredrawのフラグが立っていて、disoutがnisedl以後特殊コードのみ
     で終わっていたら、REDRAWを出して、Wnnに変換行のredrawをさせる。 */

  return (DISOUT);
}


 /* デバッグ用関数 */
#ifdef KDSP
#  ifdef MVUX

void
pridbg2 (a, b, c)
     char *a, *c;
     letter *b;
{
  printf ("%s", a);
  dump_fordbg (b);
  printf ("%s", c);
}

int
ltr_displen (l)
     letter l;
{
  while (l >= 256)
    l >>= 8;

  if (l == 0)
    return (0);
  if (l < ' ')
    return (0);
  if (168 <= l && l <= 170)
    return (1);
  return (l > 160 ? 2 : 1);
}

void
dump_fordbg (lp)
     letter *lp;
{
  while (*lp != EOLTTR)
    printf ("%x/", *lp++);
}

void
print_fordbg (lp)
     letter *lp;
{
  while (*lp != EOLTTR)
    putletter (*lp++ & ~HG1BIT);
}

void
print_ltrseq (lp)
     letter *lp;
{
  while (*lp != EOLTTR)
    print_ltr (*lp++ & ~HG1BIT);
}

void
print_ltr (l)
     letter l;
{
  letter *disptail;
  int i;

  if (!isSPCL (l))
    l &= ~HG1BIT;

  if (l == CHMSIG)
#ifdef CHMDSP
    printf ("...mode=%s\n", romkan_dispmode ())
#endif
      ;
  else
#define BEEPCH  '\007'
#define NEWLIN  '\n'
  if (l == BEEPCH || l == NISEBP)
    putchar (BEEPCH);
  else if (l == NEWLIN)
    {
      *displine = *keybuf = *urabuf = EOLTTR;
      lastoutlen = lastkbflen = 0;
      putchar (l);
    }
  else if (l == delchr || l == nisedl)
    {
      if (*displine == EOLTTR)
        putchar (BEEPCH);
      else
        {
          disptail = ltrend (displine);
          for (i = ltr_displen (*disptail); i; i--)
            printf ("\031 \031");
          *disptail = EOLTTR;
        }
    }
  else
    {
      ltr1cat (displine, l);
      putletter (l);
    }
}

#  endif /* of #ifdef MVUX */
#endif
/* デバッグ用関数終わり */


 /** DELが入力されたときの処理をする */
static void
rk_delete (input_del)
     letter input_del;
{
  if (ltrlen (keybuf) > ((flags & RK_SIMPLD) ? 0 : lastkbflen))
    {
      ltr1cut (keybuf);
      set_rubout (disout, 1, nisedl);
    }
  else
    {
      if (*urabuf != EOLTTR && !(flags & RK_SIMPLD))
        {
          ltr1cut (ltrcpy (keybuf, urabuf));
          *urabuf = EOLTTR;

          set_rubout (rk_output, lastoutlen, input_del);

          set_rubout (disout, lastkbflen, nisedl);
          bitup_ltrcat (ltrcat (disout, rk_output), keybuf);

          lastkbflen = lastoutlen = 0;
        }
      else
        {
          set_rubout (disout, 1, input_del);
          set_rubout (rk_output, 1, input_del);
        }
    }
}

 /**    letterの列 lp1 と lp2 のマッチを試みる。返値は、lp1がlp2の頭部と
        一致の時 -1、lp1またはその頭部とlp2が一致のときは一致長(lp2が
        空文字列の時を含む。この場合返値は0)、それ以外は -2。
        lp2側に式が含まれていたら評価をする。lp1側には式を含んではだめ */
static int
prefixp (lp1, lp2)
     letter *lp1, *lp2;
{
  /* 1行マッチさせるごとに、まずmatch情報をクリアしてから。
     つまり、henmatch[0] . ltrmch = EOLTTR; としておく。 */

  register int mch_len = 0, d_len;

  for (;;)
    {
      if (*lp2 == EOLTTR)
        return (mch_len);
      if (*lp1 == EOLTTR)
        return (-1);
      if ((d_len = p_eq (&lp2, &lp1)) < 0)
        return (d_len);
      mch_len += d_len;
    }
}

 /** num番目の変数が既にある文字とマッチしていると仮定して、その文字を返す */
static letter
mchedsrc (num)
     int num;
{
  struct matchpair *pairptr;

  for (pairptr = henmatch; pairptr->ltrmch != EOLTTR; pairptr++)
    {
      if (num == pairptr->hennum)
        return (pairptr->ltrmch);
    }
  return (BUGreport (8), 0);
}

 /**    num番目の変数が文字 l とマッチするか調べる。その変数がunboundだった
        場合は l にbindする。マッチしたら(bindの時を含む)1、しないと0を返す */
static int
mchsrc (num, l)
     int num;
     letter l;
{
  struct matchpair *pairptr;

  if (hen_ikisrc (num, l) == 0)
    return (0);
  for (pairptr = henmatch; pairptr->ltrmch != EOLTTR; pairptr++)
    {
      if (num == pairptr->hennum)
        return (pairptr->ltrmch == l);
    }

  pairptr->ltrmch = l;
  pairptr->hennum = num;
  (++pairptr)->ltrmch = EOLTTR;
  return (1);
}

 /**    l1pから一単位を取って評価し、文字l2と一致するかどうかを返す。評価した
        結果が一文字にならなかったら、当然一致しない。*/
static int
l_eq (l1p, l2)
     letter *l1p, l2;
{
  letter evlrsl[RSLMAX];

  switch (SHUBET (*l1p))
    {
    case 0:
      return (*l1p == l2);
    case 1:
      return (mchsrc ((int) LWRMSK (*l1p), l2));
    case 2:
      mchevl (&l1p, evlrsl);
      return (evlrsl[0] == l2 && evlrsl[1] == EOLTTR);
    default:
      return (BUGreport (1), 0);
    }
}

 /**    prefixp内で使用  但し引数の順序は逆、すなわち式が含まれうるのはl1pp側
        のみ。l1ppから一単位ぶん取って評価したものとl2ppのマッチを試みる。それ
        がl2ppまたはその頭部とマッチすれば一致長を返し(l1ppの評価結果が空文字
        列の時を含む。この場合返値は0)、逆にl1ppの評価結果の頭部とl2ppがマッ
        チした時は -1を返す。マッチが失敗したら返値は -2。*/
static int
p_eq (l1pp, l2pp)
     register letter **l1pp, **l2pp;
{
  int num;
  letter evlrsl[RSLMAX], *rslptr;
  register int retval = -2;

  /*  l2pp側には式を含まない筈 */
  if (!is_HON (**l2pp))
    {
      /*  if(is_HON(**l1pp)) retval = p_eq(l2pp, l1pp); else  */
      BUGreport (9);
    }
  else
    {
      switch (SHUBET (**l1pp))
        {
        case 0:         /* 文字同士 */
          retval = (*(*l1pp)++ == *(*l2pp)++ ? 1 : -2);
          break;
        case 1:         /* 変数と文字 */
          num = LWRMSK (*(*l1pp)++);
          retval = (mchsrc (num, *(*l2pp)++) ? 1 : -2);
          break;
        case 2:         /* 式と文字 */
          mchevl (l1pp, rslptr = evlrsl);
          for (retval = 0; *rslptr != EOLTTR; retval++)
            {
              if (**l2pp == EOLTTR)
                {
                  retval = -1;
                  break;
                }
              else if (*rslptr++ != *(*l2pp)++)
                {
                  retval = -2;
                  break;
                }
            }
          break;
        default:
          BUGreport (2);
        }
    }

  return (retval);
}

 /** l1pから一単位評価してl2pに入れる */
static void
mchevl (l1pp, l2p)
     letter **l1pp, *l2p;
{
  letter *l1p, tmpevl[RSLMAX];

  l1p = *l1pp;

  switch (SHUBET (*l1p))
    {
    case 0:
      *l2p++ = *l1p++;
      break;
    case 1:
      *l2p++ = mchedsrc ((int) LWRMSK (*l1p++));
      break;
    case 2:                     /* toupper, tolower, error, … */
      switch (LWRMSK (*l1p++))
        {
        case 2:
          mchevl (&l1p, tmpevl);
          *l2p++ = to_upper (*tmpevl);
          break;
        case 3:
          mchevl (&l1p, tmpevl);
          *l2p++ = to_lower (*tmpevl);
          break;
        case 4:
          *l2p++ = CHMSIG;
          *l2p++ = LWRMSK (*l1p++);
          *l2p++ = 0;
          break;                /* EOLではない */
        case 5:
          *l2p++ = CHMSIG;
          *l2p++ = LWRMSK (*l1p++);
          *l2p++ = 1;
          break;
        case 6:
          *l2p++ = CHMSIG;
          *l2p++ = XY2INT (2, LWRMSK (*l1p++));
          *l2p++ = 1;
          break;
        case 7:
          mchevl (&l1p, tmpevl);
          *l2p++ = to_updown (*tmpevl);
          break;
        case 8:
          mchevl (&l1p, tmpevl);
          *l2p++ = to_zenalpha (*tmpevl);
          break;
        case 9:
          mchevl (&l1p, tmpevl);
          *l2p++ = to_hira (*tmpevl);
          break;
        case 10:
          mchevl (&l1p, tmpevl);
          *l2p++ = to_kata (*tmpevl);
          break;
        case 11:
          mchevl (&l1p, tmpevl);
          to_hankata (*tmpevl, &l2p);
          break;                /* 特殊 */
        case 12:
          mchevl (&l1p, tmpevl);
          *l2p++ = to_zenhira (*tmpevl);
          break;
        case 13:
          mchevl (&l1p, tmpevl);
          *l2p++ = to_zenkata (*tmpevl);
          break;
        case 14:
          mchevl (&l1p, tmpevl);
          *l2p = *tmpevl;
          mchevl (&l1p, tmpevl);
          *l2p += *tmpevl;
          LWRCUT (*l2p++);
          break;
        case 15:
          mchevl (&l1p, tmpevl);
          *l2p = *tmpevl;
          mchevl (&l1p, tmpevl);
          *l2p -= *tmpevl;
          LWRCUT (*l2p++);
          break;
        case 16:
          mchevl (&l1p, tmpevl);
          *l2p = *tmpevl;
          mchevl (&l1p, tmpevl);
          *l2p *= *tmpevl;
          LWRCUT (*l2p++);
          break;
        case 17:
          mchevl (&l1p, tmpevl);
          *l2p = *tmpevl;
          mchevl (&l1p, tmpevl);
          if (!*tmpevl)
            *l2p = LTRHUG;
          else
            *l2p /= *tmpevl;
          LWRCUT (*l2p++);
          break;
        case 18:
          mchevl (&l1p, tmpevl);
          *l2p = *tmpevl;
          mchevl (&l1p, tmpevl);
          if (!*tmpevl)
            *l2p = LTRHUG;
          else
            *l2p %= *tmpevl;
          LWRCUT (*l2p++);
          break;
          /* 19〜21・30は、条件を満たすと空文字列、
             そうでないとUNUSDCを文字列として返す。 */
        case 19:
          mchevl (&l1p, tmpevl);
          if (lastmch () != *tmpevl)
            *l2p++ = UNUSDC;
          break;
        case 20:
          if (!modesw[LWRMSK (*l1p++)].curmode)
            *l2p++ = UNUSDC;
          break;
        case 21:
          if (modesw[LWRMSK (*l1p++)].curmode)
            *l2p++ = UNUSDC;
          break;
        case 22:
          *l2p++ = REASIG;
          break;
        case 23:
          *l2p++ = delchr;
          break;
        case 24:
          *l2p++ = CHMSIG;
          *l2p++ = XY2INT (1, 0);
          /* これで「all」を表す */
          *l2p++ = 0;
          break;
        case 25:
          *l2p++ = CHMSIG;
          *l2p++ = XY2INT (1, 0);
          *l2p++ = 1;
          break;
        case 26:
          mchevl (&l1p, tmpevl);
          *l2p = *tmpevl;
          mchevl (&l1p, tmpevl);
          *l2p &= *tmpevl;
          LWRCUT (*l2p++);
          break;
        case 27:
          mchevl (&l1p, tmpevl);
          *l2p = *tmpevl;
          mchevl (&l1p, tmpevl);
          *l2p |= *tmpevl;
          LWRCUT (*l2p++);
          break;
        case 28:
          mchevl (&l1p, tmpevl);
          *l2p = ~(*tmpevl);
          LWRCUT (*l2p++);
          break;
        case 29:
          *l2p++ = URBFCL;
          break;
        case 30:
          if (eofflg != 2 || *keybuf != EOLTTR)
            *l2p++ = UNUSDC;
          break;
        case 31:
          {
            letter code, basenum;

            mchevl (&l1p, tmpevl);
            code = *tmpevl;
            mchevl (&l1p, tmpevl);
            if ((basenum = *tmpevl) <= 1 || BASEMX < basenum)
              basenum = 10;
            to_digit (code, basenum, &l2p);
          }
          break;
        case 32:
          mchevl (&l1p, tmpevl);
          dakuadd (*tmpevl, &l2p);
          break;                /* 特殊 */
        case 33:
          mchevl (&l1p, tmpevl);
          handakuadd (*tmpevl, &l2p);
          break;                /* 特殊 */
        case 34:
          mchevl (&l1p, tmpevl);
          *l2p++ = ltov (*tmpevl);
          break;
        case 35:
          *l2p++ = ERRCOD;
          break;
          /* case 36: omitted */
        case 37:
          *l2p++ = CHMSIG;
          *l2p++ = LWRMSK (*l1p++);
          *l2p++ = *l1p++;
          break;
        case 38:
          *l2p++ = CHMSIG;
          *l2p++ = XY2INT (2, LWRMSK (*l1p++));
          *l2p++ = *l1p++;
          break;
        case 39:
          *l2p++ = CHMSIG;
          *l2p++ = XY2INT (3, LWRMSK (*l1p++));
          *l2p++ = *l1p++;
          break;
        case 40:
          {
            letter modnum;

            modnum = LWRMSK (*l1p++);
            if (modesw[modnum].curmode != *l1p++)
              *l2p++ = UNUSDC;
            break;
          }
        case 41:
          {
            letter modnum;

            modnum = LWRMSK (*l1p++);
            if (modesw[modnum].curmode == *l1p++)
              *l2p++ = UNUSDC;
            break;
          }
        case 42:
          {
            letter modnum;

            modnum = LWRMSK (*l1p++);
            if (modesw[modnum].curmode >= *l1p++)
              *l2p++ = UNUSDC;
            break;
          }
        case 43:
          {
            letter modnum;

            modnum = LWRMSK (*l1p++);
            if (modesw[modnum].curmode <= *l1p++)
              *l2p++ = UNUSDC;
            break;
          }
        case 44:
          mchevl (&l1p, tmpevl);
          *l2p++ = SENDCH;
          *l2p++ = *tmpevl;
          break;
        default:                /* case 0及び上記以外 */ ;
          BUGreport (7);
        }
    }

  *l2p = EOLTTR;
  *l1pp = l1p;
}

 /** num番目の変数の変域が文字 l を含むかどうかを返す */
static int
hen_ikisrc (num, l)
     int num;
     letter l;
{
  letter *defptr;

  defptr = hyo_n[hyonum].hensudef[num];
  if (*defptr == VARRNG)
    {
      for (defptr++; *defptr != EOLTTR;)
        if (*defptr++ <= l && l <= *defptr++)
          return (1);
      return (0);
    }

  for (; *defptr != EOLTTR; defptr++)
    {
      if (l == *defptr)
        return (1);
    }
  return (0);
}

 /**    変換のメインルーチン。本処理を行うが、ついでに後処理もやっている。
        ちなみに前処理は、romkan_getcの下位関数romkan_henkanの中で、
        この関数を呼ぶ前にやっている。
        この関数は、romkan_nextから一文字来る度に呼ばれる。呼び出された直後は
        outputは空文字列、disoutには入力コード一文字が入っている。
        この関数で得られる文字の列が、romkan_henkanに渡り、
        romkan_getcは、それを文字ごとに分解して返す。
        (error)でエラーが引き起こされた場合は0を返し、正常終了時は1を返す */
int
match ()
{
  int henkanflg = 0, okcode = 0, chm_exist;
  letter *p;
  letter urabufcreate[KBFSIZ], orgkeybuf[KBFSIZ], kbftail[KBFSIZ];
  letter *urabufjunbi, *outcutptr, *dis_end;

  if (*keybuf == EOLTTR)
    {
      *urabuf = EOLTTR;
      return (1);
    }

  ltrcpy (urabufjunbi = orgkeybuf, keybuf);
  outcutptr = rk_output;

  while ((okcode = henkan_ok ()) > 0)
    {
      henkanflg = 1;

      codeout_chg ();
      ltrcat (rk_output, codeout);

      ltrcpy (kbftail, keybuf + codein_len);
      ltrcat (ltrcpy (keybuf, remainkbf), kbftail);

      if (okcode == 2)
        {
          ltrcpy (urabufjunbi = urabufcreate, keybuf);
          totail (outcutptr);
        }
    }

  if (okcode == 0)
    {
      ltr1cut (ltrcpy (keybuf, orgkeybuf));
      ltr_to_ltrseq (disout, NISEBP);
      *rk_output = EOLTTR;
      return (0);
    }
  if (henkanflg)
    {
      ltrcpy (urabuf, urabufjunbi);

      set_rubout (disout, ltrlen (orgkeybuf) - 1, nisedl);

      dis_end = disout;
      totail (dis_end);
      ltrcpy (dis_end, rk_output);

      /* モードチェンジを直ちに知らせるため CHMSIGを出力
         (flags中のRK_CHMOUTが立ってないと、あとで一文字ずつに
         分解する時点で、CHMSIGをカット)。
         但し、rk_outputからは、CHMSIGを抜く。
         また、CHMSIGは末尾に1回しか出力しない
         (2回以上あっても、1回にまとめて、末尾に置く)。 */
      for (chm_exist = 0, p = rk_output;; p++)
        {
          while (*p == CHMSIG)
            {
              chm_exist = 1;
              if (ltrcpy (p, p + 1) < outcutptr)
                outcutptr--;
            }
          if (*p == EOLTTR)
            break;
        }
      if (chm_exist)
        {
          /* CHMSIGを1つにまとめたものをdis_endにつなげ直す。
             このif文をカットすれば、CHMSIGのとりまとめはやらない */
          ltr1cat (ltrcpy (dis_end, rk_output), CHMSIG);
        }

      bitup_ltrcat (disout, keybuf);
      lastoutlen = ltrlen (outcutptr);
      lastkbflen = ltrlen (keybuf);
    }
  return (1);
}

 /**    LTREOFが入ったときに、何か出すように指定されているか調べて、
        codeoutをその結果の文字列(指定がなかったら当然空)にポイントする。
        超急ごしらえで、特殊コード等は一切無視する。*/
static void
add_at_eof ()
{
  register struct dat *datptr;
  register int i;
  int hyoseq;
  letter evlrsl[RSLMAX], *p;

  for (hyoseq = 0; (hyonum = usehyo[hyoseq]) != -1; hyoseq++)
    {
      for (i = 0, datptr = hyo_n[hyonum].data; NULL != (p = datptr[i].code[0]); i++)
        {
          henmatch[0].ltrmch = EOLTTR;
          while (*p != EOLTTR)
            {
              switch (SHUBET (*p))
                {
                case 0: /* 文字 */
                case 1: /* 変数 */
                  /* これらがある場合は、NULLとは
                     マッチし得ない。 */
                  goto Pass;
                case 2: /* 式 */
                  mchevl (&p, evlrsl);
                  if (*evlrsl != EOLTTR)
                    goto Pass;
                  /* 入力コード部に、評価すると
                     空文字列になるものが、他にはない
                     ことが前提。 */
                }
            }

          ltrevlcpy (codeout = p = evalbuf[0][0], datptr[i].code[1]);
          while (*p != EOLTTR)
            {
              if (isSPCL (*p))
                ltrcpy (p, p + 1);
              else
                p++;
            }
          codeout_chg ();
          return;

        Pass:;
        }
    }
  codeout = nil;
}

 /**    一回マッチを試みる。返値は、マッチして確定した場合1(モードチェンジが
        混じっている場合は2)、マッチしたが未確定の時-1、マッチしなかったら0。
        実行中は、変数 l に、それまでに一致した長さの最高記録を入れており、
        より長く一致するものが見つかるごとに、これを更新する。lの値は、マッチ
        していても0になることもある。p_eq() 及び prefixp() の注釈文を参照。*/
static int
henkan_ok ()
{
  register struct dat *datptr;
  register int i, k;
  int l, j, hyoseq;
  char urabuf_clrf;             /* モードチェンジなどで、urabufをクリアする必要が
                                   生じた場合はこれが立ち、その結果、henkan_ok()
                                   が1を返すべきところで2を返す。それを見て、
                                   match()がurabufなどの調整をする。 */
  register letter *p;

  if (*keybuf == EOLTTR)
    return (-1);

  for (l = -1, hyoseq = 0; (hyonum = usehyo[hyoseq]) != -1; hyoseq++)
    {
      for (i = 0, datptr = hyo_n[hyonum].data; NULL != (p = datptr[i].code[0]); i++)
        {
          henmatch[0].ltrmch = EOLTTR;
          switch (k = prefixp (keybuf, p))
            {
            case -2:
              break;
            case -1:
              if (eofflg != 0)
                break;
              /* eofflgが立っていたら、未確定の可能性は
                 捨てる。 */

              return (-1);
            default:
              if (k > l)
                {
                  ebf_sw = !ebf_sw;
                  for (j = 1; j <= 2; j++)
                    {
                      ltrevlcpy (evalbuf[ebf_sw][j - 1], datptr[i].code[j]);
                    }

                  l = k;
                }
            }
        }
    }

  if (l >= 0)
    {
      codein_len = l;
      codeout = evalbuf[ebf_sw][0];
      remainkbf = evalbuf[ebf_sw][1];

      for (urabuf_clrf = 0, p = codeout; *p != EOLTTR;)
        {
          switch (*p)
            {
            case CHMSIG:
              p++;
              /* codeoutの1バイト目に関数のスイッチとモード番号、
                 2バイト目にモードの新状態が入ってる */
              switch (SHUBET (*p))
                {
                case 0:
                  chgmod ((int) *p, (modetyp) * (p + 1));
                  break;
                case 1:
                  allchgmod ((modetyp) * (p + 1));
                  break;
                case 2:
                  incmod ((int) LWRMSK (*p), (modetyp) * (p + 1));
                  break;
                case 3:
                  decmod ((int) LWRMSK (*p), (modetyp) * (p + 1));
                  break;
                }
              ltrcpy (p, p + 2);
              /* CHMSIGだけ残して1,2バイト目cut */

              urabuf_clrf = 1;
              break;
            case URBFCL:
              /* urabufのクリアを明示的に指定する */
              urabuf_clrf = 1;
              ltrcpy (p, p + 1);
              break;
            default:
              p++;
            }
        }

      if (*codeout == ERRCOD)
        {
          if (eofflg == 0)
            {
              rk_errstat = 1;
              return (0);
            }

          /* (error)であって、しかもeofflgが立ってたら、keybuf
             の末尾まで、そのまま出す。 */
          codein_len = ltrlen (keybuf);
          codeout = ltrcpy (evalbuf[ebf_sw][0], keybuf);
          remainkbf = nil;
          rk_errstat = 2;

          return (1);
        }
      /* (error)は単独でしか書けないので、エラー検出はこれで十分。 */

      if (*codeout == REASIG)
        {
          *codeout = (rk_rst () != 0 ? EOLTTR : CHMSIG);
          /* 再readでエラったらモードチェンジの通知はしない */

          urabuf_clrf = 1;
        }
      /* 表の再read。但し、これが起こったことを外に知らせるのはCHMSIGで
         このコードそのものは外へ出ない。(restart)は、(error)同様、
         単独でしか書けないので、検出はこれで十分。 */

      return (urabuf_clrf ? 2 : 1);
    }

  /* 表に現れていないコードはそのまま返す */

  codein_len = 1;
  *(codeout = oneletter) = *keybuf;
  remainkbf = nil;
  return (1);
}

 /* rk_rst内で使うマクロ */
#define taihi(X, Y, N) {for(i = 0; i < N; i++) X[i] = Y[i];}
#define recov(X, Y, N) taihi(Y, X, N)

 /**    表の動的再読み込みをする。現在の内部表現を全て退避し、前と同じ
        ディレクトリ(に、現在のところ限定)から表を読み込む。もし、
        読み込み中にエラーを検出すれば、もとの内部データを復活し非0を返す。*/
static int
rk_rst ()
{
  register int i;
  int j;

  letter memoryt[SIZALL];
  struct dat datat[LINALL];
  struct hyo hyo_nt[HYOMAX];
  letter *hensudefhyot[VARTOT];
  struct modestat modeswt[MODMAX];
  char hyoshut[HYOMAX];
  char *modmeibgnt[MODMAX], modmeimem_t[MODMEI];
  char *dspnambgnt[DMDMAX], dspcod_t[DMDCHR];
  int usemaehyot[HYOMAX], usehyot[HYOMAX], useatohyot[HYOMAX];
  int naibu_t[NAIBMX];
  char *dspmodt[2][2];

  taihi (memoryt, memory, SIZALL);
  taihi (datat, data, LINALL);
  taihi (hyo_nt, hyo_n, HYOMAX);
  taihi (hensudefhyot, hensudefhyo, VARTOT);
  taihi (modeswt, modesw, MODMAX);
  taihi (hyoshut, hyoshu, HYOMAX);
  taihi (modmeibgnt, modmeibgn, MODMAX);
  taihi (modmeimem_t, modmeimem_, MODMEI);
  taihi (dspnambgnt, dspnambgn, DMDMAX);
  taihi (dspcod_t, dspcod_, DMDCHR);
  taihi (usemaehyot, usemaehyo, HYOMAX);
  taihi (usehyot, usehyo, HYOMAX);
  taihi (useatohyot, useatohyo, HYOMAX);
  taihi (naibu_t, naibu_, NAIBMX);
  for (i = 0; i < 2; i++)
    for (j = 0; j < 2; j++)
      dspmodt[i][j] = dspmod[i][j];

  if (0 == romkan_init3 (prv_modfnm, delchr, nisedl, delchr2, keyin_method, bytcnt_method, kbytcnt_method, 1, flags & ~RK_VERBOS))
    return (0);                 /* 正常終了 */

  recov (memoryt, memory, SIZALL);
  recov (datat, data, LINALL);
  recov (hyo_nt, hyo_n, HYOMAX);
  recov (hensudefhyot, hensudefhyo, VARTOT);
  recov (modeswt, modesw, MODMAX);
  recov (hyoshut, hyoshu, HYOMAX);
  recov (modmeibgnt, modmeibgn, MODMAX);
  recov (modmeimem_t, modmeimem_, MODMEI);
  recov (dspnambgnt, dspnambgn, DMDMAX);
  recov (dspcod_t, dspcod_, DMDCHR);
  recov (usemaehyot, usemaehyo, HYOMAX);
  recov (usehyot, usehyo, HYOMAX);
  recov (useatohyot, useatohyo, HYOMAX);
  recov (naibu_t, naibu_, NAIBMX);
  for (i = 0; i < 2; i++)
    for (j = 0; j < 2; j++)
      dspmod[i][j] = dspmodt[i][j];
  return (1);
}

 /** lp2から評価して得た文字列をlp1にコピー */
static void
ltrevlcpy (lp1, lp2)
     letter *lp1, *lp2;
{
  while (*lp2 != EOLTTR)
    {
      mchevl (&lp2, lp1);
      totail (lp1);
    }
  *lp1 = EOLTTR;
}

static void
set_rubout (lp, n, del)
 /** lpに 「del」n個の列をセットする。ここに del は 'delchr'か'nisedl' */
     letter *lp, del;
     int n;
{
  for (; n; n--)
    *lp++ = del;
  *lp = EOLTTR;
}

 /** これが実行されたらバグ。但し実行はそのまま続く */
void
BUGreport (n)
     int n;
{
  fprintf (stderr, "\r\nromkan-Bug%d!!\r\n", n);
}

 /** 前処理(mae_in→mae_out)又は後処理(ato_in→ato_out)を行う。*/
static void
maeato_henkan (in, outp, m_a_hyo)
     letter in;                 /* 入力の一文字 */
     letter *outp;              /* 出力はここに入る */
     int *m_a_hyo;              /* どの前・後処理表が選択されているかの情報 */
{
  struct dat *datptr;
  int i, hyoseq;
  letter *curdat;

  if (isSPCL (in))
    {
      /* LTREOFやCHMSIGが来うるので、特殊コードはそのまま返すように
         細工しておかないといけない。 */
      ltr_to_ltrseq (outp, in);
      return;
    }

  for (hyoseq = 0; (hyonum = m_a_hyo[hyoseq]) != -1; hyoseq++)
    {
      for (i = 0, datptr = hyo_n[hyonum].data; NULL != (curdat = datptr[i].code[0]); i++)
        {
          henmatch[0].ltrmch = EOLTTR;
          if (!l_eq (curdat, in))
            continue;

          ltrevlcpy (outp, datptr[i].code[1]);
          return;
        }
    }
  ltr_to_ltrseq (outp, in);
}

 /** 後処理 */
static void
codeout_chg ()
{
  letter *saishu_outp;

  *(saishu_outp = saishu_out) = EOLTTR;

  for (; *codeout != EOLTTR; codeout++)
    {
      maeato_henkan (*codeout, saishu_outp, useatohyo);
      totail (saishu_outp);
    }

  codeout = saishu_out;
}

 /** 一文字プッシュ・バック */
/* *INDENT-OFF* */
letter
romkan_ungetc (l)
     letter l;
/* *INDENT-ON* */
{
  return (ungetc_buf = l);
}

 /** romkan_nextに対し一文字プッシュ・バック */
/* *INDENT-OFF* */
letter
romkan_unnext (l)
     letter l;
/* *INDENT-ON* */
{
  return (unnext_buf = l);
}

 /**    deleteとして使うキャラクタの設定(偽deleteも)。これを実行後は
        romkan_clearを実行しておかないと混乱のもとになります。*/
/*      (廃止)
void
romkan_setdel(delchr_, nisedl_)
letter  delchr_, nisedl_;
{
        delchr = delchr_;
        nisedl = nisedl_;
}
*/