ref: 80b2dc5db97b8e1d0628578d6696836693e1d563
dir: /hyph.c/
/* hyphenation */ #include <ctype.h> #include <stdio.h> #include <string.h> #include "roff.h" #include "hyen.h" #define HYPATLEN (NHYPHS * 16) /* hyphenation pattern length */ static void hcode_strcpy(char *d, char *s, int *map, int dots); static int hcode_mapchar(char *s); /* the hyphenation dictionary (.hw) */ static char hwword[HYPATLEN]; /* buffer for .hw words */ static char hwhyph[HYPATLEN]; /* buffer for .hw hyphenations */ static int hwword_len; /* used hwword[] length */ static struct dict *hwdict; /* map words to their index in hwoff[] */ static int hwoff[NHYPHS]; /* the offset of words in hwword[] */ static int hw_n; /* the number of dictionary words */ /* read a single character from s into d; return the number of characters read */ static int hy_cget(char *d, char *s) { int i = 0; if (s[0] != '\\') return utf8read(&s, d); if (s[1] == '[') { s += 2; while (*s && *s != ']' && i < GNLEN - 1) d[i++] = *s++; d[i] = '\0'; return *s ? i + 3 : i + 2; } if (s[1] == '(') { s += 2; i += utf8read(&s, d + i); i += utf8read(&s, d + i); return 2 + i; } if (s[1] == 'C') { int q = s[2]; s += 3; while (*s && *s != q && i < GNLEN - 1) d[i++] = *s++; d[i] = '\0'; return *s ? i + 4 : i + 3; } *d++ = *s++; return 1 + utf8read(&s, d); } /* append character s to d; return the number of characters written */ int hy_cput(char *d, char *s) { if (!s[0] || !s[1] || utf8one(s)) strcpy(d, s); else if (s[0] == '\\') strcpy(d, s); else if (!s[2]) snprintf(d, GNLEN, "\\[%s]", s); return strlen(d); } /* insert word s into hwword[] and hwhyph[] */ static void hw_add(char *s) { char *p = hwword + hwword_len; char *n = hwhyph + hwword_len; int len = strlen(s) + 1; int i = 0, c; if (hw_n == NHYPHS || hwword_len + len > sizeof(hwword)) return; memset(n, 0, len); while ((c = (unsigned char) *s++)) { if (c == '-') n[i] = 1; else p[i++] = c; } p[i] = '\0'; hwoff[hw_n] = hwword_len; dict_put(hwdict, hwword + hwoff[hw_n], hw_n); hwword_len += i + 1; hw_n++; } static int hw_lookup(char *word, char *hyph) { char word2[WORDLEN] = {0}; char *hyph2; int map[WORDLEN] = {0}; int off = 0; int i, j, idx = -1; hcode_strcpy(word2, word, map, 0); while (word2[off] == '.') /* skip unknown characters at the front */ off++; i = dict_prefix(hwdict, word2 + off, &idx); if (i < 0) return 1; hyph2 = hwhyph + hwoff[i]; for (j = 0; word2[j + off]; j++) if (hyph2[j]) hyph[map[j + off]] = hyph2[j]; return 0; } void tr_hw(char **args) { char word[WORDLEN]; char *c; int i; for (i = 1; i < NARGS && args[i]; i++) { char *s = args[i]; char *d = word; while (d - word < WORDLEN - GNLEN && !escread(&s, &c)) { if (strcmp("-", c)) hcode_mapchar(c); d += hy_cput(d, c); } hw_add(word); } } /* the tex hyphenation algorithm */ static int hyinit; /* hyphenation data initialized */ static char hypats[HYPATLEN]; /* hyphenation patterns */ static char hynums[HYPATLEN]; /* hyphenation pattern numbers */ static int hypats_len; /* used hypats[] and hynums[] length */ static struct dict *hydict; /* map patterns to their index in hyoff[] */ static int hyoff[NHYPHS]; /* the offset of this pattern in hypats[] */ static int hy_n; /* the number of patterns */ /* find the patterns matching s and update hyphenation values in n */ static void hy_find(char *s, char *n) { int plen; char *p, *np; int i, j; int idx = -1; while ((i = dict_prefix(hydict, s, &idx)) >= 0) { p = hypats + hyoff[i]; np = hynums + (p - hypats); plen = strlen(p) + 1; for (j = 0; j < plen; j++) if (n[j] < np[j]) n[j] = np[j]; } } /* mark the hyphenation points of word in hyph */ static void hy_dohyph(char *hyph, char *word, int flg) { char w[WORDLEN] = {0}; /* cleaned-up word[]; "Abc" -> ".abc." */ char n[WORDLEN] = {0}; /* the hyphenation value for w[] */ int c[WORDLEN]; /* start of the i-th character in w */ int wmap[WORDLEN] = {0}; /* w[i] corresponds to word[wmap[i]] */ char ch[GNLEN]; int nc = 0; int i, wlen; hcode_strcpy(w, word, wmap, 1); wlen = strlen(w); for (i = 0; i < wlen - 1; i += hy_cget(ch, w + i)) c[nc++] = i; for (i = 0; i < nc - 1; i++) hy_find(w + c[i], n + c[i]); memset(hyph, 0, wlen * sizeof(hyph[0])); for (i = 3; i < nc - 2; i++) if (n[c[i]] % 2 && w[c[i - 1]] != '.' && w[c[i]] != '.' && w[c[i - 2]] != '.' && w[c[i + 1]] != '.' && (~flg & HY_FINAL2 || w[c[i + 2]] != '.') && (~flg & HY_FIRST2 || w[c[i - 3]] != '.')) hyph[wmap[c[i]]] = 1; } /* insert pattern s into hypats[] and hynums[] */ static void hy_add(char *s) { char *p = hypats + hypats_len; char *n = hynums + hypats_len; int len = strlen(s) + 1; int i = 0, c; if (hy_n >= NHYPHS || hypats_len + len >= sizeof(hypats)) return; memset(n, 0, len); while ((c = (unsigned char) *s++)) { if (c >= '0' && c <= '9') n[i] = c - '0'; else p[i++] = c; } p[i] = '\0'; hyoff[hy_n] = hypats_len; dict_put(hydict, hypats + hyoff[hy_n], hy_n); hypats_len += i + 1; hy_n++; } /* .hcode request */ static struct dict *hcodedict; static char hcodesrc[NHCODES][GNLEN]; static char hcodedst[NHCODES][GNLEN]; static int hcode_n; /* replace the character in s after .hcode mapping; returns s's new length */ static int hcode_mapchar(char *s) { int i = dict_get(hcodedict, s); if (i >= 0) strcpy(s, hcodedst[i]); else if (!s[1]) *s = isalpha((unsigned char) *s) ? tolower((unsigned char) *s) : '.'; return strlen(s); } /* copy s to d after .hcode mappings; s[map[j]] corresponds to d[j] */ static void hcode_strcpy(char *d, char *s, int *map, int dots) { char c[GNLEN]; int di = 0, si = 0; if (dots) d[di++] = '.'; while (di < WORDLEN - GNLEN && s[si]) { map[di] = si; si += hy_cget(c, s + si); hcode_mapchar(c); di += hy_cput(d + di, c); } if (dots) d[di++] = '.'; d[di] = '\0'; } static void hcode_add(char *c1, char *c2) { int i = dict_get(hcodedict, c1); if (i >= 0) { strcpy(hcodedst[i], c2); } else if (hcode_n < NHCODES) { strcpy(hcodesrc[hcode_n], c1); strcpy(hcodedst[hcode_n], c2); dict_put(hcodedict, hcodesrc[hcode_n], hcode_n); hcode_n++; } } void tr_hcode(char **args) { char c1[GNLEN], c2[GNLEN]; char *s = args[1]; while (s && charread(&s, c1) >= 0 && charread(&s, c2) >= 0) hcode_add(c1, c2); } static void hyph_readpatterns(char *s) { char word[WORDLEN]; char *d; while (*s) { d = word; while (*s && !isspace((unsigned char) *s)) *d++ = *s++; *d = '\0'; hy_add(word); while (*s && isspace((unsigned char) *s)) s++; } } static void hyph_readexceptions(char *s) { char word[WORDLEN]; char *d; while (*s) { d = word; while (*s && !isspace((unsigned char) *s)) *d++ = *s++; *d = '\0'; hw_add(word); while (*s && isspace((unsigned char) *s)) s++; } } void hyphenate(char *hyph, char *word, int flg) { if (!hyinit) { hyinit = 1; hyph_readpatterns(en_patterns); hyph_readexceptions(en_exceptions); } if (hw_lookup(word, hyph)) hy_dohyph(hyph, word, flg); } void tr_hpfa(char **args) { char tok[ILNLEN], c1[ILNLEN], c2[ILNLEN]; FILE *filp; hyinit = 1; /* load english hyphenation patterns with no arguments */ if (!args[1]) { hyph_readpatterns(en_patterns); hyph_readexceptions(en_exceptions); } /* reading patterns */ if (args[1] && (filp = fopen(args[1], "r"))) { while (fscanf(filp, "%s", tok) == 1) if (strlen(tok) < WORDLEN) hy_add(tok); fclose(filp); } /* reading exceptions */ if (args[2] && (filp = fopen(args[2], "r"))) { while (fscanf(filp, "%s", tok) == 1) if (strlen(tok) < WORDLEN) hw_add(tok); fclose(filp); } /* reading hcode mappings */ if (args[3] && (filp = fopen(args[3], "r"))) { while (fscanf(filp, "%s", tok) == 1) { char *s = tok; if (utf8read(&s, c1) && utf8read(&s, c2) && !*s) hcode_add(c2, c1); /* inverting */ } fclose(filp); } } void hyph_init(void) { hwdict = dict_make(-1, 0, 1); hydict = dict_make(-1, 0, 1); hcodedict = dict_make(-1, 0, 1); } void hyph_done(void) { if (hwdict) dict_free(hwdict); if (hydict) dict_free(hydict); if (hcodedict) dict_free(hcodedict); } void tr_hpf(char **args) { /* reseting the patterns */ hypats_len = 0; hy_n = 0; dict_free(hydict); /* reseting the dictionary */ hwword_len = 0; hw_n = 0; dict_free(hwdict); /* reseting hcode mappings */ hcode_n = 0; dict_free(hcodedict); /* reading */ hyph_init(); tr_hpfa(args); }