diff Wnn/jserver/b_index.c @ 0:bbc77ca4def5

initial import
author Yoshiki Yazawa <yaz@cc.rim.or.jp>
date Thu, 13 Dec 2007 04:30:14 +0900
parents
children 790205f476c0
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/Wnn/jserver/b_index.c	Thu Dec 13 04:30:14 2007 +0900
@@ -0,0 +1,346 @@
+/*
+ * 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, 2002, 2003
+ *
+ * Maintainer:  FreeWnn Project   <freewnn@tomo.gr.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 of the License, 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 this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ */
+static char rcs_id[] = "$Id: b_index.c,v 1.8 2003/06/07 02:23:58 hiroo Exp $";
+
+#ifdef HAVE_CONFIG_H
+#  include <config.h>
+#endif
+
+#include <stdio.h>
+#if STDC_HEADERS
+#  include <stdlib.h>
+#else
+#  if HAVE_MALLOC_H
+#    include <malloc.h>
+#  endif
+#endif /* STDC_HEADERS */
+
+#include "commonhd.h"
+#include "de_header.h"
+#include "jdata.h"
+
+#ifdef CONVERT_by_STROKE
+/*******  Create B_dic index  *************************************/
+static int delete_b_node (struct JT *jt, w_char *yomi, int level, int p_node);
+static int creat_b_node (struct JT *jt, w_char *yomi, int level, int p_node);
+static int bnode_alloc (struct JT *jt);
+static int bnode_free (struct JT *jt, int k);
+
+static int free_bnode = -1;     /* Initially no free b_node */
+static int b_cnt;               /* Current member of b_node in b_nodes area */
+                                /* Note,  b_cnt < jt->bufsize_bnode */
+
+/*----------------------------------------------------------------+
+  SYNOPSYS: Create b_index for b_dic.
+  RETURN VALUE: success: b_cnt (current b_node); failure: -1
+ +----------------------------------------------------------------*/
+int
+create_b_index (struct JT *jt)
+{
+  int i;
+  int serial;
+  w_char *yomi;                 /* The inputed character string */
+
+  /* Make Space of b_nodes */
+  jt->bind = (struct b_node *) malloc (2 * jt->bufsize_ri1[0] * sizeof (struct b_node));
+
+  /*  If the bind size is determined in atod, we use the
+     following statement instead of the above  
+   */
+  if (jt->bind == NULL)
+    {
+      log_err ("error in creating head of b_index.");
+      return (-1);
+    }
+  /* Set the first one be a  dummy b_node */
+  jt->bind[0].pter_next = -1;
+  jt->bind[0].pter_son = -1;
+  jt->bind[0].pter = -1;
+
+/* Delete the following line when the bufsize of bnode is set at atod step */
+  jt->bufsize_bnode = 2 * jt->bufsize_ri1[0];
+
+  b_cnt = 0;
+
+        /** For each tuple in ri1[0] create b_nodes */
+  for (i = 0; i < jt->maxri1[0]; i++)
+    {
+      serial = (jt->ri1[0] + i)->pter;
+      yomi = KANJI_str (jt->ri2[serial].kanjipter + jt->kanji, 0);
+      b_index_add (jt, yomi, serial);
+    }
+  return (b_cnt);
+}
+
+/*----------------------------------------------------------------+
+  SYNOPSYS: Create a b_node for each character in a tuple.
+  RETURN VALUE:	success: 0;  failure: -1
+ +----------------------------------------------------------------*/
+int
+b_index_add (struct JT *jt, w_char *yomi, int serial)
+{
+  int k;
+  int p_node;                   /* Current b_node in No.  */
+
+  p_node = 0;
+  for (k = 0; k < Strlen (yomi); k++)
+    {
+      if ((p_node = creat_b_node (jt, yomi, k, p_node)) == -1)
+        {
+          log_err ("error in creating b_index.");
+          return (-1);
+        }
+    }
+  jt->bind[p_node].pter = serial;
+  return (0);
+}
+
+/*----------------------------------------------------------------+
+  SYNOPSYS:
+ +----------------------------------------------------------------*/
+void
+b_index_delete (struct JT *jt, int serial)
+{
+  w_char *yomi;
+  yomi = KANJI_str (jt->ri2[serial].kanjipter + jt->kanji, 0);
+  delete_b_node (jt, yomi, 0, 0);
+}
+
+/*----------------------------------------------------------------+
+  SYNOPSYS:
+  RETURN VALUE:	0  Having no son
+ 		1  Having some sons
+ +----------------------------------------------------------------*/
+static int
+delete_b_node (struct JT *jt, w_char *yomi, int level, int p_node)
+{
+  int tmp_node;
+  int buf_node1, buf_node2;
+  w_char *yo_kanji = NULL;
+
+  if (p_node == -1)
+    return (0);
+
+  buf_node1 = jt->bind[p_node].pter_son;        /*Initialize two tmp nodes */
+  buf_node2 = jt->bind[p_node].pter_son;
+
+  /* search if the node exists already */
+  while (buf_node2 != -1)
+    {
+      tmp_node = buf_node2;
+      while (jt->bind[tmp_node].pter == -1)
+        {
+          tmp_node = jt->bind[tmp_node].pter_son;
+        }
+      yo_kanji = KANJI_str (jt->ri2[jt->bind[tmp_node].pter].kanjipter + jt->kanji, 0);
+      if (yomi[level] > yo_kanji[level])
+        {
+          buf_node1 = buf_node2;
+          buf_node2 = jt->bind[buf_node2].pter_next;
+        }
+      else
+        break;
+    }
+  if (yo_kanji == NULL || yomi[level] != yo_kanji[level])
+    {
+      /* Error case */
+      log_err ("error on b_index.");
+      return (-1);
+    }
+
+  if (level == (Strlen (yomi) - 1))
+    {
+      jt->bind[buf_node2].pter = -1;    /* HERE HERE */
+      if (jt->bind[buf_node2].pter_son != -1)   /* having son */
+        return (1);
+    }
+  if (level < (Strlen (yomi) - 1))
+    {
+      if ((delete_b_node (jt, yomi, level + 1, buf_node2) != 0) || (jt->bind[buf_node2].pter != -1))
+        return (1);
+    }
+  /* 0 case: (bnode_free) below */
+  if (buf_node1 == buf_node2)
+    {                           /* only one b_node in this level */
+      bnode_free (jt, buf_node2);
+      return (0);
+    }
+  else
+    {
+      jt->bind[buf_node1].pter_next = jt->bind[buf_node2].pter_next;
+      bnode_free (jt, buf_node2);
+      return (1);
+    }
+}
+
+/*----------------------------------------------------------------+
+  SYNOPSIS: Create a son of the parent node p_node, when it does
+ 	 not exist.
+  PARAMETERS:	jt: Header of the current dic.
+		yomi: Cureent character in this level.
+		level: Level number.
+		p_node: cureent b_node.
+  RETURN VALUE:	new or existent node number whether created or existed.
+		failure: -1
+ +----------------------------------------------------------------*/
+static int
+creat_b_node (struct JT *jt, w_char *yomi, int level, int p_node)
+{
+  int new_node;
+  int buf_node1, buf_node2;
+  int tmp_node;
+  w_char *yo_kanji = NULL;
+
+  buf_node1 = jt->bind[p_node].pter_son;        /*Initialize two tmp nodes */
+  buf_node2 = jt->bind[p_node].pter_son;
+
+  /* search if the node exists already */
+  while (buf_node2 != -1)
+    {
+      tmp_node = buf_node2;
+      while (jt->bind[tmp_node].pter == -1)
+        {
+          tmp_node = jt->bind[tmp_node].pter_son;
+        }
+      yo_kanji = KANJI_str (jt->ri2[jt->bind[tmp_node].pter].kanjipter + jt->kanji, 0);
+
+      if (yomi[level] > yo_kanji[level])
+        {
+          buf_node1 = buf_node2;
+          buf_node2 = jt->bind[buf_node2].pter_next;
+        }
+      else
+        break;
+    }
+
+  if (buf_node1 == -1)
+    {                           /* the cur is the first */
+      if (buf_node2 == -1)
+        {
+          if ((new_node = bnode_alloc (jt)) == -1)
+            {
+              log_err ("error in reallocing area of b_index.");
+              return (-1);
+            }
+          jt->bind[p_node].pter_son = new_node;
+
+          jt->bind[new_node].pter_next = -1;
+          jt->bind[new_node].pter_son = -1;
+          jt->bind[new_node].pter = -1;
+        }
+      else
+        return (-1);            /* Error case.  Impossible case  */
+      return (new_node);
+
+    }
+  else if (buf_node2 == -1)
+    {                           /* insert to last */
+      new_node = bnode_alloc (jt);
+      jt->bind[buf_node1].pter_next = new_node;
+
+      jt->bind[new_node].pter_next = -1;
+      jt->bind[new_node].pter_son = -1;
+      jt->bind[new_node].pter = -1;
+      return (new_node);
+
+    }
+  else if (yo_kanji == NULL)
+    {
+      return (-1);
+    }
+  else if (yomi[level] == yo_kanji[level])      /* exist already */
+    return (buf_node2);
+
+  /* insert  in before tnp_node2 */
+  else if (yomi[level] < yo_kanji[level])
+    {
+      new_node = bnode_alloc (jt);
+      jt->bind[new_node].pter_next = buf_node2;
+      jt->bind[new_node].pter_son = -1;
+      jt->bind[new_node].pter = -1;
+
+
+      if (buf_node1 == buf_node2)       /*insert to first */
+        jt->bind[p_node].pter_son = new_node;
+      else
+        jt->bind[buf_node1].pter_next = new_node;       /*to middle */
+      return (new_node);
+    }
+  else
+    return (-1);
+}
+
+
+/*----------------------------------------------------------------+
+  SYNOPSYS: Get a b_node from free_bnode list if there is any.
+	Otherwise get a b_node sequencially by doing b_cnt++.
+	When b_cnt is greater than bufsize of b_node, rallocation
+	will be performed.
+  RETURN VALUE:	
+ +----------------------------------------------------------------*/
+static int
+bnode_alloc (struct JT *jt)
+{
+  int i;
+
+  if (free_bnode != -1)
+    {                           /* Use free b_nodes */
+      i = free_bnode;
+      free_bnode = jt->bind[free_bnode].pter_next;
+      return (i);
+    }
+  if (b_cnt++ >= jt->bufsize_bnode)     /* Use new  b_nodes */
+    if (rd_realloc_bind (jt) == NULL)	/* realloc jt->bind */
+      return (-1);
+  return (jt->bufsize_bnode = b_cnt);   /* Not re-alloc */
+}
+
+/*----------------------------------------------------------------+
+  SYNOPSIS: Free b_node[k] from jt.
+ +----------------------------------------------------------------*/
+static int
+bnode_free (struct JT *jt, int k)
+{
+  if (k <= 0 || k > jt->max_bnode)
+    {
+      log_err ("error: bnode_free()");
+      return (-1);
+    }
+  jt->bind[k].pter = -1;        /* Initial this node   */
+  jt->bind[k].pter_son = -1;
+  jt->bind[k].pter_next = -1;
+
+  if (k < jt->max_bnode)
+    {
+      jt->bind[k].pter_next = free_bnode;
+      free_bnode = k;
+    }
+  else
+    jt->max_bnode = --b_cnt;    /* case of k==b_cnt */
+  return (0);
+}
+#endif /* CONVERT_by_STROKE */