/* Copyright (C) 1989-2020 Free Software Foundation, Inc.
Written by James Clark (jjc@jclark.com)
This file is part of groff.
groff 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 3 of the License, or
(at your option) any later version.
groff 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, see . */
#include "lib.h"
#include
#include
#include
#include "errarg.h"
#include "error.h"
#include "font.h"
#include "ptable.h"
#include "itable.h"
// Every glyphinfo is actually a charinfo.
class charinfo : glyph {
public:
const char *name; // The glyph name, or a null pointer.
friend class character_indexer;
};
// PTABLE(charinfo) is a hash table mapping 'const char *' to 'charinfo *'.
declare_ptable(charinfo)
implement_ptable(charinfo)
// ITABLE(charinfo) is a hash table mapping 'int >= 0' to 'charinfo *'.
declare_itable(charinfo)
implement_itable(charinfo)
// This class is as a registry storing all named and numbered glyphs known
// so far, and assigns a unique index to each glyph.
class character_indexer {
public:
character_indexer();
~character_indexer();
// --------------------- Lookup or creation of a glyph.
glyph *ascii_char_glyph(unsigned char);
glyph *named_char_glyph(const char *);
glyph *numbered_char_glyph(int);
private:
int next_index; // Number of glyphs already allocated.
PTABLE(charinfo) table; // Table mapping name to glyph.
glyph *ascii_glyph[256]; // Shorthand table for looking up "charNNN"
// glyphs.
ITABLE(charinfo) ntable; // Table mapping number to glyph.
enum { NSMALL = 256 };
glyph *small_number_glyph[NSMALL]; // Shorthand table for looking up
// numbered glyphs with small numbers.
};
character_indexer::character_indexer()
: next_index(0)
{
int i;
for (i = 0; i < 256; i++)
ascii_glyph[i] = UNDEFINED_GLYPH;
for (i = 0; i < NSMALL; i++)
small_number_glyph[i] = UNDEFINED_GLYPH;
}
character_indexer::~character_indexer()
{
}
glyph *character_indexer::ascii_char_glyph(unsigned char c)
{
if (ascii_glyph[c] == UNDEFINED_GLYPH) {
char buf[4+3+1];
memcpy(buf, "char", 4);
strcpy(buf + 4, i_to_a(c));
charinfo *ci = new charinfo;
ci->index = next_index++;
ci->number = -1;
ci->name = strsave(buf);
ascii_glyph[c] = ci;
}
return ascii_glyph[c];
}
inline glyph *character_indexer::named_char_glyph(const char *s)
{
// Glyphs with name 'charNNN' are only stored in ascii_glyph[], not
// in the table. Therefore treat them specially here.
if (s[0] == 'c' && s[1] == 'h' && s[2] == 'a' && s[3] == 'r') {
char *val;
long n = strtol(s + 4, &val, 10);
if (val != s + 4 && *val == '\0' && n >= 0 && n < 256)
return ascii_char_glyph((unsigned char)n);
}
charinfo *ci = table.lookupassoc(&s);
if (0 == ci) {
ci = new charinfo[1];
ci->index = next_index++;
ci->number = -1;
ci->name = table.define(s, ci);
}
return ci;
}
inline glyph *character_indexer::numbered_char_glyph(int n)
{
if (n >= 0 && n < NSMALL) {
if (small_number_glyph[n] == UNDEFINED_GLYPH) {
charinfo *ci = new charinfo;
ci->index = next_index++;
ci->number = n;
ci->name = 0;
small_number_glyph[n] = ci;
}
return small_number_glyph[n];
}
charinfo *ci = ntable.lookup(n);
if (0 == ci) {
ci = new charinfo[1];
ci->index = next_index++;
ci->number = n;
ci->name = 0;
ntable.define(n, ci);
}
return ci;
}
static character_indexer indexer;
glyph *number_to_glyph(int n)
{
return indexer.numbered_char_glyph(n);
}
// troff overrides this function with its own version.
glyph *name_to_glyph(const char *s)
{
assert(s != 0 && s[0] != '\0' && s[0] != ' ');
if (s[1] == '\0')
// \200 and char128 are synonyms
return indexer.ascii_char_glyph(s[0]);
return indexer.named_char_glyph(s);
}
const char *glyph_to_name(glyph *g)
{
charinfo *ci = (charinfo *)g; // Every glyph is actually a charinfo.
return ci->name;
}
// Local Variables:
// fill-column: 72
// mode: C++
// End:
// vim: set cindent noexpandtab shiftwidth=2 textwidth=72: