Blame glib/glib-mirroring-tab/gen-mirroring-tab.c

Packit ae235b
/* gen-mirroring-tab.c - generate gmirroringtable.h for glib
Packit ae235b
 * copied from FriBidi.
Packit ae235b
 *
Packit ae235b
 * $Id$
Packit ae235b
 * $Author$
Packit ae235b
 * $Date$
Packit ae235b
 * $Revision$
Packit ae235b
 * $Source$
Packit ae235b
 *
Packit ae235b
 * Author:
Packit ae235b
 *   Behdad Esfahbod, 2001, 2002, 2004
Packit ae235b
 *
Packit ae235b
 * Copyright (C) 2004 Sharif FarsiWeb, Inc
Packit ae235b
 * Copyright (C) 2001,2002,2004 Behdad Esfahbod
Packit ae235b
 * 
Packit ae235b
 * This library is free software; you can redistribute it and/or
Packit ae235b
 * modify it under the terms of the GNU Lesser General Public
Packit ae235b
 * License as published by the Free Software Foundation; either
Packit ae235b
 * version 2.1 of the License, or (at your option) any later version.
Packit ae235b
 * 
Packit ae235b
 * This library is distributed in the hope that it will be useful,
Packit ae235b
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit ae235b
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit ae235b
 * Lesser General Public License for more details.
Packit ae235b
 * 
Packit ae235b
 * You should have received a copy of the GNU Lesser General Public License
Packit ae235b
 * along with this library; if not, see <http://www.gnu.org/licenses/>.
Packit ae235b
 * 
Packit ae235b
 * For licensing issues, contact <license@farsiweb.info>.
Packit ae235b
 */
Packit ae235b
Packit ae235b
#include <glib.h>
Packit ae235b
Packit ae235b
#include <stdlib.h>
Packit ae235b
#include <stdio.h>
Packit ae235b
Packit ae235b
#include "packtab.h"
Packit ae235b
Packit ae235b
#define appname "gen-mirroring-tab"
Packit ae235b
#define outputname "gmirroringtable.h"
Packit ae235b
Packit ae235b
static void
Packit ae235b
die (
Packit ae235b
  const char *msg
Packit ae235b
)
Packit ae235b
{
Packit ae235b
  fprintf (stderr, appname ": %s\n", msg);
Packit ae235b
  exit (1);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
die2 (
Packit ae235b
  const char *fmt,
Packit ae235b
  const char *p
Packit ae235b
)
Packit ae235b
{
Packit ae235b
  fprintf (stderr, appname ": ");
Packit ae235b
  fprintf (stderr, fmt, p);
Packit ae235b
  fprintf (stderr, "\n");
Packit ae235b
  exit (1);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
die4 (
Packit ae235b
  const char *fmt,
Packit ae235b
  unsigned long l,
Packit ae235b
  unsigned long p,
Packit ae235b
  unsigned long q
Packit ae235b
)
Packit ae235b
{
Packit ae235b
  fprintf (stderr, appname ": ");
Packit ae235b
  fprintf (stderr, fmt, l, p, q);
Packit ae235b
  fprintf (stderr, "\n");
Packit ae235b
  exit (1);
Packit ae235b
}
Packit ae235b
Packit ae235b
#define table_name "Mir"
Packit ae235b
#define macro_name "GLIB_GET_MIRRORING"
Packit ae235b
Packit ae235b
#define UNICODE_CHARS 0x110000
Packit ae235b
Packit ae235b
static signed int table[UNICODE_CHARS];
Packit ae235b
static char buf[4000];
Packit ae235b
static signed long max_dist;
Packit ae235b
Packit ae235b
static void
Packit ae235b
init (
Packit ae235b
  void
Packit ae235b
)
Packit ae235b
{
Packit ae235b
  max_dist = 0;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
clear_tab (
Packit ae235b
  void
Packit ae235b
)
Packit ae235b
{
Packit ae235b
  register gunichar c;
Packit ae235b
Packit ae235b
  for (c = 0; c < UNICODE_CHARS; c++)
Packit ae235b
    table[c] = 0;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
init_tab_mirroring_txt (
Packit ae235b
  void
Packit ae235b
)
Packit ae235b
{
Packit ae235b
  clear_tab ();
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
read_bidi_mirroring_txt (
Packit ae235b
  FILE *f
Packit ae235b
)
Packit ae235b
{
Packit ae235b
  unsigned long l;
Packit ae235b
Packit ae235b
  init_tab_mirroring_txt ();
Packit ae235b
Packit ae235b
  l = 0;
Packit ae235b
  while (fgets (buf, sizeof buf, f))
Packit ae235b
    {
Packit ae235b
      unsigned long i, j;
Packit ae235b
      signed long dist;
Packit ae235b
      int k;
Packit ae235b
      const char *s = buf;
Packit ae235b
Packit ae235b
      l++;
Packit ae235b
Packit ae235b
      while (*s == ' ')
Packit ae235b
	s++;
Packit ae235b
Packit ae235b
      if (s[0] == '#' || s[0] == '\0' || s[0] == '\n')
Packit ae235b
	continue;
Packit ae235b
Packit ae235b
      k = sscanf (s, "%lx; %lx", &i, &j);
Packit ae235b
      if (k != 2 || i >= UNICODE_CHARS || j >= UNICODE_CHARS)
Packit ae235b
	die4 ("invalid pair in input at line %lu: %04lX, %04lX", l, i, j);
Packit ae235b
      dist = ((signed long) j - (signed long) i);
Packit ae235b
      table[i] = dist;
Packit ae235b
      if (dist > max_dist)
Packit ae235b
	max_dist = dist;
Packit ae235b
      else if (-dist > max_dist)
Packit ae235b
	max_dist = -dist;
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
read_data (
Packit ae235b
  const char *data_file_type,
Packit ae235b
  const char *data_file_name
Packit ae235b
)
Packit ae235b
{
Packit ae235b
  FILE *f;
Packit ae235b
Packit ae235b
  fprintf (stderr, "Reading '%s'\n", data_file_name);
Packit ae235b
  if (!(f = fopen (data_file_name, "rt")))
Packit ae235b
    die2 ("error: cannot open '%s' for reading", data_file_name);
Packit ae235b
Packit ae235b
  if (!strcmp (data_file_type, "BidiMirroring.txt"))
Packit ae235b
    read_bidi_mirroring_txt (f);
Packit ae235b
  else
Packit ae235b
    die2 ("error: unknown data-file-type %s", data_file_type);
Packit ae235b
Packit ae235b
  fclose (f);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
gen_mirroring_tab (
Packit ae235b
  int max_depth,
Packit ae235b
  const char *data_file_type
Packit ae235b
)
Packit ae235b
{
Packit ae235b
  int key_bytes;
Packit ae235b
  const char *key_type;
Packit ae235b
Packit ae235b
  fprintf (stderr,
Packit ae235b
	   "Generating '" outputname "', it may take up to a few minutes\n");
Packit ae235b
  printf ("/* " outputname "\n * generated by " appname " "
Packit ae235b
	  "\n" " * from the file %s of */\n\n", data_file_type);
Packit ae235b
Packit ae235b
  printf ("#define PACKTAB_UINT8 guint8\n"
Packit ae235b
	  "#define PACKTAB_UINT16 guint16\n"
Packit ae235b
	  "#define PACKTAB_UINT32 guint32\n\n");
Packit ae235b
Packit ae235b
  key_bytes = max_dist <= 0x7f ? 1 : max_dist < 0x7fff ? 2 : 4;
Packit ae235b
  key_type = key_bytes == 1 ? "gint8" : key_bytes == 2 ?
Packit ae235b
    "gint16" : "gint32";
Packit ae235b
Packit ae235b
  if (!pack_table
Packit ae235b
      (table, UNICODE_CHARS, key_bytes, 0, max_depth, 1, NULL,
Packit ae235b
       key_type, table_name, macro_name "_DELTA", stdout))
Packit ae235b
    die ("error: insufficient memory, decrease max_depth");
Packit ae235b
Packit ae235b
  printf ("#undef PACKTAB_UINT8\n"
Packit ae235b
	  "#undef PACKTAB_UINT16\n" "#undef PACKTAB_UINT32\n\n");
Packit ae235b
Packit ae235b
  printf ("#define " macro_name "(x) ((x) + " macro_name "_DELTA(x))\n\n");
Packit ae235b
Packit ae235b
  printf ("/* End of generated " outputname " */\n");
Packit ae235b
}
Packit ae235b
Packit ae235b
int
Packit ae235b
main (
Packit ae235b
  int argc,
Packit ae235b
  const char **argv
Packit ae235b
)
Packit ae235b
{
Packit ae235b
  const char *data_file_type = "BidiMirroring.txt";
Packit ae235b
Packit ae235b
  if (argc < 3)
Packit ae235b
    die2 ("usage:\n  " appname " max-lookups /path/to/%s [junk...]",
Packit ae235b
	  data_file_type);
Packit ae235b
Packit ae235b
  {
Packit ae235b
    int max_depth = atoi (argv[1]);
Packit ae235b
    const char *data_file_name = argv[2];
Packit ae235b
Packit ae235b
    if (max_depth < 2)
Packit ae235b
      die ("invalid depth");
Packit ae235b
Packit ae235b
    init ();
Packit ae235b
    read_data (data_file_type, data_file_name);
Packit ae235b
    gen_mirroring_tab (max_depth, data_file_type);
Packit ae235b
  }
Packit ae235b
Packit ae235b
  return 0;
Packit ae235b
}