]> git.donarmstrong.com Git - xournal.git/commitdiff
Print via gtk-print instead of libgnomeprint
authorauroux <auroux>
Wed, 16 Sep 2009 20:36:27 +0000 (20:36 +0000)
committerauroux <auroux>
Wed, 16 Sep 2009 20:36:27 +0000 (20:36 +0000)
src/ttsubset/Makefile.am [new file with mode: 0644]
src/ttsubset/README [new file with mode: 0644]
src/ttsubset/list.c [new file with mode: 0644]
src/ttsubset/list.h [new file with mode: 0644]
src/ttsubset/sft.c [new file with mode: 0644]
src/ttsubset/sft.h [new file with mode: 0644]
src/ttsubset/ttcr.c [new file with mode: 0644]
src/ttsubset/ttcr.h [new file with mode: 0644]

diff --git a/src/ttsubset/Makefile.am b/src/ttsubset/Makefile.am
new file mode 100644 (file)
index 0000000..dabfe8e
--- /dev/null
@@ -0,0 +1,11 @@
+## Process this file with automake to produce Makefile.in
+
+INCLUDES = \
+       @PACKAGE_CFLAGS@ -DNO_MAPPERS -DNO_TYPE3 -DNO_TYPE42
+
+noinst_LIBRARIES = libttsubset.a
+
+libttsubset_a_SOURCES = \
+       sft.c sft.h \
+       list.c list.h \
+       ttcr.c ttcr.h
diff --git a/src/ttsubset/README b/src/ttsubset/README
new file mode 100644 (file)
index 0000000..27d679d
--- /dev/null
@@ -0,0 +1,21 @@
+* from libgnomeprint 2.12.1:
+
+       The files sft.c, list.c, ttcr.c, crc32.c and corresponding header files
+are yanked from STSF (Standard Type Services Framework)  text imaging and
+font handling framework for application developers.
+
+http://stsf.sourceforge.net/about.html
+
+STSF 0.4
+March 25, 2003
+
+                                               suresh.chandrasekharan@sun.com  
+                                                               06/21/04
+
+
+* for xournal purposes:
+
+These files have been adapted slightly to remove a dependency to the
+gnome-print system, and to allow TrueType subset creation in memory
+
+ auroux@math.mit.edu, 09/07/09
diff --git a/src/ttsubset/list.c b/src/ttsubset/list.c
new file mode 100644 (file)
index 0000000..0cf958e
--- /dev/null
@@ -0,0 +1,665 @@
+/*
+ * Copyright © 2002, 2003 Sun Microsystems, Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ *
+ * 3. Neither the name of Sun Microsystems, Inc. nor the names of 
+ *    contributors may be used to endorse or promote products derived from
+ *    this software without specific prior written permission.
+ *
+ * This software is provided "AS IS," without a warranty of any kind.
+ *
+ * ALL EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES,
+ * INCLUDING ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A
+ * PARTICULAR PURPOSE OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED.
+ * SUN AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES OR
+ * LIABILITIES SUFFERED BY LICENSEE AS A RESULT OF OR RELATING TO USE,
+ * MODIFICATION OR DISTRIBUTION OF THE SOFTWARE OR ITS DERIVATIVES.
+ * IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST REVENUE,
+ * PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL,
+ * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE
+ * THEORY OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE
+ * SOFTWARE, EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
+ *
+ */
+
+/* $Id$ */
+/* @(#)list.c 1.7 03/02/06 SMI */
+
+/*
+ * @file list.c
+ * @brief Bidirectional list class
+ * @author Alexander Gelfenbain
+ * @version 1.0
+ *
+ */
+
+#include <stdlib.h>
+#include <assert.h>
+#ifdef MALLOC_TRACE
+#include <stdio.h>
+#include </usr/local/include/malloc.h>
+#endif
+#include "list.h"
+
+/*- private data types */
+typedef struct _lnode {
+    struct _lnode *next;
+    struct _lnode *prev;
+
+    void *value;
+
+} lnode;
+
+struct _list {
+    lnode *head, *tail, *cptr;
+    size_t aCount;
+    void (*eDtor)(void *);
+};
+
+/*- private methods */
+
+static lnode *newNode(void *el)
+{
+    lnode *ptr = malloc(sizeof(lnode));
+    assert(ptr != 0);
+
+    ptr->value = el;
+
+    return ptr;
+}
+
+static lnode *appendPrim(list this, void *el)
+{
+    lnode *ptr = newNode(el);
+    lnode **flink, *blink;
+
+    if (this->tail != 0) {
+        flink = &(this->tail->next);
+        blink = this->tail;
+    } else {
+        flink = &this->head;
+        blink = NULL;
+        this->cptr = ptr;                         /*- list was empty - set current to this element */
+    }
+
+    *flink  = ptr;
+    this->tail = ptr;
+
+    ptr->prev = blink;
+    ptr->next = NULL;
+
+    this->aCount++;
+    return ptr;
+}
+
+static lnode *prependPrim(list this, void *el)
+{
+    lnode *ptr = newNode(el);
+    lnode *flink, **blink;
+
+    if (this->head != 0) {
+        blink = &(this->head->prev);
+        flink = this->head;
+    } else {
+        blink = &this->tail;
+        flink = NULL;
+        this->cptr  = ptr;                        /*- list was empty - set current to this element */
+    }
+
+    *blink = ptr;
+    this->head   = ptr;
+
+    ptr->next = flink;
+    ptr->prev = NULL;
+
+    this->aCount++;
+    return ptr;
+}
+
+
+/*- public methods  */
+list listNewEmpty(void)                           /*- default ctor */
+{
+    list this = malloc(sizeof(struct _list));
+    assert(this != 0);
+
+    this->aCount = 0;
+    this->eDtor = NULL;
+    this->head = this->tail = this->cptr = NULL;
+
+    return this;
+}
+
+list listNewCopy(list l)                          /*- copy ctor */
+{
+    lnode *ptr, *c;
+    list this;
+    assert(l != 0);
+
+    this = malloc(sizeof(struct _list));
+    assert(this != 0);
+
+    ptr = l->head;
+
+    this->aCount = 0;
+    this->eDtor = NULL;
+    this->head = this->tail = this->cptr = NULL;
+
+    while (ptr) {
+        c = appendPrim(this, ptr->value);
+        if (ptr == l->cptr) this->cptr = c;
+        ptr = ptr->next;
+    }
+
+    return this;
+}
+
+list listNewConcat(list lhs, list rhs)
+{
+    lnode *ptr, *c;
+    list this;
+
+    assert(lhs != 0);
+    assert(rhs != 0);
+
+    this = malloc(sizeof(struct _list));
+    assert(this != 0);
+
+
+    this->aCount = 0;
+    this->eDtor = NULL;
+    this->head = this->tail = this->cptr = NULL;
+
+    ptr = lhs->head;
+    while (ptr) {
+        c = appendPrim(this, ptr->value);
+        ptr = ptr->next;
+    }
+
+    ptr = rhs->head;
+    while (ptr) {
+        c = appendPrim(this, ptr->value);
+        ptr = ptr->next;
+    }
+
+    this->cptr = this->head;
+
+    return this;
+}
+
+    
+
+    
+    
+
+
+void listDispose(list this)                       /*- dtor */
+{
+    assert(this != 0);
+    listClear(this);
+    free(this);
+}
+
+void listSetElementDtor(list this, GDestroyNotify f)
+{
+    assert(this != 0);
+    this->eDtor = f;
+}
+    
+
+list listCopy(list to, list from)                 /*- assignment */
+{
+    lnode *ptr, *c;
+    assert(to != 0);
+    assert(from != 0);
+    
+    listClear(to);
+    ptr = from->head;
+
+    while (ptr) {
+        c = appendPrim(to, ptr->value);
+        if (ptr == from->cptr) to->cptr = c;
+        ptr = ptr->next;
+    }
+
+    return to;
+}
+    
+
+/* calling this function on an empty list is a run-time error */
+void *listCurrent(list this)
+{
+    assert(this != 0);
+    assert(this->cptr != 0);
+    return this->cptr->value;
+}
+
+int   listCount(list this)
+{
+    assert(this != 0);
+    return this->aCount;
+}
+
+int   listIsEmpty(list this)
+{
+    assert(this != 0);
+    return this->aCount == 0;
+}
+
+
+int   listAtFirst(list this)
+{
+    assert(this != 0);
+    return this->cptr == this->head;
+}
+
+int   listAtLast(list this)
+{
+    assert(this != 0);
+    return this->cptr == this->tail;
+}
+
+int   listPosition(list this)
+{
+    int res = 0;
+    lnode *ptr;
+    assert(this != 0);
+
+    ptr = this->head;
+
+    while (ptr != this->cptr) {
+        ptr = ptr->next;
+        res++;
+    }
+
+    return res;
+}
+
+int    listFind(list this, void *el)
+{
+    lnode *ptr;
+    assert(this != 0);
+    
+    ptr = this->head;
+
+    while (ptr) {
+        if (ptr->value == el) {
+            this->cptr = ptr;
+            return 1;
+        }
+        ptr = ptr->next;
+    }
+
+    return 0;
+}
+
+int    listNext(list this)
+{
+    return listSkipForward(this, 1);
+}
+
+int    listPrev(list this)
+{
+    return listSkipBackward(this, 1);
+}
+
+int    listSkipForward(list this, int n)
+{
+    int m = 0;
+    assert(this != 0);
+    
+    if (this->cptr == 0) return 0;
+
+    while (n != 0) {
+        if (this->cptr->next == 0) break;
+        this->cptr = this->cptr->next;
+        n--;
+        m++;
+    }
+    return m;
+}
+
+int    listSkipBackward(list this, int n)
+{
+    int m = 0;
+    assert(this != 0);
+    
+    if (this->cptr == 0) return 0;
+    
+    while (n != 0) {
+        if (this->cptr->prev == 0) break;
+        this->cptr = this->cptr->prev;
+        n--;
+        m++;
+    }
+    return m;
+}
+
+int    listToFirst(list this)
+{
+    assert(this != 0);
+
+    if (this->cptr != this->head) {
+        this->cptr = this->head;
+        return 1;
+    }
+    return 0;
+}
+
+int    listToLast(list this)
+{
+    assert(this != 0);
+
+    if (this->cptr != this->tail) {
+        this->cptr = this->tail;
+        return 1;
+    }
+    return 0;
+}
+
+int    listPositionAt(list this, int n)                     /*- returns the actual position number */
+{
+    int m = 0;
+    assert(this != 0);
+    
+    this->cptr = this->head;
+    while (n != 0) {
+        if (this->cptr->next == 0) break;
+        this->cptr = this->cptr->next;
+        n--;
+        m++;
+    }
+    return m;
+}
+
+list   listAppend(list this, void *el)
+{
+    assert(this != 0);
+
+    appendPrim(this, el);
+    return this;
+}
+
+list   listConcat(list lhs, list rhs)
+{
+    lnode *ptr;
+
+    assert(lhs != 0);
+    assert(rhs != 0);
+
+    ptr = rhs->head;
+    while (ptr != 0) {
+        appendPrim(lhs, ptr->value);
+        ptr = ptr->next;
+    }
+
+    return lhs;
+}
+
+    
+list   listPrepend(list this, void *el)
+{
+    assert(this != 0);
+
+    prependPrim(this, el);
+    return this;
+}
+
+list   listInsertAfter(list this, void *el)
+{
+    lnode *ptr;
+    assert(this != 0);
+
+    if (this->cptr == 0) return listAppend(this, el);
+
+    ptr = newNode(el);
+
+    ptr->prev  = this->cptr;
+    ptr->next  = this->cptr->next;
+    this->cptr->next = ptr;
+
+    if (ptr->next != 0) {
+        ptr->next->prev = ptr;
+    } else {
+        this->tail = ptr;
+    }
+    this->aCount++;
+    return this;
+}
+
+list   listInsertBefore(list this, void *el)
+{
+    lnode *ptr;
+    assert(this != 0);
+    
+    if (this->cptr == 0) return listAppend(this, el);
+
+    ptr = newNode(el);
+
+    ptr->prev  = this->cptr->prev;
+    ptr->next  = this->cptr;
+    this->cptr->prev = ptr;
+
+    if (ptr->prev != 0) {
+        ptr->prev->next = ptr;
+    } else {
+        this->head = ptr;
+    }
+    this->aCount++;
+    return this;
+}
+
+list   listRemove(list this)
+{
+    lnode *ptr = NULL;
+    if (this->cptr == 0) return this;
+
+    if (this->cptr->next != 0) {
+        ptr  = this->cptr->next;
+        this->cptr->next->prev = this->cptr->prev;
+    } else {
+        this->tail = this->cptr->prev;
+    }
+
+    if (this->cptr->prev != 0) {
+        if (ptr == 0) ptr = this->cptr->prev;
+        this->cptr->prev->next = this->cptr->next;
+    } else {
+        this->head = this->cptr->next;
+    }
+
+    if (this->eDtor) this->eDtor(this->cptr->value);        /* call the dtor callback */
+    
+    free(this->cptr);
+    this->aCount--;
+    this->cptr = ptr;
+    return this;
+}
+
+list   listClear(list this)
+{
+    lnode *node = this->head, *ptr;
+
+    while (node) {
+        ptr = node->next;
+        if (this->eDtor) this->eDtor(node->value);           /* call the dtor callback */
+        free(node);
+        this->aCount--;
+        node = ptr;
+    }
+
+    this->head = this->tail = this->cptr = NULL;
+    assert(this->aCount == 0);
+    return this;
+}
+
+void   listForAll(list this, void (*f)(void *))
+{
+    lnode *ptr = this->head;
+    while (ptr) {
+        f(ptr->value);
+        ptr = ptr->next;
+    }
+}
+
+void **listToArray(list this)
+{
+    void **res;
+    lnode *ptr = this->head;
+    int i = 0;
+    
+    assert(this->aCount != 0);
+    res = calloc(this->aCount, sizeof(void *));
+    assert(res != 0);
+
+    while (ptr) {
+        res[i++] = ptr->value;
+        ptr = ptr->next;
+    }
+    return res;
+}
+
+
+/* #define TEST */
+#ifdef TEST
+#include <stdio.h>
+
+void printlist(list l)
+{
+    int saved;
+    assert(l != 0);
+    saved = listPosition(l);
+
+    printf("[ ");
+
+    if (!listIsEmpty(l)) {
+        listToFirst(l);
+        do {
+            printf("%d ", (int) listCurrent(l));
+        } while (listNext(l));
+    }
+
+    printf("]\n");
+
+    listPositionAt(l, saved);
+}
+
+void printstringlist(list l)
+{
+    int saved;
+    assert(l != 0);
+    saved = listPosition(l);
+
+    printf("[ ");
+
+    if (!listIsEmpty(l)) {
+        listToFirst(l);
+        do {
+            printf("'%s' ", (char *) listCurrent(l));
+        } while (listNext(l));
+    }
+
+    printf("]\n");
+
+    listPositionAt(l, saved);
+}
+
+void printstat(list l)
+{
+    printf("count: %d, position: %d, isEmpty: %d, atFirst: %d, atLast: %d.\n",
+           listCount(l), listPosition(l), listIsEmpty(l), listAtFirst(l), listAtLast(l));
+}
+
+void allfunc(void *e)
+{
+    printf("%d ", e);
+}
+
+void edtor(void *ptr)
+{
+    printf("element dtor: 0x%08x\n", ptr);
+    free(ptr);
+}
+
+int main()
+{
+    list l1, l2, l3;
+    char *ptr;
+    int i;
+
+#ifdef MALLOC_TRACE
+    mal_leaktrace(1);
+    mal_debug(2);
+#endif
+
+    l1 = listNewEmpty();
+    printstat(l1);
+
+    listAppend(l1, 1);
+    printstat(l1);
+
+    listAppend(l1, 2);
+    printstat(l1);
+
+    listAppend(l1, 3);
+    printstat(l1);
+
+    printlist(l1);
+
+    listToFirst(l1);
+    listInsertBefore(l1, -5);
+    printf("l1: ");
+    printlist(l1);
+
+    l2 = listNewCopy(l1);
+    printf("l2: ");
+    printlist(l2);
+
+    l3 = listNewConcat(l1, l2);
+    
+    printf("l1 + l2: ");
+    printlist(l3);
+
+    listConcat(l3, l1);
+    printf("l3 + l1" );
+    printlist(l3);
+
+
+    listForAll(l2, allfunc);
+    printf("\n");
+
+    listClear(l1);
+    listSetElementDtor(l1, edtor);
+
+    for(i=0; i<10; i++) {
+        ptr = malloc(20);
+        sprintf(ptr, "element # %d", i);
+        listAppend(l1, ptr);
+    }
+
+    printstringlist(l1);
+
+
+    listDispose(l1);
+    listDispose(l2);
+    listDispose(l3);
+
+#ifdef MALLOC_TRACE
+    mal_dumpleaktrace(stdout);
+#endif
+
+    
+    return 0;
+}
+#endif
diff --git a/src/ttsubset/list.h b/src/ttsubset/list.h
new file mode 100644 (file)
index 0000000..98f5bf8
--- /dev/null
@@ -0,0 +1,116 @@
+/*
+ * Copyright © 2002, 2003 Sun Microsystems, Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ *
+ * 3. Neither the name of Sun Microsystems, Inc. nor the names of 
+ *    contributors may be used to endorse or promote products derived from
+ *    this software without specific prior written permission.
+ *
+ * This software is provided "AS IS," without a warranty of any kind.
+ *
+ * ALL EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES,
+ * INCLUDING ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A
+ * PARTICULAR PURPOSE OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED.
+ * SUN AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES OR
+ * LIABILITIES SUFFERED BY LICENSEE AS A RESULT OF OR RELATING TO USE,
+ * MODIFICATION OR DISTRIBUTION OF THE SOFTWARE OR ITS DERIVATIVES.
+ * IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST REVENUE,
+ * PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL,
+ * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE
+ * THEORY OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE
+ * SOFTWARE, EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
+ *
+ */
+
+/* $Id$ */
+/* @(#)list.h 1.6 03/02/06 SMI */
+
+/*
+ * @file list.h
+ * @brief Bidirectional list class header file
+ * @author Alexander Gelfenbain
+ * @version 1.0
+ *
+ */
+
+#ifndef __CLIST_H
+#define __CLIST_H
+
+#ifdef __cplusplus
+extern "C"{
+#endif
+
+#include <glib.h>
+
+/*
+ * List of void * pointers
+ */
+
+typedef struct _list *list;
+
+/*- constructors and a destructor */
+list listNewEmpty(void);
+list listNewCopy(list);
+list listNewConcat(list, list);                             /* concatenates elements in two lists and creates a new list with them */
+void listDispose(list);
+void listSetElementDtor(list, GDestroyNotify f);           /*- this function will be executed when the element is removed via listRemove() or listClear() */
+
+/*- assignment */
+list listCopy(list to, list from);
+
+/*- queries */
+void * listCurrent(list);
+int    listCount(list);
+int    listIsEmpty(list);
+int    listAtFirst(list);
+int    listAtLast(list);
+int    listPosition(list);                        /* Expensive! */
+
+/*- search */
+int    listFind(list, void *);                    /* Returns true/false */
+
+/*- positioning functions */
+/*- return the number of elements by which the current position in the list changes */
+int    listNext(list);
+int    listPrev(list);
+int    listSkipForward(list, int n);
+int    listSkipBackward(list, int n);
+int    listToFirst(list);
+int    listToLast(list);
+int    listPositionAt(list, int n);               /* Expensive! */
+
+/*- adding and removing elements */
+list   listAppend(list, void *);
+list   listPrepend(list, void *);
+list   listInsertAfter(list, void *);
+list   listInsertBefore(list, void *);
+list   listConcat(list lhs, list rhs);            /* appends all elements of rhs to lhs and returns lhs */
+                                                  
+list   listRemove(list);                          /* removes the current element */
+list   listClear(list);                           /* removes all elements */
+
+/*- forall */
+void   listForAll(list, void (*f)(void *));
+
+/*- conversion */
+void **listToArray(list);                         /* XXX listToArray does not duplicate the elements, just copies pointers to them */
+                                                  /* so you can't call listRemove(), listClear(), or listDispose() until you are done */
+                                                  /* with the array. */
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /* __CLIST_H */
diff --git a/src/ttsubset/sft.c b/src/ttsubset/sft.c
new file mode 100644 (file)
index 0000000..249fbdf
--- /dev/null
@@ -0,0 +1,3570 @@
+/*
+ * Copyright © 2002, 2003 Sun Microsystems, Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ *
+ * 3. Neither the name of Sun Microsystems, Inc. nor the names of 
+ *    contributors may be used to endorse or promote products derived from
+ *    this software without specific prior written permission.
+ *
+ * This software is provided "AS IS," without a warranty of any kind.
+ *
+ * ALL EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES,
+ * INCLUDING ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A
+ * PARTICULAR PURPOSE OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED.
+ * SUN AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES OR
+ * LIABILITIES SUFFERED BY LICENSEE AS A RESULT OF OR RELATING TO USE,
+ * MODIFICATION OR DISTRIBUTION OF THE SOFTWARE OR ITS DERIVATIVES.
+ * IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST REVENUE,
+ * PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL,
+ * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE
+ * THEORY OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE
+ * SOFTWARE, EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
+ *
+ */
+
+/* $Id$ */
+/* @(#)sft.c 1.17 03/01/08 SMI */
+
+/*
+ * @file sft.c
+ * @brief Sun Font Tools
+ * @author Alexander Gelfenbain <adg@sun.com>
+ * @version 1.0
+ */
+
+#include <assert.h>
+#include <stdlib.h>
+#include <string.h>
+#include <fcntl.h>
+#include "sft.h"
+#ifdef USE_GSUB
+#include "gsub.h"
+#endif
+#if ! (defined(NO_TTCR) && defined(NO_TYPE42))
+#include "ttcr.h"
+#endif
+#ifdef NO_LIST
+#include "list.h"             /* list.h does not get included in the sft.h */
+#endif
+#ifndef NO_MAPPERS            /* include MapChar() and MapString() */
+#include "xlat.h"
+#endif
+#if ! (defined(NO_TYPE3) && defined(NO_TYPE42))
+#include "crc32.h"
+#endif
+
+#ifdef TEST7
+#include <ctype.h>
+#endif
+
+/*- module identification */
+
+const char *modname  = "SunTypeTools-TT";
+const char *modver   = "1.0";
+const char *modextra = "gelf";
+
+/*- private functions, constants and data types */ /*FOLD00*/
+
+enum PathSegmentType {
+    PS_NOOP      = 0,
+    PS_MOVETO    = 1,
+    PS_LINETO    = 2,
+    PS_CURVETO   = 3,
+    PS_CLOSEPATH = 4
+};
+    
+typedef struct {
+    int type;
+    int x1, y1;
+    int x2, y2;
+    int x3, y3;
+} PSPathElement;
+
+
+#define HFORMAT_LINELEN 64
+
+typedef struct {
+    FILE *o;
+    char buffer[HFORMAT_LINELEN];
+    int bufpos;
+    int total;
+} HexFmt;
+
+typedef struct {
+    guint32 nGlyphs;           /* number of glyphs in the font + 1 */
+    guint32 *offs;             /* array of nGlyphs offsets */
+} GlyphOffsets;
+
+/* private tags */
+static const guint32 TTFontClassTag = 0x74746663;  /* 'ttfc' */
+
+static const guint32 T_true = 0x74727565;        /* 'true' */
+static const guint32 T_ttcf = 0x74746366;        /* 'ttcf' */
+
+/* standard TrueType table tags and their ordinal numbers */
+static const guint32 T_maxp = 0x6D617870;    static const guint32 O_maxp = 0;     /* 'maxp' */
+static const guint32 T_glyf = 0x676C7966;    static const guint32 O_glyf = 1;     /* 'glyf' */
+static const guint32 T_head = 0x68656164;    static const guint32 O_head = 2;     /* 'head' */
+static const guint32 T_loca = 0x6C6F6361;    static const guint32 O_loca = 3;     /* 'loca' */
+static const guint32 T_name = 0x6E616D65;    static const guint32 O_name = 4;     /* 'name' */
+static const guint32 T_hhea = 0x68686561;    static const guint32 O_hhea = 5;     /* 'hhea' */
+static const guint32 T_hmtx = 0x686D7478;    static const guint32 O_hmtx = 6;     /* 'hmtx' */
+static const guint32 T_cmap = 0x636D6170;    static const guint32 O_cmap = 7;     /* 'cmap' */
+static const guint32 T_vhea = 0x76686561;    static const guint32 O_vhea = 8;     /* 'vhea' */
+static const guint32 T_vmtx = 0x766D7478;    static const guint32 O_vmtx = 9;     /* 'vmtx' */
+static const guint32 T_OS2  = 0x4F532F32;    static const guint32 O_OS2  = 10;    /* 'OS/2' */
+static const guint32 T_post = 0x706F7374;    static const guint32 O_post = 11;    /* 'post' */
+static const guint32 T_kern = 0x6B65726E;    static const guint32 O_kern = 12;    /* 'kern' */
+static const guint32 T_cvt  = 0x63767420;    static const guint32 O_cvt  = 13;    /* 'cvt_' - only used in TT->TT generation */
+static const guint32 T_prep = 0x70726570;    static const guint32 O_prep = 14;    /* 'prep' - only used in TT->TT generation */
+static const guint32 T_fpgm = 0x6670676D;    static const guint32 O_fpgm = 15;    /* 'fpgm' - only used in TT->TT generation */
+static const guint32 T_gsub = 0x47535542;    static const guint32 O_gsub = 16;    /* 'GSUB' */
+#define NUM_TAGS 17
+
+#define LAST_URANGE_BIT 69
+const char *ulcodes[LAST_URANGE_BIT+2] = {
+    /*  0   */  "Basic Latin",
+    /*  1   */  "Latin-1 Supplement",
+    /*  2   */  "Latin Extended-A",
+    /*  3   */  "Latin Extended-B",
+    /*  4   */  "IPA Extensions",
+    /*  5   */  "Spacing Modifier Letters",
+    /*  6   */  "Combining Diacritical Marks",
+    /*  7   */  "Basic Greek",
+    /*  8   */  "Greek Symbols And Coptic",
+    /*  9   */  "Cyrillic",
+    /*  10  */  "Armenian",
+    /*  11  */  "Basic Hebrew",
+    /*  12  */  "Hebrew Extended (A and B blocks combined)",
+    /*  13  */  "Basic Arabic",
+    /*  14  */  "Arabic Extended",
+    /*  15  */  "Devanagari",
+    /*  16  */  "Bengali",
+    /*  17  */  "Gurmukhi",
+    /*  18  */  "Gujarati",
+    /*  19  */  "Oriya",
+    /*  20  */  "Tamil",
+    /*  21  */  "Telugu",
+    /*  22  */  "Kannada",
+    /*  23  */  "Malayalam",
+    /*  24  */  "Thai",
+    /*  25  */  "Lao",
+    /*  26  */  "Basic Georgian",
+    /*  27  */  "Georgian Extended",
+    /*  28  */  "Hangul Jamo",
+    /*  29  */  "Latin Extended Additional",
+    /*  30  */  "Greek Extended",
+    /*  31  */  "General Punctuation",
+    /*  32  */  "Superscripts And Subscripts",
+    /*  33  */  "Currency Symbols",
+    /*  34  */  "Combining Diacritical Marks For Symbols",
+    /*  35  */  "Letterlike Symbols",
+    /*  36  */  "Number Forms",
+    /*  37  */  "Arrows",
+    /*  38  */  "Mathematical Operators",
+    /*  39  */  "Miscellaneous Technical",
+    /*  40  */  "Control Pictures",
+    /*  41  */  "Optical Character Recognition",
+    /*  42  */  "Enclosed Alphanumerics",
+    /*  43  */  "Box Drawing",
+    /*  44  */  "Block Elements",
+    /*  45  */  "Geometric Shapes",
+    /*  46  */  "Miscellaneous Symbols",
+    /*  47  */  "Dingbats",
+    /*  48  */  "CJK Symbols And Punctuation",
+    /*  49  */  "Hiragana",
+    /*  50  */  "Katakana",
+    /*  51  */  "Bopomofo",
+    /*  52  */  "Hangul Compatibility Jamo",
+    /*  53  */  "CJK Miscellaneous",
+    /*  54  */  "Enclosed CJK Letters And Months",
+    /*  55  */  "CJK Compatibility",
+    /*  56  */  "Hangul",
+    /*  57  */  "Reserved for Unicode SubRanges",
+    /*  58  */  "Reserved for Unicode SubRanges",
+    /*  59  */  "CJK Unified Ideographs",
+    /*  60  */  "Private Use Area",
+    /*  61  */  "CJK Compatibility Ideographs",
+    /*  62  */  "Alphabetic Presentation Forms",
+    /*  63  */  "Arabic Presentation Forms-A",
+    /*  64  */  "Combining Half Marks",
+    /*  65  */  "CJK Compatibility Forms",
+    /*  66  */  "Small Form Variants",
+    /*  67  */  "Arabic Presentation Forms-B",
+    /*  68  */  "Halfwidth And Fullwidth Forms",
+    /*  69  */  "Specials",
+    /*70-127*/  "Reserved for Unicode SubRanges"
+};
+
+/*- inline functions */ /*FOLD01*/
+#ifdef __GNUC__
+#define _inline static __inline__
+#else
+#define _inline static
+#endif
+
+_inline void *smalloc(size_t size)
+{
+    void *res = malloc(size);
+    assert(res != 0);
+    return res;
+}
+
+_inline void *scalloc(size_t n, size_t size)
+{
+    void *res = calloc(n, size);
+    assert(res != 0);
+    return res;
+}
+
+#if !defined(STSF)
+
+_inline guint32 mkTag(guint8 a, guint8 b, guint8 c, guint8 d) {
+    return (a << 24) | (b << 16) | (c << 8) | d;
+}
+
+/*- Data access macros for data stored in big-endian or little-endian format */
+_inline gint16 GetInt16(const guint8 *ptr, size_t offset, int bigendian)
+{
+    gint16 t;
+    assert(ptr != 0);
+    
+    if (bigendian) {
+        t = (ptr+offset)[0] << 8 | (ptr+offset)[1];
+    } else {
+        t = (ptr+offset)[1] << 8 | (ptr+offset)[0];
+    }
+        
+    return t;
+}
+
+_inline guint16 GetUInt16(const guint8 *ptr, size_t offset, int bigendian)
+{
+    guint16 t;
+    assert(ptr != 0);
+
+    if (bigendian) {
+        t = (ptr+offset)[0] << 8 | (ptr+offset)[1];
+    } else {
+        t = (ptr+offset)[1] << 8 | (ptr+offset)[0];
+    }
+
+    return t;
+}
+
+_inline gint32  GetInt32(const guint8 *ptr, size_t offset, int bigendian)
+{
+    gint32 t;
+    assert(ptr != 0);
+    
+    if (bigendian) {
+        t = (ptr+offset)[0] << 24 | (ptr+offset)[1] << 16 |
+            (ptr+offset)[2] << 8  | (ptr+offset)[3];
+    } else {
+        t = (ptr+offset)[3] << 24 | (ptr+offset)[2] << 16 |
+            (ptr+offset)[1] << 8  | (ptr+offset)[0];
+    }
+        
+    return t;
+}
+
+_inline guint32 GetUInt32(const guint8 *ptr, size_t offset, int bigendian)
+{
+    guint32 t;
+    assert(ptr != 0);
+    
+
+    if (bigendian) {
+        t = (ptr+offset)[0] << 24 | (ptr+offset)[1] << 16 |
+            (ptr+offset)[2] << 8  | (ptr+offset)[3];
+    } else {
+        t = (ptr+offset)[3] << 24 | (ptr+offset)[2] << 16 |
+            (ptr+offset)[1] << 8  | (ptr+offset)[0];
+    }
+        
+    return t;
+}
+
+_inline void PutInt16(gint16 val, guint8 *ptr, size_t offset, int bigendian)
+{
+    assert(ptr != 0);
+
+    if (bigendian) {
+        ptr[offset] = (val >> 8) & 0xFF;
+        ptr[offset+1] = val & 0xFF;
+    } else {
+        ptr[offset+1] = (val >> 8) & 0xFF;
+        ptr[offset] = val & 0xFF;
+    }
+
+}
+#else
+#include "inlines.h"
+#endif
+
+#if G_BYTE_ORDER == G_BIG_ENDIAN
+#define Int16FromMOTA(a) (a)
+#else
+static guint16 Int16FromMOTA(guint16 a) {
+  return (guint16) (((guint8)((a) >> 8)) | ((guint8)(a) << 8));
+}
+#endif
+
+_inline F16Dot16 fixedMul(F16Dot16 a, F16Dot16 b)
+{
+    unsigned int a1, b1;
+    unsigned int a2, b2;
+    F16Dot16 res;
+    int sign;
+
+    sign = (a & 0x80000000) ^ (b & 0x80000000);
+    if (a < 0) a = -a;
+    if (b < 0) b = -b;
+
+    a1 = a >> 16;
+    b1 = a & 0xFFFF;
+    a2 = b >> 16;
+    b2 = b & 0xFFFF;
+
+    res = a1 * a2;
+
+    /* if (res  > 0x7FFF) assert(!"fixedMul: F16Dot16 overflow"); */
+
+    res <<= 16;
+    res += a1 * b2 + b1 * a2 + ((b1 * b2) >> 16);
+
+    return sign ? -res : res;
+}
+
+
+_inline F16Dot16 fixedDiv(F16Dot16 a, F16Dot16 b)
+{
+    unsigned int f, r;
+    F16Dot16 res;
+    int sign;
+
+    sign = (a & 0x80000000) ^ (b & 0x80000000);
+    if (a < 0) a = -a;
+    if (b < 0) b = -b;
+
+    f = a / b;
+    r = a % b;
+
+    /* if (f > 0x7FFFF) assert(!"fixedDiv: F16Dot16 overflow"); */
+
+    while (r > 0xFFFF) {
+        r >>= 1;
+        b >>= 1;
+    }
+
+    res = (f << 16) + (r << 16) / b;
+
+    return sign ? -res : res;
+}
+
+/*- returns a * b / c -*/
+/* XXX provide a real implementation that preserves accuracy */
+_inline F16Dot16 fixedMulDiv(F16Dot16 a, F16Dot16 b, F16Dot16 c)
+{
+    F16Dot16 res;
+
+    res = fixedMul(a, b);
+    return fixedDiv(res, c);
+}
+
+/*- Translate units from TT to PS (standard 1/1000) -*/
+_inline int XUnits(int unitsPerEm, int n)
+{
+    return (n * 1000) / unitsPerEm;
+}
+
+_inline const char *UnicodeRangeName(guint16 bit)
+{
+  if (bit > LAST_URANGE_BIT) bit = LAST_URANGE_BIT+1;
+  
+  return ulcodes[bit];
+}
+
+
+#if 0
+/* It would have been nice if it worked, but I found so many fonts that don't
+ * follow the TrueType spec (sorted table directory) that I have to re-write this
+ * stuff.
+ */
+
+static guint8 *getTDEntry(TrueTypeFont *ttf, guint32 tag) /*FOLD01*/
+{
+    int l = 0, r = ttf->ntables-1, i;
+    guint32 t;
+
+    do {
+        i = (l + r) >> 1;
+        t = GetUInt32(ttf->ptr + 12, i << 4, 1);
+        if (tag >= t) l = i + 1;
+        if (tag <= t) r = i - 1;
+    } while (l <= r);
+
+    if (l - r == 2) {
+        return ttf->ptr + 12 + 16 * (l - 1);
+    }
+    return 0;
+}
+    
+static guint8 *getTable(TrueTypeFont *ttf, guint32 tag) /*FOLD01*/
+{
+    guint8 *ptr = getTDEntry(ttf, tag);
+    if (!ptr) return 0;
+    
+    return ttf->ptr + GetUInt32(ptr, 8, 1);
+}
+
+static guint32 getTableSize(TrueTypeFont *ttf, guint32 tag) /*FOLD01*/
+{
+    guint8 *ptr = getTDEntry(ttf, tag);
+    if (!ptr) return 0;
+
+    return GetUInt32(ptr, 12, 1);
+}
+
+#endif
+
+_inline guint8 *getTable(TrueTypeFont *ttf, guint32 ord)
+{
+    return ttf->tables[ord];
+}
+
+_inline guint32 getTableSize(TrueTypeFont *ttf, guint32 ord)
+{
+    return ttf->tlens[ord];
+}
+
+static guint32 tagToOrd(guint32 tag)
+{
+    if (tag == T_maxp) {
+        return O_maxp;
+    } else if (tag == T_glyf) {
+        return O_glyf;
+    } else if (tag == T_head) {
+        return O_head;
+    } else if (tag == T_loca) {
+        return O_loca;
+    } else if (tag == T_name) {
+        return O_name;
+    } else if (tag == T_hhea) {
+        return O_hhea;
+    } else if (tag == T_hmtx) {
+        return O_hmtx;
+    } else if (tag == T_cmap) {
+        return O_cmap;
+    } else if (tag == T_vhea) {
+        return O_vhea;
+    } else if (tag == T_vmtx) {
+        return O_vmtx;
+    } else if (tag == T_OS2) {
+        return O_OS2;
+    } else if (tag == T_post) {
+        return O_post;
+    } else if (tag == T_kern) {
+        return O_kern;
+    } else if (tag == T_cvt) {
+        return O_cvt;
+    } else if (tag == T_prep) {
+        return O_prep;
+    } else if (tag == T_fpgm) {
+        return O_fpgm;
+    } else if (tag == T_gsub) {
+        return O_gsub;
+    }
+    return 0xFFFFFFFF;
+}
+
+#ifndef NO_TYPE42
+/* Hex Formatter functions */
+static char HexChars[] = "0123456789ABCDEF";
+
+static HexFmt *HexFmtNew(FILE *outf)
+{
+    HexFmt *res = smalloc(sizeof(HexFmt));
+    res->bufpos = res->total = 0;
+    res->o = outf;
+    return res;
+}
+
+static void HexFmtFlush(HexFmt *_this)
+{
+    if (_this->bufpos) {
+        fwrite(_this->buffer, 1, _this->bufpos, _this->o);
+        _this->bufpos = 0;
+    }
+}
+
+
+_inline void HexFmtOpenString(HexFmt *_this)
+{
+    fputs("<\n", _this->o);
+}
+
+_inline void HexFmtCloseString(HexFmt *_this)
+{
+    HexFmtFlush(_this);
+    fputs("00\n>\n", _this->o);
+}
+
+_inline void HexFmtDispose(HexFmt *_this)
+{
+    HexFmtFlush(_this);
+    free(_this);
+}
+
+static void HexFmtBlockWrite(HexFmt *_this, const void *ptr, off_t size)
+{
+    guint8 Ch;
+    off_t i;
+  
+    if (_this->total + size > 65534) {
+        HexFmtFlush(_this);
+        HexFmtCloseString(_this);
+        _this->total = 0;
+        HexFmtOpenString(_this);
+    }
+    for (i=0; i<size; i++) {
+        Ch = ((guint8 *) ptr)[i];
+        _this->buffer[_this->bufpos++] = HexChars[Ch >> 4];
+        _this->buffer[_this->bufpos++] = HexChars[Ch & 0xF];
+        if (_this->bufpos == HFORMAT_LINELEN) {
+            HexFmtFlush(_this);
+            fputc('\n', _this->o);
+        }
+
+    }
+    _this->total += size;
+}
+#endif
+
+
+
+/* Outline Extraction functions */ /*FOLD01*/
+
+/* fills the aw and lsb entries of the TTGlyphMetrics structure from hmtx table -*/
+static void GetMetrics(TrueTypeFont *ttf, guint32 glyphID, TTGlyphMetrics *metrics)
+{
+    guint8 *table = getTable(ttf, O_hmtx);
+
+    metrics->aw = metrics->lsb = metrics->ah = metrics->tsb = 0;
+    if (!table || !ttf->numberOfHMetrics) return;
+
+    if (glyphID < ttf->numberOfHMetrics) {
+        metrics->aw  = GetUInt16(table, 4 * glyphID, 1);
+        metrics->lsb = GetInt16(table, 4 * glyphID + 2, 1);
+    } else {
+        metrics->aw  = GetUInt16(table, 4 * (ttf->numberOfHMetrics - 1), 1);
+        metrics->lsb = GetInt16(table + ttf->numberOfHMetrics * 4, (glyphID - ttf->numberOfHMetrics) * 2, 1);
+    }
+
+    table = getTable(ttf, O_vmtx);
+    if (!table || !ttf->numOfLongVerMetrics) return;
+
+    if (glyphID < ttf->numOfLongVerMetrics) {
+        metrics->ah  = GetUInt16(table, 4 * glyphID, 1);
+        metrics->tsb = GetInt16(table, 4 * glyphID + 2, 1);
+    } else {
+        metrics->ah  = GetUInt16(table, 4 * (ttf->numOfLongVerMetrics - 1), 1);
+        metrics->tsb = GetInt16(table + ttf->numOfLongVerMetrics * 4, (glyphID - ttf->numOfLongVerMetrics) * 2, 1);
+    }
+}
+
+FUnitBBox *GetTTGlyphBoundingBoxes(TrueTypeFont *ttf)
+{
+    guint8 *table = getTable(ttf, O_glyf);
+    FUnitBBox *b = calloc(ttf->nglyphs, sizeof(FUnitBBox));
+    
+    if (b != NULL) {
+        int i;
+
+        for (i = 0; i< ttf->nglyphs; i++) {
+            guint8 *ptr = table + ttf->goffsets[i];
+            b[i].xMin = XUnits(ttf->unitsPerEm, GetInt16(ptr, 2, 1));
+            b[i].yMin = XUnits(ttf->unitsPerEm, GetInt16(ptr, 4, 1));
+            b[i].xMax = XUnits(ttf->unitsPerEm, GetInt16(ptr, 6, 1));
+            b[i].yMax = XUnits(ttf->unitsPerEm, GetInt16(ptr, 8, 1));
+        }
+    }
+
+    return b;
+}
+
+
+static int GetTTGlyphOutline(TrueTypeFont *, guint32 , ControlPoint **, TTGlyphMetrics *, list );
+
+/* returns the number of control points, allocates the pointArray */
+static int GetSimpleTTOutline(TrueTypeFont *ttf, guint32 glyphID, ControlPoint **pointArray, TTGlyphMetrics *metrics) /*FOLD02*/
+{
+    guint8 *table = getTable(ttf, O_glyf);
+    guint8 *ptr, *p, flag, n;
+    gint16 numberOfContours;
+    guint16 t, instLen, lastPoint=0;
+    int i, j, z;
+    ControlPoint* pa;
+
+    *pointArray = NULL;
+
+    /* printf("GetSimpleTTOutline(%d)\n", glyphID); */
+
+    if (glyphID >= ttf->nglyphs) return 0;                            /*- glyph is not present in the font */
+    ptr = table + ttf->goffsets[glyphID];
+    if ((numberOfContours = GetInt16(ptr, 0, 1)) <= 0) return 0;      /*- glyph is not simple */
+
+    if (metrics) {                                                    /*- GetCompoundTTOutline() calls this function with NULL metrics -*/
+        metrics->xMin = GetInt16(ptr, 2, 1);
+        metrics->yMin = GetInt16(ptr, 4, 1);
+        metrics->xMax = GetInt16(ptr, 6, 1);
+        metrics->yMax = GetInt16(ptr, 8, 1);
+        GetMetrics(ttf, glyphID, metrics);
+    }
+
+    /* determine the last point and be extra safe about it. But probably this code is not needed */
+    
+    for (i=0; i<numberOfContours; i++) {
+        if ((t = GetUInt16(ptr, 10+i*2, 1)) > lastPoint) lastPoint = t;
+    }
+
+    instLen = GetUInt16(ptr, 10 + numberOfContours*2, 1);
+    p = ptr + 10 + 2 * numberOfContours + 2 + instLen;
+    pa = calloc(lastPoint+1, sizeof(ControlPoint));
+
+    i = 0;
+    while (i <= lastPoint) {
+        pa[i++].flags = flag = (guint32) *p++;
+        if (flag & 8) {                                     /*- repeat flag */
+            n = *p++;
+            for (j=0; j<n; j++) {
+                if (i > lastPoint) {                        /*- if the font is really broken */
+                    free(pa);
+                    return 0;
+                }
+                pa[i++].flags = flag;
+            }
+        }
+    }
+
+    /*- Process the X coordinate */
+    z = 0;
+    for (i = 0; i <= lastPoint; i++) {
+        if (pa[i].flags & 0x02) {
+            if (pa[i].flags & 0x10) {
+                z += (int) (*p++);
+            } else {
+                z -= (int) (*p++);
+            }
+        } else if ( !(pa[i].flags & 0x10)) {
+            z += GetInt16(p, 0, 1);
+            p += 2;
+        }
+        pa[i].x = z;
+    }
+
+    /*- Process the Y coordinate */
+    z = 0;
+    for (i = 0; i <= lastPoint; i++) {
+        if (pa[i].flags & 0x04) {
+            if (pa[i].flags & 0x20) {
+                z += *p++;
+            } else {
+                z -= *p++;
+            }
+        } else if ( !(pa[i].flags & 0x20)) {
+            z += GetInt16(p, 0, 1);
+            p += 2;
+        }
+        pa[i].y = z;
+    }
+
+    for (i=0; i<numberOfContours; i++) {
+        pa[GetUInt16(ptr, 10 + i * 2, 1)].flags |= 0x00008000;      /*- set the end contour flag */
+    }
+
+    *pointArray = pa;
+    return lastPoint + 1;
+}
+
+static int GetCompoundTTOutline(TrueTypeFont *ttf, guint32 glyphID, ControlPoint **pointArray, TTGlyphMetrics *metrics, list glyphlist) /*FOLD02*/
+{
+    guint16 flags, index;
+    gint16 e, f, numberOfContours;
+    guint8 *table = getTable(ttf, O_glyf);
+    guint8 *ptr;
+    list myPoints;
+    ControlPoint *nextComponent, *pa;
+    int i, np;
+    F16Dot16 a = 0x10000, b = 0, c = 0, d = 0x10000, m, n, abs1, abs2, abs3;
+
+    *pointArray = NULL;
+    /* printf("GetCompoundTTOutline(%d)\n", glyphID); */
+
+    if (glyphID >= ttf->nglyphs) {                          /*- incorrect glyphID */
+        return 0;
+    }
+    ptr = table + ttf->goffsets[glyphID];
+    if ((numberOfContours = GetInt16(ptr, 0, 1)) != -1) {   /*- glyph is not compound */
+        return 0;
+    }
+
+    myPoints = listNewEmpty();
+    listSetElementDtor(myPoints, free);
+
+    if (metrics) {
+        metrics->xMin = GetInt16(ptr, 2, 1);
+        metrics->yMin = GetInt16(ptr, 4, 1);
+        metrics->xMax = GetInt16(ptr, 6, 1);
+        metrics->yMax = GetInt16(ptr, 8, 1);
+        GetMetrics(ttf, glyphID, metrics);
+    }
+
+    ptr += 10;
+
+    do {
+        flags = GetUInt16(ptr, 0, 1);
+        /* printf("flags: 0x%X\n", flags); */
+        index = GetUInt16(ptr, 2, 1);
+        ptr += 4;
+
+        if (listFind(glyphlist, (void *) (int) index)) {
+#ifdef DEBUG
+            fprintf(stderr, "Endless loop found in a compound glyph.\n");
+            fprintf(stderr, "%d -> ", index);
+            listToFirst(glyphlist);
+            fprintf(stderr," [");
+            do {
+                fprintf(stderr,"%d ", (int) listCurrent(glyphlist));
+            } while (listNext(glyphlist));
+            fprintf(stderr,"]\n");
+        /**/
+#endif
+        }
+
+        listAppend(glyphlist, (void *) (int) index);
+
+#ifdef DEBUG2
+        fprintf(stderr,"glyphlist: += %d\n", index);
+#endif
+           
+        if ((np = GetTTGlyphOutline(ttf, index, &nextComponent, NULL, glyphlist)) == 0) {
+            /* XXX that probably indicates a corrupted font */
+#ifdef DEBUG
+            fprintf(stderr, "An empty compound!\n");
+            /* assert(!"An empty compound"); */
+#endif
+        }
+
+        listToLast(glyphlist);
+#ifdef DEBUG2
+        listToFirst(glyphlist);
+        fprintf(stderr,"%d [", listCount(glyphlist));
+        if (!listIsEmpty(glyphlist)) {
+            do {
+                fprintf(stderr,"%d ", (int) listCurrent(glyphlist));
+            } while (listNext(glyphlist));
+        }
+        fprintf(stderr, "]\n");
+        fprintf(stderr, "glyphlist: -= %d\n", (int) listCurrent(glyphlist));
+        
+#endif
+        listRemove(glyphlist);
+
+        if (flags & USE_MY_METRICS) {
+            if (metrics) GetMetrics(ttf, index, metrics);
+        }
+
+        if (flags & ARG_1_AND_2_ARE_WORDS) {
+            e = GetInt16(ptr, 0, 1);
+            f = GetInt16(ptr, 2, 1);
+            /* printf("ARG_1_AND_2_ARE_WORDS: %d %d\n", e & 0xFFFF, f & 0xFFFF); */
+            ptr += 4;
+        } else {
+            if (flags & ARGS_ARE_XY_VALUES) {     /* args are signed */
+                e = (gint8) *ptr++;
+                f = (gint8) *ptr++;
+                /* printf("ARGS_ARE_XY_VALUES: %d %d\n", e & 0xFF, f & 0xFF); */
+            } else {                              /* args are unsigned */
+                /* printf("!ARGS_ARE_XY_VALUES\n"); */
+                e = *ptr++;
+                f = *ptr++;
+            }
+
+        }
+
+        a = d = 0x10000;
+        b = c = 0;
+
+        if (flags & WE_HAVE_A_SCALE) {
+#ifdef DEBUG2
+            fprintf(stderr, "WE_HAVE_A_SCALE\n");
+#endif
+            a = GetInt16(ptr, 0, 1) << 2;
+            d = a;
+            ptr += 2;
+        } else if (flags & WE_HAVE_AN_X_AND_Y_SCALE) {
+#ifdef DEBUG2
+            fprintf(stderr, "WE_HAVE_AN_X_AND_Y_SCALE\n");
+#endif
+            a = GetInt16(ptr, 0, 1) << 2;
+            d = GetInt16(ptr, 2, 1) << 2;
+            ptr += 4;
+        } else if (flags & WE_HAVE_A_TWO_BY_TWO) {
+#ifdef DEBUG2
+            fprintf(stderr, "WE_HAVE_A_TWO_BY_TWO\n");
+#endif
+            a = GetInt16(ptr, 0, 1) << 2;
+            b = GetInt16(ptr, 2, 1) << 2;
+            c = GetInt16(ptr, 4, 1) << 2;
+            d = GetInt16(ptr, 6, 1) << 2;
+            ptr += 8;
+        }
+
+        abs1 = (a < 0) ? -a : a;
+        abs2 = (b < 0) ? -b : b;
+        m    = (abs1 > abs2) ? abs1 : abs2;
+        abs3 = abs1 - abs2;
+        if (abs3 < 0) abs3 = -abs3;
+        if (abs3 <= 33) m *= 2;
+
+        abs1 = (c < 0) ? -c : c;
+        abs2 = (d < 0) ? -d : d;
+        n    = (abs1 > abs2) ? abs1 : abs2;
+        abs3 = abs1 - abs2;
+        if (abs3 < 0) abs3 = -abs3;
+        if (abs3 <= 33) n *= 2;
+
+        if (!ARGS_ARE_XY_VALUES) {      /* match the points */
+            assert(!"ARGS_ARE_XY_VALUES is not implemented!!!\n");
+        }
+
+#ifdef DEBUG2
+        fprintf(stderr, "a: %f, b: %f, c: %f, d: %f, e: %f, f: %f, m: %f, n: %f\n",
+                ((double) a) / 65536,
+                ((double) b) / 65536,
+                ((double) c) / 65536,
+                ((double) d) / 65536,
+                ((double) e) / 65536,
+                ((double) f) / 65536,
+                ((double) m) / 65536,
+                ((double) n) / 65536);
+#endif
+
+        for (i=0; i<np; i++) {
+            F16Dot16 t;
+            ControlPoint *cp = malloc(sizeof(ControlPoint));
+            cp->flags = nextComponent[i].flags;
+            t = fixedMulDiv(a, nextComponent[i].x << 16, m) + fixedMulDiv(c, nextComponent[i].y << 16, m) + (e << 16);
+            cp->x = fixedMul(t, m) >> 16;
+            t = fixedMulDiv(b, nextComponent[i].x << 16, n) + fixedMulDiv(d, nextComponent[i].y << 16, n) + (f << 16);
+            cp->y = fixedMul(t, n) >> 16;
+
+#ifdef DEBUG2
+            fprintf(stderr, "( %d %d ) -> ( %d %d )\n", nextComponent[i].x, nextComponent[i].y, cp->x, cp->y);
+#endif
+            
+            listAppend(myPoints, cp);
+        }
+            
+        free(nextComponent);
+
+    } while (flags & MORE_COMPONENTS);
+
+    
+
+    np = listCount(myPoints);
+
+    pa = calloc(np, sizeof(ControlPoint));
+    assert(pa != 0);
+    listToFirst(myPoints);
+    for (i=0; i<np; i++) {
+        memcpy(pa+i, listCurrent(myPoints), sizeof(ControlPoint));
+        listNext(myPoints);
+    }
+    listDispose(myPoints);
+
+    *pointArray = pa;
+    return np;
+}
+
+/* NOTE: GetTTGlyphOutline() returns -1 if the glyphID is incorrect, 
+ * but Get{Simple|Compound}GlyphOutline returns 0 in such a case.
+ *
+ * NOTE: glyphlist is the stack of glyphs traversed while constructing
+ * a composite glyph. This is a safequard against endless recursion
+ * in corrupted fonts.
+ */
+static int GetTTGlyphOutline(TrueTypeFont *ttf, guint32 glyphID, ControlPoint **pointArray, TTGlyphMetrics *metrics, list glyphlist)
+{
+    guint8 *ptr, *table = getTable(ttf, O_glyf);
+    gint16 numberOfContours;
+    int length;
+    int res;
+    *pointArray = NULL;
+
+    if (metrics) {
+        memset(metrics, 0, sizeof(TTGlyphMetrics));         /*- metrics is initialized to all zeroes */
+    }
+
+    if (glyphID >= ttf->nglyphs) return -1;                 /**/
+    
+    ptr = table + ttf->goffsets[glyphID];
+    length = ttf->goffsets[glyphID+1] - ttf->goffsets[glyphID];
+
+    if (length == 0) {                                      /*- empty glyphs still have hmtx and vmtx metrics values */
+        if (metrics) GetMetrics(ttf, glyphID, metrics);
+        return 0;
+    }
+    
+    numberOfContours = GetInt16(ptr, 0, 1);
+
+    if (numberOfContours >= 0) {
+        res=GetSimpleTTOutline(ttf, glyphID, pointArray, metrics);
+    } else {
+        int glyphlistFlag = 0;
+        if (!glyphlist) {
+            glyphlistFlag = 1;
+            glyphlist = listNewEmpty();
+            listAppend(glyphlist, (void *) glyphID);
+        }
+        res = GetCompoundTTOutline(ttf, glyphID, pointArray, metrics, glyphlist);
+        if (glyphlistFlag) {
+            glyphlistFlag = 0;
+            listDispose(glyphlist);
+            glyphlist = NULL;
+        }
+    }
+
+#ifdef DEBUG3
+    {
+        int i;
+        FILE *out = fopen("points.dat", "ab");
+        assert(out != 0);
+        fprintf(out, "Glyph: %d\nPoints: %d\n", glyphID, res);
+        for (i=0; i<res; i++) {
+            fprintf(out, "%c ", ((*pointArray)[i].flags & 0x8000) ? 'X' : '.');
+            fprintf(out, "%c ", ((*pointArray)[i].flags & 1) ? '+' : '-');
+            fprintf(out, "%d %d\n", (*pointArray)[i].x, (*pointArray)[i].y);
+        }
+        fclose(out);
+    }
+#endif
+
+    return res;
+}
+
+#ifndef NO_TYPE3
+
+static PSPathElement *newPSPathElement(int t)
+{
+    PSPathElement *p = malloc(sizeof(PSPathElement));
+    assert(p != 0);
+
+    p->type = t;
+    return p;
+}
+
+/*- returns the number of items in the path -*/
+
+static int BSplineToPSPath(ControlPoint *srcA, int srcCount, PSPathElement **path)
+{
+    list pList = listNewEmpty();
+    int i = 0, pCount = 0;
+    PSPathElement *p;
+
+    int x0, y0, x1, y1, x2, y2, curx, cury;
+    int lastOff = 0;                                        /*- last point was off-contour */
+    int scflag = 1;                                         /*- start contour flag */
+    int ecflag = 0;                                         /*- end contour flag */
+    int cp = 0;                                             /*- current point */
+
+    listSetElementDtor(pList, free);
+    *path = 0;
+
+    /* if (srcCount > 0) for(;;) */
+    while (srcCount > 0) {                                  /*- srcCount does not get changed inside the loop. */
+        int StartContour, EndContour;
+
+        if (scflag) {
+            int l = cp;
+            StartContour = cp;
+            while (!(srcA[l].flags & 0x8000)) l++;
+            EndContour = l;
+            if (StartContour == EndContour) {
+                if (cp + 1 < srcCount) {
+                    cp++;
+                    continue;
+                } else {
+                    break;
+                }
+            }
+            p = newPSPathElement(PS_MOVETO);
+            if (!(srcA[cp].flags & 1)) {
+                if (!(srcA[EndContour].flags & 1)) {
+                    p->x1 = x0 = (srcA[cp].x + srcA[EndContour].x + 1) / 2;
+                    p->y1 = y0 = (srcA[cp].y + srcA[EndContour].y + 1) / 2;
+                } else {
+                    p->x1 = x0 = srcA[EndContour].x;
+                    p->y1 = y0 = srcA[EndContour].y;
+                }
+            } else {
+                p->x1 = x0 = srcA[cp].x;
+                p->y1 = y0 = srcA[cp].y;
+                cp++;
+            }
+            listAppend(pList, p);
+            lastOff = 0;
+            scflag = 0;
+        }
+
+        curx = srcA[cp].x;
+        cury = srcA[cp].y;
+
+        if (srcA[cp].flags & 1) {
+            if (lastOff) {
+                p = newPSPathElement(PS_CURVETO);
+                p->x1 = x0 + (2 * (x1 - x0) + 1) / 3;
+                p->y1 = y0 + (2 * (y1 - y0) + 1) / 3;
+                p->x2 = x1 + (curx - x1 + 1) / 3;
+                p->y2 = y1 + (cury - y1 + 1) / 3;
+                p->x3 = curx;
+                p->y3 = cury;
+                listAppend(pList, p);
+            } else {
+                if (!(x0 == curx && y0 == cury)) {                              /* eliminate empty lines */
+                    p = newPSPathElement(PS_LINETO);
+                    p->x1 = curx;
+                    p->y1 = cury;
+                    listAppend(pList, p);
+                }
+            }
+                
+            x0 = curx; y0 = cury; lastOff = 0;
+        } else {
+            if (lastOff) {
+                x2 = (x1 + curx + 1) / 2;
+                y2 = (y1 + cury + 1) / 2;
+                p = newPSPathElement(PS_CURVETO);
+                p->x1 = x0 + (2 * (x1 - x0) + 1) / 3;
+                p->y1 = y0 + (2 * (y1 - y0) + 1) / 3;
+                p->x2 = x1 + (x2 - x1 + 1) / 3;
+                p->y2 = y1 + (y2 - y1 + 1) / 3;
+                p->x3 = x2;
+                p->y3 = y2;
+                listAppend(pList, p);
+                x0 = x2; y0 = y2;
+                x1 = curx; y1 = cury;
+            } else {
+                x1 = curx; y1 = cury;
+            }
+            lastOff = true;
+        }
+
+        if (ecflag) {
+            listAppend(pList, newPSPathElement(PS_CLOSEPATH));
+            scflag = 1;
+            ecflag = 0;
+            cp = EndContour + 1;
+            if (cp >= srcCount) break;
+            continue;
+        }
+
+        if (cp == EndContour) {
+            cp = StartContour;
+            ecflag = true;
+        } else {
+            cp++;
+        }
+    }
+
+    if ((pCount = listCount(pList)) > 0) {
+        p = calloc(pCount, sizeof(PSPathElement));
+        assert(p != 0);
+        listToFirst(pList);
+        for (i=0; i<pCount; i++) {
+            memcpy(p + i, listCurrent(pList), sizeof(PSPathElement));
+            listNext(pList);
+        }
+        listDispose(pList);
+        *path = p;
+    }
+    
+    return pCount;
+}
+
+#endif
+
+/*- Extracts a string from the name table and allocates memory for it -*/
+
+static char *nameExtract(guint8 *name, int n, int dbFlag, guint16** ucs2result )
+{
+    int i;
+    char *res;
+    guint8 *ptr =  name + GetUInt16(name, 4, 1) + GetUInt16(name + 6, 12 * n + 10, 1);
+    int len = GetUInt16(name+6, 12 * n + 8, 1);
+
+    if (ucs2result) *ucs2result = NULL;
+    
+    if (dbFlag) {
+        res = malloc(1 + len/2);
+        assert(res != 0);
+        for (i = 0; i < len/2; i++) res[i] = *(ptr + i * 2 + 1);
+        res[len/2] = 0;
+        if (ucs2result) {
+            *ucs2result = malloc(len + 2);
+            for (i = 0; i < len/2; i++ ) (*ucs2result)[i] = GetUInt16(ptr, 2*i, 1);
+            (*ucs2result)[len/2] = 0;
+        }
+    } else {
+        res = malloc(1 + len);
+        assert(res != 0);
+        memcpy(res, ptr, len);
+        res[len] = 0;
+    }
+
+    return res;
+}
+
+static int findname(guint8 *name, guint16 n, guint16 platformID, guint16 encodingID, guint16 languageID, guint16 nameID)
+{
+    int l = 0, r = n-1, i;
+    guint32 t1, t2;
+    guint32 m1, m2;
+
+    if (n == 0) return -1;
+
+    m1 = (platformID << 16) | encodingID;
+    m2 = (languageID << 16) | nameID;
+
+    do {
+        i = (l + r) >> 1;
+        t1 = GetUInt32(name + 6, i * 12 + 0, 1);
+        t2 = GetUInt32(name + 6, i * 12 + 4, 1);
+
+        if (! ((m1 < t1) || ((m1 == t1) && (m2 < t2)))) l = i + 1;
+        if (! ((m1 > t1) || ((m1 == t1) && (m2 > t2)))) r = i - 1;
+    } while (l <= r);
+
+    if (l - r == 2) {
+        return l - 1;
+    }
+
+    return -1;
+}
+
+/* XXX marlett.ttf uses (3, 0, 1033) instead of (3, 1, 1033) and does not have any Apple tables. 
+ * Fix: if (3, 1, 1033) is not found - need to check for (3, 0, 1033)
+ *
+ * /d/fonts/ttzh_tw/Big5/Hanyi/ma6b5p uses (1, 0, 19) for English strings, instead of (1, 0, 0)
+ * and does not have (3, 1, 1033)
+ * Fix: if (1, 0, 0) and (3, 1, 1033) are not found need to look for (1, 0, *) - that will
+ * require a change in algorithm
+ *
+ * /d/fonts/fdltest/Korean/h2drrm has unsorted names and a an unknown (to me) Mac LanguageID,
+ * but (1, 0, 1042) strings usable
+ * Fix: change algorithm, and use (1, 0, *) if both standard Mac and MS strings are not found
+ */
+
+static void GetNames(TrueTypeFont *t)
+{
+    guint8 *table = getTable(t, O_name);
+    guint16 n = GetUInt16(table, 2, 1);
+    int i, r;
+
+    /* PostScript name: preferred Microsoft */
+    if ((r = findname(table, n, 3, 1, 0x0409, 6)) != -1) {
+        t->psname = nameExtract(table, r, 1, NULL);
+    } else if ((r = findname(table, n, 1, 0, 0, 6)) != -1) {
+        t->psname = nameExtract(table, r, 0, NULL);
+    } else {
+        char* pReverse = t->fname + strlen(t->fname);
+        /* take only last token of filename */
+        while(pReverse != t->fname && *pReverse != '/') pReverse--;
+        if(*pReverse == '/') pReverse++;
+        t->psname = strdup(pReverse);
+        assert(t->psname != 0);
+        for (i=strlen(t->psname) - 1; i > 0; i--) {                                /*- Remove the suffix  -*/
+            if (t->psname[i] == '.' ) {
+                t->psname[i] = 0;
+                break;
+            }
+        }
+    }
+
+    /* Font family and subfamily names: preferred Apple */
+    if ((r = findname(table, n, 0, 0, 0, 1)) != -1) {
+        t->family = nameExtract(table, r, 1, &t->ufamily);
+    } else if ((r = findname(table, n, 3, 1, 0x0409, 1)) != -1) {
+        t->family = nameExtract(table, r, 1, &t->ufamily);
+    } else if ((r = findname(table, n, 1, 0, 0, 1)) != -1) {
+        t->family = nameExtract(table, r, 0, NULL);
+    } else if ((r = findname(table, n, 3, 1, 0x0411, 1)) != -1) {
+        t->family = nameExtract(table, r, 1, &t->ufamily);
+    } else {
+        t->family = strdup(t->psname);
+        assert(t->family != 0);
+    }
+
+    if ((r = findname(table, n, 1, 0, 0, 2)) != -1) {
+        t->subfamily = nameExtract(table, r, 0, NULL);
+    } else if ((r = findname(table, n, 3, 1, 0x0409, 2)) != -1) {
+        t->subfamily = nameExtract(table, r, 1, NULL);
+    } else {
+        t->subfamily = strdup("");
+        assert(t->family != 0);
+    }
+
+}
+        
+enum cmapType {
+    CMAP_NOT_USABLE           = -1,
+    CMAP_MS_Symbol            = 10,
+    CMAP_MS_Unicode           = 11,
+    CMAP_MS_ShiftJIS          = 12,
+    CMAP_MS_Big5              = 13,
+    CMAP_MS_PRC               = 14,
+    CMAP_MS_Wansung           = 15,
+    CMAP_MS_Johab             = 16
+};
+
+#define MISSING_GLYPH_INDEX 0
+
+/*
+ * All getGlyph?() functions and freinds are implemented by:
+ * @author Manpreet Singh
+ */
+static guint16 getGlyph0(const guint8* cmap, guint16 c) {
+    if (c <= 255) {
+        return *(cmap + 6 + c);
+    } else {
+        return MISSING_GLYPH_INDEX;
+    }
+}
+
+typedef struct _subHeader2 {
+    guint16 firstCode;
+    guint16 entryCount;
+    guint16 idDelta;
+    guint16 idRangeOffset;
+} subHeader2;
+
+static guint16 getGlyph2(const guint8 *cmap, guint16 c) {
+    guint16 *CMAP2 = (guint16 *) cmap;
+    guint8 theHighByte;
+    guint8 theLowByte;
+    subHeader2* subHeader2s; 
+    guint16* subHeader2Keys;
+    guint16 firstCode;
+    int k;
+    int ToReturn;
+
+    theHighByte = (guint8)((c >> 8) & 0x00ff);
+    theLowByte = (guint8)(c & 0x00ff);
+    subHeader2Keys = CMAP2 + 3;
+    subHeader2s = (subHeader2 *)(subHeader2Keys + 256);
+    k = Int16FromMOTA(subHeader2Keys[theHighByte]) / 8;
+    
+    if(k == 0) {
+        firstCode = Int16FromMOTA(subHeader2s[k].firstCode);
+        if(theLowByte >= firstCode && theLowByte < (firstCode + Int16FromMOTA(subHeader2s[k].entryCount))) {
+            return *((&(subHeader2s[0].idRangeOffset))
+                     + (Int16FromMOTA(subHeader2s[0].idRangeOffset)/2)             /* + offset        */
+                     + theLowByte                                                  /* + to_look       */
+                     - Int16FromMOTA(subHeader2s[0].firstCode)
+                     );
+        } else {
+            return MISSING_GLYPH_INDEX;
+        }
+    } else if (k > 0) {
+        firstCode = Int16FromMOTA(subHeader2s[k].firstCode);
+        if(theLowByte >= firstCode && theLowByte < (firstCode + Int16FromMOTA(subHeader2s[k].entryCount))) {
+            ToReturn = *((&(subHeader2s[k].idRangeOffset))
+                         + (Int16FromMOTA(subHeader2s[k].idRangeOffset)/2)
+                         + theLowByte - firstCode);
+            if(ToReturn == 0) {
+                return MISSING_GLYPH_INDEX;
+            } else {
+                return (guint16)((ToReturn + Int16FromMOTA(subHeader2s[k].idDelta)) % 0xFFFF);
+            }
+        } else {
+            return MISSING_GLYPH_INDEX;
+        }
+    } else {
+        return MISSING_GLYPH_INDEX;
+    }
+}
+
+static guint16 getGlyph6(const guint8 *cmap, guint16 c) {
+    guint16 firstCode;
+    guint16 *CMAP6 = (guint16 *) cmap;
+    
+    firstCode = *(CMAP6 + 3);
+    if (c < firstCode ||  c > (firstCode + (*(CMAP6 + 4))/*entryCount*/ - 1)) {
+        return MISSING_GLYPH_INDEX;
+    } else {
+        return *((CMAP6 + 5)/*glyphIdArray*/ + (c - firstCode));
+    }
+}
+
+static guint16 GEbinsearch(guint16 *ar, guint16 length, guint16 toSearch) {
+    signed int low, mid, high, lastfound = 0xffff;
+    guint16 res;
+    if(length == (guint16)0 || length == (guint16)0xFFFF) {
+        return (guint16)0xFFFF;
+    }
+    low = 0;
+    high = length - 1;
+    while(high >= low) {
+        mid = (high + low)/2;
+        res = Int16FromMOTA(*(ar+mid));
+        if(res >= toSearch) {
+            lastfound = mid;
+            high = --mid; 
+        } else {
+            low = ++mid;
+        }
+    }
+    return lastfound;
+}
+
+
+static guint16 getGlyph4(const guint8 *cmap, guint16 c) {
+    guint16  i;
+    int ToReturn;
+    guint16  segCount;
+    guint16 * startCode;
+    guint16 * endCode;
+    guint16 * idDelta;
+    /* guint16 * glyphIdArray; */
+    guint16 * idRangeOffset;
+    guint16 * glyphIndexArray;
+    guint16  *CMAP4 = (guint16 *) cmap;
+    /* guint16  GEbinsearch(guint16 *ar, guint16 length, guint16 toSearch); */
+
+    segCount = Int16FromMOTA(*(CMAP4 + 3))/2;
+    endCode = CMAP4 + 7;
+    i = GEbinsearch(endCode, segCount, c); 
+
+    if (i == (guint16) 0xFFFF) {
+        return MISSING_GLYPH_INDEX;
+    }
+    startCode = endCode + segCount + 1;
+    
+    if(Int16FromMOTA(startCode[i]) > c) {
+        return MISSING_GLYPH_INDEX;
+    }
+    idDelta = startCode + segCount;
+    idRangeOffset = idDelta + segCount;
+    glyphIndexArray = idRangeOffset + segCount;
+
+    if(Int16FromMOTA(idRangeOffset[i]) != 0) {
+        ToReturn = Int16FromMOTA(*(&(idRangeOffset[i]) + (Int16FromMOTA(idRangeOffset[i])/2 + (c - Int16FromMOTA(startCode[i])))));
+    } else {
+        ToReturn = (Int16FromMOTA(idDelta[i]) + c)%65536;
+    }
+    return ToReturn;
+}
+
+static void FindCmap(TrueTypeFont *ttf)
+{
+    guint8 *table = getTable(ttf, O_cmap);
+    guint16 ncmaps = GetUInt16(table, 2, 1);
+    int i;
+    guint32 ThreeZero  = 0;              /* MS Symbol            */
+    guint32 ThreeOne   = 0;              /* MS Unicode           */
+    guint32 ThreeTwo   = 0;              /* MS ShiftJIS          */
+    guint32 ThreeThree = 0;              /* MS Big5              */
+    guint32 ThreeFour  = 0;              /* MS PRC               */
+    guint32 ThreeFive  = 0;              /* MS Wansung           */
+    guint32 ThreeSix   = 0;              /* MS Johab             */
+
+    for (i = 0; i < ncmaps; i++) {
+        guint32 offset;
+        guint16 pID, eID;
+
+        pID = GetUInt16(table, 4 + i * 8, 1);
+        eID = GetUInt16(table, 6 + i * 8, 1);
+        offset = GetUInt32(table, 8 + i * 8, 1);
+
+        if (pID == 3) {
+            switch (eID) {
+              case 0: ThreeZero  = offset; break;
+              case 1: ThreeOne   = offset; break;
+              case 2: ThreeTwo   = offset; break;
+              case 3: ThreeThree = offset; break;
+              case 4: ThreeFour  = offset; break;
+              case 5: ThreeFive  = offset; break;
+              case 6: ThreeSix   = offset; break;
+            }
+        }
+    }
+
+    if (ThreeOne) {
+        ttf->cmapType = CMAP_MS_Unicode;
+        ttf->cmap = table + ThreeOne;
+    } else if (ThreeTwo) {
+        ttf->cmapType = CMAP_MS_ShiftJIS;
+        ttf->cmap = table + ThreeTwo;
+    } else if (ThreeThree) {
+        ttf->cmapType = CMAP_MS_Big5;
+        ttf->cmap = table + ThreeThree;
+    } else if (ThreeFour) {
+        ttf->cmapType = CMAP_MS_PRC;
+        ttf->cmap = table + ThreeFour;
+    } else if (ThreeFive) {
+        ttf->cmapType = CMAP_MS_Wansung;
+        ttf->cmap = table + ThreeFive;
+    } else if (ThreeSix) {
+        ttf->cmapType = CMAP_MS_Johab;
+        ttf->cmap = table + ThreeSix;
+    } else if (ThreeZero) {
+        ttf->cmapType = CMAP_MS_Symbol;
+        ttf->cmap = table + ThreeZero;
+    } else {
+        ttf->cmapType = CMAP_NOT_USABLE;
+        ttf->cmap = NULL; 
+    }
+
+    if (ttf->cmapType != CMAP_NOT_USABLE) {
+        
+        switch (GetUInt16(ttf->cmap, 0, 1)) {
+          case 0: ttf->mapper = getGlyph0; break;
+          case 2: ttf->mapper = getGlyph2; break;
+          case 4: ttf->mapper = getGlyph4; break;
+          case 6: ttf->mapper = getGlyph6; break;
+          default:
+#ifdef DEBUG
+                /*- if the cmap table is really broken */
+            printf("%s: %d is not a recognized cmap format.\n", ttf->fname, GetUInt16(ttf->cmap, 0, 1));
+#endif
+            ttf->cmapType = CMAP_NOT_USABLE;
+            ttf->cmap = NULL;
+            ttf->mapper = NULL;
+        }
+    }
+}
+
+static void GetKern(TrueTypeFont *ttf)
+{
+    guint8 *table = getTable(ttf, O_kern);
+    guint8 *ptr;
+    int i;
+
+    if (!table) goto badtable;
+
+    if (getTableSize(ttf, O_kern) < 32) goto badtable;
+
+    if (GetUInt16(table, 0, 1) == 0) {                                /* Traditional Microsoft style table with USHORT version and nTables fields */
+        ttf->nkern = GetUInt16(table, 2, 1);
+        ttf->kerntables = calloc(ttf->nkern, sizeof(guint8 *));
+        assert(ttf->kerntables != 0);
+        memset(ttf->kerntables, 0, ttf->nkern * sizeof(guint8 *));
+        ttf->kerntype = KT_MICROSOFT;
+        ptr = table + 4;
+        for (i=0; i < ttf->nkern; i++) {
+            ttf->kerntables[i] = ptr;
+            ptr += GetUInt16(ptr, 2, 1);
+            /* sanity check */
+            if( ptr > ttf->ptr+ttf->fsize ) {
+                free( ttf->kerntables );
+                goto badtable;
+            }
+        }
+        return;
+    }
+
+    if (GetUInt32(table, 0, 1) == 0x00010000) {                       /* MacOS style kern tables: fixed32 version and guint32 nTables fields */
+        ttf->nkern = GetUInt32(table, 4, 1);
+        ttf->kerntables = calloc(ttf->nkern, sizeof(guint8 *));
+        assert(ttf->kerntables != 0);
+        memset(ttf->kerntables, 0, ttf->nkern * sizeof(guint8 *));
+        ttf->kerntype = KT_APPLE_NEW;
+        ptr = table + 8;
+        for (i = 0; i < ttf->nkern; i++) {
+            ttf->kerntables[i] = ptr;
+            ptr += GetUInt32(ptr, 0, 1);
+            /* sanity check; there are some fonts that are broken in this regard */
+            if( ptr > ttf->ptr+ttf->fsize ) {
+                free( ttf->kerntables );
+                goto badtable;
+            }
+        }
+        return;
+    }
+
+  badtable:
+    ttf->kerntype = KT_NONE;
+    ttf->kerntables = NULL;
+    
+    return;
+}
+
+/* KernGlyphsPrim?() functions expect the caller to ensure the validity of their arguments and
+ * that x and y elements of the kern array are initialized to zeroes
+ */
+static void KernGlyphsPrim1(TrueTypeFont *ttf, guint16 *glyphs, int nglyphs, int wmode, KernData *kern)
+{
+    fprintf(stderr, "MacOS kerning tables have not been implemented yet!\n");
+}
+
+static void KernGlyphsPrim2(TrueTypeFont *ttf, guint16 *glyphs, int nglyphs, int wmode, KernData *kern)
+{
+    int i, j;
+    guint32 gpair;
+
+    for (i = 0; i < nglyphs - 1; i++) {
+        gpair = (glyphs[i] << 16) | glyphs[i+1];
+#ifdef DEBUG2
+        /* All fonts with MS kern table that I've seen so far contain just one kern subtable.
+         * MS kern documentation is very poor and I doubt that font developers will be using
+         * several subtables. I expect them to be using OpenType tables instead.
+         * According to MS documention, format 2 subtables are not supported by Windows and OS/2.
+         */
+        if (ttf->nkern > 1) {
+            fprintf(stderr, "KernGlyphsPrim2: %d kern tables found.\n", ttf->nkern);
+        }
+#endif
+        for (j = 0; j < ttf->nkern; j++) {
+            guint16 coverage = GetUInt16(ttf->kerntables[j], 4, 1);
+            guint8 *ptr;
+            int npairs;
+            guint32 t;
+            int l, r, k;
+            
+            if (! ((coverage & 1) ^ wmode)) continue;
+            if ((coverage & 0xFFFE) != 0) {
+#ifdef DEBUG2
+                fprintf(stderr, "KernGlyphsPrim2: coverage flags are not supported: %04X.\n", coverage);
+#endif
+                continue;
+            }
+            ptr = ttf->kerntables[j];
+            npairs = GetUInt16(ptr, 6, 1);
+            ptr += 14;
+            l = 0;
+            r = npairs;
+            do {
+                k = (l + r) >> 1;
+                t = GetUInt32(ptr, k * 6, 1);
+                if (gpair >= t) l = k + 1;
+                if (gpair <= t) r = k - 1;
+            } while (l <= r);
+            if (l - r == 2) {
+                if (!wmode) {
+                    kern[i].x = XUnits(ttf->unitsPerEm, GetInt16(ptr, 4 + (l-1) * 6, 1));
+                } else {
+                    kern[i].y = XUnits(ttf->unitsPerEm, GetInt16(ptr, 4 + (l-1) * 6, 1));
+                }
+                /* !wmode ? kern[i].x : kern[i].y = GetInt16(ptr, 4 + (l-1) * 6, 1); */
+            }
+        }
+    }
+}
+
+static void KernGlyphPairPrim1(guint32 nkern, guint8 **kerntables, int upem, int wmode, guint32 a, guint32 b, int *x, int *y)
+{
+    fprintf(stderr, "MacOS kerning tables have not been implemented yet!\n");
+}
+
+static void KernGlyphPairPrim2(guint32 nkern, guint8 **kerntables, int upem, int wmode, guint32 a, guint32 b, int *x, int *y)
+{
+    int j;
+    guint32 gpair;
+
+    if (a > 0xFFFF || b > 0xFFFF) return;                   /* 32 bit glyphs are not supported by 'kern' */
+    gpair = a << 16 | b;
+
+    for (j = 0; j < nkern; j++) {
+        guint16 coverage = GetUInt16(kerntables[j], 4, 1);
+        guint8 *ptr;
+        int npairs;
+        guint32 t;
+        int l, r, k;
+            
+        if (! ((coverage & 1) ^ wmode)) continue;
+        if ((coverage & 0xFFFE) != 0) {
+#ifdef DEBUG2
+            fprintf(stderr, "KernGlyphPairPrim2: coverage flags are not supported: %04X.\n", coverage);
+#endif
+            continue;
+        }
+        ptr = kerntables[j];
+        npairs = GetUInt16(ptr, 6, 1);
+        ptr += 14;
+        l = 0;
+        r = npairs;
+        do {
+            k = (l + r) >> 1;
+            t = GetUInt32(ptr, k * 6, 1);
+            if (gpair >= t) l = k + 1;
+            if (gpair <= t) r = k - 1;
+        } while (l <= r);
+        if (l - r == 2) {
+            if (!wmode) {
+                *x = XUnits(upem, GetInt16(ptr, 4 + (l-1) * 6, 1));
+            } else {
+                *y = XUnits(upem, GetInt16(ptr, 4 + (l-1) * 6, 1));
+            }
+        }
+    }
+}
+
+        
+/*- Public functions */ /*FOLD00*/
+
+int CountTTCFonts(const char* fname)
+{
+    int nFonts = 0;
+    guint8 buffer[12];
+    int fd = open(fname, O_RDONLY);
+    if( fd != -1 ) {
+        if (read(fd, buffer, 12) == 12) {
+            if(GetUInt32(buffer, 0, 1) == T_ttcf )
+                nFonts = GetUInt32(buffer, 8, 1);
+        }
+        close(fd);
+    }
+    return nFonts;
+}
+
+
+int OpenTTFont(const char *fname, guint32 facenum, TrueTypeFont** ttf)
+{
+    TrueTypeFont *t;
+    int ret, i, fd = -1;
+    guint32 version;
+    guint8 *table, *offset;
+    guint32 length, tag;
+    gsize filelength;
+#if 0
+    guint32 tdoffset = 0;               /* offset to TableDirectory in a TTC file. For TTF files is 0 */
+#endif
+
+    int indexfmt, k;
+
+    *ttf = NULL;
+
+    if (!fname || !*fname) return SF_BADFILE;
+
+    t = calloc(1,sizeof(TrueTypeFont));
+    assert(t != 0);
+    t->tag = 0;
+    t->fname = NULL;
+    t->fsize = -1;
+    t->ptr = NULL;
+    t->nglyphs = 0xFFFFFFFF;
+
+#ifdef USE_GSUB
+    t->pGSubstitution = 0;
+#endif
+
+    t->fname = strdup(fname);
+    assert(t->fname != 0);
+
+    if (!g_file_get_contents(t->fname, (gchar **)&t->ptr, &filelength, NULL)) {
+        ret = SF_FILEIO;
+        goto cleanup;
+    }
+    t->fsize = filelength;
+
+    if (t->ptr == NULL) {
+        ret = SF_MEMORY;
+        goto cleanup;
+    }
+
+    version = GetInt32(t->ptr, 0, 1);
+
+    if ((version == 0x00010000) || (version == T_true)) {
+        t->tdoffset = 0;
+    } else if (version == T_ttcf) {                         /*- TrueType collection */
+        if (GetUInt32(t->ptr, 4, 1) != 0x00010000) {
+            CloseTTFont(t);
+            return SF_TTFORMAT;
+        }
+        if (facenum >= GetUInt32(t->ptr, 8, 1)) {
+            CloseTTFont(t);
+            return SF_FONTNO;
+        }
+        t->tdoffset = GetUInt32(t->ptr, 12 + 4 * facenum, 1);
+    } else {
+        CloseTTFont(t);
+        return SF_TTFORMAT;
+    }
+
+#ifdef DEBUG2
+    fprintf(stderr, "t->tdoffset: %d\n", t->tdoffset);
+#endif
+
+    /* magic number */
+    t->tag = TTFontClassTag;
+
+    t->ntables = GetUInt16(t->ptr + t->tdoffset, 4, 1);
+
+    t->tables = calloc(NUM_TAGS, sizeof(void *));
+    assert(t->tables != 0);
+    t->tlens = calloc(NUM_TAGS, sizeof(guint32));
+    assert(t->tlens != 0);
+
+    memset(t->tables, 0, NUM_TAGS * sizeof(void *));
+    memset(t->tlens, 0, NUM_TAGS * sizeof(guint32));
+    
+
+    /* parse the tables */
+    for (i=0; i<t->ntables; i++) {
+        tag = GetUInt32(t->ptr + t->tdoffset + 12, 16 * i, 1);
+        offset = t->ptr + GetUInt32(t->ptr + t->tdoffset + 12, 16 * i + 8, 1);
+        length = GetUInt32(t->ptr + t->tdoffset + 12, 16 * i + 12, 1);
+        
+        if (tag == T_maxp) { t->tables[O_maxp] = offset; t->tlens[O_maxp] = length; continue; }
+        if (tag == T_glyf) { t->tables[O_glyf] = offset; t->tlens[O_glyf] = length; continue; }
+        if (tag == T_head) { t->tables[O_head] = offset; t->tlens[O_head] = length; continue; }
+        if (tag == T_loca) { t->tables[O_loca] = offset; t->tlens[O_loca] = length; continue; }
+        if (tag == T_name) { t->tables[O_name] = offset; t->tlens[O_name] = length; continue; }
+        if (tag == T_hhea) { t->tables[O_hhea] = offset; t->tlens[O_hhea] = length; continue; }
+        if (tag == T_hmtx) { t->tables[O_hmtx] = offset; t->tlens[O_hmtx] = length; continue; }
+        if (tag == T_cmap) { t->tables[O_cmap] = offset; t->tlens[O_cmap] = length; continue; }
+        if (tag == T_vhea) { t->tables[O_vhea] = offset; t->tlens[O_vhea] = length; continue; }
+        if (tag == T_vmtx) { t->tables[O_vmtx] = offset; t->tlens[O_vmtx] = length; continue; }
+        if (tag == T_OS2 ) { t->tables[O_OS2 ] = offset; t->tlens[O_OS2 ] = length; continue; }
+        if (tag == T_post) { t->tables[O_post] = offset; t->tlens[O_post] = length; continue; }
+        if (tag == T_kern) { t->tables[O_kern] = offset; t->tlens[O_kern] = length; continue; }
+        if (tag == T_cvt ) { t->tables[O_cvt ] = offset; t->tlens[O_cvt ] = length; continue; }
+        if (tag == T_prep) { t->tables[O_prep] = offset; t->tlens[O_prep] = length; continue; }
+        if (tag == T_fpgm) { t->tables[O_fpgm] = offset; t->tlens[O_fpgm] = length; continue; }
+        if (tag == T_gsub) { t->tables[O_gsub] = offset; t->tlens[O_gsub] = length; continue; }
+    }
+    
+
+    /* At this point TrueTypeFont is constructed, now need to verify the font format
+       and read the basic font properties */
+
+    /* The following tables are absolutely required:
+     * maxp, head, glyf, loca, name, cmap
+     */
+
+    if (!(getTable(t, O_maxp) && getTable(t, O_head) && getTable(t, O_glyf) && getTable(t, O_loca) && getTable(t, O_name) && getTable(t, O_cmap) )) {
+        CloseTTFont(t);
+        return SF_TTFORMAT;
+    }
+     
+    table = getTable(t, O_maxp);
+    t->nglyphs = GetUInt16(table, 4, 1);
+
+    table = getTable(t, O_head);
+    t->unitsPerEm = GetUInt16(table, 18, 1);
+    indexfmt = GetInt16(table, 50, 1);  
+
+    if (!((indexfmt == 0) || indexfmt == 1)) {
+        CloseTTFont(t);
+        return SF_TTFORMAT;
+    }
+
+    k = (getTableSize(t, O_loca) / (indexfmt ? 4 : 2)) - 1;
+    if (k < t->nglyphs) t->nglyphs = k;       /* Hack for broken Chinese fonts */
+
+    table = getTable(t, O_loca);
+
+    t->goffsets = (guint32 *) calloc(1+t->nglyphs, sizeof(guint32));
+    assert(t->goffsets != 0);
+    
+    for (i = 0; i <= t->nglyphs; i++) {
+        t->goffsets[i] = indexfmt ? GetUInt32(table, i << 2, 1) : GetUInt16(table, i << 1, 1) << 1;
+    }
+
+    table = getTable(t, O_hhea);
+    t->numberOfHMetrics = (table != 0) ? GetUInt16(table, 34, 1) : 0;
+
+    table = getTable(t, O_vhea);
+    t->numOfLongVerMetrics = (table != 0) ? GetUInt16(table, 34, 1) : 0;
+
+    GetNames(t);
+    FindCmap(t);
+    GetKern(t);
+#ifdef USE_GSUB
+    ReadGSUB(t,t->tables[O_gsub],0,0);
+#endif
+
+    *ttf = t;
+    return SF_OK;
+
+  cleanup:
+    /*- t and t->fname have been allocated! */
+    free(t->fname);
+    free(t);
+    if (fd != -1) close(fd);
+    return ret;
+}
+
+
+
+void CloseTTFont(TrueTypeFont *ttf) /*FOLD01*/
+{
+    if (ttf->tag != TTFontClassTag) return;
+
+    free(ttf->ptr);
+    free(ttf->fname);
+    free(ttf->goffsets);
+    free(ttf->psname);
+    free(ttf->family);
+    if( ttf->ufamily )
+        free( ttf->ufamily );
+    free(ttf->subfamily);
+    free(ttf->tables);
+    free(ttf->tlens);
+    free(ttf->kerntables);
+    free(ttf);
+    return;
+}
+
+int GetTTGlyphPoints(TrueTypeFont *ttf, guint32 glyphID, ControlPoint **pointArray)
+{
+    return GetTTGlyphOutline(ttf, glyphID, pointArray, NULL, NULL);
+}
+
+#ifdef NO_LIST
+static
+#endif
+int GetTTGlyphComponents(TrueTypeFont *ttf, guint32 glyphID, list glyphlist)
+{
+    guint8 *ptr, *glyf = getTable(ttf, O_glyf);
+    int n = 1;
+
+    if (glyphID >= ttf->nglyphs) return 0;
+    ptr = glyf + ttf->goffsets[glyphID];
+
+    listAppend(glyphlist, (void *) glyphID);
+
+    if (GetInt16(ptr, 0, 1) == -1) {
+        guint16 flags, index;
+        ptr += 10;
+        do {
+            flags = GetUInt16(ptr, 0, 1);
+            index = GetUInt16(ptr, 2, 1);
+
+            ptr += 4;
+            n += GetTTGlyphComponents(ttf, index, glyphlist);
+
+            if (flags & ARG_1_AND_2_ARE_WORDS) {
+                ptr += 4;
+            } else {
+                ptr += 2;
+            }
+
+            if (flags & WE_HAVE_A_SCALE) {
+                ptr += 2;
+            } else if (flags & WE_HAVE_AN_X_AND_Y_SCALE) {
+                ptr += 4;
+            } else if (flags & WE_HAVE_A_TWO_BY_TWO) {
+                ptr += 8;
+            }
+        } while (flags & MORE_COMPONENTS);
+    }
+
+    return n;
+}
+
+#ifndef NO_TYPE3
+int  CreateT3FromTTGlyphs(TrueTypeFont *ttf, FILE *outf, const char *fname, /*FOLD00*/
+                          guint16 *glyphArray, guint8 *encoding, int nGlyphs,
+                          int wmode)
+{
+    ControlPoint *pa;
+    PSPathElement *path;
+    int i, j, r, n;
+    guint8 *table = getTable(ttf, O_head);
+    TTGlyphMetrics metrics;
+    int UPEm = ttf->unitsPerEm;
+    
+    const char *h01 = "%%!PS-AdobeFont-%d.%d-%d.%d\n";
+    const char *h02 = "%%%%Creator: %s %s %s\n";
+    const char *h03 = "%%%%Title: %s\n";
+    const char *h04 = "%%%%CreationDate: %s\n";
+    const char *h05 = "%%%%Pages: 0\n";
+    const char *h06 = "%%%%EndComments\n";
+    const char *h07 = "%%%%BeginResource: font %s\n";
+    const char *h08 = "%%%%EndResource\n";
+    const char *h09 = "%% Original font name: %s\n";
+
+    const char *h10 =
+        "30 dict begin\n"
+        "/PaintType 0 def\n"
+        "/FontType 3 def\n"
+        "/StrokeWidth 0 def\n";
+
+    const char *h11 = "/FontName /%s def\n";
+
+    /*
+      const char *h12 = "%/UniqueID %d def\n";
+    */
+    const char *h13 = "/FontMatrix [.001 0 0 .001 0 0] def\n";
+    const char *h14 = "/FontBBox [%d %d %d %d] def\n";
+
+    const char *h15=
+        "/Encoding 256 array def\n"
+        "    0 1 255 {Encoding exch /.notdef put} for\n";
+
+    const char *h16 = "    Encoding %d /glyph%d put\n";
+    const char *h17 = "/XUID [103 0 0 16#%08X %d 16#%08X 16#%08X] def\n";
+
+    const char *h30 = "/CharProcs %d dict def\n";
+    const char *h31 = "  CharProcs begin\n";
+    const char *h32 = "    /.notdef {} def\n";
+    const char *h33 = "    /glyph%d {\n";
+    const char *h34 = "    } bind def\n";
+    const char *h35 = "  end\n";
+
+    const char *h40 =
+        "/BuildGlyph {\n"
+        "  exch /CharProcs get exch\n"
+        "  2 copy known not\n"
+        "    {pop /.notdef} if\n"
+        "  get exec\n"
+        "} bind def\n"
+        "/BuildChar {\n"
+        "  1 index /Encoding get exch get\n"
+        "  1 index /BuildGlyph get exec\n"
+        "} bind def\n"
+        "currentdict end\n";
+
+    const char *h41 = "/%s exch definefont pop\n";
+
+    
+    if (!((nGlyphs > 0) && (nGlyphs <= 256))) return SF_GLYPHNUM;
+    if (!glyphArray) return SF_BADARG;
+    if (!fname) fname = ttf->psname;
+
+    fprintf(outf, h01, GetInt16(table, 0, 1), GetUInt16(table, 2, 1), GetInt16(table, 4, 1), GetUInt16(table, 6, 1));
+    fprintf(outf, h02, modname, modver, modextra);
+    fprintf(outf, h03, fname);
+    fprintf(outf, h04, " ");
+    fprintf(outf, h05);
+    fprintf(outf, h06);
+    fprintf(outf, h07, fname);
+    fprintf(outf, h09, ttf->psname);
+
+    fprintf(outf, h10);
+    fprintf(outf, h11, fname);
+/*    fprintf(outf, h12, 4000000); */
+
+    /* XUID generation:
+     * 103 0 0 C1 C2 C3 C4
+     * C1 - CRC-32 of the entire source TrueType font
+     * C2 - number of glyphs in the subset
+     * C3 - CRC-32 of the glyph array
+     * C4 - CRC-32 of the encoding array
+     *
+     * All CRC-32 numbers are presented as hexadecimal numbers
+     */
+
+    fprintf(outf, h17, crc32(ttf->ptr, ttf->fsize), nGlyphs, crc32(glyphArray, nGlyphs * 2), crc32(encoding, nGlyphs));
+    fprintf(outf, h13);
+    fprintf(outf, h14, XUnits(UPEm, GetInt16(table, 36, 1)), XUnits(UPEm, GetInt16(table, 38, 1)), XUnits(UPEm, GetInt16(table, 40, 1)), XUnits(UPEm, GetInt16(table, 42, 1)));
+    fprintf(outf, h15);
+
+    for (i = 0; i < nGlyphs; i++) {
+        fprintf(outf, h16, encoding[i], i);
+    }
+
+    fprintf(outf, h30, nGlyphs+1);
+    fprintf(outf, h31);
+    fprintf(outf, h32);
+
+    for (i = 0; i < nGlyphs; i++) {
+        fprintf(outf, h33, i);
+        r = GetTTGlyphOutline(ttf, glyphArray[i] < ttf->nglyphs ? glyphArray[i] : 0, &pa, &metrics, 0);
+        
+        if (r > 0) {
+            n =  BSplineToPSPath(pa, r, &path);
+        } else {
+            n = 0;                      /* glyph might have zero contours but valid metrics ??? */
+            path = 0;
+            if (r < 0) {                /* glyph is not present in the font - pa array was not allocated, so no need to free it */
+                continue;
+            }
+        }
+        fprintf(outf, "\t%d %d %d %d %d %d setcachedevice\n",
+                wmode == 0 ? XUnits(UPEm, metrics.aw) : 0,
+                wmode == 0 ? 0 : -XUnits(UPEm, metrics.ah),
+                XUnits(UPEm, metrics.xMin),
+                XUnits(UPEm, metrics.yMin),
+                XUnits(UPEm, metrics.xMax),
+                XUnits(UPEm, metrics.yMax));
+        
+        for (j = 0; j < n; j++) {
+            switch (path[j].type) {
+                case PS_MOVETO:
+                  fprintf(outf, "\t%d %d moveto\n", XUnits(UPEm, path[j].x1), XUnits(UPEm, path[j].y1));
+                  break;
+
+                case PS_LINETO:
+                  fprintf(outf, "\t%d %d lineto\n", XUnits(UPEm, path[j].x1), XUnits(UPEm, path[j].y1));
+                  break;
+
+                case PS_CURVETO:
+                  fprintf(outf, "\t%d %d %d %d %d %d curveto\n", XUnits(UPEm, path[j].x1), XUnits(UPEm, path[j].y1), XUnits(UPEm, path[j].x2), XUnits(UPEm, path[j].y2), XUnits(UPEm, path[j].x3), XUnits(UPEm, path[j].y3));
+                  break;
+
+                case PS_CLOSEPATH:
+                  fprintf(outf, "\tclosepath\n");
+                  break;
+            }
+        }
+        if (n > 0) fprintf(outf, "\tfill\n");     /* if glyph is not a whitespace character */
+
+        fprintf(outf, h34);
+
+        free(pa);
+        free(path);
+    }
+    fprintf(outf, h35);
+
+    fprintf(outf, h40);
+    fprintf(outf, h41, fname);
+
+    fprintf(outf, h08);
+    return SF_OK;
+}
+#endif
+
+#ifndef NO_TTCR
+int  CreateTTFromTTGlyphs(TrueTypeFont  *ttf,
+                          const char    *fname,
+                          guint16        *glyphArray,
+                          guint8          *encoding,
+                          int            nGlyphs,
+                          int            nNameRecs,
+                          NameRecord    *nr,
+                          guint32        flags)
+{
+    TrueTypeCreator *ttcr;
+    TrueTypeTable *head = NULL, *hhea = NULL, *maxp = NULL, *cvt = NULL, *prep = NULL, *glyf = NULL, *fpgm = NULL, *cmap = NULL, *name = NULL, *post = NULL, *os2 = NULL;
+    guint8 *p;
+    int i;
+    int res;
+    guint32 *gID;
+
+    TrueTypeCreatorNewEmpty(T_true, &ttcr);
+
+    /**                       name                         **/
+
+    if (flags & TTCF_AutoName) {
+        /* not implemented yet
+           NameRecord *names;
+           NameRecord newname;
+           int n = GetTTNameRecords(ttf, &names);
+           int n1 = 0, n2 = 0, n3 = 0, n4 = 0, n5 = 0, n6 = 0;
+           guint8 *cp1;
+           guint8 suffix[32];
+           guint32 c1 = crc32(glyphArray, nGlyphs * 2);
+           guint32 c2 = crc32(encoding, nGlyphs);
+           int len;
+           snprintf(suffix, 31, "S%08X%08X-%d", c1, c2, nGlyphs);
+        
+           name = TrueTypeTableNew_name(0, 0);
+           for (i = 0; i < n; i++) {
+           if (names[i].platformID == 1 && names[i].encodingID == 0 && names[i].languageID == 0 && names[i].nameID == 1) {
+
+           memcpy(newname, names+i, sizeof(NameRecord));
+           newname.slen = name[i].slen + strlen(suffix);
+        */
+        const guint8 ptr[] = {0,'T',0,'r',0,'u',0,'e',0,'T',0,'y',0,'p',0,'e',0,'S',0,'u',0,'b',0,'s',0,'e',0,'t'};
+        NameRecord n1 = {1, 0, 0, 6, 14, (guint8 *) "TrueTypeSubset"};
+        NameRecord n2 = {3, 1, 1033, 6, 28, NULL};
+        n2.sptr = (guint8 *) ptr;
+        name = TrueTypeTableNew_name(0, NULL);
+        nameAdd(name, &n1);
+        nameAdd(name, &n2);
+    } else {
+        if (nNameRecs == 0) {
+            NameRecord *names;
+            int n = GetTTNameRecords(ttf, &names);
+            name = TrueTypeTableNew_name(n, names);
+            DisposeNameRecords(names, n);
+        } else {
+            name = TrueTypeTableNew_name(nNameRecs, nr);
+        }
+    }
+
+    /**                       maxp                         **/
+    maxp = TrueTypeTableNew_maxp(getTable(ttf, O_maxp), getTableSize(ttf, O_maxp));
+
+    /**                       hhea                         **/
+    p = getTable(ttf, O_hhea);
+    if (p) {
+        hhea = TrueTypeTableNew_hhea(GetUInt16(p, 4, 1), GetUInt16(p, 6, 1), GetUInt16(p, 8, 1), GetUInt16(p, 18, 1), GetUInt16(p, 20, 1));
+    } else {
+        hhea = TrueTypeTableNew_hhea(0, 0, 0, 0, 0);
+    }
+
+    /**                       head                         **/
+
+    p = getTable(ttf, O_head);
+    assert(p != 0);
+    head = TrueTypeTableNew_head(GetUInt32(p, 4, 1),
+                                 GetUInt16(p, 16, 1),
+                                 GetUInt16(p, 18, 1),
+                                 p+20,
+                                 GetUInt16(p, 44, 1),
+                                 GetUInt16(p, 46, 1),
+                                 GetInt16(p, 48, 1));
+
+
+    /**                       glyf                          **/
+
+    glyf = TrueTypeTableNew_glyf();
+    gID = scalloc(nGlyphs, sizeof(guint32));
+    
+    for (i = 0; i < nGlyphs; i++) {
+        gID[i] = glyfAdd(glyf, GetTTRawGlyphData(ttf, glyphArray[i]), ttf);
+    }
+        
+
+    /**                       cmap                          **/
+    cmap = TrueTypeTableNew_cmap();
+
+    for (i=0; i < nGlyphs; i++) {
+        cmapAdd(cmap, 0x0100, encoding[i], gID[i]);
+    }
+
+    /**                       cvt                           **/
+    if ((p = getTable(ttf, O_cvt)) != 0) {
+        cvt = TrueTypeTableNew(T_cvt, getTableSize(ttf, O_cvt), p);
+    }
+
+    /**                       prep                          **/
+    if ((p = getTable(ttf, O_prep)) != 0) {
+        prep = TrueTypeTableNew(T_prep, getTableSize(ttf, O_prep), p);
+    }
+
+    /**                       fpgm                          **/
+    if ((p = getTable(ttf, O_fpgm)) != 0) {
+        fpgm = TrueTypeTableNew(T_fpgm, getTableSize(ttf, O_fpgm), p);
+    }
+
+    /**                       post                          **/
+    if ((p = getTable(ttf, O_post)) != 0) {
+        post = TrueTypeTableNew_post(0x00030000,
+                                     GetUInt32(p, 4, 1),
+                                     GetUInt16(p, 8, 1),
+                                     GetUInt16(p, 10, 1),
+                                     GetUInt16(p, 12, 1));
+    } else {
+        post = TrueTypeTableNew_post(0x00030000, 0, 0, 0, 0);
+    }
+
+    if (flags & TTCF_IncludeOS2) {
+        if ((p = getTable(ttf, O_OS2)) != 0) {
+            os2 = TrueTypeTableNew(T_OS2, getTableSize(ttf, O_OS2), p);
+        }
+    }
+
+    AddTable(ttcr, name); AddTable(ttcr, maxp); AddTable(ttcr, hhea);
+    AddTable(ttcr, head); AddTable(ttcr, glyf); AddTable(ttcr, cmap);
+    AddTable(ttcr, cvt ); AddTable(ttcr, prep); AddTable(ttcr, fpgm);
+    AddTable(ttcr, post); AddTable(ttcr, os2);
+    
+    if ((res = StreamToFile(ttcr, fname)) != SF_OK) {
+#ifdef DEBUG
+        fprintf(stderr, "StreamToFile: error code: %d.\n", res);
+#endif
+    }
+
+    TrueTypeCreatorDispose(ttcr);
+    free(gID);
+
+    return res;
+}
+
+
+// THE SAME, BUT TO MEMORY
+
+int  CreateTTFromTTGlyphs_tomemory
+                         (TrueTypeFont  *ttf,
+                          guint8        **out_buf,
+                          guint32       *out_len,
+                          guint16        *glyphArray,
+                          guint8          *encoding,
+                          int            nGlyphs,
+                          int            nNameRecs,
+                          NameRecord    *nr,
+                          guint32        flags)
+{
+    TrueTypeCreator *ttcr;
+    TrueTypeTable *head = NULL, *hhea = NULL, *maxp = NULL, *cvt = NULL, *prep = NULL, *glyf = NULL, *fpgm = NULL, *cmap = NULL, *name = NULL, *post = NULL, *os2 = NULL;
+    guint8 *p;
+    int i;
+    int res;
+    guint32 *gID;
+
+    TrueTypeCreatorNewEmpty(T_true, &ttcr);
+
+    /**                       name                         **/
+
+    if (flags & TTCF_AutoName) {
+        /* not implemented yet
+           NameRecord *names;
+           NameRecord newname;
+           int n = GetTTNameRecords(ttf, &names);
+           int n1 = 0, n2 = 0, n3 = 0, n4 = 0, n5 = 0, n6 = 0;
+           guint8 *cp1;
+           guint8 suffix[32];
+           guint32 c1 = crc32(glyphArray, nGlyphs * 2);
+           guint32 c2 = crc32(encoding, nGlyphs);
+           int len;
+           snprintf(suffix, 31, "S%08X%08X-%d", c1, c2, nGlyphs);
+        
+           name = TrueTypeTableNew_name(0, 0);
+           for (i = 0; i < n; i++) {
+           if (names[i].platformID == 1 && names[i].encodingID == 0 && names[i].languageID == 0 && names[i].nameID == 1) {
+
+           memcpy(newname, names+i, sizeof(NameRecord));
+           newname.slen = name[i].slen + strlen(suffix);
+        */
+        const guint8 ptr[] = {0,'T',0,'r',0,'u',0,'e',0,'T',0,'y',0,'p',0,'e',0,'S',0,'u',0,'b',0,'s',0,'e',0,'t'};
+        NameRecord n1 = {1, 0, 0, 6, 14, (guint8 *) "TrueTypeSubset"};
+        NameRecord n2 = {3, 1, 1033, 6, 28, NULL};
+        n2.sptr = (guint8 *) ptr;
+        name = TrueTypeTableNew_name(0, NULL);
+        nameAdd(name, &n1);
+        nameAdd(name, &n2);
+    } else {
+        if (nNameRecs == 0) {
+            NameRecord *names;
+            int n = GetTTNameRecords(ttf, &names);
+            name = TrueTypeTableNew_name(n, names);
+            DisposeNameRecords(names, n);
+        } else {
+            name = TrueTypeTableNew_name(nNameRecs, nr);
+        }
+    }
+
+    /**                       maxp                         **/
+    maxp = TrueTypeTableNew_maxp(getTable(ttf, O_maxp), getTableSize(ttf, O_maxp));
+
+    /**                       hhea                         **/
+    p = getTable(ttf, O_hhea);
+    if (p) {
+        hhea = TrueTypeTableNew_hhea(GetUInt16(p, 4, 1), GetUInt16(p, 6, 1), GetUInt16(p, 8, 1), GetUInt16(p, 18, 1), GetUInt16(p, 20, 1));
+    } else {
+        hhea = TrueTypeTableNew_hhea(0, 0, 0, 0, 0);
+    }
+
+    /**                       head                         **/
+
+    p = getTable(ttf, O_head);
+    assert(p != 0);
+    head = TrueTypeTableNew_head(GetUInt32(p, 4, 1),
+                                 GetUInt16(p, 16, 1),
+                                 GetUInt16(p, 18, 1),
+                                 p+20,
+                                 GetUInt16(p, 44, 1),
+                                 GetUInt16(p, 46, 1),
+                                 GetInt16(p, 48, 1));
+
+
+    /**                       glyf                          **/
+
+    glyf = TrueTypeTableNew_glyf();
+    gID = scalloc(nGlyphs, sizeof(guint32));
+    
+    for (i = 0; i < nGlyphs; i++) {
+        gID[i] = glyfAdd(glyf, GetTTRawGlyphData(ttf, glyphArray[i]), ttf);
+    }
+        
+
+    /**                       cmap                          **/
+    cmap = TrueTypeTableNew_cmap();
+
+    for (i=0; i < nGlyphs; i++) {
+        cmapAdd(cmap, 0x0100, encoding[i], gID[i]);
+    }
+
+    /**                       cvt                           **/
+    if ((p = getTable(ttf, O_cvt)) != 0) {
+        cvt = TrueTypeTableNew(T_cvt, getTableSize(ttf, O_cvt), p);
+    }
+
+    /**                       prep                          **/
+    if ((p = getTable(ttf, O_prep)) != 0) {
+        prep = TrueTypeTableNew(T_prep, getTableSize(ttf, O_prep), p);
+    }
+
+    /**                       fpgm                          **/
+    if ((p = getTable(ttf, O_fpgm)) != 0) {
+        fpgm = TrueTypeTableNew(T_fpgm, getTableSize(ttf, O_fpgm), p);
+    }
+
+    /**                       post                          **/
+    if ((p = getTable(ttf, O_post)) != 0) {
+        post = TrueTypeTableNew_post(0x00030000,
+                                     GetUInt32(p, 4, 1),
+                                     GetUInt16(p, 8, 1),
+                                     GetUInt16(p, 10, 1),
+                                     GetUInt16(p, 12, 1));
+    } else {
+        post = TrueTypeTableNew_post(0x00030000, 0, 0, 0, 0);
+    }
+
+    if (flags & TTCF_IncludeOS2) {
+        if ((p = getTable(ttf, O_OS2)) != 0) {
+            os2 = TrueTypeTableNew(T_OS2, getTableSize(ttf, O_OS2), p);
+        }
+    }
+
+    AddTable(ttcr, name); AddTable(ttcr, maxp); AddTable(ttcr, hhea);
+    AddTable(ttcr, head); AddTable(ttcr, glyf); AddTable(ttcr, cmap);
+    AddTable(ttcr, cvt ); AddTable(ttcr, prep); AddTable(ttcr, fpgm);
+    AddTable(ttcr, post); AddTable(ttcr, os2);
+    
+    if ((res = StreamToMemory(ttcr, out_buf, out_len)) != SF_OK) {
+#ifdef DEBUG
+        fprintf(stderr, "StreamToMemory: error code: %d.\n", res);
+#endif
+    }
+
+    TrueTypeCreatorDispose(ttcr);
+    free(gID);
+
+    return res;
+}
+
+
+#endif
+
+
+#ifndef NO_TYPE42
+static GlyphOffsets *GlyphOffsetsNew(guint8 *sfntP)
+{
+    GlyphOffsets *res = smalloc(sizeof(GlyphOffsets));
+    guint16 i, numTables = GetUInt16(sfntP, 4, 1);
+    guint32 locaLen = 0;
+    gint16 indexToLocFormat = 1;
+    guint8 *loca = sfntP + 12 + 16*numTables;
+
+    for (i = 0; i < numTables; i++) {
+        guint32 tag = GetUInt32(sfntP + 12, 16 * i, 1);
+        guint32 off = GetUInt32(sfntP + 12, 16 * i + 8, 1);
+        guint32 len = GetUInt32(sfntP + 12, 16 * i + 12, 1);
+
+        if (tag == T_loca) {
+            loca = sfntP + off;
+            locaLen = len;
+        } else if (tag == T_head) {
+            indexToLocFormat = GetInt16(sfntP + off, 50, 1);
+        }
+    }
+
+    res->nGlyphs = locaLen / ((indexToLocFormat == 1) ? 4 : 2);
+    assert(res->nGlyphs != 0);
+    res->offs = scalloc(res->nGlyphs, sizeof(guint32));
+
+    for (i = 0; i < res->nGlyphs; i++) {
+        if (indexToLocFormat == 1) {
+            res->offs[i] = GetUInt32(loca, i * 4, 1);
+        } else {
+            res->offs[i] = GetUInt16(loca, i * 2, 1) << 1;
+        }
+    }
+    return res;
+}
+
+static void GlyphOffsetsDispose(GlyphOffsets *_this)
+{
+    if (_this) {
+        free(_this->offs);
+        free(_this);
+    }
+}
+
+static void DumpSfnts(FILE *outf, guint8 *sfntP)
+{
+    HexFmt *h = HexFmtNew(outf);
+    guint16 i, numTables = GetUInt16(sfntP, 4, 1);
+    guint32 j, *offs, *len;
+    GlyphOffsets *go = GlyphOffsetsNew(sfntP);
+    guint8 pad[] = {0,0,0,0};                     /* zeroes                       */
+
+    assert(numTables <= 9);                                 /* Type42 has 9 required tables */
+
+    offs = scalloc(numTables, sizeof(guint32));
+    len = scalloc(numTables, sizeof(guint32));
+
+    fputs("/sfnts [", outf);
+    HexFmtOpenString(h);
+    HexFmtBlockWrite(h, sfntP, 12);                         /* stream out the Offset Table    */
+    HexFmtBlockWrite(h, sfntP+12, 16 * numTables);          /* stream out the Table Directory */
+
+    for (i=0; i<numTables; i++) {
+        guint32 tag = GetUInt32(sfntP + 12, 16 * i, 1);
+        guint32 off = GetUInt32(sfntP + 12, 16 * i + 8, 1);
+        guint32 len = GetUInt32(sfntP + 12, 16 * i + 12, 1);
+
+        if (tag != T_glyf) {
+            HexFmtBlockWrite(h, sfntP + off, len);
+        } else {
+            guint8 *glyf = sfntP + off;
+            guint32 o, l;
+            for (j = 0; j < go->nGlyphs - 1; j++) {
+                o = go->offs[j];
+                l = go->offs[j + 1] - o;
+                HexFmtBlockWrite(h, glyf + o, l);
+            }
+        }
+        HexFmtBlockWrite(h, pad, (4 - (len & 3)) & 3);
+    }
+    HexFmtCloseString(h);
+    fputs("] def\n", outf);
+    GlyphOffsetsDispose(go);
+    HexFmtDispose(h);
+    free(offs);
+    free(len);
+}
+
+int  CreateT42FromTTGlyphs(TrueTypeFont  *ttf,
+                           FILE          *outf,
+                           const char    *psname,
+                           guint16        *glyphArray,
+                           guint8          *encoding,
+                           int            nGlyphs)
+{
+    TrueTypeCreator *ttcr;
+    TrueTypeTable *head = NULL, *hhea = NULL, *maxp = NULL, *cvt = NULL, *prep = NULL, *glyf = NULL, *fpgm = NULL;
+    guint8 *p;
+    int i;
+    int res;
+
+    guint32 ver, rev;
+    guint8 *headP;
+
+    guint8 *sfntP;
+    guint32 sfntLen;
+    int UPEm = ttf->unitsPerEm;
+
+    guint16 *gID;
+
+    /*
+    if (nGlyphs >= 256) return SF_GLYPHNUM;
+    */
+    
+    assert(psname != 0);
+
+    TrueTypeCreatorNewEmpty(T_true, &ttcr);
+
+    /*                        head                          */
+    headP = p = getTable(ttf, O_head);
+    assert(p != 0);
+    head = TrueTypeTableNew_head(GetUInt32(p, 4, 1), GetUInt16(p, 16, 1), GetUInt16(p, 18, 1), p+20, GetUInt16(p, 44, 1), GetUInt16(p, 46, 1), GetInt16(p, 48, 1));
+    ver = GetUInt32(p, 0, 1);
+    rev = GetUInt32(p, 4, 1);
+
+    /**                       hhea                         **/
+    p = getTable(ttf, O_hhea);
+    if (p) {
+        hhea = TrueTypeTableNew_hhea(GetUInt16(p, 4, 1), GetUInt16(p, 6, 1), GetUInt16(p, 8, 1), GetUInt16(p, 18, 1), GetUInt16(p, 20, 1));
+    } else {
+        hhea = TrueTypeTableNew_hhea(0, 0, 0, 0, 0);
+    }
+
+    /**                       maxp                         **/
+    maxp = TrueTypeTableNew_maxp(getTable(ttf, O_maxp), getTableSize(ttf, O_maxp));
+    
+    /**                       cvt                           **/
+    if ((p = getTable(ttf, O_cvt)) != 0) {
+        cvt = TrueTypeTableNew(T_cvt, getTableSize(ttf, O_cvt), p);
+    }
+
+    /**                       prep                          **/
+    if ((p = getTable(ttf, O_prep)) != 0) {
+        prep = TrueTypeTableNew(T_prep, getTableSize(ttf, O_prep), p);
+    }
+
+    /**                       fpgm                          **/
+    if ((p = getTable(ttf, O_fpgm)) != 0) {
+        fpgm = TrueTypeTableNew(T_fpgm, getTableSize(ttf, O_fpgm), p);
+    }
+
+    /**                       glyf                          **/
+    glyf = TrueTypeTableNew_glyf();
+    gID = scalloc(nGlyphs, sizeof(guint32));
+    
+    for (i = 0; i < nGlyphs; i++) {
+        gID[i] = glyfAdd(glyf, GetTTRawGlyphData(ttf, glyphArray[i]), ttf);
+    }
+
+    AddTable(ttcr, head); AddTable(ttcr, hhea); AddTable(ttcr, maxp); AddTable(ttcr, cvt);
+    AddTable(ttcr, prep); AddTable(ttcr, glyf); AddTable(ttcr, fpgm);
+
+    if ((res = StreamToMemory(ttcr, &sfntP, &sfntLen)) != SF_OK) {
+        TrueTypeCreatorDispose(ttcr);
+        free(gID);
+        return res;
+    }
+
+    fprintf(outf, "%%!PS-TrueTypeFont-%d.%d-%d.%d\n", ver>>16, ver & 0xFFFF, rev>>16, rev & 0xFFFF);
+    fprintf(outf, "%%%%Creator: %s %s %s\n", modname, modver, modextra);
+    fprintf(outf, "%%- Font subset generated from a source font file: '%s'\n", ttf->fname);
+    fprintf(outf, "%%- Original font name: %s\n", ttf->psname);
+    fprintf(outf, "%%- Original font family: %s\n", ttf->family);
+    fprintf(outf, "%%- Original font sub-family: %s\n", ttf->subfamily);
+    fprintf(outf, "11 dict begin\n");
+    fprintf(outf, "/FontName /%s def\n", psname);
+    fprintf(outf, "/PaintType 0 def\n");
+    fprintf(outf, "/FontMatrix [1 0 0 1 0 0] def\n");
+    fprintf(outf, "/FontBBox [%d %d %d %d] def\n", XUnits(UPEm, GetInt16(headP, 36, 1)), XUnits(UPEm, GetInt16(headP, 38, 1)), XUnits(UPEm, GetInt16(headP, 40, 1)), XUnits(UPEm, GetInt16(headP, 42, 1)));
+    fprintf(outf, "/FontType 42 def\n");
+    fprintf(outf, "/Encoding 256 array def\n");
+    fprintf(outf, "    0 1 255 {Encoding exch /.notdef put} for\n");
+
+    for (i = 1; i<nGlyphs; i++) {
+        fprintf(outf, "Encoding %d /glyph%d put\n", encoding[i], gID[i]);
+    }
+    fprintf(outf, "/XUID [103 0 1 16#%08X %d 16#%08X 16#%08X] def\n", crc32(ttf->ptr, ttf->fsize), nGlyphs, crc32(glyphArray, nGlyphs * 2), crc32(encoding, nGlyphs));
+
+    DumpSfnts(outf, sfntP);
+    /* dump charstrings */
+    fprintf(outf, "/CharStrings %d dict dup begin\n", nGlyphs);
+    fprintf(outf, "/.notdef 0 def\n");
+    for (i = 1; i < glyfCount(glyf); i++) {
+        fprintf(outf,"/glyph%d %d def\n", i, i);
+    }
+    fprintf(outf, "end readonly def\n");
+
+    fprintf(outf, "FontName currentdict end definefont pop\n");
+    TrueTypeCreatorDispose(ttcr);
+    free(gID);
+    free(sfntP);
+    return SF_OK;
+}
+#endif
+    
+
+#ifndef NO_MAPPERS
+#ifdef USE_GSUB
+int MapString(TrueTypeFont *ttf, guint16 *str, int nchars, guint16 *glyphArray, int bvertical)
+#else
+int MapString(TrueTypeFont *ttf, guint16 *str, int nchars, guint16 *glyphArray)
+#endif
+{
+    int i;
+    guint16 *cp;
+
+    if (ttf->cmapType == CMAP_NOT_USABLE ) return -1;
+    if (!nchars) return 0;
+
+    if (glyphArray == 0) {
+        cp = str;
+    } else {
+        cp = glyphArray;
+    }
+
+    switch (ttf->cmapType) {
+        case CMAP_MS_Symbol:
+            if( ttf->mapper == getGlyph0 ) {
+                guint16 aChar;
+                for( i = 0; i < nchars; i++ ) {
+                    aChar = str[i];
+                    if( ( aChar & 0xf000 ) == 0xf000 ) {
+                        aChar &= 0x00ff;
+                    }
+                    cp[i] = aChar;
+                }
+            }
+            else if( glyphArray )
+                memcpy(glyphArray, str, nchars * 2);
+            break;
+
+        case CMAP_MS_Unicode:
+            if (glyphArray != 0) {
+                memcpy(glyphArray, str, nchars * 2);
+            }
+            break;
+        
+        case CMAP_MS_ShiftJIS:  TranslateString12(str, cp, nchars); break;
+        case CMAP_MS_Big5:      TranslateString13(str, cp, nchars); break;
+        case CMAP_MS_PRC:       TranslateString14(str, cp, nchars); break;
+        case CMAP_MS_Wansung:   TranslateString15(str, cp, nchars); break;
+        case CMAP_MS_Johab:     TranslateString16(str, cp, nchars); break;
+    }
+
+    for (i = 0; i < nchars; i++) {
+        cp[i] = ttf->mapper(ttf->cmap, cp[i]);
+#ifdef USE_GSUB
+        if (cp[i]!=0 && bvertical!=0)
+            cp[i] = UseGSUB(ttf,cp[i],bvertical);
+#endif
+    }
+    return nchars;
+}
+#ifdef USE_GSUB
+guint16 MapChar(TrueTypeFont *ttf, guint16 ch, int bvertical)
+#else
+guint16 MapChar(TrueTypeFont *ttf, guint16 ch)
+#endif
+
+{
+    switch (ttf->cmapType) {
+      case CMAP_MS_Symbol:
+        if( ttf->mapper == getGlyph0 && ( ch & 0xf000 ) == 0xf000 ) {
+            ch &= 0x00ff;
+        }
+        return ttf->mapper(ttf->cmap, ch );
+
+        case CMAP_MS_Unicode:   break;
+        case CMAP_MS_ShiftJIS:  ch = TranslateChar12(ch); break;
+        case CMAP_MS_Big5:      ch = TranslateChar13(ch); break;
+        case CMAP_MS_PRC:       ch = TranslateChar14(ch); break;
+        case CMAP_MS_Wansung:   ch = TranslateChar15(ch); break;
+        case CMAP_MS_Johab:     ch = TranslateChar16(ch); break;
+        default:                return 0;
+    }
+    ch = ttf->mapper(ttf->cmap, ch);
+#ifdef USE_GSUB
+    if (ch != 0 && bvertical != 0) {
+        ch = UseGSUB(ttf,ch,bvertical);
+    }
+#endif
+    return ch;
+}
+#endif
+
+void GetTTGlyphMetrics(TrueTypeFont *ttf, guint32 glyphID, TTGlyphMetrics *metrics)
+{
+    GetMetrics(ttf, glyphID, metrics);
+}
+    
+TTSimpleGlyphMetrics *GetTTSimpleGlyphMetrics(TrueTypeFont *ttf, guint16 *glyphArray, int nGlyphs, int mode)
+{
+    guint8 *table;
+    TTSimpleGlyphMetrics *res;
+    int i;
+    guint16 glyphID;
+    int n;
+    int UPEm = ttf->unitsPerEm;
+
+    if (mode == 0) {
+        table = getTable(ttf, O_hmtx);
+        n = ttf->numberOfHMetrics;
+    } else {
+        table = getTable(ttf, O_vmtx);
+        n = ttf->numOfLongVerMetrics;
+    }
+
+    if (!nGlyphs || !glyphArray) return NULL;        /* invalid parameters */
+    if (!n || !table) return NULL;                   /* the font does not contain the requested metrics */
+    
+    res = calloc(nGlyphs, sizeof(TTSimpleGlyphMetrics));
+    assert(res != 0);
+        
+    for (i=0; i<nGlyphs; i++) {
+        glyphID = glyphArray[i];
+
+        if (glyphID < n) {
+            res[i].adv = XUnits(UPEm, GetUInt16(table, 4 * glyphID, 1));
+            res[i].sb  = XUnits(UPEm, GetInt16(table, 4 * glyphID + 2, 1));
+        } else {
+            res[i].adv = XUnits(UPEm, GetUInt16(table, 4 * (n - 1), 1));
+            if( glyphID-n < ttf->nglyphs ) {
+                res[i].sb = XUnits(UPEm, GetInt16(table + n * 4, (glyphID - n) * 2, 1));
+            } else {                              /* font is broken */
+                res[i].sb = XUnits(UPEm, GetInt16(table, 4*(n-1) + 2, 1));
+            }
+        }
+    }
+    
+    return res;
+}
+
+#ifndef NO_MAPPERS
+TTSimpleGlyphMetrics *GetTTSimpleCharMetrics(TrueTypeFont * ttf, guint16 firstChar, int nChars, int mode)
+{
+    TTSimpleGlyphMetrics *res = 0;
+    guint16 *str;
+    int i, n;
+
+    str = malloc(nChars * 2);
+    assert(str != 0);
+
+    for (i=0; i<nChars; i++) str[i] = firstChar + i;
+
+    if ((n = MapString(ttf, str, nChars, 0
+#ifdef USE_GSUB
+                       , mode
+#endif
+                      )) != -1) {
+
+        res = GetTTSimpleGlyphMetrics(ttf, str, n, mode);
+    }
+
+    free(str);
+
+    return res;
+}
+#endif
+
+void GetTTGlobalFontInfo(TrueTypeFont *ttf, TTGlobalFontInfo *info)
+{
+    guint8 *table;
+    int UPEm = ttf->unitsPerEm;
+
+    memset(info, 0, sizeof(TTGlobalFontInfo));
+    
+    info->family = ttf->family;
+    info->ufamily = ttf->ufamily;
+    info->subfamily = ttf->subfamily;
+    info->psname = ttf->psname;
+    info->symbolEncoded = ttf->cmapType == CMAP_MS_Symbol ? 1 : 0;
+
+    table = getTable(ttf, O_OS2);
+    if (table) {
+        info->weight = GetUInt16(table, 4, 1);
+        info->width  = GetUInt16(table, 6, 1);
+        info->fsSelection = GetUInt16(table, 62, 1);
+
+        /* There are 3 different versions of OS/2 table: original (68 bytes long),
+         * Microsoft old (78 bytes long) and Microsoft new (86 bytes long,)
+         * Apple's documentation recommends looking at the table length.
+         */
+        if (getTableSize(ttf, O_OS2) > 68) {
+            info->typoAscender = XUnits(UPEm,GetInt16(table, 68, 1));
+            info->typoDescender = XUnits(UPEm, GetInt16(table, 70, 1));
+            info->typoLineGap = XUnits(UPEm, GetInt16(table, 72, 1));
+            info->winAscent = XUnits(UPEm, GetUInt16(table, 74, 1));
+            info->winDescent = XUnits(UPEm, GetUInt16(table, 76, 1));
+        }
+        if (ttf->cmapType == CMAP_MS_Unicode) {
+            info->rangeFlag = 1;
+            info->ur1 = GetUInt32(table, 42, 1);
+            info->ur2 = GetUInt32(table, 46, 1);
+            info->ur3 = GetUInt32(table, 50, 1);
+            info->ur4 = GetUInt32(table, 54, 1);
+        }
+        memcpy(info->panose, table + 32, 10);
+        info->typeFlags = GetUInt16( table, 8, 1 );
+    }
+
+
+    table = getTable(ttf, O_post);
+    if (table) {
+        info->pitch  = GetUInt32(table, 12, 1);
+        info->italicAngle = GetInt32(table, 4, 1);
+    }
+
+    table = getTable(ttf, O_head);      /* 'head' tables is always there */
+    info->xMin = XUnits(UPEm, GetInt16(table, 36, 1));
+    info->yMin = XUnits(UPEm, GetInt16(table, 38, 1));
+    info->xMax = XUnits(UPEm, GetInt16(table, 40, 1));
+    info->yMax = XUnits(UPEm, GetInt16(table, 42, 1));
+
+    table = getTable(ttf, O_hhea);
+    if (table) {
+        info->ascender  = XUnits(UPEm, GetInt16(table, 4, 1));
+        info->descender = XUnits(UPEm, GetInt16(table, 6, 1));
+        info->linegap   = XUnits(UPEm, GetInt16(table, 8, 1));
+    }
+
+    table = getTable(ttf, O_vhea);
+    if (table) {
+        info->vascent  = XUnits(UPEm, GetInt16(table, 4, 1));
+        info->vdescent = XUnits(UPEm, GetInt16(table, 6, 1));
+    }
+}
+
+#if 0
+
+guint8 *ExtractCmap(TrueTypeFont *ttf)
+{
+    guint8 *ptr = 0;
+    guint32 s;
+
+    if ((s = getTableSize(ttf, O_cmap)) != 0) {
+        ptr = smalloc(s);
+        memcpy(ptr, getTable(ttf, O_cmap), s);
+    }
+
+    return ptr;
+}
+
+#endif
+
+guint8 *ExtractTable(TrueTypeFont *ttf, guint32 tag)
+{
+    guint8 *ptr = NULL;
+    int o = tagToOrd(tag);
+    guint32 s;
+
+    if (o != 0xFFFFFFFF) {                        /* Tag is one of the predefined tags */
+        if ((s = getTableSize(ttf, o)) != 0) {
+            ptr = smalloc(s);
+            memcpy(ptr, getTable(ttf, o), s);
+        }
+    } else {                                      /* Need to do everything by hand */
+        int i;
+        guint32 t;
+        
+        for (i=0; i<ttf->ntables; i++) {
+            guint8 *table;
+            
+            t = GetUInt32(ttf->ptr + ttf->tdoffset + 12, 16 * i, 1);
+            if (t == tag) {
+                table = ttf->ptr + GetUInt32(ttf->ptr + ttf->tdoffset + 12, 16 * i + 8, 1);
+                s = GetUInt32(ttf->ptr + ttf->tdoffset + 12, 16 * i + 12, 1);
+
+                ptr = smalloc(s);
+                memcpy(ptr, table, s);
+                break;
+            }
+        }
+    }
+
+    return ptr;
+}
+
+const guint8 *GetTable(TrueTypeFont *ttf, guint32 tag)
+{
+    guint8 *ptr = NULL;
+    int o = tagToOrd(tag);
+    guint32 s;
+
+    if (o != 0xFFFFFFFF) {                        /* Tag is one of the predefined tags */
+        if ((s = getTableSize(ttf, o)) != 0) {
+            ptr = getTable(ttf, o);
+        }
+    } else {                                      /* Need to do everything by hand */
+        int i;
+        guint32 t;
+        
+        for (i=0; i<ttf->ntables; i++) {
+            t = GetUInt32(ttf->ptr + ttf->tdoffset + 12, 16 * i, 1);
+            if (t == tag) {
+                ptr = ttf->ptr + GetUInt32(ttf->ptr + ttf->tdoffset + 12, 16 * i + 8, 1);
+                break;
+            }
+        }
+    }
+
+    return ptr;
+}
+
+
+void KernGlyphs(TrueTypeFont *ttf, guint16 *glyphs, int nglyphs, int wmode, KernData *kern)
+{
+    int i;
+    
+    if (!nglyphs || !glyphs || !kern) return;
+
+    for (i = 0; i < nglyphs-1; i++) kern[i].x = kern[i].y = 0;
+    
+    switch (ttf->kerntype) {
+      case KT_APPLE_NEW: KernGlyphsPrim1(ttf, glyphs, nglyphs, wmode, kern);    return;
+      case KT_MICROSOFT: KernGlyphsPrim2(ttf, glyphs, nglyphs, wmode, kern);    return;
+      default: return;
+    }
+}
+
+GlyphData *GetTTRawGlyphData(TrueTypeFont *ttf, guint32 glyphID)
+{
+    guint8 *glyf = getTable(ttf, O_glyf);
+    guint8 *hmtx = getTable(ttf, O_hmtx);
+    guint8 *ptr;
+    guint32 length;
+    GlyphData *d;
+    ControlPoint *cp;
+    int i, n, m;
+
+    if (glyphID >= ttf->nglyphs) return NULL;
+
+    ptr = glyf + ttf->goffsets[glyphID];
+    length = ttf->goffsets[glyphID+1] - ttf->goffsets[glyphID];
+
+    d = malloc(sizeof(GlyphData)); assert(d != 0);
+
+    if (length) {
+        d->ptr = malloc((length + 1) & ~1); assert(d->ptr != 0);
+        memcpy(d->ptr, ptr, length);
+        if (GetInt16(ptr, 0, 1) >= 0) {
+            d->compflag = 0;
+        } else {
+            d->compflag = 1;
+        }
+    } else {
+        d->ptr = NULL;
+        d->compflag = 0;
+    }
+
+    d->glyphID = glyphID;
+    d->nbytes = (length + 1) & ~1;
+
+    /* now calculate npoints and ncontours */
+    n = GetTTGlyphPoints(ttf, glyphID, &cp);
+    if (n != -1) {
+        m = 0;
+        for (i = 0; i < n; i++) {
+            if (cp[i].flags & 0x8000) m++;
+        }
+        d->npoints = n;
+        d->ncontours = m;
+        free(cp);
+    } else {
+        d->npoints = 0;
+        d->ncontours = 0;
+    }
+
+    /* get adwance width and left sidebearing */
+    if (glyphID < ttf->numberOfHMetrics) {
+        d->aw = GetUInt16(hmtx, 4 * glyphID, 1);
+        d->lsb = GetInt16(hmtx, 4 * glyphID + 2, 1);
+    } else {
+        d->aw = GetUInt16(hmtx, 4 * (ttf->numberOfHMetrics - 1), 1);
+        d->lsb  = GetInt16(hmtx + ttf->numberOfHMetrics * 4, (glyphID - ttf->numberOfHMetrics) * 2, 1);
+    }
+
+    return d;
+}
+
+int GetTTNameRecords(TrueTypeFont *ttf, NameRecord **nr)
+{
+    guint8 *table = getTable(ttf, O_name);
+    guint16 n = GetUInt16(table, 2, 1);
+    NameRecord *rec;
+    guint16 i;
+
+    *nr = NULL;
+    if (n == 0) return 0;
+
+    rec = calloc(n, sizeof(NameRecord));
+
+    for (i = 0; i < n; i++) {
+        rec[i].platformID = GetUInt16(table + 6, 12 * i, 1);
+        rec[i].encodingID = GetUInt16(table + 6, 2 + 12 * i, 1);
+        rec[i].languageID = GetUInt16(table + 6, 4 + 12 * i, 1);
+        rec[i].nameID = GetUInt16(table + 6, 6 + 12 * i, 1);
+        rec[i].slen = GetUInt16(table + 6, 8 + 12 * i, 1);
+        if (rec[i].slen) {
+            rec[i].sptr = (guint8 *) malloc(rec[i].slen); assert(rec[i].sptr != 0);
+            memcpy(rec[i].sptr, table + GetUInt16(table, 4, 1) + GetUInt16(table + 6, 10 + 12 * i, 1), rec[i].slen);
+        } else {
+            rec[i].sptr = NULL;
+        }
+    }
+
+    *nr = rec;
+    return n;
+}
+
+void DisposeNameRecords(NameRecord* nr, int n)
+{
+    int i;
+    for (i = 0; i < n; i++) {
+        if (nr[i].sptr) free(nr[i].sptr);
+    }
+    free(nr);
+}
+
+// int ExtractSimpleGlyphMetrics(guint8 *table, int numberOfMetrics, int nglyphs, int UPEm
+
+TTFullSimpleGlyphMetrics *ReadGlyphMetrics(guint8 *hmtx, guint8 *vmtx, int hcount, int vcount, int gcount, int UPEm, guint16 *glyphArray, int nGlyphs)
+{
+    TTFullSimpleGlyphMetrics *res;
+    guint16 glyphID;
+    int i;
+
+    if (!nGlyphs || !glyphArray) return NULL;        /* invalid parameters */
+
+    //printf("ReadGlyphMetrics: hmtx: %x, vmtx: %x, hcount: %d, vcount: %d\n", hmtx, vmtx, hcount, vcount);
+
+    res = calloc(nGlyphs, sizeof(TTFullSimpleGlyphMetrics)); assert(res != 0);
+    for (i = 0; i < nGlyphs; i++) {
+        glyphID = glyphArray[i];
+
+        res[i].aw = res[i].ah = res[i].lsb = res[i].tsb = 0;
+
+        /* horizontal metrics */
+        if (hmtx != 0 && hcount > 0) {
+
+            if (glyphID < hcount) {
+                res[i].aw = XUnits(UPEm, GetUInt16(hmtx, 4 * glyphID, 1));
+                res[i].lsb  = XUnits(UPEm, GetInt16(hmtx, 4 * glyphID + 2, 1));
+            } else {
+                res[i].aw = XUnits(UPEm, GetUInt16(hmtx, 4 * (hcount - 1), 1));
+                if( glyphID-hcount < gcount ) {
+                    res[i].lsb = XUnits(UPEm, GetInt16(hmtx + hcount * 4, (glyphID - hcount) * 2, 1));
+                } else {                              /* font is broken */
+                    res[i].lsb = XUnits(UPEm, GetInt16(hmtx, 4*(hcount-1) + 2, 1));
+                }
+            }
+        }
+
+        /* vertical metrics */
+        if (vmtx != 0 && vcount > 0) {
+
+            if (glyphID < vcount) {
+                res[i].ah = XUnits(UPEm, GetUInt16(vmtx, 4 * glyphID, 1));
+                res[i].tsb  = XUnits(UPEm, GetInt16(vmtx, 4 * glyphID + 2, 1));
+            } else {
+                res[i].ah = XUnits(UPEm, GetUInt16(vmtx, 4 * (vcount - 1), 1));
+                if( glyphID-hcount < gcount ) {
+                    res[i].tsb = XUnits(UPEm, GetInt16(vmtx + vcount * 4, (glyphID - vcount) * 2, 1));
+                } else {                              /* font is broken */
+                    res[i].tsb = XUnits(UPEm, GetInt16(vmtx, 4*(vcount-1) + 2, 1));
+                }
+            }
+        }
+    }
+
+    return res;
+}
+
+void ReadSingleGlyphMetrics(guint8 *hmtx, guint8 *vmtx, int hcount, int vcount, int gcount, int UPEm, guint16 glyphID, TTFullSimpleGlyphMetrics *metrics)
+{
+
+    memset(metrics, 0, sizeof(TTFullSimpleGlyphMetrics));
+
+    /* horizontal metrics */
+    if (hmtx != 0 && hcount > 0) {
+        if (glyphID < hcount) {
+            metrics->aw = XUnits(UPEm, GetUInt16(hmtx, 4 * glyphID, 1));
+            metrics->lsb  = XUnits(UPEm, GetInt16(hmtx, 4 * glyphID + 2, 1));
+        } else {
+            metrics->aw = XUnits(UPEm, GetUInt16(hmtx, 4 * (hcount - 1), 1));
+            if( glyphID-hcount < gcount ) {
+                metrics->lsb = XUnits(UPEm, GetInt16(hmtx + hcount * 4, (glyphID - hcount) * 2, 1));
+            } else {                              /* font is broken */
+                metrics->lsb = XUnits(UPEm, GetInt16(hmtx, 4*(hcount-1) + 2, 1));
+            }
+        }
+    }
+
+    /* vertical metrics */
+    if (vmtx != 0 && vcount > 0) {
+        if (glyphID < vcount) {
+            metrics->ah = XUnits(UPEm, GetUInt16(vmtx, 4 * glyphID, 1));
+            metrics->tsb  = XUnits(UPEm, GetInt16(vmtx, 4 * glyphID + 2, 1));
+        } else {
+            metrics->ah = XUnits(UPEm, GetUInt16(vmtx, 4 * (vcount - 1), 1));
+            if( glyphID-hcount < gcount ) {
+                metrics->tsb = XUnits(UPEm, GetInt16(vmtx + vcount * 4, (glyphID - vcount) * 2, 1));
+            } else {                              /* font is broken */
+                metrics->tsb = XUnits(UPEm, GetInt16(vmtx, 4*(vcount-1) + 2, 1));
+            }
+        }
+    }
+}
+
+
+
+
+
+guint32 GetKernSubtableLength(guint8 *kern)
+{
+    guint32 r = 0;
+
+    if (kern != NULL) {
+        r = GetUInt16(kern, 2, 1);
+    }
+
+    return r;
+}
+
+void KernGlyphPair(int kerntype, guint32 nkern, guint8 **kern, int unitsPerEm, int wmode, guint32 a, guint32 b, int *x, int *y)
+{
+    if (x == NULL || y == NULL) return;
+    *x  = *y = 0;
+
+    if (nkern == 0 || kern == NULL) return;
+
+    switch (kerntype) {
+      case KT_APPLE_NEW: KernGlyphPairPrim1(nkern, kern, unitsPerEm, wmode, a, b, x, y);  return;
+      case KT_MICROSOFT: KernGlyphPairPrim2(nkern, kern, unitsPerEm, wmode, a, b, x, y);  return;
+    }
+}
+
+#ifdef TEST1
+/* This example creates a subset of a TrueType font with two encoded characters */
+int main(int ac, char **av)
+{
+    TrueTypeFont *fnt;
+    int r;
+
+    /* Array of Unicode source characters */
+    guint16 chars[2];
+
+    /* Encoding vector maps character encoding to the ordinal number
+     * of the glyph in the output file */
+    guint8 encoding[2];
+
+    /* This array is for glyph IDs that  source characters map to */
+    guint16 g[2];
+
+
+    if (ac < 2) return 0;
+
+    if ((r = OpenTTFont(av[1], 0, &fnt)) != SF_OK) {
+        fprintf(stderr, "Error %d opening font file: `%s`.\n", r, av[1]);
+        return 0;
+    }
+
+
+    /* We want to create the output file that only contains two Unicode characters:
+     * L'a' and L'A' */
+    
+    chars[0] = L'a';
+    chars[1] = L'A';
+
+    /* Figure out what glyphs do these characters map in our font */
+    MapString(fnt, chars, 2, g);
+
+    /* Encode the characters. Value of encoding[i] is the number 0..255 which maps to glyph i of the
+     * newly generated font */
+    encoding[0] = chars[0];
+    encoding[1] = chars[1];
+    
+
+    /* Generate a subset */
+    CreateT3FromTTGlyphs(fnt, stdout, 0, g, encoding, 2, 0);
+
+    /* Now call the dtor for the font */
+    CloseTTFont(fnt);
+    return 0;
+}
+#endif
+
+#ifdef TEST2
+/* This example extracts first 224 glyphs from a TT fonts and encodes them starting at 32 */
+int main(int ac, char **av)
+{
+    TrueTypeFont *fnt;
+    int i, r;
+
+    /* Array of Unicode source characters */
+    guint16 glyphs[224];
+
+    /* Encoding vector maps character encoding to the ordinal number
+     * of the glyph in the output file */
+    guint8 encoding[224];
+
+
+    
+    for (i=0; i<224; i++) {
+        glyphs[i] = i;
+        encoding[i] = 32 + i;
+    }
+
+    if (ac < 2) return 0;
+
+    if ((r = OpenTTFont(av[1], 0, &fnt)) != SF_OK) {
+        fprintf(stderr, "Error %d opening font file: `%s`.\n", r, av[1]);
+        return 0;
+    }
+
+
+    /* Encode the characters. Value of encoding[i] is the number 0..255 which maps to glyph i of the
+     * newly generated font */
+
+    /* Generate a subset */
+    CreateT3FromTTGlyphs(fnt, stdout, 0, glyphs, encoding, 224, 0);
+
+    /* Now call the dtor for the font */
+    CloseTTFont(fnt);
+    return 0;
+}
+#endif
+
+#ifdef TEST3
+/* Glyph metrics example */
+int main(int ac, char **av)
+{
+    TrueTypeFont *fnt;
+    int i, r;
+    guint16 glyphs[224];
+    TTSimpleGlyphMetrics *m;
+
+    for (i=0; i<224; i++) {
+        glyphs[i] = i;
+    }
+
+    if (ac < 2) return 0;
+
+    if ((r = OpenTTFont(av[1], 0, &fnt)) != SF_OK) {
+        fprintf(stderr, "Error %d opening font file: `%s`.\n", r, av[1]);
+        return 0;
+    }
+
+    if ((m = GetTTSimpleGlyphMetrics(fnt, glyphs, 224, 0)) == 0) {
+        printf("Requested metrics is not available\n");
+    } else {
+        for (i=0; i<224; i++) {
+            printf("%d. advWid: %5d, LSBear: %5d\n", i, m[i].adv, m[i].sb);
+        }
+    }
+
+    /* Now call the dtor for the font */
+    free(m);
+    CloseTTFont(fnt);
+    return 0;
+}
+#endif
+
+#ifdef TEST4
+int main(int ac, char **av)
+{
+    TrueTypeFont *fnt;
+    TTGlobalFontInfo info;
+    int i, r;
+
+
+    if ((r = OpenTTFont(av[1], 0, &fnt)) != SF_OK) {
+        fprintf(stderr, "Error %d opening font file: `%s`.\n", r, av[1]);
+        return 0;
+    }
+
+    printf("Font file: %s\n", av[1]);
+
+#ifdef PRINT_KERN
+    switch (fnt->kerntype) {
+      case KT_MICROSOFT:
+        printf("\tkern: MICROSOFT, ntables: %d.", fnt->nkern);
+        if (fnt->nkern) {
+            printf(" [");
+            for (i=0; i<fnt->nkern; i++) {
+                printf("%04X ", GetUInt16(fnt->kerntables[i], 4, 1));
+            }
+            printf("]");
+        }
+        printf("\n");
+        break;
+
+      case KT_APPLE_NEW:
+        printf("\tkern: APPLE_NEW, ntables: %d.", fnt->nkern);
+        if (fnt->nkern) {
+            printf(" [");
+            for (i=0; i<fnt->nkern; i++) {
+                printf("%04X ", GetUInt16(fnt->kerntables[i], 4, 1));
+            }
+            printf("]");
+        }
+        printf("\n");
+        break;
+
+      case KT_NONE:
+        printf("\tkern: none.\n");
+        break;
+
+      default:
+        printf("\tkern: unrecoginzed.\n");
+        break;
+    }
+    printf("\n");
+#endif
+
+    GetTTGlobalFontInfo(fnt, &info);
+    printf("\tfamily name: `%s`\n", info.family);
+    printf("\tsubfamily name: `%s`\n", info.subfamily);
+    printf("\tpostscript name: `%s`\n", info.psname);
+    printf("\tweight: %d\n", info.weight);
+    printf("\twidth: %d\n", info.width);
+    printf("\tpitch: %d\n", info.pitch);
+    printf("\titalic angle: %d\n", info.italicAngle);
+    printf("\tbouding box: [%d %d %d %d]\n", info.xMin, info.yMin, info.xMax, info.yMax);
+    printf("\tascender: %d\n", info.ascender);
+    printf("\tdescender: %d\n", info.descender);
+    printf("\tlinegap: %d\n", info.linegap);
+    printf("\tvascent: %d\n", info.vascent);
+    printf("\tvdescent: %d\n", info.vdescent);
+    printf("\ttypoAscender: %d\n", info.typoAscender);
+    printf("\ttypoDescender: %d\n", info.typoDescender);
+    printf("\ttypoLineGap: %d\n", info.typoLineGap);
+    printf("\twinAscent: %d\n", info.winAscent);
+    printf("\twinDescent: %d\n", info.winDescent);
+    printf("\tUnicode ranges:\n");
+    for (i = 0; i < 32; i++) {
+        if ((info.ur1 >> i) & 1) {
+            printf("\t\t\t%s\n", UnicodeRangeName(i));
+        }
+    }
+    for (i = 0; i < 32; i++) {
+        if ((info.ur2 >> i) & 1) {
+            printf("\t\t\t%s\n", UnicodeRangeName(i+32));
+        }
+    }
+    for (i = 0; i < 32; i++) {
+        if ((info.ur3 >> i) & 1) {
+            printf("\t\t\t%s\n", UnicodeRangeName(i+64));
+        }
+    }
+    for (i = 0; i < 32; i++) {
+        if ((info.ur4 >> i) & 1) {
+            printf("\t\t\t%s\n", UnicodeRangeName(i+96));
+        }
+    }
+
+    CloseTTFont(fnt);
+    return 0;
+}
+#endif
+
+#ifdef TEST5
+/* Kerning example */
+int main(int ac, char **av)
+{
+    TrueTypeFont *fnt;
+    guint16 g[224];
+    KernData d[223];
+    int r, i, k = 0;
+
+    g[k++] = 57;
+    g[k++] = 36;
+    g[k++] = 57;
+    g[k++] = 98;
+    g[k++] = 11;
+    g[k++] = 144;
+    g[k++] = 41;
+    g[k++] = 171;
+    g[k++] = 51;
+    g[k++] = 15;
+
+    if (ac < 2) return 0;
+
+    if ((r = OpenTTFont(av[1], 0, &fnt)) != SF_OK) {
+        fprintf(stderr, "Error %d opening font file: `%s`.\n", r, av[1]);
+        return 0;
+    }
+
+    KernGlyphs(fnt, g, k, 0, d);
+
+    for (i = 0; i < k-1; i++) {
+        printf("%3d %3d: [%3d %3d]\n", g[i], g[i+1], d[i].x, d[i].y);
+
+    }
+
+    for (i=0; i<k-1; i++) {
+        int x, y;
+        KernGlyphPair(fnt->kerntype, fnt->nkern, fnt->kerntables, fnt->unitsPerEm, 0, g[i], g[i+1], &x, &y);
+        printf("KernGlyphPair (wmode: 0) :  %3d %3d: [%3d %3d]\n", g[i], g[i+1], x, y);
+        KernGlyphPair(fnt->kerntype, fnt->nkern, fnt->kerntables, fnt->unitsPerEm, 1, g[i], g[i+1], &x, &y);
+        printf("KernGlyphPair (wmode: 1) :  %3d %3d: [%3d %3d]\n", g[i], g[i+1], x, y);
+    }
+
+    CloseTTFont(fnt);
+    return 0;
+}
+#endif
+
+
+
+#ifdef TEST6
+/* This example extracts a single glyph from a font */
+int main(int ac, char **av)
+{
+    TrueTypeFont *fnt;
+    int r, i;
+
+    guint16 glyphs[256];
+    guint8 encoding[256];
+
+    for (i=0; i<256; i++) {
+        glyphs[i] = 512 + i;
+        encoding[i] = i;
+    }
+
+#if 0
+    i=0;
+    glyphs[i++] = 2001;
+    glyphs[i++] = 2002;
+    glyphs[i++] = 2003;
+    glyphs[i++] = 2004;
+    glyphs[i++] = 2005;
+    glyphs[i++] = 2006;
+    glyphs[i++] = 2007;
+    glyphs[i++] = 2008;
+    glyphs[i++] = 2009;
+    glyphs[i++] = 2010;
+    glyphs[i++] = 2011;
+    glyphs[i++] = 2012;
+    glyphs[i++] = 2013;
+    glyphs[i++] = 2014;
+    glyphs[i++] = 2015;
+    glyphs[i++] = 2016;
+    glyphs[i++] = 2017;
+    glyphs[i++] = 2018;
+    glyphs[i++] = 2019;
+    glyphs[i++] = 2020;
+
+
+    r = 97;
+    i = 0;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+    encoding[i++] = r++;
+#endif
+    
+    if (ac < 2) return 0;
+
+    if ((r = OpenTTFont(av[1], 0, &fnt)) != SF_OK) {
+        fprintf(stderr, "Error %d opening font file: `%s`.\n", r, av[1]);
+        return 0;
+    }
+
+    /* Generate a subset */
+    CreateT3FromTTGlyphs(fnt, stdout, 0, glyphs, encoding, 256, 0);
+
+    fprintf(stderr, "UnitsPerEm: %d.\n", fnt->unitsPerEm);
+
+    /* Now call the dtor for the font */
+    CloseTTFont(fnt);
+    return 0;
+}
+#endif
+
+#ifdef TEST7
+/* NameRecord extraction example */
+int main(int ac, char **av)
+{
+    TrueTypeFont *fnt;
+    int r, i, j,  n;
+    NameRecord *nr;
+
+    if ((r = OpenTTFont(av[1], 0, &fnt)) != SF_OK) {
+        fprintf(stderr, "Error %d opening font file: `%s`.\n", r, av[1]);
+        return 0;
+    }
+
+    if ((n = GetTTNameRecords(fnt, &nr)) == 0) {
+        fprintf(stderr, "No name records in the font.\n");
+        return 0;
+    }
+
+    printf("Number of name records: %d.\n", n);
+    for (i = 0; i < n; i++) {
+        printf("%d %d %04X %d [", nr[i].platformID, nr[i].encodingID, nr[i].languageID, nr[i].nameID);
+        for (j=0; j<nr[i].slen; j++) {
+            printf("%c", isprint(nr[i].sptr[j]) ? nr[i].sptr[j] : '.');
+        }
+        printf("]\n");
+    }
+
+    
+    DisposeNameRecords(nr, n);
+    CloseTTFont(fnt);
+    return 0;
+}
+#endif
+
+#ifdef TEST8
+/* TrueType -> TrueType subsetting */
+int main(int ac, char **av)
+{
+    TrueTypeFont *fnt;
+    guint16 glyphArray[] = { 0,  98,  99,  22,  24, 25, 26,  27,  28,  29, 30, 31, 1270, 1289, 34};
+    guint8 encoding[]     = {32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46};
+    int r;
+
+    if ((r = OpenTTFont(av[1], 0, &fnt)) != SF_OK) {
+        fprintf(stderr, "Error %d opening font file: `%s`.\n", r, av[1]);
+        return 0;
+    }
+
+    CreateTTFromTTGlyphs(fnt, "subfont.ttf", glyphArray, encoding, 15, 0, 0, TTCF_AutoName | TTCF_IncludeOS2);
+
+    CloseTTFont(fnt);
+
+    return 0;
+}
+#endif
+
+#ifdef TEST9
+/* TrueType -> Type42 subsetting */
+int main(int ac, char **av)
+{
+    TrueTypeFont *fnt;
+    /*
+      guint16 glyphArray[] = { 0,  20,  21,  22,  24, 25, 26,  27,  28,  29, 30, 31, 32, 33, 34};
+      guint8 encoding[]     = {32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46};
+    guint16 glyphArray[] = { 0,  6711,  6724,  11133,  11144, 14360, 26,  27,  28,  29, 30, 31, 1270, 1289, 34};
+    guint8 encoding[]     = {32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46};
+    */
+    guint16 glyphArray[1000];
+    guint8 encoding[1000];
+    int r, i;
+
+    for (i=0;i<1000;i++) {
+       glyphArray[i]=2000 + i;
+       encoding[i] =i;
+    }
+    if ((r = OpenTTFont(av[1], 0, &fnt)) != SF_OK) {
+        fprintf(stderr, "Error %d opening font file: `%s`.\n", r, av[1]);
+        return 0;
+    }
+
+    CreateT42FromTTGlyphs(fnt, stdout, "testfont", glyphArray, encoding, 1000);
+
+    CloseTTFont(fnt);
+
+    return 0;
+}
+#endif
+
+#ifdef TEST10
+/* Component glyph test */
+int main(int ac, char **av)
+{
+    TrueTypeFont *fnt;
+    int r, i;
+    list glyphlist = listNewEmpty();
+
+    if ((r = OpenTTFont(av[1], 0, &fnt)) != SF_OK) {
+        fprintf(stderr, "Error %d opening font file: `%s`.\n", r, av[1]);
+        return 0;
+    }
+
+    for (i = 0; i < fnt->nglyphs; i++) {
+        r = GetTTGlyphComponents(fnt, i, glyphlist);
+        if (r > 1) {
+            printf("%d -> ", i);
+            listToFirst(glyphlist);
+            do {
+                printf("%d ", (int) listCurrent(glyphlist));
+            } while (listNext(glyphlist));
+            printf("\n");
+        } else {
+            printf("%d: single glyph.\n", i);
+        }
+        listClear(glyphlist);
+    }
+    
+    CloseTTFont(fnt);
+    listDispose(glyphlist);
+
+    return 0;
+}
+#endif
+
+
diff --git a/src/ttsubset/sft.h b/src/ttsubset/sft.h
new file mode 100644 (file)
index 0000000..09319d7
--- /dev/null
@@ -0,0 +1,794 @@
+/*
+ * Copyright © 2002, 2003 Sun Microsystems, Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ *
+ * 3. Neither the name of Sun Microsystems, Inc. nor the names of 
+ *    contributors may be used to endorse or promote products derived from
+ *    this software without specific prior written permission.
+ *
+ * This software is provided "AS IS," without a warranty of any kind.
+ *
+ * ALL EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES,
+ * INCLUDING ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A
+ * PARTICULAR PURPOSE OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED.
+ * SUN AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES OR
+ * LIABILITIES SUFFERED BY LICENSEE AS A RESULT OF OR RELATING TO USE,
+ * MODIFICATION OR DISTRIBUTION OF THE SOFTWARE OR ITS DERIVATIVES.
+ * IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST REVENUE,
+ * PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL,
+ * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE
+ * THEORY OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE
+ * SOFTWARE, EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
+ *
+ */
+
+/* $Id$ */
+/* @(#)sft.h 1.17 03/01/08 SMI */
+
+/*
+ * @file sft.h
+ * @brief Sun Font Tools
+ * @author Alexander Gelfenbain <adg@sun.com>
+ * @version 1.0
+ */
+
+/*
+ *        If NO_MAPPERS is defined, MapChar() and MapString() and consequently GetTTSimpleCharMetrics()
+ *        don't get compiled in. This is done to avoid including a large chunk of code (TranslateXY() from
+ *        xlat.c in the projects that don't require it.
+ *
+ *        If NO_TYPE3 is defined CreateT3FromTTGlyphs() does not get compiled in.
+ *        If NO_TYPE42 is defined Type42-related code is excluded
+ *        If NO_TTCR is defined TrueType creation related code is excluded\
+ *        If NO_LIST is defined list.h and piblic functions that use it don't get compiled
+ *        If USE_GSUB is *not* defined Philipp's GSUB code does not get included
+ *
+ *        When STSF is defined several data types are defined elsewhere
+ */
+
+/*
+ *        Generated fonts contain an XUID entry in the form of:
+ *
+ *                  103 0 T C1 N C2 C3
+ *
+ *        103 - Sun's Adobe assigned XUID number. Contact person: Alexander Gelfenbain <gelf@eng.sun.com>
+ *
+ *        T  - font type. 0: Type 3, 1: Type 42
+ *        C1 - CRC-32 of the entire source TrueType font
+ *        N  - number of glyphs in the subset
+ *        C2 - CRC-32 of the array of glyph IDs used to generate the subset
+ *        C3 - CRC-32 of the array of encoding numbers used to generate the subset
+ *
+ */
+
+#ifndef __SUBFONT_H
+#define __SUBFONT_H
+
+#include <sys/types.h>
+#ifdef HAVE_UNISTD_H
+#include <unistd.h>
+#endif
+#include <stdio.h>
+#ifndef NO_LIST
+#include "list.h"
+#endif
+
+#ifdef STSF
+#include <sttypes.h>
+#endif
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#if 0  /* Use glib's G_BYTE_ORDER == G_BIG_ENDIAN instead */
+#ifdef __sparc
+#ifndef G_BIG_ENDIAN
+#define G_BIG_ENDIAN
+#endif
+#endif
+
+#if defined(__powerpc__) || defined(POWERPC)
+#ifndef G_BIG_ENDIAN
+#define G_BIG_ENDIAN
+#endif
+#endif
+
+#ifdef __i386
+#ifndef G_LITTLE_ENDIAN
+#define G_LITTLE_ENDIAN
+#endif
+#endif
+
+#ifdef __mips
+#ifndef G_BIG_ENDIAN
+#define G_BIG_ENDIAN
+#endif
+#endif
+
+#ifdef __BIG_ENDIAN__
+#define G_BIG_ENDIAN
+#endif
+
+#ifdef __LITTLE_ENDIAN__
+#define G_LITTLE_ENDIAN
+#endif
+
+#if !defined(G_BIG_ENDIAN) && !defined(G_LITTLE_ENDIAN)
+#error "Either G_BIG_ENDIAN or G_LITTLE_ENDIAN should be defined."
+#endif
+
+#if defined(G_BIG_ENDIAN) && defined(G_LITTLE_ENDIAN)
+#error "This is bizarre"
+#endif
+#endif
+
+#if 0  /* These should be defined in the makefile */
+#define DEBUG      /* Generate debugging output */
+#define DEBUG2     /* More detailed debugging output */
+#define DEBUG3     /* Dump of TrueType outlines */
+#endif
+
+    
+
+
+/*@{*/
+#define false 0               /**< standard false value */
+#define true  1               /**< standard true value */
+/*@}*/
+
+#ifndef STSF
+/* glib already deals with different compilers */
+#include <glibconfig.h>
+
+/*@{*/
+    typedef gint16       F2Dot14;            /**< fixed: 2.14 */
+    typedef gint32       F16Dot16;           /**< fixed: 16.16 */
+/*@}*/
+
+    typedef struct {
+        guint16 s;
+        guint16 d;
+    } uint16pair;
+#endif
+
+/** Return value of OpenTTFont() and CreateT3FromTTGlyphs() */
+    enum SFErrCodes {
+        SF_OK,                              /**< no error                                     */
+        SF_BADFILE,                         /**< file not found                               */
+        SF_FILEIO,                          /**< file I/O error                               */
+        SF_MEMORY,                          /**< memory allocation error                      */
+        SF_GLYPHNUM,                        /**< incorrect number of glyphs                   */
+        SF_BADARG,                          /**< incorrect arguments                          */
+        SF_TTFORMAT,                        /**< incorrect TrueType font format               */
+        SF_TABLEFORMAT,                     /**< incorrect format of a TrueType table         */
+        SF_FONTNO                           /**< incorrect logical font number of a TTC font  */
+    };
+
+#ifndef FW_THIN /* WIN32 compilation would conflict */
+/** Value of the weight member of the TTGlobalFontInfo struct */
+    enum WeightClass {
+        FW_THIN = 100,                      /**< Thin                               */
+        FW_EXTRALIGHT = 200,                /**< Extra-light (Ultra-light)          */
+        FW_LIGHT = 300,                     /**< Light                              */
+        FW_NORMAL = 400,                    /**< Normal (Regular)                   */
+        FW_MEDIUM = 500,                    /**< Medium                             */
+        FW_SEMIBOLD = 600,                  /**< Semi-bold (Demi-bold)              */
+        FW_BOLD = 700,                      /**< Bold                               */
+        FW_EXTRABOLD = 800,                 /**< Extra-bold (Ultra-bold)            */
+        FW_BLACK = 900                      /**< Black (Heavy)                      */
+    };
+
+/** Value of the width member of the TTGlobalFontInfo struct */
+    enum WidthClass {
+        FWIDTH_ULTRA_CONDENSED = 1,         /**< 50% of normal                      */
+        FWIDTH_EXTRA_CONDENSED = 2,         /**< 62.5% of normal                    */
+        FWIDTH_CONDENSED = 3,               /**< 75% of normal                      */
+        FWIDTH_SEMI_CONDENSED = 4,          /**< 87.5% of normal                    */
+        FWIDTH_NORMAL = 5,                  /**< Medium, 100%                       */
+        FWIDTH_SEMI_EXPANDED = 6,           /**< 112.5% of normal                   */
+        FWIDTH_EXPANDED = 7,                /**< 125% of normal                     */
+        FWIDTH_EXTRA_EXPANDED = 8,          /**< 150% of normal                     */
+        FWIDTH_ULTRA_EXPANDED = 9           /**< 200% of normal                     */
+    };
+#endif /* FW_THIN */
+
+/** Type of the 'kern' table, stored in _TrueTypeFont::kerntype */
+    enum KernType {
+
+        KT_NONE         = 0,                /**< no kern table                      */
+        KT_APPLE_NEW    = 1,                /**< new Apple kern table               */
+        KT_MICROSOFT    = 2                 /**< Microsoft table                    */
+    };
+
+/* Composite glyph flags definition */
+    enum CompositeFlags {
+        ARG_1_AND_2_ARE_WORDS     = 1,
+        ARGS_ARE_XY_VALUES        = 1<<1,
+        ROUND_XY_TO_GRID          = 1<<2,
+        WE_HAVE_A_SCALE           = 1<<3,
+        MORE_COMPONENTS           = 1<<5,
+        WE_HAVE_AN_X_AND_Y_SCALE  = 1<<6,
+        WE_HAVE_A_TWO_BY_TWO      = 1<<7,
+        WE_HAVE_INSTRUCTIONS      = 1<<8,
+        USE_MY_METRICS            = 1<<9,
+        OVERLAP_COMPOUND          = 1<<10
+    };
+
+#ifndef NO_TTCR
+/** Flags for TrueType generation */
+    enum TTCreationFlags {
+        TTCF_AutoName = 1,                  /**< Automatically generate a compact 'name' table.
+                                               If this flag is not set, name table is generated
+                                               either from an array of NameRecord structs passed as
+                                               arguments or if the array is NULL, 'name' table
+                                               of the generated TrueType file will be a copy
+                                               of the name table of the original file.
+                                               If this flag is set the array of NameRecord structs
+                                               is ignored and a very compact 'name' table is automatically
+                                               generated. */
+
+        TTCF_IncludeOS2 = 2                 /** If this flag is set OS/2 table from the original font will be
+                                                copied to the subset */
+    };
+#endif
+
+    /** Structure used by GetTTGlyphMetrics() */
+    /*- In horisontal writing mode right sidebearing is calculated using this formula
+     *- rsb = aw - (lsb + xMax - xMin) -*/
+     typedef struct {
+         gint16  xMin;
+         gint16  yMin;
+         gint16  xMax;
+         gint16  yMax;
+         guint16 aw;                /*- Advance Width (horisontal writing mode)    */
+         gint16  lsb;               /*- Left sidebearing (horisontal writing mode) */
+         guint16 ah;                /*- advance height (vertical writing mode)     */
+         gint16  tsb;               /*- top sidebearing (vertical writing mode)    */
+     } TTGlyphMetrics;
+
+
+    /** Structure used by GetTTSimpleGlyphMetrics() and GetTTSimpleCharMetrics() functions */
+    typedef struct {
+        guint16 adv;                         /**< advance width or height            */
+        gint16 sb;                           /**< left or top sidebearing            */
+    } TTSimpleGlyphMetrics;
+
+    /** Structure returned by ReadGlyphMetrics() */
+    typedef struct {
+        guint16 aw, ah;
+        gint16 lsb, tsb;
+    } TTFullSimpleGlyphMetrics;
+
+
+/** Structure used by the TrueType Creator and GetRawGlyphData() */
+
+    typedef struct {
+        guint32 glyphID;                     /**< glyph ID                           */
+        guint16 nbytes;                      /**< number of bytes in glyph data      */
+        guint8  *ptr;                         /**< pointer to glyph data              */
+        guint16 aw;                          /**< advance width                      */
+        gint16  lsb;                         /**< left sidebearing                   */
+        guint16 compflag;                    /**< 0- if non-composite, 1- otherwise  */
+        guint16 npoints;                     /**< number of points                   */
+        guint16 ncontours;                   /**< number of contours                 */
+        /* */
+        guint32 newID;                       /**< used internally by the TTCR        */
+    } GlyphData;
+
+
+#ifndef STSF
+    /* STSF defines NameRecord and FUnitBBox structures in its own include file sttypes.h */
+
+    typedef struct {
+        gint16 xMin;
+        gint16 yMin;
+        gint16 xMax;
+        gint16 yMax;
+    } FUnitBBox;
+
+/** Structure used by the TrueType Creator and CreateTTFromTTGlyphs() */
+    typedef struct {
+        guint16 platformID;                  /**< Platform ID                                            */
+        guint16 encodingID;                  /**< Platform-specific encoding ID                          */
+        guint16 languageID;                  /**< Language ID                                            */
+        guint16 nameID;                      /**< Name ID                                                */
+        guint16 slen;                        /**< String length in bytes                                 */
+        guint8  *sptr;                        /**< Pointer to string data (not zero-terminated!)          */
+    } NameRecord;
+#endif
+
+
+
+/** Return value of GetTTGlobalFontInfo() */
+
+    typedef struct {
+        char *family;             /**< family name                                             */
+        guint16 *ufamily;                /**< family name UCS2                                         */
+        char *subfamily;          /**< subfamily name                                          */
+        char *psname;             /**< PostScript name                                         */
+        int   weight;             /**< value of WeightClass or 0 if can't be determined        */
+        int   width;              /**< value of WidthClass or 0 if can't be determined         */
+        int   pitch;              /**< 0: proportianal font, otherwise: monospaced             */
+        int   italicAngle;        /**< in counter-clockwise degrees * 65536                    */
+        guint16 fsSelection;       /**< fsSelection field of OS/2 table                         */
+        int   xMin;               /**< global bounding box: xMin                               */
+        int   yMin;               /**< global bounding box: yMin                               */
+        int   xMax;               /**< global bounding box: xMax                               */
+        int   yMax;               /**< global bounding box: yMax                               */
+        int   ascender;           /**< typographic ascent.                                     */
+        int   descender;          /**< typographic descent.                                    */
+        int   linegap;            /**< typographic line gap.\ Negative values are treated as
+                                     zero in Win 3.1, System 6 and System 7.                 */
+        int   vascent;            /**< typographic ascent for vertical writing mode            */
+        int   vdescent;           /**< typographic descent for vertical writing mode           */
+        int   typoAscender;       /**< OS/2 portable typographic ascender                      */
+        int   typoDescender;      /**< OS/2 portable typographic descender                     */
+        int   typoLineGap;        /**< OS/2 portable typographc line gap                       */
+        int   winAscent;          /**< ascender metric for Windows                             */
+        int   winDescent;         /**< descender metric for Windows                            */
+        int   symbolEncoded;      /**< 1: MS symbol encoded 0: not symbol encoded              */
+        int   rangeFlag;          /**< if set to 1 Unicode Range flags are applicable          */
+        guint32 ur1;               /**< bits 0 - 31 of Unicode Range flags                      */
+        guint32 ur2;               /**< bits 32 - 63 of Unicode Range flags                     */
+        guint32 ur3;               /**< bits 64 - 95 of Unicode Range flags                     */
+        guint32 ur4;               /**< bits 96 - 127 of Unicode Range flags                    */
+        guint8   panose[10];        /**< PANOSE classification number                            */
+        guint16 typeFlags;               /**< type flags (copyright information)                      */
+    } TTGlobalFontInfo;
+
+/** Structure used by KernGlyphs()      */
+    typedef struct {
+        int x;                    /**< positive: right, negative: left                        */
+        int y;                    /**< positive: up, negative: down                           */
+    } KernData;
+
+
+/** ControlPoint structure used by GetTTGlyphPoints() */
+    typedef struct {
+        guint32 flags;             /**< 00000000 00000000 e0000000 bbbbbbbb */
+        /**< b - guint8 flags from the glyf array  */
+        /**< e == 0 - regular point              */
+        /**< e == 1 - end contour                */
+        gint16 x;                  /**< X coordinate in EmSquare units      */
+        gint16 y;                  /**< Y coordinate in EmSquare units      */
+    } ControlPoint;
+
+    typedef struct _TrueTypeFont TrueTypeFont;
+
+/*
+ * @defgroup sft Sun Font Tools Exported Functions
+ */
+
+
+/*
+ * Get the number of fonts contained in a TrueType collection
+ * @param  fname - file name
+ * @return number of fonts or zero, if file is not a TTC file.
+ * @ingroup sft
+ */
+    int CountTTCFonts(const char* fname);
+
+
+/*
+ * TrueTypeFont constructor. 
+ * Reads the font file and allocates the memory for the structure.
+ * @param  facenum - logical font number within a TTC file. This value is ignored
+ *                   for TrueType fonts
+ * @return value of SFErrCodes enum
+ * @ingroup sft
+ */
+    int  OpenTTFont(const char *fname, guint32 facenum, TrueTypeFont**);
+
+/*
+ * TrueTypeFont destructor. Deallocates the memory.
+ * @ingroup sft
+ */
+    void CloseTTFont(TrueTypeFont *);
+
+/*
+ * Extracts TrueType control points, and stores them in an allocated array pointed to
+ * by *pointArray. This function returns the number of extracted points.
+ *
+ * @param ttf         pointer to the TrueTypeFont structure
+ * @param glyphID     Glyph ID
+ * @param pointArray  Return value - address of the pointer to the first element of the array
+ *                    of points allocated by the function
+ * @return            Returns the number of points in *pointArray or -1 if glyphID is
+ *                    invalid.
+ * @ingroup sft
+ *
+ */
+int GetTTGlyphPoints(TrueTypeFont *ttf, guint32 glyphID, ControlPoint **pointArray);
+
+/*
+ * Extracts bounding boxes in normalized FUnits (1000/em) for all glyphs in the
+ * font and allocates an array of FUnitBBox structures. There are ttf->nglyphs
+ * elements in the array.
+ *
+ * @param ttf     pointer to the TrueTypeFont structure
+ *
+ * @return        an array of FUnitBBox structures with values normalized to 1000 UPEm
+ *
+ * @ingroup sft
+ */
+FUnitBBox *GetTTGlyphBoundingBoxes(TrueTypeFont *ttf);
+
+/*
+ * Extracts raw glyph data from the 'glyf' table and returns it in an allocated
+ * GlyphData structure.
+ *
+ * @param ttf         pointer to the TrueTypeFont structure
+ * @param glyphID     Glyph ID
+ *
+ * @return            pointer to an allocated GlyphData structure or NULL if
+ *                    glyphID is not present in the font
+ * @ingroup sft
+ *
+ */
+    GlyphData *GetTTRawGlyphData(TrueTypeFont *ttf, guint32 glyphID);
+
+#ifndef NO_LIST
+/*
+ * For a specified glyph adds all component glyphs IDs to the list and
+ * return their number. If the glyph is a single glyph it has one component
+ * glyph (which is added to the list) and the function returns 1.
+ * For a composite glyphs it returns the number of component glyphs
+ * and adds all of them to the list.
+ *
+ * @param ttf         pointer to the TrueTypeFont structure
+ * @param glyphID     Glyph ID
+ * @param glyphlist   list of glyphs
+ *
+ * @return            number of component glyphs
+ * @ingroup sft
+ *
+ */
+    int GetTTGlyphComponents(TrueTypeFont *ttf, guint32 glyphID, list glyphlist);
+#endif
+
+/*
+ * Extracts all Name Records from the font and stores them in an allocated
+ * array of NameRecord structs
+ *
+ * @param ttf       pointer to the TrueTypeFont struct
+ * @param nr        pointer to the array of NameRecord structs
+ *
+ * @return          number of NameRecord structs
+ * @ingroup sft
+ */
+
+    int GetTTNameRecords(TrueTypeFont *ttf, NameRecord **nr);
+
+/*
+ * Deallocates previously allocated array of NameRecords.
+ *
+ * @param nr        array of NameRecord structs
+ * @param n         number of elements in the array
+ *
+ * @ingroup sft
+ */
+    void DisposeNameRecords(NameRecord* nr, int n);
+
+
+#ifndef NO_TYPE3
+/*
+ * Generates a new PostScript Type 3 font and dumps it to <b>outf</b> file.
+ * This functions subsititues glyph 0 for all glyphIDs that are not found in the font.
+ * @param ttf         pointer to the TrueTypeFont structure
+ * @param outf        the resulting font is written to this stream
+ * @param fname       font name for the new font. If it is NULL the PostScript name of the
+ *                    original font will be used
+ * @param glyphArray  pointer to an array of glyphs that are to be extracted from ttf
+ * @param encoding    array of encoding values. encoding[i] specifies the position of the glyph
+ *                    glyphArray[i] in the encoding vector of the resulting Type3 font
+ * @param nGlyphs     number of glyph IDs in glyphArray and encoding values in encoding
+ * @param wmode       writing mode for the output file: 0 - horizontal, 1 - vertical
+ * @return            return the value of SFErrCodes enum
+ * @see               SFErrCodes
+ * @ingroup sft
+ *
+ */
+    int  CreateT3FromTTGlyphs(TrueTypeFont *ttf, FILE *outf, const char *fname, guint16 *glyphArray, guint8 *encoding, int nGlyphs, int wmode);
+#endif
+
+#ifndef NO_TTCR
+/*
+ * Generates a new TrueType font and dumps it to <b>outf</b> file.
+ * This functions subsititues glyph 0 for all glyphIDs that are not found in the font.
+ * @param ttf         pointer to the TrueTypeFont structure
+ * @param fname       file name for the output TrueType font file
+ * @param glyphArray  pointer to an array of glyphs that are to be extracted from ttf. The first
+ *                    element of this array has to be glyph 0 (default glyph)
+ * @param encoding    array of encoding values. encoding[i] specifies character code for
+ *                    the glyphID glyphArray[i]. Character code 0 usually points to a default
+ *                    glyph (glyphID 0)
+ * @param nGlyphs     number of glyph IDs in glyphArray and encoding values in encoding
+ * @param nNameRecs   number of NameRecords for the font, if 0 the name table from the
+ *                    original font will be used
+ * @param nr          array of NameRecords
+ * @param flags       or'ed TTCreationFlags
+ * @return            return the value of SFErrCodes enum
+ * @see               SFErrCodes
+ * @ingroup sft
+ *
+ */
+    int  CreateTTFromTTGlyphs(TrueTypeFont  *ttf,
+                              const char    *fname,
+                              guint16        *glyphArray,
+                              guint8          *encoding,
+                              int            nGlyphs,
+                              int            nNameRecs,
+                              NameRecord    *nr,
+                              guint32        flags);
+
+// the same, but to memory (output to out_buf, out_len)
+    int  CreateTTFromTTGlyphs_tomemory
+                             (TrueTypeFont  *ttf,
+                              guint8        **out_buf,
+                              guint32       *out_len,
+                              guint16        *glyphArray,
+                              guint8          *encoding,
+                              int            nGlyphs,
+                              int            nNameRecs,
+                              NameRecord    *nr,
+                              guint32        flags);
+
+#endif
+
+#ifndef NO_TYPE42
+/*
+ * Generates a new PostScript Type42 font and dumps it to <b>outf</b> file.
+ * This functions subsititues glyph 0 for all glyphIDs that are not found in the font.
+ * @param ttf         pointer to the TrueTypeFont structure
+ * @param outf        output stream for a resulting font
+ * @param psname      PostScript name of the resulting font
+ * @param glyphArray  pointer to an array of glyphs that are to be extracted from ttf. The first
+ *                    element of this array has to be glyph 0 (default glyph)
+ * @param encoding    array of encoding values. encoding[i] specifies character code for
+ *                    the glyphID glyphArray[i]. Character code 0 usually points to a default
+ *                    glyph (glyphID 0)
+ * @param nGlyphs     number of glyph IDs in glyphArray and encoding values in encoding
+ * @return            SF_OK - no errors
+ *                    SF_GLYPHNUM - too many glyphs (> 255)
+ *                    SF_TTFORMAT - corrupted TrueType fonts
+ *
+ * @see               SFErrCodes
+ * @ingroup sft
+ *
+ */
+    int  CreateT42FromTTGlyphs(TrueTypeFont  *ttf,
+                               FILE          *outf,
+                               const char    *psname,
+                               guint16        *glyphArray,
+                               guint8          *encoding,
+                               int            nGlyphs);
+#endif
+
+/*
+ * Queries full glyph metrics for one glyph
+ */
+void GetTTGlyphMetrics(TrueTypeFont *ttf, guint32 glyphID, TTGlyphMetrics *metrics);
+
+
+/*
+ * Queries glyph metrics. Allocates an array of TTSimpleGlyphMetrics structs and returns it.
+ *
+ * @param ttf         pointer to the TrueTypeFont structure
+ * @param glyphArray  pointer to an array of glyphs that are to be extracted from ttf
+ * @param nGlyphs     number of glyph IDs in glyphArray and encoding values in encoding
+ * @param mode        writing mode: 0 - horizontal, 1 - vertical
+ * @ingroup sft
+ *
+ */
+TTSimpleGlyphMetrics *GetTTSimpleGlyphMetrics(TrueTypeFont *ttf, guint16 *glyphArray, int nGlyphs, int mode);
+
+#ifndef NO_MAPPERS
+/*
+ * Queries character metrics. Allocates an array of TTSimpleGlyphMetrics structs and returns it.
+ * This function behaves just like GetTTSimpleGlyphMetrics() but it takes a range of Unicode
+ * characters instead of an array of glyphs.
+ *
+ * @param ttf         pointer to the TrueTypeFont structure
+ * @param firstChar   Unicode value of the first character in the range
+ * @param nChars      number of Unicode characters in the range
+ * @param mode        writing mode: 0 - horizontal, 1 - vertical
+ *
+ * @see GetTTSimpleGlyphMetrics
+ * @ingroup sft
+ *
+ */
+    TTSimpleGlyphMetrics *GetTTSimpleCharMetrics(TrueTypeFont *ttf, guint16 firstChar, int nChars, int mode);
+
+/*
+ * Maps a Unicode (UCS-2) string to a glyph array. Returns the number of glyphs in the array,
+ * which for TrueType fonts is always the same as the number of input characters.
+ *
+ * @param ttf         pointer to the TrueTypeFont structure
+ * @param str         pointer to a UCS-2 string
+ * @param nchars      number of characters in <b>str</b>
+ * @param glyphArray  pointer to the glyph array where glyph IDs are to be recorded.
+ *
+ * @return MapString() returns -1 if the TrueType font has no usable 'cmap' tables.
+ *         Otherwise it returns the number of characters processed: <b>nChars</b>
+ *
+ * glyphIDs of TrueType fonts are 2 guint8 positive numbers. glyphID of 0 denotes a missing
+ * glyph and traditionally defaults to an empty square.
+ * glyphArray should be at least sizeof(guint16) * nchars bytes long. If glyphArray is NULL
+ * MapString() replaces the UCS-2 characters in str with glyphIDs.
+ * @ingroup sft
+ */
+#ifdef USE_GSUB
+    int MapString(TrueTypeFont *ttf, guint16 *str, int nchars, guint16 *glyphArray, int bvertical);
+#else
+    int MapString(TrueTypeFont *ttf, guint16 *str, int nchars, guint16 *glyphArray);
+#endif
+
+/*
+ * Maps a Unicode (UCS-2) character to a glyph ID and returns it. Missing glyph has
+ * a glyphID of 0 so this function can be used to test if a character is encoded in the font.
+ *
+ * @param ttf         pointer to the TrueTypeFont structure
+ * @param ch          Unicode (UCS-2) character
+ * @return glyph ID, if the character is missing in the font, the return value is 0.
+ * @ingroup sft
+ */
+#ifdef USE_GSUB
+    guint16 MapChar(TrueTypeFont *ttf, guint16 ch, int bvertical);
+#else
+    guint16 MapChar(TrueTypeFont *ttf, guint16 ch);
+#endif
+#endif
+
+/*
+ * Returns global font information about the TrueType font.
+ * @see TTGlobalFontInfo
+ *
+ * @param ttf         pointer to a TrueTypeFont structure
+ * @param info        pointer to a TTGlobalFontInfo structure
+ * @ingroup sft
+ *
+ */
+    void GetTTGlobalFontInfo(TrueTypeFont *ttf, TTGlobalFontInfo *info);
+
+/*
+ * Returns kerning information for an array of glyphs.
+ * Kerning is not cumulative.
+ * kern[i] contains kerning information for a pair of glyphs at positions i and i+1
+ *
+ * @param ttf         pointer to a TrueTypeFont structure
+ * @param glyphs      array of source glyphs
+ * @param nglyphs     number of glyphs in the array
+ * @param wmode       writing mode: 0 - horizontal, 1 - vertical
+ * @param kern        array of KernData structures. It should contain nglyphs-1 elements
+ * @see KernData
+ * @ingroup sft
+ *
+ */
+void KernGlyphs(TrueTypeFont *ttf, guint16 *glyphs, int nglyphs, int wmode, KernData *kern);
+
+/*
+ * Returns nonzero if font is a symbol encoded font
+ */
+int CheckSymbolEncoding(TrueTypeFont* ttf);
+
+/*
+ * Extracts a 'cmap' table from a font, allocates memory for it and returns a pointer to it.
+ * DEPRECATED - use ExtractTable instead
+ */
+#if 0
+guint8 *ExtractCmap(TrueTypeFont *ttf);
+#endif
+
+/*
+ * Extracts a table from a font, allocates memort for it and returns a pointer to it.
+ */
+guint8 *ExtractTable(TrueTypeFont *ttf, guint32 tag);
+
+/*
+ * Returns a pointer to the table but does not allocate memory for it.
+ */
+const guint8 *GetTable(TrueTypeFont *ttf, guint32 tag);
+
+/*
+ * Functions that do not use TrueTypeFont structure
+ * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+ */
+
+/*
+ * Reads full (vertical and horisontal) glyph metrics for an array of glyphs from hmtx and vmtx tables
+ *
+ * @param hmtx       TrueType hmtx table
+ * @param vmtx       TrueType vmtx table
+ * @param hcount     numberOfHMetrics value
+ * @param vcount     numOfLongVerMetrics value
+ * @param gcount     total number of glyphs in the font
+ * @param UPEm       units per Em value
+ * @param glyphArray array of source glyph IDs
+ * @param nGlyphs    number of glyphs in the glyphArray array
+ *
+ * @return           array of TTFullSimpleGlyphMetrics data structures
+ *
+ */
+
+TTFullSimpleGlyphMetrics *ReadGlyphMetrics(guint8 *hmtx, guint8 *vmtx, int hcount, int vcount, int gcount, int UPEm, guint16 *glyphArray, int nGlyphs);
+
+void ReadSingleGlyphMetrics(guint8 *hmtx, guint8 *vmtx, int hcount, int vcount, int gcount, int UPEm, guint16 glyphID, TTFullSimpleGlyphMetrics *metrics);
+
+
+/*
+ * Returns the length of the 'kern' subtable
+ *
+ * @param kern       pointer to the 'kern' subtable
+ *
+ * @return           number of bytes in it
+ */
+guint32 GetKernSubtableLength(guint8 *kern);
+
+/*
+ *  Kerns a pair of glyphs.
+ *
+ * @param kerntype   type of the kern table
+ * @param nkern      number of kern subtables
+ * @param kern       array of pointers to kern subtables
+ * @pram unitsPerEm  units per Em value
+ * @param wmode      writing mode: 0 - horizontal, 1 - vertical
+ * @param a          ID of the first glyoh
+ * @param b          ID of the second glyoh
+ * @param x          X-axis kerning value is returned here
+ * @param y          Y-axis kerning value is returned here
+ */
+void KernGlyphPair(int kerntype, guint32 nkern, guint8 **kern, int unitsPerEm, int wmode, guint32 a, guint32 b, int *x, int *y);
+
+
+
+/*- private definitions */ /*FOLD00*/
+
+struct _TrueTypeFont {
+    guint32 tag;
+
+    char   *fname;
+    off_t  fsize;
+    guint8   *ptr;
+
+    char   *psname;
+    char   *family;
+    guint16  *ufamily;
+    char   *subfamily;
+
+    guint32 ntables;
+    guint32 tdoffset;                              /* offset to the table directory (!= 0 for TrueType collections)      */
+    guint32 *goffsets;
+    int    nglyphs;
+    int    unitsPerEm;
+    int    numberOfHMetrics;
+    int    numOfLongVerMetrics;                   /* if this number is not 0, font has vertical metrics information */
+    guint8   *cmap;
+    int    cmapType;
+    guint16 (*mapper)(const guint8 *, guint16);       /* character to glyphID translation function                          */
+    void   **tables;                              /* array of pointers to tables                                        */
+    guint32 *tlens;                                /* array of table lengths                                             */
+    int    kerntype;                              /* Defined in the KernType enum                                       */
+    guint32 nkern;                                 /* number of kern subtables                                           */
+    guint8   **kerntables;                          /* array of pointers to kern subtables                                */
+#ifdef USE_GSUB
+    void   *pGSubstitution;                       /* info provided by GSUB for UseGSUB()                                */
+#endif
+};
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __SUBFONT_H */
diff --git a/src/ttsubset/ttcr.c b/src/ttsubset/ttcr.c
new file mode 100644 (file)
index 0000000..eecfcad
--- /dev/null
@@ -0,0 +1,1651 @@
+/*
+ * Copyright © 2002, 2003 Sun Microsystems, Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ *
+ * 3. Neither the name of Sun Microsystems, Inc. nor the names of 
+ *    contributors may be used to endorse or promote products derived from
+ *    this software without specific prior written permission.
+ *
+ * This software is provided "AS IS," without a warranty of any kind.
+ *
+ * ALL EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES,
+ * INCLUDING ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A
+ * PARTICULAR PURPOSE OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED.
+ * SUN AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES OR
+ * LIABILITIES SUFFERED BY LICENSEE AS A RESULT OF OR RELATING TO USE,
+ * MODIFICATION OR DISTRIBUTION OF THE SOFTWARE OR ITS DERIVATIVES.
+ * IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST REVENUE,
+ * PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL,
+ * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE
+ * THEORY OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE
+ * SOFTWARE, EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
+ *
+ */
+
+/* $Id$ */
+/* @(#)ttcr.c 1.7 03/01/08 SMI */
+
+/*
+ * @file ttcr.c
+ * @brief TrueTypeCreator method implementation
+ * @author Alexander Gelfenbain <adg@sun.com>
+ * @version 1.3
+ *
+ */
+
+#include <sys/types.h>
+#ifdef HAVE_UNISTD_H
+#include <unistd.h>
+#endif
+#include <fcntl.h>
+#include <stdlib.h>
+#ifdef sun
+#include <strings.h> /* bzero() only in strings.h on Solaris */
+#else
+#include <string.h>
+#endif
+#include <assert.h>
+#include "ttcr.h"
+
+#ifndef O_BINARY
+#define O_BINARY 0
+#endif
+
+/* These must be #defined so that they can be used in initializers */
+#define T_maxp  0x6D617870
+#define T_glyf  0x676C7966
+#define T_head  0x68656164
+#define T_loca  0x6C6F6361
+#define T_name  0x6E616D65
+#define T_hhea  0x68686561
+#define T_hmtx  0x686D7478
+#define T_cmap  0x636D6170
+#define T_vhea  0x76686561
+#define T_vmtx  0x766D7478
+#define T_OS2   0x4F532F32
+#define T_post  0x706F7374
+#define T_kern  0x6B65726E
+#define T_cvt   0x63767420
+
+typedef struct {
+    guint32 tag;
+    guint32 length;
+    guint8  *data;
+} TableEntry;
+
+/*
+ * this is a duplicate code from sft.c but it is left here for performance reasons
+ */
+#ifdef __GNUC__
+#define _inline static __inline__
+#else
+#define _inline static
+#endif
+
+_inline guint32 mkTag(guint8 a, guint8 b, guint8 c, guint8 d) {
+    return (a << 24) | (b << 16) | (c << 8) | d;
+}
+
+/*- Data access macros for data stored in big-endian or little-endian format */
+_inline gint16 GetInt16(const guint8 *ptr, size_t offset, int bigendian)
+{
+    gint16 t;
+    assert(ptr != 0);
+    
+    if (bigendian) {
+        t = (ptr+offset)[0] << 8 | (ptr+offset)[1];
+    } else {
+        t = (ptr+offset)[1] << 8 | (ptr+offset)[0];
+    }
+        
+    return t;
+}
+
+_inline guint16 GetUInt16(const guint8 *ptr, size_t offset, int bigendian)
+{
+    guint16 t;
+    assert(ptr != 0);
+
+    if (bigendian) {
+        t = (ptr+offset)[0] << 8 | (ptr+offset)[1];
+    } else {
+        t = (ptr+offset)[1] << 8 | (ptr+offset)[0];
+    }
+
+    return t;
+}
+
+_inline gint32  GetInt32(const guint8 *ptr, size_t offset, int bigendian)
+{
+    gint32 t;
+    assert(ptr != 0);
+    
+    if (bigendian) {
+        t = (ptr+offset)[0] << 24 | (ptr+offset)[1] << 16 |
+            (ptr+offset)[2] << 8  | (ptr+offset)[3];
+    } else {
+        t = (ptr+offset)[3] << 24 | (ptr+offset)[2] << 16 |
+            (ptr+offset)[1] << 8  | (ptr+offset)[0];
+    }
+        
+    return t;
+}
+
+_inline guint32 GetUInt32(const guint8 *ptr, size_t offset, int bigendian)
+{
+    guint32 t;
+    assert(ptr != 0);
+    
+
+    if (bigendian) {
+        t = (ptr+offset)[0] << 24 | (ptr+offset)[1] << 16 |
+            (ptr+offset)[2] << 8  | (ptr+offset)[3];
+    } else {
+        t = (ptr+offset)[3] << 24 | (ptr+offset)[2] << 16 |
+            (ptr+offset)[1] << 8  | (ptr+offset)[0];
+    }
+        
+    return t;
+}
+
+
+_inline void PutInt16(gint16 val, guint8 *ptr, size_t offset, int bigendian)
+{
+    assert(ptr != 0);
+
+    if (bigendian) {
+        ptr[offset] = (val >> 8) & 0xFF;
+        ptr[offset+1] = val & 0xFF;
+    } else {
+        ptr[offset+1] = (val >> 8) & 0xFF;
+        ptr[offset] = val & 0xFF;
+    }
+
+}
+
+_inline void PutUInt16(guint16 val, guint8 *ptr, size_t offset, int bigendian)
+{
+    assert(ptr != 0);
+
+    if (bigendian) {
+        ptr[offset] = (val >> 8) & 0xFF;
+        ptr[offset+1] = val & 0xFF;
+    } else {
+        ptr[offset+1] = (val >> 8) & 0xFF;
+        ptr[offset] = val & 0xFF;
+    }
+
+}
+
+
+_inline void PutUInt32(guint32 val, guint8 *ptr, size_t offset, int bigendian)
+{
+    assert(ptr != 0);
+
+    if (bigendian) {
+        ptr[offset] = (val >> 24) & 0xFF;
+        ptr[offset+1] = (val >> 16) & 0xFF;
+        ptr[offset+2] = (val >> 8) & 0xFF;
+        ptr[offset+3] = val & 0xFF;
+    } else {
+        ptr[offset+3] = (val >> 24) & 0xFF;
+        ptr[offset+2] = (val >> 16) & 0xFF;
+        ptr[offset+1] = (val >> 8) & 0xFF;
+        ptr[offset] = val & 0xFF;
+    }
+
+}
+
+
+_inline void PutInt32(gint32 val, guint8 *ptr, size_t offset, int bigendian)
+{
+    assert(ptr != 0);
+
+    if (bigendian) {
+        ptr[offset] = (val >> 24) & 0xFF;
+        ptr[offset+1] = (val >> 16) & 0xFF;
+        ptr[offset+2] = (val >> 8) & 0xFF;
+        ptr[offset+3] = val & 0xFF;
+    } else {
+        ptr[offset+3] = (val >> 24) & 0xFF;
+        ptr[offset+2] = (val >> 16) & 0xFF;
+        ptr[offset+1] = (val >> 8) & 0xFF;
+        ptr[offset] = val & 0xFF;
+    }
+
+}
+
+static int TableEntryCompareF(const void *l, const void *r)
+{
+    return ((const TableEntry *) l)->tag - ((const TableEntry *) r)->tag;
+}
+
+static int NameRecordCompareF(const void *l, const void *r)
+{
+    NameRecord *ll = (NameRecord *) l;
+    NameRecord *rr = (NameRecord *) r;
+
+    if (ll->platformID != rr->platformID) {
+        return ll->platformID - rr->platformID;
+    } else if (ll->encodingID != rr->encodingID) {
+        return ll->encodingID - rr->encodingID;
+    } else if (ll->languageID != rr->languageID) {
+        return ll->languageID - rr->languageID;
+    } else if (ll->nameID != rr->nameID) {
+        return ll->nameID - rr->nameID;
+    }
+    return 0;
+}
+    
+
+static guint32 CheckSum(guint32 *ptr, guint32 length)
+{
+    guint32 sum = 0;
+    guint32 *endptr = ptr + ((length + 3) & (guint32) ~3) / 4;
+
+    while (ptr < endptr) sum += *ptr++;
+
+    return sum;
+}
+
+_inline void *smalloc(size_t size)
+{
+    void *res = malloc(size);
+    assert(res != 0);
+    return res;
+}
+
+_inline void *scalloc(size_t n, size_t size)
+{
+    void *res = calloc(n, size);
+    assert(res != 0);
+    return res;
+}
+
+/*
+ * Public functions
+ */
+
+void TrueTypeCreatorNewEmpty(guint32 tag, TrueTypeCreator **_this)
+{
+    TrueTypeCreator *ptr = smalloc(sizeof(TrueTypeCreator));
+
+    ptr->tables = listNewEmpty();
+    listSetElementDtor(ptr->tables, (GDestroyNotify)TrueTypeTableDispose);
+
+    ptr->tag = tag;
+
+    *_this = ptr;
+}
+
+void TrueTypeCreatorDispose(TrueTypeCreator *_this)
+{
+    listDispose(_this->tables);
+    free(_this);
+}
+
+int AddTable(TrueTypeCreator *_this, TrueTypeTable *table)
+{
+    if (table != 0) {
+        listAppend(_this->tables, table);
+    }
+    return SF_OK;
+}
+
+void RemoveTable(TrueTypeCreator *_this, guint32 tag)
+{
+    int done = 0;
+    
+    if (listCount(_this->tables)) {
+        listToFirst(_this->tables);
+        do {
+            if (((TrueTypeTable *) listCurrent(_this->tables))->tag == tag) {
+                listRemove(_this->tables);
+            } else {
+                if (listNext(_this->tables)) {
+                    done = 1;
+                }
+            }
+        } while (!done);
+    }
+}
+
+static void ProcessTables(TrueTypeCreator *);
+
+int StreamToMemory(TrueTypeCreator *_this, guint8 **ptr, guint32 *length)
+{
+    guint16 numTables, searchRange=1, entrySelector=0, rangeShift;
+    guint32 s, offset, checkSumAdjustment = 0;
+    guint32 *p;
+    guint8 *ttf;
+    int i=0, n;
+    TableEntry *te;
+    guint8 *head = NULL;     /* saved pointer to the head table data for checkSumAdjustment calculation */
+    
+    if ((n = listCount(_this->tables)) == 0) return SF_TTFORMAT;
+
+    ProcessTables(_this);
+
+    /* ProcessTables() adds 'loca' and 'hmtx' */
+    
+    n = listCount(_this->tables);
+    numTables = (guint16) n;
+    
+
+    te = scalloc(n, sizeof(TableEntry));
+
+    listToFirst(_this->tables);
+    for (i = 0; i < n; i++) {
+        GetRawData((TrueTypeTable *) listCurrent(_this->tables), &te[i].data, &te[i].length, &te[i].tag);
+        listNext(_this->tables);
+    }
+
+    qsort(te, n, sizeof(TableEntry), TableEntryCompareF);
+    
+    do {
+        searchRange *= 2;
+        entrySelector++;
+    } while (searchRange <= numTables);
+
+    searchRange *= 8;
+    entrySelector--;
+    rangeShift = numTables * 16 - searchRange;
+
+    s = offset = 12 + 16 * n;
+
+    for (i = 0; i < n; i++) {
+        s += (te[i].length + 3) & (guint32) ~3;
+        /* if ((te[i].length & 3) != 0) s += (4 - (te[i].length & 3)) & 3; */
+    }
+
+    ttf = smalloc(s);
+
+    /* Offset Table */
+    PutUInt32(_this->tag, ttf, 0, 1);
+    PutUInt16(numTables, ttf, 4, 1);
+    PutUInt16(searchRange, ttf, 6, 1);
+    PutUInt16(entrySelector, ttf, 8, 1);
+    PutUInt16(rangeShift, ttf, 10, 1);
+
+    /* Table Directory */
+    for (i = 0; i < n; i++) {
+        PutUInt32(te[i].tag, ttf + 12, 16 * i, 1);
+        PutUInt32(CheckSum((guint32 *) te[i].data, te[i].length), ttf + 12, 16 * i + 4, 1);
+        PutUInt32(offset, ttf + 12, 16 * i + 8, 1);
+        PutUInt32(te[i].length, ttf + 12, 16 * i + 12, 1);
+
+        if (te[i].tag == T_head) {
+            head = ttf + offset;
+        }
+
+        memcpy(ttf+offset, te[i].data, (te[i].length + 3) & (guint32) ~3 );
+        offset += (te[i].length + 3) & (guint32) ~3;
+        /* if ((te[i].length & 3) != 0) offset += (4 - (te[i].length & 3)) & 3; */
+    }
+
+    free(te);
+
+    p = (guint32 *) ttf;
+    for (i = 0; i < s / 4; i++) checkSumAdjustment += p[i];
+    PutUInt32(0xB1B0AFBA - checkSumAdjustment, head, 8, 1);
+
+    *ptr = ttf;
+    *length = s;
+    
+    return SF_OK;
+}
+
+int StreamToFile(TrueTypeCreator *_this, const char* fname)
+{
+    guint8 *ptr;
+    guint32 length;
+    int fd, r;
+
+    if (!fname) return SF_BADFILE;
+    if ((fd = open(fname, O_WRONLY | O_BINARY | O_CREAT | O_TRUNC, 0644)) == -1) return SF_BADFILE;
+
+    if ((r = StreamToMemory(_this, &ptr, &length)) != SF_OK) return r;
+
+    if (write(fd, ptr, length) != length) {
+        r = SF_FILEIO;
+    } else {
+        r = SF_OK;
+    }
+
+    close(fd);
+    free(ptr);
+    return r;
+}
+
+
+
+/*
+ * TrueTypeTable private methods
+ */
+
+#define TABLESIZE_head 54
+#define TABLESIZE_hhea 36
+#define TABLESIZE_maxp 32
+
+
+
+/*    Table         data points to
+ * --------------------------------------------
+ *    generic       tdata_generic struct
+ *    'head'        TABLESIZE_head bytes of memory
+ *    'hhea'        TABLESIZE_hhea bytes of memory
+ *    'loca'        tdata_loca struct
+ *    'maxp'        TABLESIZE_maxp bytes of memory
+ *    'glyf'        list of GlyphData structs (defined in sft.h)
+ *    'name'        list of NameRecord structs (defined in sft.h)
+ *    'post'        tdata_post struct
+ *
+ */
+
+
+#define CMAP_SUBTABLE_INIT 10
+#define CMAP_SUBTABLE_INCR 10
+#define CMAP_PAIR_INIT 500
+#define CMAP_PAIR_INCR 500
+
+typedef struct {
+    guint32  id;                         /* subtable ID (platform/encoding ID)    */
+    guint32  n;                          /* number of used translation pairs      */
+    guint32  m;                          /* number of allocated translation pairs */
+    guint32 *xc;                         /* character array                       */
+    guint32 *xg;                         /* glyph array                           */
+} CmapSubTable;
+
+typedef struct {
+    guint32 n;                           /* number of used CMAP sub-tables       */
+    guint32 m;                           /* number of allocated CMAP sub-tables  */
+    CmapSubTable *s;                    /* sotred array of sub-tables           */
+} table_cmap;
+
+typedef struct {
+    guint32 tag;
+    guint32 nbytes;
+    guint8 *ptr;
+} tdata_generic;
+
+typedef struct {
+    guint32 nbytes;                      /* number of bytes in loca table */
+    guint8 *ptr;                          /* pointer to the data */
+} tdata_loca;
+
+typedef struct {
+    guint32 format;
+    guint32 italicAngle;
+    gint16  underlinePosition;
+    gint16  underlineThickness;
+    guint32 isFixedPitch;
+    void   *ptr;                        /* format-specific pointer */
+} tdata_post;
+    
+
+/* allocate memory for a TT table */
+static guint8 *ttmalloc(guint32 nbytes)
+{
+    guint32 n; 
+    guint8 *res;
+
+    n = (nbytes + 3) & (guint32) ~3;
+    res = malloc(n);
+    assert(res != 0);
+    bzero(res, n);
+
+    return res;
+}
+    
+static void FreeGlyphData(void *ptr)
+{
+    GlyphData *p = (GlyphData *) ptr;
+    if (p->ptr) free(p->ptr);
+    free(p);
+}
+
+static void TrueTypeTableDispose_generic(TrueTypeTable *_this)
+{
+    if (_this) {
+        if (_this->data) {
+            tdata_generic *pdata = (tdata_generic *) _this->data;
+            if (pdata->nbytes) free(pdata->ptr);
+            free(_this->data);
+        }
+        free(_this);
+    }
+}
+
+static void TrueTypeTableDispose_head(TrueTypeTable *_this)
+{
+    if (_this) {
+        if (_this->data) free(_this->data);
+        free(_this);
+    }
+}
+
+static void TrueTypeTableDispose_hhea(TrueTypeTable *_this)
+{
+    if (_this) {
+        if (_this->data) free(_this->data);
+        free(_this);
+    }
+}
+
+static void TrueTypeTableDispose_loca(TrueTypeTable *_this)
+{
+    if (_this) {
+        if (_this->data) {
+            tdata_loca *p = (tdata_loca *) _this->data;
+            if (p->ptr) free(p->ptr);
+            free(_this->data);
+        }
+        free(_this);
+    }
+}
+
+static void TrueTypeTableDispose_maxp(TrueTypeTable *_this)
+{
+    if (_this) {
+        if (_this->data) free(_this->data);
+        free(_this);
+    }
+}
+
+static void TrueTypeTableDispose_glyf(TrueTypeTable *_this)
+{
+    if (_this) {
+        if (_this->data) listDispose((list) _this->data);
+        free(_this);
+    }
+}
+
+static void TrueTypeTableDispose_cmap(TrueTypeTable *_this)
+{
+    table_cmap *t;
+    CmapSubTable *s;
+    int i;
+    
+    if (_this) {
+        t = (table_cmap *) _this->data;
+        if (t) {
+            s = t->s;
+            if (s) {
+                for (i = 0; i < t->m; i++) {
+                    if (s[i].xc) free(s[i].xc);
+                    if (s[i].xg) free(s[i].xg);
+                }
+                free(s);
+            }
+            free(t);
+        }
+        free(_this);
+    }
+}
+
+static void TrueTypeTableDispose_name(TrueTypeTable *_this)
+{
+    if (_this) {
+        if (_this->data) listDispose((list) _this->data);
+        free(_this);
+    }
+}
+
+static void TrueTypeTableDispose_post(TrueTypeTable *_this)
+{
+    if (_this) {
+        tdata_post *p = (tdata_post *) _this->data;
+        if (p) {
+            if (p->format == 0x00030000) {
+                /* do nothing */
+            } else {
+                fprintf(stderr, "Unsupported format of a 'post' table: %08X.\n", p->format);
+            }
+            free(p);
+        }
+        free(_this);
+    }
+}
+
+/* destructor vtable */
+
+static struct {
+    guint32 tag;
+    void (*f)(TrueTypeTable *);
+} vtable1[] =
+{
+    {0,      TrueTypeTableDispose_generic},
+    {T_head, TrueTypeTableDispose_head},
+    {T_hhea, TrueTypeTableDispose_hhea},
+    {T_loca, TrueTypeTableDispose_loca},
+    {T_maxp, TrueTypeTableDispose_maxp},
+    {T_glyf, TrueTypeTableDispose_glyf},
+    {T_cmap, TrueTypeTableDispose_cmap},
+    {T_name, TrueTypeTableDispose_name},
+    {T_post, TrueTypeTableDispose_post}
+    
+};
+
+static int GetRawData_generic(TrueTypeTable *_this, guint8 **ptr, guint32 *len, guint32 *tag)
+{
+    assert(_this != 0);
+    assert(_this->data != 0);
+
+    *ptr = ((tdata_generic *) _this->data)->ptr;
+    *len = ((tdata_generic *) _this->data)->nbytes;
+    *tag = ((tdata_generic *) _this->data)->tag;
+
+    return TTCR_OK;
+}
+
+
+static int GetRawData_head(TrueTypeTable *_this, guint8 **ptr, guint32 *len, guint32 *tag)
+{
+    *len = TABLESIZE_head;
+    *ptr = (guint8 *) _this->data;
+    *tag = T_head;
+    
+    return TTCR_OK;
+}
+
+static int GetRawData_hhea(TrueTypeTable *_this, guint8 **ptr, guint32 *len, guint32 *tag)
+{
+    *len = TABLESIZE_hhea;
+    *ptr = (guint8 *) _this->data;
+    *tag = T_hhea;
+    
+    return TTCR_OK;
+}
+
+static int GetRawData_loca(TrueTypeTable *_this, guint8 **ptr, guint32 *len, guint32 *tag)
+{
+    tdata_loca *p;
+
+    assert(_this->data != 0);
+
+    p = (tdata_loca *) _this->data;
+
+    if (p->nbytes == 0) return TTCR_ZEROGLYPHS;
+
+    *ptr = p->ptr;
+    *len = p->nbytes;
+    *tag = T_loca;
+
+    return TTCR_OK;
+}
+
+static int GetRawData_maxp(TrueTypeTable *_this, guint8 **ptr, guint32 *len, guint32 *tag)
+{
+    *len = TABLESIZE_maxp;
+    *ptr = (guint8 *) _this->data;
+    *tag = T_maxp;
+    
+    return TTCR_OK;
+}
+
+static int GetRawData_glyf(TrueTypeTable *_this, guint8 **ptr, guint32 *len, guint32 *tag)
+{
+    guint32 n, nbytes = 0;
+    list l = (list) _this->data;
+    /* guint16 curID = 0;    */               /* to check if glyph IDs are sequential and start from zero */
+    guint8 *p;
+
+    *ptr = NULL;
+    *len = 0;
+    *tag = 0;
+
+    if (listCount(l) == 0) return TTCR_ZEROGLYPHS;
+
+    listToFirst(l);
+    do {
+        /* if (((GlyphData *) listCurrent(l))->glyphID != curID++) return TTCR_GLYPHSEQ; */
+        nbytes += ((GlyphData *) listCurrent(l))->nbytes;
+    } while (listNext(l));
+
+    p = _this->rawdata = ttmalloc(nbytes);
+
+    listToFirst(l);
+    do {
+        n = ((GlyphData *) listCurrent(l))->nbytes;
+        if (n != 0) {
+            memcpy(p, ((GlyphData *) listCurrent(l))->ptr, n);
+            p += n;
+        }
+    } while (listNext(l));
+
+    *len = nbytes;
+    *ptr = _this->rawdata;
+    *tag = T_glyf;
+
+    return TTCR_OK;
+}
+
+/* cmap packers */
+static guint8 *PackCmapType0(CmapSubTable *s, guint32 *length)
+{
+    guint8 *ptr = smalloc(262);
+    guint8 *p = ptr + 6;
+    int i, j;
+    guint16 g;
+
+    PutUInt16(0, ptr, 0, 1);
+    PutUInt16(262, ptr, 2, 1);
+    PutUInt16(0, ptr, 4, 1);
+
+    for (i = 0; i < 256; i++) {
+        g = 0;
+        for (j = 0; j < s->n; j++) {
+            if (s->xc[j] == i) {
+                g = (guint16) s->xg[j];
+            }
+        }
+        p[i] = (guint8) g;
+    }
+    *length = 262;
+    return ptr;
+}
+            
+
+/* XXX it only handles Format 0 encoding tables */
+static guint8 *PackCmap(CmapSubTable *s, guint32 *length)
+{
+    return PackCmapType0(s, length);
+}
+
+static int GetRawData_cmap(TrueTypeTable *_this, guint8 **ptr, guint32 *len, guint32 *tag)
+{
+    table_cmap *t;
+    guint8 **subtables;
+    guint32 *sizes;            /* of subtables */
+    int i;
+    guint32 tlen = 0;
+    guint32 l;
+    guint32 cmapsize;
+    guint8 *cmap;
+    guint32 coffset;
+
+    assert(_this != 0);
+    t = (table_cmap *) _this->data;
+    assert(t != 0);
+    assert(t->n != 0);
+
+    subtables = scalloc(t->n, sizeof(guint8 *));
+    sizes = scalloc(t->n, sizeof(guint32));
+
+    for (i = 0; i < t->n; i++) {
+        subtables[i] = PackCmap(t->s+i, &l);
+        sizes[i] = l;
+        tlen += l;
+    }
+
+    cmapsize = tlen + 4 + 8 * t->n;
+    _this->rawdata = cmap = ttmalloc(cmapsize);
+
+    PutUInt16(0, cmap, 0, 1);
+    PutUInt16(t->n, cmap, 2, 1);
+    coffset = 4 + t->n * 8;
+
+    for (i = 0; i < t->n; i++) {
+        PutUInt16(t->s[i].id >> 16, cmap + 4, i * 8, 1);
+        PutUInt16(t->s[i].id & 0xFF, cmap + 4, 2 + i * 8, 1);
+        PutUInt32(coffset, cmap + 4, 4 + i * 8, 1);
+        memcpy(cmap + coffset, subtables[i], sizes[i]);
+        free(subtables[i]);
+        coffset += sizes[i];
+    }
+
+    free(subtables);
+    free(sizes);
+
+    *ptr = cmap;
+    *len = cmapsize;
+    *tag = T_cmap;
+
+    return TTCR_OK;
+}
+
+
+static int GetRawData_name(TrueTypeTable *_this, guint8 **ptr, guint32 *len, guint32 *tag)
+{
+    list l;
+    NameRecord *nr;
+    gint16 i=0, n;                          /* number of Name Records */
+    guint8 *name;
+    guint16 nameLen;
+    int stringLen = 0;
+    guint8 *p1, *p2;
+
+    *ptr = NULL;
+    *len = 0;
+    *tag = 0;
+
+    assert(_this != 0);
+    l = (list) _this->data;
+    assert(l != 0);
+
+    if ((n = listCount(l)) == 0) return TTCR_NONAMES;
+
+    nr = scalloc(n, sizeof(NameRecord));
+
+    listToFirst(l);
+
+    do {
+        memcpy(nr+i, listCurrent(l), sizeof(NameRecord));
+        stringLen += nr[i].slen;
+        i++;
+    } while (listNext(l));
+
+    if (stringLen > 65535) {
+        free(nr);
+        return TTCR_NAMETOOLONG;
+    }
+
+    qsort(nr, n, sizeof(NameRecord), NameRecordCompareF);
+
+    nameLen = stringLen + 12 * n + 6;
+    name = ttmalloc(nameLen); 
+
+    PutUInt16(0, name, 0, 1);
+    PutUInt16(n, name, 2, 1);
+    PutUInt16(6 + 12 * n, name, 4, 1);
+
+    p1 = name + 6;
+    p2 = p1 + 12 * n;
+
+    for (i = 0; i < n; i++) {
+        PutUInt16(nr[i].platformID, p1, 0, 1);
+        PutUInt16(nr[i].encodingID, p1, 2, 1);
+        PutUInt16(nr[i].languageID, p1, 4, 1);
+        PutUInt16(nr[i].nameID, p1, 6, 1);
+        PutUInt16(nr[i].slen, p1, 8, 1);
+        PutUInt16(p2 - (name + 6 + 12 * n), p1, 10, 1);
+        memcpy(p2, nr[i].sptr, nr[i].slen);
+        /* {int j; for(j=0; j<nr[i].slen; j++) printf("%c", nr[i].sptr[j]); printf("\n"); }; */
+        p2 += nr[i].slen;
+        p1 += 12;
+    }
+
+    free(nr);
+    _this->rawdata = name;
+
+    *ptr = name;
+    *len = nameLen;
+    *tag = T_name;
+
+     /*{int j; for(j=0; j<nameLen; j++) printf("%c", name[j]); }; */
+
+    return TTCR_OK;
+}
+
+static int GetRawData_post(TrueTypeTable *_this, guint8 **ptr, guint32 *len, guint32 *tag)
+{
+    tdata_post *p = (tdata_post *) _this->data;
+    guint8 *post = NULL;
+    guint32 postLen = 0;
+    int ret;
+
+    if (_this->rawdata) free(_this->rawdata);
+
+    if (p->format == 0x00030000) {
+        postLen = 32;
+        post = ttmalloc(postLen);
+        PutUInt32(0x00030000, post, 0, 1);
+        PutUInt32(p->italicAngle, post, 4, 1);
+        PutUInt16(p->underlinePosition, post, 8, 1);
+        PutUInt16(p->underlineThickness, post, 10, 1);
+        PutUInt16(p->isFixedPitch, post, 12, 1);
+        ret = TTCR_OK;
+    } else {
+        fprintf(stderr, "Unrecognized format of a post table: %08X.\n", p->format);
+        ret = TTCR_POSTFORMAT;
+    }
+
+    *ptr = _this->rawdata = post;
+    *len = postLen;
+    *tag = T_post;
+
+    return ret;
+}
+
+    
+
+    
+
+static struct {
+    guint32 tag;
+    int (*f)(TrueTypeTable *, guint8 **, guint32 *, guint32 *);
+} vtable2[] =
+{
+    {0,      GetRawData_generic},
+    {T_head, GetRawData_head},
+    {T_hhea, GetRawData_hhea},
+    {T_loca, GetRawData_loca},
+    {T_maxp, GetRawData_maxp},
+    {T_glyf, GetRawData_glyf},
+    {T_cmap, GetRawData_cmap},
+    {T_name, GetRawData_name},
+    {T_post, GetRawData_post}
+    
+    
+};
+/*
+ * TrueTypeTable public methods
+ */
+
+/* Note: Type42 fonts only need these tables:
+ *        head, hhea, loca, maxp, cvt, prep, glyf, hmtx, fpgm
+ *
+ * Microsoft required tables
+ *        cmap, glyf, head, hhea, hmtx, loca, maxp, name, post, OS/2
+ *
+ * Apple required tables
+ *        cmap, glyf, head, hhea, hmtx, loca, maxp, name, post
+ *
+ */
+
+TrueTypeTable *TrueTypeTableNew(guint32 tag,
+                                guint32 nbytes,
+                                guint8 *ptr)
+{
+    TrueTypeTable *table;
+    tdata_generic *pdata;
+
+    table = smalloc(sizeof(TrueTypeTable));
+    pdata = (tdata_generic *) smalloc(sizeof(tdata_generic)); 
+    pdata->nbytes = nbytes;
+    pdata->tag = tag;
+    if (nbytes) {
+        pdata->ptr = ttmalloc(nbytes); 
+        memcpy(pdata->ptr, ptr, nbytes);
+    } else {
+        pdata->ptr = NULL;
+    }
+
+    table->tag = 0;
+    table->data = pdata;
+    table->rawdata = NULL;
+
+    return table;
+}
+    
+TrueTypeTable *TrueTypeTableNew_head(guint32 fontRevision,
+                                     guint16 flags,
+                                     guint16 unitsPerEm,
+                                     guint8  *created,
+                                     guint16 macStyle,
+                                     guint16 lowestRecPPEM,
+                                     gint16  fontDirectionHint)
+{
+    TrueTypeTable *table;
+    guint8 *ptr;
+
+    assert(created != 0);
+
+    table  = smalloc(sizeof(TrueTypeTable));  
+    ptr  = ttmalloc(TABLESIZE_head);
+
+
+    PutUInt32(0x00010000, ptr, 0, 1);             /* version */
+    PutUInt32(fontRevision, ptr, 4, 1);
+    PutUInt32(0x5F0F3CF5, ptr, 12, 1);            /* magic number */
+    PutUInt16(flags, ptr, 16, 1);
+    PutUInt16(unitsPerEm, ptr, 18, 1);
+    memcpy(ptr+20, created, 8);                   /* Created Long Date */
+    bzero(ptr+28, 8);                             /* Modified Long Date */
+    PutUInt16(macStyle, ptr, 44, 1);
+    PutUInt16(lowestRecPPEM, ptr, 46, 1);
+    PutUInt16(fontDirectionHint, ptr, 48, 1);
+    PutUInt16(0, ptr, 52, 1);                     /* glyph data format: 0 */
+
+    table->data = (void *) ptr;
+    table->tag = T_head;
+    table->rawdata = NULL;
+
+    return table;
+}
+
+TrueTypeTable *TrueTypeTableNew_hhea(gint16  ascender,
+                                     gint16  descender,
+                                     gint16  linegap,
+                                     gint16  caretSlopeRise,
+                                     gint16  caretSlopeRun)
+{
+    TrueTypeTable *table;
+    guint8 *ptr;
+
+    table  = smalloc(sizeof(TrueTypeTable));
+    ptr  = ttmalloc(TABLESIZE_hhea);
+
+    PutUInt32(0x00010000, ptr, 0, 1);             /* version */
+    PutUInt16(ascender, ptr, 4, 1);
+    PutUInt16(descender, ptr, 6, 1);
+    PutUInt16(linegap, ptr, 8, 1);
+    PutUInt16(caretSlopeRise, ptr, 18, 1);
+    PutUInt16(caretSlopeRun, ptr, 20, 1);
+    PutUInt16(0, ptr, 22, 1);                     /* reserved 1 */
+    PutUInt16(0, ptr, 24, 1);                     /* reserved 2 */
+    PutUInt16(0, ptr, 26, 1);                     /* reserved 3 */
+    PutUInt16(0, ptr, 28, 1);                     /* reserved 4 */
+    PutUInt16(0, ptr, 30, 1);                     /* reserved 5 */
+    PutUInt16(0, ptr, 32, 1);                     /* metricDataFormat */
+    
+    table->data = (void *) ptr;
+    table->tag = T_hhea;
+    table->rawdata = NULL;
+
+    return table;
+}
+
+TrueTypeTable *TrueTypeTableNew_loca(void)
+{
+    TrueTypeTable *table = smalloc(sizeof(TrueTypeTable));
+    table->data = smalloc(sizeof(tdata_loca));
+
+    ((tdata_loca *)table->data)->nbytes = 0;
+    ((tdata_loca *)table->data)->ptr = NULL;
+
+    table->tag = T_loca;
+    table->rawdata = NULL;
+
+    return table;
+}
+
+TrueTypeTable *TrueTypeTableNew_maxp(guint8 *maxp, int size)
+{
+    TrueTypeTable *table = smalloc(sizeof(TrueTypeTable));
+    table->data = ttmalloc(TABLESIZE_maxp);
+
+    if (maxp && size == TABLESIZE_maxp) {
+        memcpy(table->data, maxp, TABLESIZE_maxp);
+    }
+    
+    table->tag = T_maxp;
+    table->rawdata = NULL;
+
+    return table;
+}
+
+TrueTypeTable *TrueTypeTableNew_glyf(void)
+{
+    TrueTypeTable *table = smalloc(sizeof(TrueTypeTable));
+    list l = listNewEmpty();
+    
+    assert(l != 0);
+
+    listSetElementDtor(l, FreeGlyphData);
+
+    table->data = l;
+    table->rawdata = NULL;
+    table->tag = T_glyf;
+
+    return table;
+}
+
+TrueTypeTable *TrueTypeTableNew_cmap(void)
+{
+    TrueTypeTable *table = smalloc(sizeof(TrueTypeTable));
+    table_cmap *cmap = smalloc(sizeof(table_cmap));
+    
+    cmap->n = 0;
+    cmap->m = CMAP_SUBTABLE_INIT;
+    cmap->s = (CmapSubTable *) scalloc(CMAP_SUBTABLE_INIT, sizeof(CmapSubTable));
+    memset(cmap->s, 0, sizeof(CmapSubTable) * CMAP_SUBTABLE_INIT);
+    
+    table->data = (table_cmap *) cmap;
+
+    table->rawdata = NULL;
+    table->tag = T_cmap;
+
+    return table;
+}
+
+static void DisposeNameRecord(void *ptr)
+{
+    if (ptr != 0) {
+        NameRecord *nr = (NameRecord *) ptr;
+        if (nr->sptr) free(nr->sptr);
+        free(ptr);
+    }
+}
+
+static NameRecord* NameRecordNewCopy(NameRecord *nr)
+{
+    NameRecord *p = smalloc(sizeof(NameRecord));
+
+    memcpy(p, nr, sizeof(NameRecord));
+
+    if (p->slen) {
+        p->sptr = smalloc(p->slen);
+        memcpy(p->sptr, nr->sptr, p->slen);
+    }
+
+    return p;
+}
+
+TrueTypeTable *TrueTypeTableNew_name(int n, NameRecord *nr)
+{
+    TrueTypeTable *table = smalloc(sizeof(TrueTypeTable));
+    list l = listNewEmpty();
+    
+    assert(l != 0);
+
+    listSetElementDtor(l, DisposeNameRecord);
+
+    if (n != 0) {
+        int i;
+        for (i = 0; i < n; i++) {
+            listAppend(l, NameRecordNewCopy(nr+i));
+        }
+    }
+
+    table->data = l;
+    table->rawdata = NULL;
+    table->tag = T_name;
+
+    return table;
+}
+
+TrueTypeTable *TrueTypeTableNew_post(guint32 format,
+                                     guint32 italicAngle,
+                                     gint16 underlinePosition,
+                                     gint16 underlineThickness,
+                                     guint32 isFixedPitch)
+{
+    TrueTypeTable *table;
+    tdata_post *post;
+
+    assert(format == 0x00030000);                 /* Only format 3.0 is supported at this time */
+    table = smalloc(sizeof(TrueTypeTable));
+    post = smalloc(sizeof(tdata_post));
+
+    post->format = format;
+    post->italicAngle = italicAngle;
+    post->underlinePosition = underlinePosition;
+    post->underlineThickness = underlineThickness;
+    post->isFixedPitch = isFixedPitch;
+    post->ptr = NULL;
+
+    table->data = post;
+    table->rawdata = NULL;
+    table->tag = T_post;
+
+    return table;
+}
+
+
+
+void TrueTypeTableDispose(TrueTypeTable *_this)
+{
+    /* XXX do a binary search */
+    int i;
+
+    assert(_this != 0);
+
+    if (_this->rawdata) free(_this->rawdata);
+
+    for(i=0; i < sizeof(vtable1)/sizeof(*vtable1); i++) {
+        if (_this->tag == vtable1[i].tag) {
+            vtable1[i].f(_this);
+            return;
+        }
+    }
+    assert(!"Unknown TrueType table.\n");
+}
+
+int GetRawData(TrueTypeTable *_this, guint8 **ptr, guint32 *len, guint32 *tag)
+{
+    /* XXX do a binary search */
+    int i;
+
+    assert(_this != 0);
+    assert(ptr != 0);
+    assert(len != 0);
+    assert(tag != 0);
+
+    *ptr = NULL; *len = 0; *tag = 0;
+
+    if (_this->rawdata) {
+        free(_this->rawdata);
+        _this->rawdata = NULL;
+    }
+
+    for(i=0; i < sizeof(vtable2)/sizeof(*vtable2); i++) {
+        if (_this->tag == vtable2[i].tag) {
+            return vtable2[i].f(_this, ptr, len, tag);
+        }
+    }
+
+    assert(!"Unknwon TrueType table.\n");
+    return TTCR_UNKNOWN;
+}
+    
+void cmapAdd(TrueTypeTable *table, guint32 id, guint32 c, guint32 g)
+{
+    int i, found;
+    table_cmap *t;
+    CmapSubTable *s;
+
+    assert(table != 0);
+    assert(table->tag == T_cmap);
+    t = (table_cmap *) table->data; assert(t != 0);
+    s = t->s; assert(s != 0);
+
+    found = 0;
+
+    for (i = 0; i < t->n; i++) {
+        if (s[i].id == id) {
+            found = 1;
+            break;
+        }
+    }
+
+    if (!found) {
+        if (t->n == t->m) {
+            CmapSubTable *tmp;
+            tmp = scalloc(t->m + CMAP_SUBTABLE_INCR, sizeof(CmapSubTable)); 
+            memset(tmp, 0, t->m + CMAP_SUBTABLE_INCR * sizeof(CmapSubTable));
+            memcpy(tmp, s, sizeof(CmapSubTable) * t->m);
+            t->m += CMAP_SUBTABLE_INCR;
+            free(s);
+            s = tmp;
+            t->s = s;
+        }
+
+        for (i = 0; i < t->n; i++) {
+            if (s[i].id > id) break;
+        }
+
+        if (i < t->n) {
+            memmove(s+i+1, s+i, t->n-i);
+        }
+
+        t->n++;
+
+        s[i].id = id;
+        s[i].n = 0;
+        s[i].m = CMAP_PAIR_INIT;
+        s[i].xc = scalloc(CMAP_PAIR_INIT, sizeof(guint32));
+        s[i].xg = scalloc(CMAP_PAIR_INIT, sizeof(guint32));
+    }
+
+    if (s[i].n == s[i].m) {
+        guint32 *tmp1 = scalloc(s[i].m + CMAP_PAIR_INCR, sizeof(guint32));
+        guint32 *tmp2 = scalloc(s[i].m + CMAP_PAIR_INCR, sizeof(guint32));
+        assert(tmp1 != 0);
+        assert(tmp2 != 0);
+        memcpy(tmp1, s[i].xc, sizeof(guint32) * s[i].m);
+        memcpy(tmp2, s[i].xg, sizeof(guint32) * s[i].m);
+        s[i].m += CMAP_PAIR_INCR;
+        free(s[i].xc);
+        free(s[i].xg);
+        s[i].xc = tmp1;
+        s[i].xg = tmp2;
+    }
+
+    s[i].xc[s[i].n] = c;
+    s[i].xg[s[i].n] = g;
+    s[i].n++;
+}
+
+guint32 glyfAdd(TrueTypeTable *table, GlyphData *glyphdata, TrueTypeFont *fnt)
+{
+    list l;
+    guint32 currentID;
+    int ret, n, ncomponents;
+    list glyphlist;
+    GlyphData *gd;
+
+    assert(table != 0);
+    assert(table->tag == T_glyf);
+
+    if (!glyphdata) return -1;
+
+    glyphlist = listNewEmpty();
+
+    ncomponents = GetTTGlyphComponents(fnt, glyphdata->glyphID, glyphlist);
+
+    l = (list) table->data;
+    if (listCount(l) > 0) {
+        listToLast(l);
+        ret = n = ((GlyphData *) listCurrent(l))->newID + 1;
+    } else {
+        ret = n = 0;
+    }
+    glyphdata->newID = n++;
+    listAppend(l, glyphdata);
+
+    if (ncomponents > 1) {
+        listPositionAt(glyphlist, 1);       /* glyphData->glyphID is always the first glyph on the list */
+        do {
+            int found = 0;
+            currentID = (guint32) listCurrent(glyphlist);
+            /* XXX expensive! should be rewritten with sorted arrays! */
+            listToFirst(l);
+            do {
+                if (((GlyphData *) listCurrent(l))->glyphID == currentID) {
+                    found = 1;
+                    break;
+                }
+            } while (listNext(l));
+
+            if (!found) {
+                gd = GetTTRawGlyphData(fnt, currentID);
+                gd->newID = n++;
+                listAppend(l, gd);
+            }
+        } while (listNext(glyphlist));
+    }
+
+    listDispose(glyphlist);
+    return ret;
+}
+
+guint32 glyfCount(const TrueTypeTable *table)
+{
+    assert(table != 0);
+    assert(table->tag == T_glyf);
+    return listCount((list) table->data);
+}
+    
+
+void nameAdd(TrueTypeTable *table, NameRecord *nr)
+{
+    list l;
+
+    assert(table != 0);
+    assert(table->tag == T_name);
+
+    l = (list) table->data;
+
+    listAppend(l, NameRecordNewCopy(nr));
+}
+               
+static TrueTypeTable *FindTable(TrueTypeCreator *tt, guint32 tag)
+{
+    if (listIsEmpty(tt->tables)) return NULL;
+
+    listToFirst(tt->tables);
+
+    do {
+        if (((TrueTypeTable *) listCurrent(tt->tables))->tag == tag) {
+            return listCurrent(tt->tables);
+        }
+    } while (listNext(tt->tables));
+
+    return NULL;
+}
+
+/* This function processes all the tables and synchronizes them before creating
+ * the output TrueType stream.
+ *
+ * *** It adds two TrueType tables to the font: 'loca' and 'hmtx' ***
+ *
+ * It does:
+ *
+ * - Re-numbers glyph IDs and creates 'glyf', 'loca', and 'hmtx' tables.
+ * - Calculates xMin, yMin, xMax, and yMax and stores values in 'head' table.
+ * - Stores indexToLocFormat in 'head'
+ * - updates 'maxp' table
+ * - Calculates advanceWidthMax, minLSB, minRSB, xMaxExtent and numberOfHMetrics
+ *   in 'hhea' table
+ *
+ */
+static void ProcessTables(TrueTypeCreator *tt)
+{
+    TrueTypeTable *glyf, *loca, *head, *maxp, *hhea;
+    list glyphlist;
+    guint32 nGlyphs, locaLen = 0, glyfLen = 0;
+    gint16 xMin = 0, yMin = 0, xMax = 0, yMax = 0;
+    int i = 0;
+    gint16 indexToLocFormat;
+    guint8 *glyfPtr, *locaPtr, *hmtxPtr, *hheaPtr;
+    guint32 hmtxSize;
+    guint8 *p1, *p2;
+    guint16 maxPoints = 0, maxContours = 0, maxCompositePoints = 0, maxCompositeContours = 0;
+    TTSimpleGlyphMetrics *met;
+    int nlsb = 0;
+    guint32 *gid;                        /* array of old glyphIDs */
+
+    glyf = FindTable(tt, T_glyf);
+    glyphlist = (list) glyf->data;
+    nGlyphs = listCount(glyphlist);
+    assert(nGlyphs != 0);
+    gid = scalloc(nGlyphs, sizeof(guint32));
+
+    RemoveTable(tt, T_loca);
+    RemoveTable(tt, T_hmtx);
+
+    /* XXX Need to make sure that composite glyphs do not break during glyph renumbering */
+
+    listToFirst(glyphlist);
+    do {
+        GlyphData *gd = (GlyphData *) listCurrent(glyphlist);
+        gint16 z;
+        glyfLen += gd->nbytes;
+        /* XXX if (gd->nbytes & 1) glyfLen++; */
+
+
+        assert(gd->newID == i);
+        gid[i++] = gd->glyphID;
+        /* gd->glyphID = i++; */
+
+        /* printf("IDs: %d %d.\n", gd->glyphID, gd->newID); */
+
+        if (gd->nbytes != 0) {
+            z = GetInt16(gd->ptr, 2, 1);
+            if (z < xMin) xMin = z;
+
+            z = GetInt16(gd->ptr, 4, 1);
+            if (z < yMin) yMin = z;
+        
+            z = GetInt16(gd->ptr, 6, 1);
+            if (z > xMax) xMax = z;
+        
+            z = GetInt16(gd->ptr, 8, 1);
+            if (z > yMax) yMax = z;
+        }
+
+        if (gd->compflag == 0) {                            /* non-composite glyph */
+            if (gd->npoints > maxPoints) maxPoints = gd->npoints;
+            if (gd->ncontours > maxContours) maxContours = gd->ncontours;
+        } else {                                            /* composite glyph */
+            if (gd->npoints > maxCompositePoints) maxCompositePoints = gd->npoints;
+            if (gd->ncontours > maxCompositeContours) maxCompositeContours = gd->ncontours;
+        }
+        
+    } while (listNext(glyphlist));
+
+    indexToLocFormat = (glyfLen / 2 > 0xFFFF) ? 1 : 0;
+    locaLen = indexToLocFormat ?  (nGlyphs + 1) << 2 : (nGlyphs + 1) << 1;
+
+    glyfPtr = ttmalloc(glyfLen);
+    locaPtr = ttmalloc(locaLen); 
+    met = scalloc(nGlyphs, sizeof(TTSimpleGlyphMetrics));
+    i = 0;
+
+    listToFirst(glyphlist);
+    p1 = glyfPtr;
+    p2 = locaPtr;
+    do {
+        GlyphData *gd = (GlyphData *) listCurrent(glyphlist);
+        
+        if (gd->compflag) {                       /* re-number all components */
+            guint16 flags, index;
+            guint8 *ptr = gd->ptr + 10;
+            do {
+                int j;
+                flags = GetUInt16(ptr, 0, 1);
+                index = GetUInt16(ptr, 2, 1);
+                /* XXX use the sorted array of old to new glyphID mapping and do a binary search */
+                for (j = 0; j < nGlyphs; j++) {
+                    if (gid[j] == index) {
+                        break;
+                    }
+                }
+                /* printf("X: %d -> %d.\n", index, j); */
+
+                PutUInt16((guint16) j, ptr, 2, 1);
+
+                ptr += 4;
+
+                if (flags & ARG_1_AND_2_ARE_WORDS) {
+                    ptr += 4;
+                } else {
+                    ptr += 2;
+                }
+
+                if (flags & WE_HAVE_A_SCALE) {
+                    ptr += 2;
+                } else if (flags & WE_HAVE_AN_X_AND_Y_SCALE) {
+                    ptr += 4;
+                } else if (flags & WE_HAVE_A_TWO_BY_TWO) {
+                    ptr += 8;
+                }
+            } while (flags & MORE_COMPONENTS);
+        }
+
+        if (gd->nbytes != 0) {
+            memcpy(p1, gd->ptr, gd->nbytes);
+        }
+        if (indexToLocFormat == 1) {
+            PutUInt32(p1 - glyfPtr, p2, 0, 1);
+            p2 += 4;
+        } else {
+            PutUInt16((p1 - glyfPtr) >> 1, p2, 0, 1);
+            p2 += 2;
+        }
+        p1 += gd->nbytes;
+
+        /* fill the array of metrics */
+        met[i].adv = gd->aw;
+        met[i].sb  = gd->lsb;
+        i++;
+    } while (listNext(glyphlist));
+
+    free(gid);
+
+    if (indexToLocFormat == 1) {
+        PutUInt32(p1 - glyfPtr, p2, 0, 1);
+    } else {
+        PutUInt16((p1 - glyfPtr) >> 1, p2, 0, 1);
+    }
+
+    glyf->rawdata = glyfPtr;
+
+    loca = TrueTypeTableNew_loca(); assert(loca != 0);
+    ((tdata_loca *) loca->data)->ptr = locaPtr;
+    ((tdata_loca *) loca->data)->nbytes = locaLen;
+
+    AddTable(tt, loca);
+    
+    head = FindTable(tt, T_head);
+    PutInt16(xMin, head->data, 36, 1);
+    PutInt16(yMin, head->data, 38, 1);
+    PutInt16(xMax, head->data, 40, 1);
+    PutInt16(yMax, head->data, 42, 1);
+    PutInt16(indexToLocFormat, head->data,  50, 1);
+
+    maxp = FindTable(tt, T_maxp);
+
+    PutUInt16(nGlyphs, maxp->data, 4, 1);
+    PutUInt16(maxPoints, maxp->data, 6, 1);
+    PutUInt16(maxContours, maxp->data, 8, 1);
+    PutUInt16(maxCompositePoints, maxp->data, 10, 1);
+    PutUInt16(maxCompositeContours, maxp->data, 12, 1);
+
+#if 0
+    /* XXX do not overwrite the existing data. Fix: re-calculate these numbers here */
+    PutUInt16(2, maxp->data, 14, 1);                        /* maxZones is always 2       */
+    PutUInt16(0, maxp->data, 16, 1);                        /* maxTwilightPoints          */
+    PutUInt16(0, maxp->data, 18, 1);                        /* maxStorage                 */
+    PutUInt16(0, maxp->data, 20, 1);                        /* maxFunctionDefs            */
+    PutUint16(0, maxp->data, 22, 1);                        /* maxInstructionDefs         */
+    PutUint16(0, maxp->data, 24, 1);                        /* maxStackElements           */
+    PutUint16(0, maxp->data, 26, 1);                        /* maxSizeOfInstructions      */
+    PutUint16(0, maxp->data, 28, 1);                        /* maxComponentElements       */
+    PutUint16(0, maxp->data, 30, 1);                        /* maxComponentDepth          */
+#endif
+
+    /*
+     * Generate an htmx table and update hhea table
+     */
+    hhea = FindTable(tt, T_hhea); assert(hhea != 0);
+    hheaPtr = (guint8 *) hhea->data;
+    if (nGlyphs > 2) {
+        for (i = nGlyphs - 1; i > 0; i--) {
+            if (met[i].adv != met[i-1].adv) break;
+        }
+        nlsb = nGlyphs - 1 - i;
+    }
+    hmtxSize = (nGlyphs - nlsb) * 4 + nlsb * 2;
+    hmtxPtr = ttmalloc(hmtxSize);
+    p1 = hmtxPtr;
+
+    for (i = 0; i < nGlyphs; i++) {
+        if (i < nGlyphs - nlsb) {
+            PutUInt16(met[i].adv, p1, 0, 1);
+            PutUInt16(met[i].sb, p1, 2, 1);
+            p1 += 4;
+        } else {
+            PutUInt16(met[i].sb, p1, 0, 1);
+            p1 += 2;
+        }
+    }
+
+    AddTable(tt, TrueTypeTableNew(T_hmtx, hmtxSize, hmtxPtr));
+    PutUInt16(nGlyphs - nlsb, hheaPtr, 34, 1);
+    free(hmtxPtr);
+    free(met);
+}
+
+#ifdef TEST_TTCR
+int main(void)
+{
+    TrueTypeCreator *ttcr;
+    guint8 *t1, *t2, *t3, *t4, *t5, *t6, *t7;
+
+    TrueTypeCreatorNewEmpty(mkTag('t','r','u','e'), &ttcr);
+
+    t1 = malloc(1000); memset(t1, 'a', 1000);
+    t2 = malloc(2000); memset(t2, 'b', 2000);
+    t3 = malloc(3000); memset(t3, 'c', 3000);
+    t4 = malloc(4000); memset(t4, 'd', 4000);
+    t5 = malloc(5000); memset(t5, 'e', 5000);
+    t6 = malloc(6000); memset(t6, 'f', 6000);
+    t7 = malloc(7000); memset(t7, 'g', 7000);
+
+    AddTable(ttcr, TrueTypeTableNew(0x6D617870, 1000, t1));
+    AddTable(ttcr, TrueTypeTableNew(0x4F532F32, 2000, t2));
+    AddTable(ttcr, TrueTypeTableNew(0x636D6170, 3000, t3));
+    AddTable(ttcr, TrueTypeTableNew(0x6C6F6361, 4000, t4));
+    AddTable(ttcr, TrueTypeTableNew(0x68686561, 5000, t5));
+    AddTable(ttcr, TrueTypeTableNew(0x676C7966, 6000, t6));
+    AddTable(ttcr, TrueTypeTableNew(0x6B65726E, 7000, t7));
+
+    free(t1);
+    free(t2);
+    free(t3);
+    free(t4);
+    free(t5);
+    free(t6);
+    free(t7);
+
+
+    StreamToFile(ttcr, "ttcrout.ttf");
+
+    TrueTypeCreatorDispose(ttcr);
+    return 0;
+}
+#endif
diff --git a/src/ttsubset/ttcr.h b/src/ttsubset/ttcr.h
new file mode 100644 (file)
index 0000000..4e7a80d
--- /dev/null
@@ -0,0 +1,287 @@
+/*
+ * Copyright © 2002, 2003 Sun Microsystems, Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ *
+ * 3. Neither the name of Sun Microsystems, Inc. nor the names of 
+ *    contributors may be used to endorse or promote products derived from
+ *    this software without specific prior written permission.
+ *
+ * This software is provided "AS IS," without a warranty of any kind.
+ *
+ * ALL EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES,
+ * INCLUDING ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A
+ * PARTICULAR PURPOSE OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED.
+ * SUN AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES OR
+ * LIABILITIES SUFFERED BY LICENSEE AS A RESULT OF OR RELATING TO USE,
+ * MODIFICATION OR DISTRIBUTION OF THE SOFTWARE OR ITS DERIVATIVES.
+ * IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST REVENUE,
+ * PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL,
+ * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE
+ * THEORY OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE
+ * SOFTWARE, EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
+ *
+ */
+
+/* $Id$ */
+/* @(#)ttcr.h 1.6 03/01/08 SMI */
+
+/*
+ * @file ttcr.h
+ * @brief TrueType font creator
+ * @author Alexander Gelfenbain <adg@sun.com>
+ * @version 1.1
+ *
+ */
+
+
+#ifndef __TTCR_H
+#define __TTCR_H
+
+#include "sft.h"
+#include "list.h"
+
+#ifdef __cplusplus
+extern "C"{
+#endif
+
+typedef struct _TrueTypeCreator TrueTypeCreator;
+
+/* TrueType data types */
+typedef struct {
+    guint16 aw;
+    gint16  lsb;
+} longHorMetrics;
+
+/* A generic base class for all TrueType tables */
+typedef struct {
+    guint32 tag;                         /* table tag                                                */
+    guint8   *rawdata;                    /* raw data allocated by GetRawData_*()                     */
+    void   *data;                       /* table specific data                                      */
+} TrueTypeTable;
+
+/** Error codes for most functions */
+enum TTCRErrCodes {                     
+    TTCR_OK = 0,                        /**< no error                                               */
+    TTCR_ZEROGLYPHS = 1,                /**< At least one glyph should be defined                   */
+    TTCR_UNKNOWN = 2,                   /**< Unknown TrueType table                                 */
+    TTCR_GLYPHSEQ = 3,                  /**< Glyph IDs are not sequential in the glyf table         */
+    TTCR_NONAMES = 4,                   /**< 'name' table does not contain any names                */
+    TTCR_NAMETOOLONG = 5,               /**< 'name' table is too long (string data > 64K)           */
+    TTCR_POSTFORMAT = 6                 /**< unsupported format of a 'post' table                   */
+};
+
+/* ============================================================================
+ *
+ * TrueTypeCreator methods
+ *
+ * ============================================================================ */
+
+/*
+ * TrueTypeCreator constructor.
+ * Allocates all internal structures.
+ */
+void TrueTypeCreatorNewEmpty(guint32 tag, TrueTypeCreator **_this);
+
+/*
+ * TrueTypeCreator destructor. It calls destructors for all TrueTypeTables added to it.
+ */
+void TrueTypeCreatorDispose(TrueTypeCreator *_this);
+
+/*
+ * Adds a TrueType table to the TrueType creator.
+ * SF_TABLEFORMAT value.
+ * @return value of SFErrCodes type
+ */
+int AddTable(TrueTypeCreator *_this, TrueTypeTable *table);
+
+/*
+ * Removes a TrueType table from the TrueType creator if it is stored there.
+ * It also calls a TrueTypeTable destructor.
+ * Note: all generic tables (with tag 0) will be removed if this function is
+ * called with the second argument of 0.
+ * @return value of SFErrCodes type
+ */
+void RemoveTable(TrueTypeCreator *_this, guint32 tag);
+
+
+
+/*
+ * Writes a TrueType font generated by the TrueTypeCreator to a segment of
+ * memory that this method allocates. When it is not needed anymore the caller
+ * is supposed to call free() on it.
+ * @return value of SFErrCodes type
+ */
+int StreamToMemory(TrueTypeCreator *_this, guint8 **ptr, guint32 *length);
+
+/*
+ * Writes a TrueType font  generated by the TrueTypeCreator to a file
+ * @return value of SFErrCodes type
+ */
+int StreamToFile(TrueTypeCreator *_this, const char* fname);
+
+
+/* ============================================================================
+ *
+ * TrueTypeTable methods
+ *
+ * ============================================================================ */
+
+/*
+ * Destructor for the TrueTypeTable object.
+ */
+void TrueTypeTableDispose(TrueTypeTable *);
+
+/*
+ * This function converts the data of a TrueType table to a raw array of bytes.
+ * It may allocates the memory for it and returns the size of the raw data in bytes.
+ * If memory is allocated it does not need to be freed by the caller of this function,
+ * since the pointer to it is stored in the TrueTypeTable and it is freed by the destructor
+ * @return TTCRErrCode
+ *
+ */
+
+int GetRawData(TrueTypeTable *, guint8 **ptr, guint32 *len, guint32 *tag);
+
+/*
+ *
+ * Creates a new raw TrueType table. The difference between this constructor and
+ * TrueTypeTableNew_tag constructors is that the latter create structured tables
+ * while this constructor just copies memory pointed to by ptr to its buffer
+ * and stores its length. This constructor is suitable for data that is not
+ * supposed to be processed in any way, just written to the resulting TTF file.
+ */
+TrueTypeTable *TrueTypeTableNew(guint32 tag,
+                                guint32 nbytes,
+                                guint8 *ptr);
+
+/*
+ * Creates a new 'head' table for a TrueType font.
+ * Allocates memory for it. Since a lot of values in the 'head' table depend on the
+ * rest of the tables in the TrueType font this table should be the last one added
+ * to the font.
+ */
+TrueTypeTable *TrueTypeTableNew_head(guint32 fontRevision,
+                                     guint16 flags,
+                                     guint16 unitsPerEm,
+                                     guint8  *created,
+                                     guint16 macStyle,
+                                     guint16 lowestRecPPEM,
+                                     gint16  fontDirectionHint);
+
+/*
+ * Creates a new 'hhea' table for a TrueType font.
+ * Allocates memory for it and stores it in the hhea pointer.
+ */
+TrueTypeTable *TrueTypeTableNew_hhea(gint16  ascender,
+                                     gint16  descender,
+                                     gint16  linegap,
+                                     gint16  caretSlopeRise,
+                                     gint16  caretSlopeRun);
+
+/*
+ * Creates a new empty 'loca' table for a TrueType font.
+ *
+ * INTERNAL: gets called only from ProcessTables();
+ */
+TrueTypeTable *TrueTypeTableNew_loca(void);
+
+/*
+ * Creates a new 'maxp' table based on an existing maxp table.
+ * If maxp is 0, a new empty maxp table is created
+ * size specifies the size of existing maxp table for
+ * error-checking purposes
+ */
+TrueTypeTable *TrueTypeTableNew_maxp(guint8 *maxp, int size);
+
+/*
+ * Creates a new empty 'glyf' table.
+ */
+TrueTypeTable *TrueTypeTableNew_glyf(void);
+
+/*
+ * Creates a new empty 'cmap' table.
+ */
+TrueTypeTable *TrueTypeTableNew_cmap(void);
+
+/*
+ * Creates a new 'name' table. If n != 0 the table gets populated by
+ * the Name Records stored in the nr array. This function allocates
+ * memory for its own copy of NameRecords, so nr array has to
+ * be explicitly deallocated when it is not needed.
+ */
+TrueTypeTable *TrueTypeTableNew_name(int n, NameRecord *nr);
+
+/*
+ * Creates a new 'post' table of one of the supported formats
+ */
+TrueTypeTable *TrueTypeTableNew_post(guint32 format,
+                                     guint32 italicAngle,
+                                     gint16 underlinePosition,
+                                     gint16 underlineThickness,
+                                     guint32 isFixedPitch);
+
+
+/*------------------------------------------------------------------------------
+ *
+ *  Table manipulation functions
+ *
+ *------------------------------------------------------------------------------*/
+
+/*
+ * Add a character/glyph pair to a cmap table
+ */
+void cmapAdd(TrueTypeTable *, guint32 id, guint32 c, guint32 g);
+
+/*
+ * Add a glyph to a glyf table.
+ *
+ * @return glyphID of the glyph in the new font
+ *
+ * NOTE: This function does not duplicate GlyphData, so memory will be
+ * deallocated in the table destructor
+ */
+guint32 glyfAdd(TrueTypeTable *, GlyphData *glyphdata, TrueTypeFont *fnt);
+
+/*
+ * Query the number of glyphs currently stored in the 'glyf' table
+ *
+ */
+guint32 glyfCount(const TrueTypeTable *);
+
+/*
+ * Add a Name Record to a name table.
+ * NOTE: This function duplicates NameRecord, so the argument
+ * has to be deallocated by the caller (unlike glyfAdd)
+ */
+void nameAdd(TrueTypeTable *, NameRecord *nr);
+
+                   
+
+/*
+ * Private Data Types
+ */
+
+struct _TrueTypeCreator {
+    guint32 tag;                         /**< TrueType file tag */
+    list   tables;                      /**< List of table tags and pointers */
+};
+
+
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __TTCR_H */