view Wnn/etc/msg.c @ 7:6ab41ec6f895

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

/*
 *  $Id: msg.c,v 1.13 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, 2002
 *
 * 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.
 */

/*
        struct msg_cat msg_open(name, nlspath, lang)
        char *name;
        char *nlspath;
        char *lang;

        char * msg_get(cd, id, s, lang)
        struct msg_cat cd;
        int id;
        char *s;

        void msg_close(cd)
        struct msg_cat cd;

        format of message file
            <message id>\t<message>
*/
#ifdef HAVE_CONFIG_H
#  include <config.h>
#endif

#include <stdio.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 */

#include "commonhd.h"
#include "wnn_os.h"
#include "msg.h"

static char *
getlang (lang)
     char *lang;
{
  static char tmp[32];
  char *p;
  int i;

  if (lang == NULL || *lang == '\0')
    {
#ifdef  HAS_SETLOCALE
      lang = setlocale (LC_ALL, NULL);
      if (lang == NULL || *lang == '\0')
#endif
        {
          lang = getenv ("LC_MESSAGES");
          if (lang == NULL || *lang == '\0')
            {
              lang = getenv ("LANG");
              if (lang == NULL || *lang == '\0')
                {
                  lang = DEF_LANG;
                }
            }
        }
    }
  for (i = 0, p = lang; *p && *p != '.'; i++, p++)
    {
      tmp[i] = *p;
    }
  tmp[i] = '\0';
  return (tmp);
  /*
     return(lang);
   */
}

static int
_search (id, bd)
    int *id;
    struct msg_bd *bd;
{
  return (*id - bd->msg_id);
}

static void
_escape (op, ip)
     register char *op, *ip;
{
  for (; *ip != 0; ip++, op++)
    {
      if (*ip == '\\')
        {
          switch (*++ip)
            {
            case 'n':
              *op = '\n';
              break;
            case 't':
              *op = '\t';
              break;
            case 'b':
              *op = '\b';
              break;
            case 'r':
              *op = '\r';
              break;
            case 'f':
              *op = '\f';
              break;
            case 'v':
              *op = '\v';
              break;
            case '0':
              *op = 0;
              break;
              /*
                 case 'a':
                 *op = '\a';
                 break;
                 case 'e':
                 case 'E':
                 case 'o':
                 case 'd':
                 case 'x':
                 break;
               */
            default:
              *op = *ip;
              break;
            }
        }
      else
        {
          if (*ip == '\n')
            {
              *op = '\0';
            }
          else
            {
              *op = *ip;
            }
        }
    }
  *op = 0;
}

static char *
get_msg_bd (cd, id)
     struct msg_cat *cd;
     int id;
{
  register struct msg_bd *bd;
  if (cd->msg_bd == 0 || cd->msg_cnt == 0)
    return (NULL);
  bd = (struct msg_bd *) bsearch (&id, cd->msg_bd, cd->msg_cnt, sizeof (struct msg_bd), _search);
  if (bd == NULL)
    return (NULL);
  return (bd->msg);
}

/* expand
        %N: the value of the name parameter passed to msg_open()
        %L: the value of LANG
        %l: the language element from LANG
        %t: the territory element from LANG
        %c: the codeset element from LANG
        %%: a single % charctor 
*/
static int
expand (op, ip, name, lang)
     register char *op, *ip, *name, *lang;
{
  if (!ip || !*ip)
    return (-1);
  for (; *ip != 0; ip++)
    {
      if (*ip == '%')
        {
          switch (*++ip)
            {
            case 'N':
              if (!name || !*name)
                return (-1);
              strcpy (op, name);
              op += strlen (name);
              break;
            case 'L':
              if (!lang || !*lang)
                return (-1);
              strcpy (op, lang);
              op += strlen (lang);
              break;
              /*
                 case 'l':
                 strcpy(op, language);
                 op += strlen(language);
                 break;
                 case 't':
                 strcpy(op, terr);
                 op += strlen(terr);
                 break;
                 case 'c':
                 strcpy(op, code);
                 op += strlen(code);
                 break;
                 case '%':
                 strcpy(op, "%");
                 op += strlen("%");
                 break;
               */
            default:
              break;
            }
        }
      else
        {
          *op = *ip;
          op++;
        }
    }
  *op = '\0';
  return (0);
}


struct msg_cat *
msg_open (name, nlspath, lang)
     char *name;
     char *nlspath;
     char *lang;
{
  struct msg_cat *cd;

  char fn[MAXPATHLEN];
  FILE *fp;
  char data[1024];
  char save[1024];
  int msg_cnt = 0;
  int msg_byte = 0;
  register char *dp;
  register struct msg_bd *bd;
  register char *msg, *l;

  l = getlang (lang);
  if (name && *name == '/')
    {
      strcpy (fn, name);
    }
  else
    {
      if (expand (fn, nlspath, name, l) == -1)
        {
          return (NULL);
        }
    }

  if (!(cd = (struct msg_cat *) malloc (sizeof (struct msg_cat))))
    return (NULL);

  strcpy (cd->name, name);
  strcpy (cd->lang, l);
  strcpy (cd->nlspath, nlspath);
  cd->nextp = NULL;
  cd->msg_cnt = 0;

  if ((fp = fopen (fn, "r")) == NULL)
    {
      /* message file not found */
      cd->msg_bd = 0;
      return (cd);
    }
  for (;;)
    {
      /* first: count bytes */
      if (fgets (data, 1024, fp) == NULL)
        break;
      if (*data == '#')
        continue;               /* comment */
      for (dp = data; *dp && *dp != '\t'; dp++);        /* msg_id:message\n */
      if (*dp == '\0')
        continue;
      dp++;
      msg_byte += strlen (dp);
      msg_cnt++;
    }
  rewind (fp);

  cd->msg_cnt = msg_cnt;
  if (!(bd = cd->msg_bd = (struct msg_bd *) malloc ((sizeof (struct msg_bd)) * msg_cnt + msg_byte + 1)))
    {
      fclose (fp);
      free (cd);
      return (NULL);
    }
  msg = (char *) bd + (sizeof (struct msg_bd)) * msg_cnt;

  for (;;)
    {
      /* second : get message */
      if (fgets (data, 1024, fp) == NULL)
        break;
      if (*data == '#')
        continue;               /* comment */
      for (dp = data; *dp && *dp != '\t'; dp++);        /* msg_id:message\n */
      if (*dp == '\0')
        continue;
      *dp = 0;
      dp++;
      bd->msg_id = atoi (data);
      bd->msg = msg;
      bd++;
      _escape (save, dp);
      strcpy (msg, save);
      msg += strlen (save);
      *msg = 0;
      msg++;
    }
  fclose (fp);
  return (cd);
}

char *
msg_get (catd, id, msg, lang)
     struct msg_cat *catd;
     int id;
     char *msg;
     register char *lang;
{
  register struct msg_cat *cd;
  register char *msg_bd;

  if (catd == 0)
    goto error;
  cd = catd;
  if (lang == 0 || *lang == '\0')
    {
      lang = cd->lang;
    }
  else
    {
      for (;; cd = cd->nextp)
        {
          if (strcmp (lang, cd->lang) == 0)
            break;
          if (cd->nextp == 0)
            {
              cd->nextp = msg_open (cd->name, cd->nlspath, lang);
              cd = cd->nextp;
              break;
            }
        }
    }

  if ((msg_bd = get_msg_bd (cd, id)))
    return (msg_bd);
error:
  if (msg != 0 && *msg != '\0')
    return (msg);
  {
    static char ret[128];
    sprintf (ret, "mes_id = %d: %s", id, DEF_MSG);
    return (ret);
  }
}

void
msg_close (cd)
     register struct msg_cat *cd;
{
  if (cd->nextp)
    msg_close (cd->nextp);
  if (cd->msg_bd)
    free (cd->msg_bd);
  if (cd)
    free (cd);
}

#ifdef  not_use
/* test */
main ()
{
  struct msg_cat *cd;

  cd = msg_open ("msg", "%L", "ja_JP");

  printf (msg_get (cd, 5, "message not found\n", "ja_JP"), 555);
  printf (msg_get (cd, 6, "message not found\n", "zh_CN"));
  printf (msg_get (cd, -1, "", "ja_JP"), 555);
  printf (msg_get (cd, 2, "message not found\n", "ja_JP"), "abc");
  printf (msg_get (cd, 100, "message not found\n", "zh_CN"), "abc");
}
#endif /* not_use */