home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Geek Gadgets 1
/
ADE-1.bin
/
ade-dist
/
fontutils-0.6-base.tgz
/
fontutils-0.6-base.tar
/
fsf
/
fontutils
/
charspace
/
symtab.c
< prev
next >
Wrap
C/C++ Source or Header
|
1992-06-14
|
8KB
|
323 lines
/* symtab.c: create and update a symbol table. We use a simple linear
representation, since there will be a few hundred entries at
the very most.
The numbers which get stored in the symbol table should all be in
pixels. That's what the rest of the program expects. (Unfortunately
there's no way to check this at the time of definition.)
Copyright (C) 1992 Free Software Foundation, Inc.
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, 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., 675 Mass Ave, Cambridge, MA 02139, USA. */
#include "config.h"
#include "kern.h"
#include "main.h"
#include "symtab.h"
/* Every identifier in the CMI files -- character names and fontdimen
names as well as `define'd quantities -- gets put into our symbol
table. We do not resolve any definitions until they are actually
needed, so that later definitions will replace earlier ones. */
struct symbol
{
string key;
symval_type value;
struct symbol *next;
};
/* These all work on pointers instead of structures, since that's what
we need to pass around in this file. */
#define SYM_KEY(s) ((s)->key)
#define SYM_NEXT(s) ((s)->next)
#define SYM_VAL(s) ((s)->value)
typedef struct symbol symbol_type;
/* We have only one symbol table. Here it is. */
static symbol_type *symbol_table = NULL;
static symbol_type *symtab_find (string);
static boolean resolve_string (symval_type *, real);
/* Routines to create symval nodes. */
symval_type
symtab_char_node (symval_type lsb, symval_type rsb)
{
symval_type sv;
SYMVAL_TAG (sv) = symval_char;
SYMVAL_CHAR (sv) = init_char ();
CHAR_LSB (SYMVAL_CHAR (sv)) = XTALLOC1 (sidebearing_type);
CHAR_LSB_TAG (SYMVAL_CHAR (sv)) = SYMVAL_TAG (lsb);
CHAR_LSB_VALUE (SYMVAL_CHAR (sv)) = SYMVAL_REAL_STRING (lsb);
CHAR_RSB (SYMVAL_CHAR (sv)) = XTALLOC1 (sidebearing_type);
CHAR_RSB_TAG (SYMVAL_CHAR (sv)) = SYMVAL_TAG (rsb);
CHAR_RSB_VALUE (SYMVAL_CHAR (sv)) = SYMVAL_REAL_STRING (rsb);
return sv;
}
symval_type
symtab_real_node (real r)
{
symval_type sv;
SYMVAL_TAG (sv) = symval_real;
SYMVAL_REAL (sv) = r;
return sv;
}
symval_type
symtab_real_string_node (real r, string s)
{
symval_type sv;
SYMVAL_TAG (sv) = symval_real_string;
SYMVAL_REAL (sv) = r;
SYMVAL_STRING (sv) = s;
return sv;
}
symval_type
symtab_string_node (string s)
{
symval_type sv;
SYMVAL_TAG (sv) = symval_string;
SYMVAL_STRING (sv) = s;
return sv;
}
/* Define the identifier KEY to be the value V. We do no checking on V,
we just plop it in. Overwrite any previous definition of KEY. If
KEY was not previously defined, we malloc a new symbol. */
void
symtab_define (string key, symval_type v)
{
symbol_type *s = symtab_find (key);
if (s == NULL)
{
s = XTALLOC1 (symbol_type);
/* Make a private copy of KEY so outside assignments or
deallocations don't affect the symbol table. */
SYM_KEY (s) = xstrdup (key);
SYM_NEXT (s) = symbol_table;
symbol_table = s;
}
/* Whether we just created it or not, put in the new value. */
SYM_VAL (s) = v;
}
/* Define a kern of K pixels between the characters named LEFT and
RIGHT. If no kern for that pair previously existed, we create it.
If the character LEFT was not previously in the symbol table, we add
it. But if LEFT was in the symbol table, we have to preserve any
kerns or side bearing information that have already been stored. */
void
symtab_define_kern (string left, string right, symval_type k)
{
symval_type sv;
symbol_type *s = symtab_find (left);
if (s == NULL || SYMVAL_TAG (SYM_VAL (s)) != symval_char)
{
list_type *kern_list_ptr;
char_kern_type *new_kern;
SYMVAL_TAG (sv) = symval_char;
SYMVAL_CHAR (sv) = init_char ();
kern_list_ptr = &CHAR_KERNS (SYMVAL_CHAR (sv));
new_kern = LIST_TAPPEND (kern_list_ptr, char_kern_type);
new_kern->character = right;
new_kern->kern = k;
}
else
{
sv = SYM_VAL (s);
char_set_kern (&CHAR_KERNS (SYMVAL_CHAR (sv)), right, k);
}
symtab_define (left, sv);
}
/* Look up KEY in `symbol_table'. If found, return a pointer to the
corresponding symval, else NULL. */
symval_type *
symtab_lookup (string key)
{
symbol_type *s = symtab_find (key);
symval_type *sv = s ? &SYM_VAL (s) : NULL;
return sv;
}
/* Look up KEY in `symbol_table', resolve its definition to a real, and
return the result. If the value cannot be resolved, or if KEY isn't
defined, give a fatal error. */
real
symtab_lookup_real (string key)
{
real r;
symval_type *sv = symtab_lookup (key);
if (sv == NULL)
FATAL1 ("%s: Undefined symbol", key);
if (symval_resolve (sv))
r = SYMVAL_REAL (*sv);
else
FATAL1 ("%s: Cannot be resolved to a real", key);
return r;
}
/* Look up KEY in `symbol_table'. If found, return the containing
symbol, otherwise NULL. */
static symbol_type *
symtab_find (string key)
{
symbol_type *s;
assert (key != NULL);
for (s = symbol_table; s != NULL && !STREQ (SYM_KEY (s), key);
s = SYM_NEXT (s))
;
return s;
}
/* Resolve (to a real) the symbol value SV. Change SV if we succeed.
Return success. */
boolean
symval_resolve (symval_type *sv)
{
boolean ok;
if (sv == NULL)
return false;
switch (SYMVAL_TAG (*sv))
{
case symval_char:
ok = false;
break;
case symval_real:
ok = true;
break;
case symval_real_string:
ok = resolve_string (sv, SYMVAL_REAL (*sv));
break;
case symval_string:
ok = resolve_string (sv, 1.0);
break;
default:
abort (); /* We have listed all the cases. */
}
return ok;
}
/* Assume SV is non-null, and that its SYMVAL_STRING field is
meaningful. Look up that string, resolve the result. If that
succeeds, multiply by FACTOR, change SV, and return success. */
static boolean
resolve_string (symval_type *sv, real factor)
{
boolean ok = false;
symval_type *sv_aux = symtab_lookup (SYMVAL_STRING (*sv));
if (symval_resolve (sv_aux))
{
SYMVAL_TAG (*sv) = symval_real;
SYMVAL_REAL (*sv) = SYMVAL_REAL (*sv_aux) * factor;
ok = true;
}
return ok;
}
/* Return a description of SV as a string. */
string
symval_as_string (symval_type sv)
{
string desc;
switch (SYMVAL_TAG (sv))
{
case symval_char:
{
string charcode = CHAR_BITMAP_INFO (SYMVAL_CHAR (sv)) != NULL
? concat (" ", utoa (CHAR_CHARCODE (SYMVAL_CHAR (sv))))
: "";
desc = concat ("character", charcode);
if (*charcode != 0)
free (charcode);
}
break;
case symval_real:
desc = dtoa (SYMVAL_REAL (sv));
break;
case symval_real_string:
desc = concat (dtoa (SYMVAL_REAL (sv)), SYMVAL_STRING (sv));
break;
case symval_string:
desc = xstrdup (SYMVAL_STRING (sv));
break;
default:
abort (); /* We have listed all the cases. */
}
return desc;
}