Mercurial > freewnn
view Wnn/jlib/js.c @ 18:e7e2aba67cb3
disabled build for cWnn and kWnn by default
author | Yoshiki Yazawa <yaz@cc.rim.or.jp> |
---|---|
date | Mon, 14 Apr 2008 17:33:53 +0900 |
parents | 1f16ab4b33e5 |
children | a7ccf412ba02 |
line wrap: on
line source
/* * $Id: js.c,v 1.16 2005/04/10 15:26:37 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, 2001, 2002, 2003 * * 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. */ /* Nihongo Henkan Library */ /* entry functions js_open_lang js_close js_change_current_jserver js_connect_lang js_disconnect js_isconnect js_param_get js_param_set js_access js_mkdir js_get_lang js_set_lang extern Variables int wnn_errorno; */ #if defined(HAVE_CONFIG_H) #include <config.h> #endif #include <stdio.h> #include <ctype.h> #include <errno.h> #include <signal.h> #if STDC_HEADERS # include <stdlib.h> # include <string.h> #else # if HAVE_MALLOC_H # include <malloc.h> # endif # if HAVE_STRINGS_H # include <strings.h> # endif #endif /* STDC_HEADERS */ #if HAVE_FCNTL_H # include <fcntl.h> #endif #include <pwd.h> #if HAVE_UNISTD_H # include <unistd.h> #endif #include <sys/stat.h> #include <sys/types.h> #include "jd_sock.h" #include "commonhd.h" #include "demcom.h" #include "wnn_config.h" #include "wnnerror.h" #include "jslib.h" #include "jh.h" #include "msg.h" #define JS /* For include ../etc/bdic.c */ #include "../etc/bdic.c" #include "../etc/pwd.c" /* defined in wnn_os.h & should be modified in the future */ /* #define bzero(adr,n) memset((adr),0,(n)) */ int wnn_errorno = 0; struct msg_cat *wnn_msg_cat = NULL; /* j Lib. */ static int current_sd; /** ソケットfd **/ static WNN_JSERVER_ID *current_js = NULL; /* Packet Buffers */ static unsigned char snd_buf[S_BUF_SIZ]; /** 送信 **/ static int sbp = 0; /** 送信バッファーポインター **/ static int rbc = -1; /** 受信バッファーポインター **/ #if defined(EAGAIN) # if defined(EWOULDBLOCK) # define ERRNO_CHECK(no) ((no) == EAGAIN || (no) == EWOULDBLOCK) # else /* defined(EWOULDBLOCK) */ # define ERRNO_CHECK(no) ((no) == EAGAIN) # endif /* defined(EWOULDBLOCK) */ #else /* defined(EAGAIN) */ # if defined(EWOULDBLOCK) # define ERRNO_CHECK(no) ((no) == EWOULDBLOCK) # else /* defined(EWOULDBLOCK) */ # define ERRNO_CHECK(no) (0) # endif /* defined(EWOULDBLOCK) */ #endif /* defined(EAGAIN) */ static void connect_timeout (); static int _get_server_name (); static int writen (); static char *get_unixdomain_of_serv_defs (), *get_service_of_serv_defs (); static int get_port_num_of_serv_defs (); #if DEBUG void xerror (); #endif /********* V4 *****************/ /*** jserver_dead Macro ***/ static jmp_buf current_jserver_dead; #define handler_of_jserver_dead(err_val) \ { \ if (current_js) { \ if(current_js->js_dead){wnn_errorno=WNN_JSERVER_DEAD;return err_val;}\ if(setjmp(current_jserver_dead)){ \ wnn_errorno=WNN_JSERVER_DEAD; \ return err_val; \ } \ wnn_errorno = 0; /* here initialize wnn_errorno; */ \ } \ } static void set_current_js (register WNN_JSERVER_ID *server) { current_js = server; current_sd = current_js->sd; } /** デーモンが死んだ時のための後始末 **/ static void daemon_dead (void) { current_js->js_dead = -1; wnn_errorno = WNN_JSERVER_DEAD; shutdown (current_sd, 2); #ifdef HAVE_CLOSESOCKET closesocket (current_sd); #else close (current_sd); #endif #if DEBUG fprintf (stderr, "jslib:JSERVER %s is Dead\n", current_js->js_name); #endif if (current_js->js_dead_env_flg) { longjmp (current_js->js_dead_env, 666); } longjmp (current_jserver_dead, 666); /* never reach */ } /** ソケットをオープンしてcurrent_sdにソケットfdを返す (cdというのはコミュニケーションデバイスの名残) **/ static int cd_open_un (register char *lang) { #ifdef AF_UNIX int sd; struct sockaddr_un saddr; /** ソケット **/ char *sock_name = NULL; saddr.sun_family = AF_UNIX; /* find socket name from table by lang */ if (lang && *lang) { if ((sock_name = get_unixdomain_of_serv_defs (lang)) == NULL) { sock_name = sockname; } } else { sock_name = sockname; /* Jserver */ } strcpy (saddr.sun_path, sock_name); if ((sd = socket (AF_UNIX, SOCK_STREAM, 0)) == ERROR) { #if DEBUG xerror ("jslib:Can't create unix domain socket.\n"); #endif return -1; } #if !defined(SUN_LEN) # define SUN_LEN(su) (sizeof(*(su)) - sizeof((su)->sun_path) + strlen((su)->sun_path)) #endif if (connect (sd, (struct sockaddr *) &saddr, SUN_LEN (&saddr)) == ERROR) { #if DEBUG xerror ("jslib:Can't connect unix domain socket.\n"); #endif close (sd); return -1; } return sd; #else return -1; #endif } static int cd_open_in (register char *server, register char *lang, register int timeout) { int sd; #ifdef INET6 struct addrinfo hints, *res, *res0; int error; char sport[6]; #else struct sockaddr_in saddr_in; /** ソケット **/ register struct hostent *hp; #endif struct servent *sp = NULL; int serverNO, port_num; int ret; char pserver[64]; char sserver[64]; char *serv_name = NULL; serverNO = _get_server_name (server, pserver); /* find service name from table by lang */ if (lang && *lang) { if ((serv_name = get_service_of_serv_defs (lang)) == NULL) { strcpy (sserver, SERVERNAME); } else { strcpy (sserver, serv_name); } } else { /* serv_name = SERVERNAME; */ strcpy (sserver, SERVERNAME); } /* if ((sp = getservbyname(serv_name,"tcp")) != NULL) { */ if ((sp = getservbyname (sserver, "tcp")) != NULL) { serverNO += ntohs (sp->s_port); } else { if ((port_num = get_port_num_of_serv_defs (lang)) == -1) { serverNO += WNN_PORT_IN; } else { serverNO += port_num; } } #ifdef INET6 memset(&hints, 0, sizeof(hints)); hints.ai_family = PF_UNSPEC; hints.ai_socktype = SOCK_STREAM; sprintf(sport, "%d", serverNO); error = getaddrinfo(pserver, sport, &hints, &res0); if (error) { #if DEBUG xerror (gai_strerror(error)); #endif return -1; } for (res = res0; res ; res = res->ai_next) { if (res->ai_family == AF_INET || res->ai_family == AF_INET6){ if ((sd = socket(res->ai_family, res->ai_socktype, res->ai_protocol)) != ERROR){ #else if ((hp = gethostbyname (pserver)) == NULL) { return -1; } bzero ((char *) &saddr_in, sizeof (saddr_in)); bcopy (hp->h_addr, (char *) &saddr_in.sin_addr, hp->h_length); saddr_in.sin_family = AF_INET; saddr_in.sin_port = htons (serverNO); if ((sd = socket (AF_INET, SOCK_STREAM, 0)) == ERROR) { #if DEBUG xerror ("jslib:Can't create inet socket.\n"); #endif return -1; } #endif if (timeout != 0 && timeout > 0) { signal (SIGALRM, connect_timeout); alarm (timeout); } #ifdef INET6 ret = connect (sd, res->ai_addr, res->ai_addrlen); #else ret = connect (sd, (struct sockaddr *) &saddr_in, sizeof (saddr_in)); #endif if (timeout != 0 && timeout > 0) { alarm (0); signal (SIGALRM, SIG_IGN); } if (ret == ERROR) { #if DEBUG #ifdef INET6 if (res->ai_family == AF_INET) xerror ("jslib:Can't connect inet socket.\n"); else if (res->ai_family == AF_INET6) xerror ("jslib:Can't connect inet6 socket.\n"); #else xerror ("jslib:Can't connect inet socket.\n"); #endif #endif #ifdef HAVE_CLOSESOCKET closesocket (sd); #else close (sd); #endif #ifdef INET6 sd = ERROR; } else break; } else { #if DEBUG if (res->ai_family == AF_INET) xerror ("jslib:Can't create inet socket.\n"); else if (res->ai_family == AF_INET6) xerror ("jslib:Can't create inet6 socket.\n"); #endif } } } freeaddrinfo(res0); if (sd == ERROR) { #endif return -1; } return sd; } static void connect_timeout () { } /* get server name and return serverNo */ static int _get_server_name (char *server, char *pserver) { register char *p; #ifdef INET6 int len; if (server[0] == '[') { p = strchr(server++, ']'); if (p) { len = p-server<64-1?p-server:64-1; strncpy(pserver, server, len); pserver[len] = '\0'; } else pserver[0] = '\0'; } else { #endif /* Workaround for pserver buffer overrun : Nov 11,1999 by T.Aono */ /* assumes pserver[64]. variable length string is not supported. */ strncpy(pserver, server, 64-1); pserver[64-1] = '\0'; p = pserver; #ifdef INET6 } #endif for (; *p && *p != ':'; p++); if (!*p) return (0); /* does not have a colon */ *p = 0; return (atoi (++p)); } /* Packet SND/RCV subroutines */ static void put4com (); /** パケットのヘッダーを送る **/ static void snd_head (int cmd) { sbp = 0; put4com (cmd); rbc = -1; } /** パケットのヘッダーを送る **/ static int snd_env_head (register struct wnn_env *env, int cmd) { snd_head (cmd); put4com (env->env_id); return 0; } /** パケットのヘッダーを送る **/ static int snd_server_head (register WNN_JSERVER_ID *server, int cmd) { snd_head (cmd); return 0; } /** 送信バッファをフラッシュする **/ static int snd_flush (void) { if (sbp == 0) return (-1); writen (sbp); sbp = 0; return (0); } static int writen (int n) { int cc, x; for (cc = 0; cc < n;) { errno = 0; #ifdef HAVE_SEND x = send (current_sd, &snd_buf[cc], n - cc, 0); #else x = write (current_sd, &snd_buf[cc], n - cc); #endif if (x < 0) { if (ERRNO_CHECK (errno) || errno == EINTR) { continue; } else { daemon_dead (); return -1; } } cc += x; } #if DEBUG fprintf (stderr, "jslib:writen=%d\n", x); /* dmp(snd_buf,x); */ #endif return (0); } /** サーバへ1バイト送る **/ static void put1com (int c) { snd_buf[sbp++] = c; if (sbp >= S_BUF_SIZ) { writen (S_BUF_SIZ); sbp = 0; } } /** サーバへ2バイト送る **/ static void put2com (int c) { put1com (c >> (8 * 1)); put1com (c); } /** サーバへ4バイト送る **/ static void put4com (int c) { put1com (c >> (8 * 3)); put1com (c >> (8 * 2)); put1com (c >> (8 * 1)); put1com (c); } /** サーバへ文字列を送る **/ static void putwscom (register w_char *p) { if (p == NULL) { put2com (0); return; } while (*p) put2com (*p++); put2com (0); } /** サーバへ文字列を送る **/ static void putscom (register char *p) { if (p == NULL) { put1com (0); return; } while (*p) put1com (*p++); put1com (0); } /** サーバから1バイト受ける **/ static int get1com (void) { static int rbp; static unsigned char rcv_buf[R_BUF_SIZ]; /** 受信 **/ if (rbc <= 0) { while (1) { errno = 0; #ifdef HAVE_RECV rbc = recv (current_sd, rcv_buf, R_BUF_SIZ, 0); #else rbc = read (current_sd, rcv_buf, R_BUF_SIZ); #endif if (rbc <= 0) { if (ERRNO_CHECK (errno)) { continue; } else if (rbc == 0) { daemon_dead (); return -1; } else { /* cc == -1 */ if (errno != EINTR) { daemon_dead (); return -1; } continue; } } rbp = 0; #if DEBUG fprintf (stderr, "jslib:read:rbc=%d\n", rbc); /* dmp(rcv_buf,rbc); */ #endif break; } } rbc--; return rcv_buf[rbp++] & 0xFF; } /** サーバから2バイト受ける **/ static int get2com (void) { register int h; h = get1com (); return (h << 8) | get1com (); } /** サーバから4バイト受ける **/ static int get4com (void) { register int h1, h2, h3; h1 = get1com () << 24; h2 = get1com () << 16; h3 = get1com () << 8; return h1 | h2 | h3 | get1com (); } /** サーバへ文字列を送る **/ static void getscom (register char *p) { while ((*p++ = get1com ())) ; } /** サーバへ文字列を送る **/ static void getwscom (w_char register *p) { while ((*p++ = get2com ())) ; } #ifdef nodef /* Moved from ../etc/string.c */ /** **/ static int Strlen (register w_char *s) { register int n; for (n = 0; *s++ != 0; n++); return n; } /** **/ static w_char * Strcpy (register w_char *s1, register w_char *s2) { register w_char *d; for (d = s1; (*d++ = *s2++) != 0;); return s1; } #endif /* Debug Subroutines */ #if DEBUG void xerror (char *s) { fprintf (stderr, "%s\n", s); } void dmp (p, c) char *p; { int i, j; for (i = 0;; i += 16) { for (j = 0; j < 16; j++) { if (c <= 0) { fprintf (stderr, "\n"); return; } fprintf (stderr, "%02x ", p[i + j] & 0xFF); c--; } fprintf (stderr, "\n"); } } #endif /* get login name form /etc/passwd file */ static char * getlogname (void) { struct passwd *getpwuid (); return getpwuid (getuid ())->pw_name; } /* * Lib. Functions * raw lib. */ /*** js ・global ***/ /** jserver と接続する。jserver_id を返す。 **/ WNN_JSERVER_ID * js_open_lang (register char *server, register char *lang, register int timeout) { char *new_js; char host[WNN_HOSTLEN], user[WNN_ENVNAME_LEN]; int x; if (wnn_msg_cat == NULL) { char nlspath[MAXPATHLEN]; strcpy (nlspath, LIBDIR); strcat (nlspath, "/%L/%N"); wnn_msg_cat = msg_open ("libwnn.msg", nlspath, lang); if (wnn_msg_cat == NULL) { fprintf (stderr, "libwnn: Cannot open message file for libwnn.a\n"); } } sbp = 0; /* init sndBufPointer */ if (!(new_js = (char *) malloc (sizeof (WNN_JSERVER_ID)))) { wnn_errorno = WNN_ALLOC_FAIL; return NULL; } current_js = (WNN_JSERVER_ID *) new_js; if (server == NULL) { current_js->js_name[0] = '\0'; } else { strncpy (current_js->js_name, server, sizeof (current_js->js_name) - 1); current_js->js_name[sizeof (current_js->js_name) - 1] = '\0'; } current_js->js_dead = 0; current_js->js_dead_env_flg = 0; /* if(user == NULL || 0==strcmp(user,"")) */ strncpy (user, getlogname (), WNN_ENVNAME_LEN); user[WNN_ENVNAME_LEN - 1] = '\0'; /* truncate by WNN_ENVNAME_LEN */ if (server == NULL || 0 == strcmp (server, "") || 0 == strcmp (server, "unix")) { strcpy (host, "unix"); if ((current_sd = cd_open_un (lang)) == -1) { wnn_errorno = WNN_SOCK_OPEN_FAIL; free ((char *) current_js); current_js = NULL; return NULL; } } else { gethostname (host, WNN_HOSTLEN); host[WNN_HOSTLEN - 1] = '\0'; /* truncate by WNN_HOSTLEN */ if ((current_sd = cd_open_in (server, lang, timeout)) == -1) { wnn_errorno = WNN_SOCK_OPEN_FAIL; free ((char *) current_js); current_js = NULL; return NULL; } } current_js->sd = current_sd; handler_of_jserver_dead (NULL); snd_head (JS_OPEN); put4com (JLIB_VERSION); /* H.T. */ putscom (host); putscom (user); snd_flush (); if (get4com () == -1) { x = wnn_errorno = get4com (); js_close (current_js); /* H.T. */ current_js = NULL; wnn_errorno = x; return NULL; } return current_js; } /** ソケットをクローズする **/ /** jserver との接続を close する。 **/ int js_close (WNN_JSERVER_ID *server) { register int x; WNN_JSERVER_ID tmp_js_id; if (server == 0) return (-1); tmp_js_id = *server; free ((char *) server); current_js = &tmp_js_id; current_sd = current_js->sd; /* handler of jserver dead */ handler_of_jserver_dead (-1); snd_head (JS_CLOSE); snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); #ifdef HAVE_CLOSESOCKET closesocket (current_sd); #else close (current_sd); #endif return x; } /* jserver との間に connection を張り、同時に jserver の内部に環 境を作る。env_name に既に存在する環境を指定した時にはその環境を 返し、NULL を指定した時には新しい環境を作って返す。 */ struct wnn_env * js_connect_lang (WNN_JSERVER_ID *server, register char *env_name, char *lang) { register int e_id; register struct wnn_env *env; void js_set_lang (); set_current_js (server); if (!(env = (struct wnn_env *) malloc (sizeof (struct wnn_env)))) { wnn_errorno = WNN_ALLOC_FAIL; return NULL; } handler_of_jserver_dead (NULL); snd_head (JS_CONNECT); putscom (env_name); snd_flush (); e_id = get4com (); if (e_id == -1) { wnn_errorno = get4com (); free (env); return NULL; } env->env_id = e_id; env->js_id = server; strcpy (env->lang, lang); /* set language name */ return env; } #ifdef nodef /* set language value to env */ void js_set_lang (struct wnn_env *env, register char *lang) { register char *p; extern char *getenv (); /* if not specified language , use $LANG */ if (lang == 0 || *lang == 0) lang = getenv ("LANG"); if (lang == 0 || *lang == 0) lang = WNN_DEFAULT_LANG; for (p = env->lang; *lang != '.' && *lang != '@' && *lang != 0; lang++, p++) *p = *lang; *p = 0; } #endif /* get language value from env */ char * js_get_lang (struct wnn_env *env) { return (env->lang); } int js_env_exist (register WNN_JSERVER_ID *server, register char *env_name) { set_current_js (server); handler_of_jserver_dead (-1); snd_head (JS_ENV_EXIST); putscom (env_name); snd_flush (); return (get4com ()); } int js_env_sticky (register struct wnn_env *env) { if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_ENV_STICKY); snd_flush (); return (get4com ()); } int js_env_un_sticky (register struct wnn_env *env) { if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_ENV_UN_STICKY); snd_flush (); return (get4com ()); } /** env で示される環境を無くす。 **/ int js_disconnect (register struct wnn_env *env) { register int x; struct wnn_env tmp_env; if (env == 0) return (-1); tmp_env = *env; /* 本来は、free しなきゃあかんのだけど、リソース管理が出来ないし、 まあ、8バイトだから、ゴミが残るけどいいだろう。 free((char *)env); */ set_current_js (env->js_id); handler_of_jserver_dead (0); snd_env_head (&tmp_env, JS_DISCONNECT); snd_flush (); x = get4com (); if (x == -1) { wnn_errorno = get4com (); } return x; } /** サーバとコネクトしているか **/ int js_isconnect (register struct wnn_env *env) { if (env && env->js_id) return (env->js_id->js_dead); return (-1); } /** env の 環境 との通信バッファを flush する。 **/ void js_flush (struct wnn_env *env) { /* no operation */ } /* Parameter set/get */ /** 変換 parameter を設定する。 **/ /** js_param_set **/ int js_param_set (env, para) struct wnn_env *env; register struct wnn_param *para; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_PARAM_SET); put4com (para->n); /* N(大)文節解析のN */ put4com (para->nsho); /* 大文節中の小文節の最大数 */ put4com (para->p1); /* 幹語の頻度のパラメータ */ put4com (para->p2); /* 小文節長のパラメータ */ put4com (para->p3); /* 幹語長のパラメータ */ put4com (para->p4); /* 今使ったよビットのパラメータ */ put4com (para->p5); /* 辞書のパラメータ */ put4com (para->p6); /* 小文節の評価値のパラメータ */ put4com (para->p7); /* 大文節長のパラメータ */ put4com (para->p8); /* 小文節数のパラメータ */ put4com (para->p9); /* 疑似品詞 数字の頻度 */ put4com (para->p10); /* 疑似品詞 カナの頻度 *//* CWNN 英数の頻度 */ put4com (para->p11); /* 疑似品詞 英数の頻度 *//* CWNN 記号の頻度 */ put4com (para->p12); /* 疑似品詞 記号の頻度 *//* CWNN 開括弧の頻度 */ put4com (para->p13); /* 疑似品詞 閉括弧の頻度 *//* CWNN 閉括弧の頻度 */ put4com (para->p14); /* 疑似品詞 付属語の頻度 *//* BWNN No of koho */ put4com (para->p15); /* 疑似品詞 開括弧の頻度 *//* CWNN Not used */ snd_flush (); x = get4com (); if (x == -1) { wnn_errorno = get4com (); return -1; } return 0; } /** js_param_get **/ /** env で示される環境の変換 parameter を取り出す。 **/ int js_param_get (env, para) struct wnn_env *env; register struct wnn_param *para; { if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_PARAM_GET); snd_flush (); if (get4com () == -1) { wnn_errorno = get4com (); return -1; } para->n = get4com (); /* N(大)文節解析のN */ para->nsho = get4com (); /* 大文節中の小文節の最大数 */ para->p1 = get4com (); /* 幹語の頻度のパラメータ */ para->p2 = get4com (); /* 小文節長のパラメータ */ para->p3 = get4com (); /* 幹語長のパラメータ */ para->p4 = get4com (); /* 今使ったよビットのパラメータ */ para->p5 = get4com (); /* 辞書のパラメータ */ para->p6 = get4com (); /* 小文節の評価値のパラメータ */ para->p7 = get4com (); /* 大文節長のパラメータ */ para->p8 = get4com (); /* 小文節数のパラメータ */ para->p9 = get4com (); /* 疑似品詞 数字の頻度 */ para->p10 = get4com (); /* 疑似品詞 カナの頻度 */ para->p11 = get4com (); /* 疑似品詞 英数の頻度 */ para->p12 = get4com (); /* 疑似品詞 記号の頻度 */ para->p13 = get4com (); /* 疑似品詞 閉括弧の頻度 */ para->p14 = get4com (); /* 疑似品詞 付属語の頻度 */ para->p15 = get4com (); /* 疑似品詞 開括弧の頻度 */ return 0; } /* global File Operation */ /** js_mkdir **/ int js_mkdir (env, path) struct wnn_env *env; char *path; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_MKDIR); putscom (path); snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } /** js_access **/ int js_access (env, path, amode) struct wnn_env *env; char *path; int amode; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_ACCESS); put4com (amode); putscom (path); snd_flush (); x = get4com (); return x; } /** js_file_list_all **/ static int rcv_file_list (); int js_file_list_all (WNN_JSERVER_ID *server, struct wnn_ret_buf *ret) { set_current_js (server); handler_of_jserver_dead (-1); snd_server_head (server, JS_FILE_LIST_ALL); snd_flush (); return rcv_file_list (ret); } /** js_file_list **/ int js_file_list (env, ret) struct wnn_env *env; struct wnn_ret_buf *ret; { if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_FILE_LIST); snd_flush (); return rcv_file_list (ret); } static void re_alloc (); static int rcv_file_list (ret) struct wnn_ret_buf *ret; { register int i, count; WNN_FILE_INFO_STRUCT *files; count = get4com (); re_alloc (ret, sizeof (WNN_FILE_INFO_STRUCT) * count); files = (WNN_FILE_INFO_STRUCT *) ret->buf; for (i = 0; i < count; i++) { files->fid = get4com (); files->localf = get4com (); files->ref_count = get4com (); files->type = get4com (); getscom (files->name); files++; } return count; } /** js_file_stat **/ int js_file_stat (struct wnn_env *env, char *path, WNN_FILE_STAT *s) { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_FILE_STAT); putscom (path); snd_flush (); x = get4com (); s->type = x; return x; } /** js_file_info **/ int js_file_info (struct wnn_env *env, int fid, register WNN_FILE_INFO_STRUCT *file) { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_FILE_INFO); put4com (fid); snd_flush (); file->fid = fid; x = get4com (); if (x == -1) { wnn_errorno = get4com (); return (-1); } getscom (file->name); file->localf = get4com (); file->ref_count = get4com (); file->type = get4com (); return 0; } /** js_file_loaded **/ int js_file_loaded (WNN_JSERVER_ID *server, char *path) { register int x; set_current_js (server); handler_of_jserver_dead (-1); snd_server_head (server, JS_FILE_LOADED); putscom (path); snd_flush (); x = get4com (); return x; } /** js_file_loaded_local **/ static int check_local_file (); static int file_loaded_local (); int js_file_loaded_local (WNN_JSERVER_ID *server, char *path) { int x; set_current_js (server); handler_of_jserver_dead (-1); if (check_local_file (path) == -1) return (-1); snd_server_head (server, JS_FILE_LOADED_LOCAL); x = file_loaded_local (path); if (x == -1) { return -1; } return x; } static int check_local_file (path) char *path; { register FILE *f; register int x; struct wnn_file_head fh; #ifdef WRITE_CHECK check_backup (path); #endif /* WRITE_CHECK */ f = fopen (path, "r"); if (f == NULL) { wnn_errorno = WNN_OPENF_ERR; return -1; } x = input_file_header (f, &fh); if (x == -1) { fclose (f); wnn_errorno = WNN_NOT_A_FILE; return -1; } if (check_inode (f, &fh) == -1) { change_file_uniq (&fh, path); #ifdef WRITE_CHECK fclose (f); f = fopen (path, "r"); if (f == NULL) { wnn_errorno = WNN_OPENF_ERR; return (-1); } #endif /* WRITE_CHECK */ if (check_inode (f, &fh) == -1) { fclose (f); wnn_errorno = WNN_INODE_CHECK_ERROR; return (-1); } } fclose (f); return 0; } static int file_loaded_local (path) char *path; { register int x, i; FILE *f; struct wnn_file_head fh; #ifdef WRITE_CHECK check_backup (path); #endif /* WRITE_CHECK */ f = fopen (path, "r"); if (f == NULL) { wnn_errorno = WNN_OPENF_ERR; return -1; } x = input_file_header (f, &fh); if (x == -1) { fclose (f); wnn_errorno = WNN_NOT_A_FILE; return -1; } put4com (fh.file_uniq.time); put4com (fh.file_uniq.dev); put4com (fh.file_uniq.inode); for (i = 0; i < WNN_HOSTLEN; i++) { put1com (fh.file_uniq.createhost[i]); } snd_flush (); x = get4com (); fclose (f); return x; } /** js_hindo_file_create **/ int js_hindo_file_create (struct wnn_env *env, int fid, char *fn, w_char *comment, char *hpasswd) { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_HINDO_FILE_CREATE); put4com (fid); putscom (fn); putwscom (comment); putscom (hpasswd); snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } /** js_dic_file_create **/ int js_dic_file_create (struct wnn_env *env, char *fn, int type, w_char *comment, char *passwd, char *hpasswd) { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_DIC_FILE_CREATE); putscom (fn); putwscom (comment); putscom (passwd); putscom (hpasswd); put4com (type); snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } /** js_file_discard **/ int js_file_discard (env, fid) struct wnn_env *env; int fid; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_FILE_DISCARD); put4com (fid); snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } /** js_file_read **/ int js_file_read (env, fn) struct wnn_env *env; char *fn; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_FILE_READ); putscom (fn); snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } /** js_file_write **/ int js_file_write (env, fid, fn) struct wnn_env *env; int fid; char *fn; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_FILE_WRITE); put4com (fid); putscom (fn); snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } /** js_file_receive **/ static int xget1com (); static void xput1com (); int js_file_receive (env, fid, fn) struct wnn_env *env; int fid; char *fn; { register int mode, x; char file_name[1024]; char buf[1024]; FILE *f; int n; struct wnn_file_head fh; int i; #ifdef WRITE_CHECK char *tmp, *backup = NULL, tmp_x; int tmp_err = 0; #endif /* WRITE_CHECK */ if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_FILE_RECEIVE); put4com (fid); snd_flush (); /**/ getscom (file_name); if (fn == NULL || strcmp (fn, "") == 0) { gethostname (buf, 1024); n = strlen (buf); buf[n] = C_LOCAL; buf[n + 1] = 0; if (strncmp (buf, file_name, n + 1) == 0) { fn = file_name + n + 1; } } #ifdef WRITE_CHECK check_backup (fn); #endif /* WRITE_CHECK */ if ((f = fopen (fn, "r")) == NULL) { /* New File */ fh.file_uniq.time = fh.file_uniq.dev = fh.file_uniq.inode = 0; } else { /* Old File Exists */ if (input_file_header (f, &fh) == -1) { wnn_errorno = WNN_NOT_A_FILE; fclose (f); put4com (-1); snd_flush (); sleep (1); /* enssure handshake */ return (-1); } fclose (f); } put4com (0); /* Ack */ put4com (fh.file_uniq.time); put4com (fh.file_uniq.dev); put4com (fh.file_uniq.inode); for (i = 0; i < WNN_HOSTLEN; i++) { put1com (fh.file_uniq.createhost[i]); } snd_flush (); if ((mode = get4com ()) == -1) { /* check stat */ wnn_errorno = get4com (); return -1; } else if (mode == 0) { return 0; /* need not saving */ } else if (mode == 1 || mode == 3) { /* mode == 3 means the file is a new one. */ #ifdef WRITE_CHECK backup = make_backup_file (fn); if ((tmp = make_tmp_file (fn, 0, &f)) == NULL) { delete_tmp_file (backup); #else /* WRITE_CHECK */ if ((f = fopen (fn, "w+")) == NULL) { #endif /* WRITE_CHECK */ wnn_errorno = WNN_FILE_WRITE_ERROR; put4com (-1); snd_flush (); sleep (1); /* enssure handshake */ return (-1); } } else if (mode == 2) { #ifdef WRITE_CHECK backup = make_backup_file (fn); if ((tmp = make_tmp_file (fn, 1, &f)) == NULL) { delete_tmp_file (backup); #else /* WRITE_CHECK */ if ((f = fopen (fn, "r+")) == NULL) { /* New File */ #endif /* WRITE_CHECK */ wnn_errorno = WNN_FILE_WRITE_ERROR; put4com (-1); snd_flush (); sleep (1); /* enssure handshake */ return (-1); } } put4com (0); snd_flush (); /* ACK */ for (;;) { if ((x = xget1com ()) == -1) break; /* EOF */ #ifdef WRITE_CHECK tmp_x = (char) x; if (fwrite (&tmp_x, sizeof (char), 1, f) == -1) tmp_err = 1; #else /* WRITE_CHECK */ fputc (x, f); #endif /* WRITE_CHECK */ } fclose (f); #ifdef WRITE_CHECK if (tmp_err == 0) { move_tmp_to_org (tmp, fn, 1); } else { delete_tmp_file (tmp); } delete_tmp_file (backup); #endif /* WRITE_CHECK */ x = get4com (); if (x == -1) wnn_errorno = get4com (); #ifdef WRITE_CHECK if (tmp_err) { wnn_errorno = WNN_FILE_WRITE_ERROR; return (-1); } #endif /* WRITE_CHECK */ return x; } static int xget1com () { register int x; if ((x = get1com ()) != 0xFF) return x; if (get1com () == 0xFF) return -1; /* EOF */ return 0xFF; } /** js_file_send **/ int js_file_send (env, fn) struct wnn_env *env; char *fn; { register int x; FILE *f; int n; char buf[1024], *b; register int cc, i; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); if (check_local_file (fn) == -1) return (-1); snd_env_head (env, JS_FILE_SEND); x = file_loaded_local (fn); if (x != -1) { /* file is already loaded */ if (get4com () == -1) { wnn_errorno = get4com (); return (-1); } return x; } x = get4com (); if (x == -1) { wnn_errorno = get4com (); return -1; } gethostname (buf, 1024); n = strlen (buf); buf[n] = C_LOCAL; strcpy (buf + n + 1, fn); putscom (buf); #ifdef WRITE_CHECK check_backup (fn); #endif /* WRITE_CHECK */ if ((f = fopen (fn, "r")) == NULL) { xput1com (-1); /* EOF */ return -1; } /* send contents of file */ for (;;) { cc = fread (buf, 1, 1024, f); if (cc <= 0) break; /* EOF */ for (b = buf, i = 0; i < cc; i++) { xput1com ((int) *b++ & 0xff); } } fclose (f); xput1com (-1); /* EOF */ snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } static void xput1com (d) int d; { if (d == -1) { put1com (0xFF); put1com (0xFF); return; /* EOF */ } put1com (d); if (d == 0xFF) { put1com (0x00); } } /*** Dic. Operation for Env. ***/ /** js_dic_add **/ int js_dic_add (env, fid, hfid, rev, jnice, rw, hrw, pw1, pw2) struct wnn_env *env; int fid, hfid, rev, jnice, rw, hrw; char *pw1, *pw2; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_DIC_ADD); put4com (fid); put4com (hfid); put4com (jnice); put4com (rw); put4com (hrw); putscom (pw1); putscom (pw2); put4com (rev); /* rev is to add it as reverse dict */ snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } /** js_dic_delete **/ int js_dic_delete (env, dicno) struct wnn_env *env; int dicno; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_DIC_DELETE); put4com (dicno); snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } /** js_dic_use **/ int js_dic_use (env, dic_no, flag) struct wnn_env *env; int dic_no, flag; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_DIC_USE); put4com (dic_no); put4com (flag); snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } /** js_fuzokugo_set **/ int js_fuzokugo_set (env, fid) struct wnn_env *env; int fid; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_FUZOKUGO_SET); put4com (fid); snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } /** js_fuzokugo_get **/ int js_fuzokugo_get (env) struct wnn_env *env; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_FUZOKUGO_GET); snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } /** js_dic_list_all **/ static int rcv_dic_list (); static void get_dic_info (); int js_dic_list_all (WNN_JSERVER_ID *server, struct wnn_ret_buf *ret) { set_current_js (server); handler_of_jserver_dead (-1); snd_server_head (server, JS_DIC_LIST_ALL); snd_flush (); return rcv_dic_list (ret); } /** js_dic_list **/ int js_dic_list (env, ret) struct wnn_env *env; struct wnn_ret_buf *ret; { if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_DIC_LIST); snd_flush (); return rcv_dic_list (ret); } static int rcv_dic_list (ret) struct wnn_ret_buf *ret; { register int i, count; register WNN_DIC_INFO *dic; count = get4com (); re_alloc (ret, sizeof (WNN_DIC_INFO) * (count + 1)); dic = (WNN_DIC_INFO *) ret->buf; for (i = 0; i < count; i++) { get_dic_info (dic); dic++; } dic->dic_no = -1; return count; } static void get_dic_info (register WNN_DIC_INFO *dic) { dic->dic_no = get4com (); /* dic_No */ dic->body = get4com (); /* body fid */ dic->hindo = get4com (); /* hindo fid */ dic->rw = get4com (); /* r/w */ dic->hindo_rw = get4com (); /* hindo r/w */ dic->enablef = get4com (); /* enable/disable */ dic->nice = get4com (); /* nice */ dic->rev = get4com (); /* added H.T */ getwscom (dic->comment); getscom (dic->fname); getscom (dic->hfname); getscom (dic->passwd); getscom (dic->hpasswd); dic->type = get4com (); dic->gosuu = get4com (); dic->localf = get4com (); dic->hlocalf = get4com (); } /*** Dic. Operation by dic_No. ***/ /** js_word_add **/ int js_word_add (struct wnn_env *env, int dic_no, w_char *yomi, w_char *kanji, w_char *comment, int hinshi, int init_hindo) { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_WORD_ADD); put4com (dic_no); putwscom (yomi); putwscom (kanji); putwscom (comment); put4com (hinshi); put4com (init_hindo); snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } /** js_word_delete **/ int js_word_delete (env, dic_no, entry) struct wnn_env *env; int dic_no; int entry; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_WORD_DELETE); put4com (dic_no); put4com (entry); snd_flush (); x = get4com (); if (x == -1) wnn_errorno = get4com (); return x; } /** js_word_search **/ static int rcv_word_data (); int js_word_search (struct wnn_env *env, int dic_no, w_char *yomi, struct wnn_ret_buf *ret) { if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_WORD_SEARCH); put4com (dic_no); putwscom (yomi); snd_flush (); return (rcv_word_data (ret, yomi)); } /** js_word_search_by_env **/ int js_word_search_by_env (struct wnn_env *env, w_char *yomi, struct wnn_ret_buf *ret) { if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_WORD_SEARCH_BY_ENV); putwscom (yomi); snd_flush (); return (rcv_word_data (ret, yomi)); } /** js_word_info **/ int js_word_info (env, dic_no, entry, ret) struct wnn_env *env; int dic_no, entry; struct wnn_ret_buf *ret; { register int x; w_char yomi[LENGTHYOMI]; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_WORD_INFO); put4com (dic_no); put4com (entry); snd_flush (); x = get4com (); if (x == -1) { wnn_errorno = get4com (); return (-1); } getwscom (yomi); rcv_word_data (ret, yomi); return (0); } int js_word_comment_set (struct wnn_env *env, int dic_no, int entry, w_char *comment) { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_WORD_COMMENT_SET); put4com (dic_no); put4com (entry); putwscom (comment); snd_flush (); x = get4com (); if (x == -1) { wnn_errorno = get4com (); return (-1); } return (0); } /** rcv for word_search **/ static int rcv_word_data (struct wnn_ret_buf *ret, w_char *yomi) /* Yomi is not sent from server (at least for the time being). */ { register int x, j_c, k_c; w_char *k; register struct wnn_jdata *jd; register int cnt; j_c = get4com (); k_c = get4com (); re_alloc (ret, sizeof (struct wnn_jdata) * (j_c + 1) + sizeof (w_char) * (k_c + j_c * 3 + j_c * wnn_Strlen (yomi))); jd = (struct wnn_jdata *) ret->buf; for (cnt = 0;; cnt++) { jd->dic_no = x = get4com (); if (x == -1) break; jd->serial = get4com (); jd->hinshi = get4com (); jd->hindo = get4com (); jd->ima = get4com (); jd->int_hindo = get4com (); jd->int_ima = get4com (); jd++; } jd++; k = (w_char *) jd; jd = (struct wnn_jdata *) ret->buf; for (;;) { if (jd->dic_no == -1) break; jd->yomi = k; /* Copy Yomi */ wnn_Strcpy (k, yomi); k += wnn_Strlen (k) + 1; jd->kanji = k; /* Get Kanji */ getwscom (k); k += wnn_Strlen (k) + 1; jd->com = k; /* Get Comment */ getwscom (k); k += wnn_Strlen (k) + 1; jd++; } return cnt; } /** js_dic_info **/ int js_dic_info (struct wnn_env *env, int dic_no, register WNN_DIC_INFO *ret) { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_DIC_INFO); put4com (dic_no); snd_flush (); x = get4com (); if (x == -1) { return get4com (); } get_dic_info (ret); return dic_no; } /** js_who **/ int js_who (WNN_JSERVER_ID *server, struct wnn_ret_buf *ret) { register int i, j, c; WNN_JWHO *w; set_current_js (server); handler_of_jserver_dead (-1); snd_server_head (server, JS_WHO); snd_flush (); c = get4com (); if (c == -1) { wnn_errorno = get4com (); return -1; } re_alloc (ret, sizeof (WNN_JWHO) * c); w = (WNN_JWHO *) ret->buf; for (i = 0; i < c; i++) { w->sd = get4com (); getscom (w->user_name); getscom (w->host_name); for (j = 0; j < WNN_MAX_ENV_OF_A_CLIENT; j++) { (w->env)[j] = get4com (); } w++; } return (c); } /** jserver 中の全ての環境に関する情報を得る。 (ウラ技) **/ int js_env_list (WNN_JSERVER_ID *server, struct wnn_ret_buf *ret) { register int i, j, c; WNN_ENV_INFO *w; set_current_js (server); handler_of_jserver_dead (-1); snd_server_head (server, JS_ENV_LIST); snd_flush (); c = get4com (); if (c == -1) { wnn_errorno = get4com (); return -1; } re_alloc (ret, sizeof (WNN_ENV_INFO) * c); w = (WNN_ENV_INFO *) ret->buf; for (i = 0; i < c; i++) { w->env_id = get4com (); getscom (w->env_name); w->ref_count = get4com (); w->fzk_fid = get4com (); w->jishomax = get4com (); for (j = 0; j < WNN_MAX_JISHO_OF_AN_ENV; j++) { (w->jisho)[j] = get4com (); } for (j = 0; j < WNN_MAX_FILE_OF_AN_ENV; j++) { (w->file)[j] = get4com (); } w++; } return (c); } /**** ****/ int js_hindo_set (env, dic, entry, ima, hindo) struct wnn_env *env; int dic, entry, ima, hindo; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_HINDO_SET); put4com (dic); put4com (entry); put4com (ima); put4com (hindo); snd_flush (); if ((x = get4com ()) == -1) { wnn_errorno = get4com (); return -1; } return x; } /**** Henkan ****/ static void put_fzk_vec (int hinsi, w_char *fzk, int vec, int vec1) { put4com (hinsi); putwscom (fzk); put4com (vec); put4com (vec1); } /** kanren **/ static int rcv_dai (); static void rcv_sho_x (); static void rcv_sho_kanji (); int js_kanren (struct wnn_env *env, w_char *yomi, int hinsi, w_char *fzk, int vec, int vec1, int vec2, struct wnn_ret_buf *rb) { if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_KANREN); putwscom (yomi); put_fzk_vec (hinsi, fzk, vec, vec1); put4com (vec2); snd_flush (); return rcv_dai (rb); } /* rcv dai */ static int rcv_dai (ret) struct wnn_ret_buf *ret; { int dai_cnt, sho_sum, kanji_sum, d_size, s_size, k_size, x; register int i; struct wnn_dai_bunsetsu *dai_list; register struct wnn_dai_bunsetsu *dp; struct wnn_sho_bunsetsu *sho_list; register struct wnn_sho_bunsetsu *sp; w_char *kanji, *kp; dai_cnt = get4com (); if (dai_cnt == -1) { /* error dayo */ wnn_errorno = get4com (); return -1; } sho_sum = get4com (); kanji_sum = get4com (); d_size = sizeof (struct wnn_dai_bunsetsu) * dai_cnt; s_size = sizeof (struct wnn_sho_bunsetsu) * sho_sum; k_size = sizeof (w_char) * kanji_sum; /* re_alloc(ret, d_size+s_size+k_size); Seems This cause Bug?? H.T.*/ re_alloc (ret, d_size + s_size + k_size); dai_list = (struct wnn_dai_bunsetsu *) ret->buf; sho_list = (struct wnn_sho_bunsetsu *) ((char *) ret->buf + d_size); kanji = (w_char *) ((char *) ret->buf + d_size + s_size); for (dp = dai_list, i = 0; i < dai_cnt; i++) { dp->end = get4com (); dp->start = get4com (); dp->sbncnt = get4com (); dp->hyoka = get4com (); dp++; } for (dp = dai_list, sp = sho_list, i = 0; i < dai_cnt; i++) { dp->sbn = sp; x = dp->sbncnt; rcv_sho_x (sp, x); sp += x; dp++; } for (dp = dai_list, kp = kanji, i = 0; i < dai_cnt; i++) { rcv_sho_kanji (dp->sbn, dp->sbncnt, &kp); dp++; } return dai_cnt; } /* rcv sho routines */ static void rcv_sho_x (sho_list, sho_cnt) register struct wnn_sho_bunsetsu *sho_list; int sho_cnt; { register int i; for (i = 0; i < sho_cnt; i++) { sho_list->end = get4com (); sho_list->start = get4com (); sho_list->jiriend = get4com (); sho_list->dic_no = get4com (); sho_list->entry = get4com (); sho_list->hindo = get4com (); sho_list->ima = get4com (); sho_list->hinsi = get4com (); sho_list->status = get4com (); sho_list->status_bkwd = get4com (); sho_list->kangovect = get4com (); sho_list->hyoka = get4com (); sho_list++; } } static void rcv_sho_kanji (sho_list, sho_cnt, kanji) struct wnn_sho_bunsetsu *sho_list; int sho_cnt; w_char **kanji; { register w_char *k; register int i, x; k = *kanji; for (i = 0; i < sho_cnt; i++) { sho_list->kanji = k; getwscom (k); x = wnn_Strlen (k); k += x + 1; sho_list->yomi = k; getwscom (k); x = wnn_Strlen (k); k += x + 1; sho_list->fuzoku = k; getwscom (k); x = wnn_Strlen (k); k += x + 1; sho_list++; } *kanji = k; } static int rcv_sho (ret) struct wnn_ret_buf *ret; { register int sho_sum, kanji_sum, s_size, k_size; struct wnn_sho_bunsetsu *sho_list; w_char *kanji, *kp; sho_sum = get4com (); if (sho_sum == -1) { /* error dayo */ wnn_errorno = get4com (); return -1; } kanji_sum = get4com (); s_size = sizeof (struct wnn_sho_bunsetsu) * sho_sum; k_size = sizeof (w_char) * kanji_sum; re_alloc (ret, s_size + k_size); sho_list = (struct wnn_sho_bunsetsu *) ((char *) ret->buf); kanji = (w_char *) ((char *) ret->buf + s_size); rcv_sho_x (sho_list, sho_sum); kp = kanji; rcv_sho_kanji (sho_list, sho_sum, &kp); return sho_sum; } /** kantan **/ int js_kantan_dai (struct wnn_env *env, w_char *yomi, int hinsi, w_char *fzk, int vec, int vec1, struct wnn_ret_buf *rb) { if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_KANTAN_DAI); putwscom (yomi); put_fzk_vec (hinsi, fzk, vec, vec1); snd_flush (); return rcv_dai (rb); } int js_kantan_sho (struct wnn_env *env, w_char *yomi, int hinsi, w_char *fzk, int vec, int vec1, struct wnn_ret_buf *rb) { int sbncnt; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_KANTAN_SHO); putwscom (yomi); put_fzk_vec (hinsi, fzk, vec, vec1); snd_flush (); sbncnt = rcv_sho (rb); return sbncnt; } /** kanzen **/ int js_kanzen_dai (struct wnn_env *env, w_char *yomi, int hinsi, w_char *fzk, int vec, int vec1, struct wnn_ret_buf *rb) { if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_KANZEN_DAI); putwscom (yomi); put_fzk_vec (hinsi, fzk, vec, vec1); snd_flush (); return rcv_dai (rb); } int js_kanzen_sho (struct wnn_env *env, w_char *yomi, int hinsi, w_char *fzk, int vec, int vec1, struct wnn_ret_buf *rb) { int sbncnt; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_KANZEN_SHO); putwscom (yomi); put_fzk_vec (hinsi, fzk, vec, vec1); snd_flush (); sbncnt = rcv_sho (rb); return sbncnt; } /** js_version **/ int js_version (WNN_JSERVER_ID *server, int *serv, int *libv) { set_current_js (server); handler_of_jserver_dead (-1); snd_server_head (server, JS_VERSION); snd_flush (); *libv = JLIB_VERSION; return *serv = get4com (); } static void re_alloc (register struct wnn_ret_buf *ret, int size) { if (ret->size < size) { if (ret->buf) free ((char *) ret->buf); ret->buf = malloc (size); ret->size = size; } } int js_kill (WNN_JSERVER_ID *server) { int x; set_current_js (server); handler_of_jserver_dead (-1); snd_server_head (server, JS_KILL); snd_flush (); x = get4com (); return (x); } int js_file_remove (WNN_JSERVER_ID *server, char *n, char *pwd) { register int x; set_current_js (server); handler_of_jserver_dead (-1); snd_server_head (server, JS_FILE_REMOVE); putscom (n); putscom (pwd); snd_flush (); if ((x = get4com ()) == -1) { wnn_errorno = get4com (); return -1; } return (x); } int js_file_remove_client (WNN_JSERVER_ID *server, char *n, char *pwd) { struct wnn_file_head fh; register FILE *fp; set_current_js (server); handler_of_jserver_dead (-1); if (js_file_loaded_local (server, n) != -1) { wnn_errorno = WNN_FILE_IN_USE; return (-1); } #ifdef WRITE_CHECK check_backup (n); #endif /* WRITE_CHECK */ if ((fp = fopen (n, "r")) == NULL) { wnn_errorno = WNN_FILE_READ_ERROR; return (-1); } if (input_file_header (fp, &fh) == -1) { fclose (fp); wnn_errorno = WNN_NOT_A_FILE; return (-1); } fclose (fp); if (!check_pwd (pwd, fh.file_passwd)) { wnn_errorno = WNN_INCORRECT_PASSWD; return (-1); } if (unlink (n) == -1) { wnn_errorno = WNN_UNLINK; return (-1); } return (0); } /** js_dic_file_create_client **/ int js_dic_file_create_client (struct wnn_env *env, char *fn, int type, w_char *com, char *passwd, char *hpasswd) { int x; if (type != WNN_REV_DICT && type != CWNN_REV_DICT && type != BWNN_REV_DICT && type != WNN_UD_DICT) { wnn_errorno = WNN_NOT_A_UD; return (-1); } x = create_null_dic (fn, com, passwd, hpasswd, type); if (x == -1) { wnn_errorno = WNN_FILE_CREATE_ERROR; return (-1); } return (0); } /** js_hindo_file_create_client **/ int js_hindo_file_create_client (struct wnn_env *env, int fid, char *fn, w_char *com, char *hpasswd) { register int x; struct wnn_file_uniq funiq; int serial; register int i; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_HINDO_FILE_CREATE_CLIENT); put4com (fid); snd_flush (); x = get4com (); if (x == -1) { wnn_errorno = get4com (); return (-1); } serial = get4com (); funiq.time = get4com (); funiq.dev = get4com (); funiq.inode = get4com (); for (i = 0; i < WNN_HOSTLEN; i++) { funiq.createhost[i] = get1com (); } /* getscom(funiq.createhost); */ if (create_hindo_file (&funiq, fn, com, hpasswd, serial) == -1) { wnn_errorno = WNN_FILE_CREATE_ERROR; return (-1); } return (0); } int js_file_comment_set (struct wnn_env *env, int fid, w_char *comment) { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_FILE_COMMENT_SET); put4com (fid); putwscom (comment); snd_flush (); x = get4com (); if (x == -1) { wnn_errorno = get4com (); return (-1); } return (0); } /* * Hinsi Primitives */ int js_hinsi_name (WNN_JSERVER_ID *server, int no, struct wnn_ret_buf *rb) { register int size; set_current_js (server); handler_of_jserver_dead (-1); snd_server_head (server, JS_HINSI_NAME); put4com (no); snd_flush (); if ((size = get4com ()) == -1) { wnn_errorno = get4com (); return (-1); } re_alloc (rb, sizeof (w_char) * (size + 1)); getwscom ((w_char *) rb->buf); return (0); } int js_hinsi_number (WNN_JSERVER_ID *server, w_char *name) { register int no; set_current_js (server); handler_of_jserver_dead (-1); snd_server_head (server, JS_HINSI_NUMBER); putwscom (name); snd_flush (); if ((no = get4com ()) == -1) { wnn_errorno = get4com (); return (-1); } return (no); } int js_hinsi_list (struct wnn_env *env, int dic_no, w_char *name, struct wnn_ret_buf *rb) { int size; int count; register w_char *s; register w_char **r; register int k; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_HINSI_LIST); put4com (dic_no); putwscom (name); snd_flush (); if ((count = get4com ()) == -1) { wnn_errorno = get4com (); return (-1); } size = get4com (); re_alloc (rb, sizeof (w_char) * (size + 1) + count * sizeof (w_char *)); r = (w_char **) rb->buf; s = (w_char *) ((w_char **) rb->buf + count); for (k = 0; k < count; k++) { *r++ = s; getwscom (s); s += wnn_Strlen (s) + 1; } return (count); } int js_hinsi_dicts (env, no, rb) struct wnn_env *env; int no; struct wnn_ret_buf *rb; { register int k, count; int *p; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_HINSI_DICTS); put4com (no); snd_flush (); if ((count = get4com ()) == -1) { wnn_errorno = get4com (); return (-1); } re_alloc (rb, sizeof (int) * (count + 1)); p = (int *) rb->buf; for (k = 0; k < count; k++) { *p++ = get4com (); } return (count); } char *wnn_dic_types[] = { "", "固定", "登録", "逆引", "正規" }; char *cwnn_dic_types[] = { "", "耕協", "鞠村", "憧咄", "屎号" }; char *bwnn_dic_types[] = { "", "耕協", "鞠村", "永侏", "屎号" }; char *kwnn_dic_types[] = { "", "壱舛", "去系", "蝕遂", "舛鋭" }; /* 由鑷 竒巵 羹齎 閨豫 */ /* New primitives 9/8 */ int js_file_password_set (env, fid, which, old, new) struct wnn_env *env; int fid; int which; /* 1 for dic, 2 for hindo 3(0) for both */ char *old, *new; { register int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_FILE_PASSWORD_SET); put4com (fid); put4com (which); putscom (old); putscom (new); snd_flush (); x = get4com (); if (x == -1) { wnn_errorno = get4com (); return (-1); } return (0); } int js_hinsi_table_set (struct wnn_env *env, int dic_no, w_char *hinsi_table) { int x; if (env == 0) return (-1); set_current_js (env->js_id); handler_of_jserver_dead (-1); snd_env_head (env, JS_HINSI_TABLE_SET); put4com (dic_no); putwscom (hinsi_table); snd_flush (); x = get4com (); if (x == -1) { wnn_errorno = get4com (); return (-1); } return (0); } #define SERVERDEFS_FILE "/serverdefs" #define MACHINE_NAME 1 #define UNIXDOMAIN_NAME 2 #define SERVICE_NAME 3 #define SERVICE_PORT_NUM 4 static char * get_serv_defs (lang, cnt) char *lang; int cnt; { FILE *fp; static char s[7][EXPAND_PATH_LENGTH]; char serv_defs[EXPAND_PATH_LENGTH]; char data[1024]; int num; strcpy (serv_defs, LIBDIR); strcat (serv_defs, SERVERDEFS_FILE); if ((fp = fopen (serv_defs, "r")) == NULL) { return (NULL); } while (fgets (data, 1024, fp) != NULL) { num = sscanf (data, "%s %s %s %s %s %s %s", s[0], s[1], s[2], s[3], s[4], s[5], s[6]); if ((num < 4) || s[0][0] == ';') { continue; } if (!strncmp (lang, s[0], strlen (s[0]))) { fclose (fp); if (cnt == SERVICE_PORT_NUM && num < 5) return (NULL); if (strlen (s[cnt]) == strlen ("NULL") && !strcmp (s[cnt], "NULL")) { return (NULL); } else { return (s[cnt]); } } } fclose (fp); return (NULL); } char * _wnn_get_machine_of_serv_defs (lang) char *lang; { return (get_serv_defs (lang, MACHINE_NAME)); } static char * get_unixdomain_of_serv_defs (lang) char *lang; { return (get_serv_defs (lang, UNIXDOMAIN_NAME)); } static char * get_service_of_serv_defs (lang) char *lang; { return (get_serv_defs (lang, SERVICE_NAME)); } static int get_port_num_of_serv_defs (lang) char *lang; { char *port_char; if ((port_char = get_serv_defs (lang, SERVICE_PORT_NUM)) == NULL) { return (-1); } else { return (atoi (port_char)); } }