diff --git a/.gitattributes b/.gitattributes
index 896d40bc14af5150b691428167c7ca7d364768ba..3533aa9bb156cd58a9ebafc8417d76f009527a68 100644
--- a/.gitattributes
+++ b/.gitattributes
@@ -310,6 +310,10 @@ testfont binary
 /src/modules/_Image_JPEG/acconfig.h foreign_ident
 /src/modules/_Image_JPEG/configure.in foreign_ident
 /src/modules/_Image_JPEG/image_jpeg.c foreign_ident
+/src/modules/_Image_TTF/Makefile.in foreign_ident
+/src/modules/_Image_TTF/acconfig.h foreign_ident
+/src/modules/_Image_TTF/configure.in foreign_ident
+/src/modules/_Image_TTF/image_ttf.c foreign_ident
 /src/modules/_Image_XFace/Makefile.in foreign_ident
 /src/modules/_Image_XFace/acconfig.h foreign_ident
 /src/modules/_Image_XFace/configure.in foreign_ident
diff --git a/src/modules/_Image_TTF/Makefile.in b/src/modules/_Image_TTF/Makefile.in
new file mode 100644
index 0000000000000000000000000000000000000000..b1926e677ad2f7587538bc4c3578d7d1aa799cd9
--- /dev/null
+++ b/src/modules/_Image_TTF/Makefile.in
@@ -0,0 +1,8 @@
+# $Id: Makefile.in,v 1.1 1998/10/31 19:07:53 mirar Exp $
+SRCDIR=@srcdir@
+VPATH=@srcdir@:@srcdir@/../..:../..
+OBJS=image_ttf.o
+MODULE_LDFLAGS=@LDFLAGS@ @LIBS@
+
+@dynamic_module_makefile@
+@dependencies@
diff --git a/src/modules/_Image_TTF/acconfig.h b/src/modules/_Image_TTF/acconfig.h
new file mode 100644
index 0000000000000000000000000000000000000000..9b92f40ae993d7213ef930ecc469bf6cab8974dc
--- /dev/null
+++ b/src/modules/_Image_TTF/acconfig.h
@@ -0,0 +1,11 @@
+/*
+ * $Id: acconfig.h,v 1.1 1998/10/31 19:07:54 mirar Exp $
+ */
+
+@TOP@
+@BOTTOM@
+
+/* Define this if you have -ljpeg */
+#undef HAVE_LIBTTF
+
+
diff --git a/src/modules/_Image_TTF/configure.in b/src/modules/_Image_TTF/configure.in
new file mode 100644
index 0000000000000000000000000000000000000000..646e4223938645af06ac04a6774d44b4b05e53b3
--- /dev/null
+++ b/src/modules/_Image_TTF/configure.in
@@ -0,0 +1,23 @@
+#
+# $Id: configure.in,v 1.1 1998/10/31 19:07:54 mirar Exp $
+#
+AC_INIT(image_ttf.c)
+AC_CONFIG_HEADER(config.h)
+AC_ARG_WITH(jpeglib,     [  --with(out)-ttflib       Support TTF via ttf.lib (freetype) (Image.TTF)],[],[with_ttflib=yes])
+
+sinclude(../module_configure.in)
+
+if test x$with_ttflib = xyes ; then
+  AC_CHECK_HEADERS(freetype.h)
+  if test $ac_cv_header_freetype_h = yes ; then
+    AC_CHECK_LIB(ttf, Load_TrueType_Names, [
+      AC_DEFINE(HAVE_LIBTTF) 
+      LIBS="${LIBS-} -lttf"
+      ], 
+      AC_CHECK_LIB(ttf, Load_TrueType_Names))
+  fi
+fi
+
+AC_OUTPUT(Makefile,echo FOO >stamp-h )
+
+
diff --git a/src/modules/_Image_TTF/image_ttf.c b/src/modules/_Image_TTF/image_ttf.c
new file mode 100644
index 0000000000000000000000000000000000000000..7bd427cde83dc0c8a8da2b7f08ae96e42aa7b6d8
--- /dev/null
+++ b/src/modules/_Image_TTF/image_ttf.c
@@ -0,0 +1,1152 @@
+/*
+ * $Id: image_ttf.c,v 1.1 1998/10/31 19:07:55 mirar Exp $
+ */
+
+#include "config.h"
+
+
+#include "global.h"
+RCSID("$Id: image_ttf.c,v 1.1 1998/10/31 19:07:55 mirar Exp $");
+
+#ifdef HAVE_LIBTTF
+#include <freetype.h>
+#endif /* HAVE_LIBTTF */
+
+#include "pike_macros.h"
+#include "object.h"
+#include "constants.h"
+#include "interpret.h"
+#include "svalue.h"
+#include "threads.h"
+#include "array.h"
+#include "mapping.h"
+#include "error.h"
+#include "stralloc.h"
+#include "builtin_functions.h"
+
+#ifdef HAVE_LIBTTF
+#include "../Image/image.h"
+
+#ifdef DYNAMIC_MODULE
+static struct program *image_program=NULL;
+#else
+extern struct program *image_program; 
+/* Image module is probably linked static too. */
+#endif
+
+static TT_Engine engine;
+
+#endif /* HAVE_LIBTTF */
+
+static struct pike_string *param_baseline;
+static struct pike_string *param_quality;
+
+#ifdef HAVE_LIBTTF
+
+/*
+**! module Image
+**! submodule TTF
+**!
+**!	This module adds TTF (Truetype font) capability   
+**!	to the Image module.    
+**!
+**! note
+**!	This module needs the <tt>libttf</tt> "Freetype" library
+**!
+*/
+
+void my_tt_error(char *where,char *extra,int err)
+{
+   char *errc="Unknown";
+
+   if (err==TT_Err_Invalid_Face_Handle)
+      errc="TT_Err_Invalid_Face_Handle";
+   else if (err==TT_Err_Invalid_Instance_Handle)
+      errc="TT_Err_Invalid_Instance_Handle";
+   else if (err==TT_Err_Invalid_Glyph_Handle)
+      errc="TT_Err_Invalid_Glyph_Handle";
+   else if (err==TT_Err_Invalid_CharMap_Handle)
+      errc="TT_Err_Invalid_CharMap_Handle";
+   else if (err==TT_Err_Invalid_Result_Address)
+      errc="TT_Err_Invalid_Result_Address";
+   else if (err==TT_Err_Invalid_Glyph_Index)
+      errc="TT_Err_Invalid_Glyph_Index";
+   else if (err==TT_Err_Invalid_Argument)
+      errc="TT_Err_Invalid_Argument";
+   else if (err==TT_Err_Could_Not_Open_File)
+      errc="TT_Err_Could_Not_Open_File";
+   else if (err==TT_Err_File_Is_Not_Collection)
+      errc="TT_Err_File_Is_Not_Collection";
+   else if (err==TT_Err_Table_Missing)
+      errc="TT_Err_Table_Missing";
+   else if (err==TT_Err_Invalid_Horiz_Metrics)
+      errc="TT_Err_Invalid_Horiz_Metrics";
+   else if (err==TT_Err_Invalid_CharMap_Format)
+      errc="TT_Err_Invalid_CharMap_Format";
+   else if (err==TT_Err_Invalid_PPem)
+      errc="TT_Err_Invalid_PPem";
+   else if (err==TT_Err_Invalid_File_Format)
+      errc="TT_Err_Invalid_File_Format";
+   else if (err==TT_Err_Invalid_Engine)
+      errc="TT_Err_Invalid_Engine";
+   else if (err==TT_Err_Too_Many_Extensions)
+      errc="TT_Err_Too_Many_Extensions";
+   else if (err==TT_Err_Extensions_Unsupported)
+      errc="TT_Err_Extensions_Unsupported";
+   else if (err==TT_Err_Invalid_Extension_Id)
+      errc="TT_Err_Invalid_Extension_Id";
+   else if (err==TT_Err_Max_Profile_Missing)
+      errc="TT_Err_Max_Profile_Missing";
+   else if (err==TT_Err_Header_Table_Missing)
+      errc="TT_Err_Header_Table_Missing";
+   else if (err==TT_Err_Horiz_Header_Missing)
+      errc="TT_Err_Horiz_Header_Missing";
+   else if (err==TT_Err_Locations_Missing)
+      errc="TT_Err_Locations_Missing";
+   else if (err==TT_Err_Name_Table_Missing)
+      errc="TT_Err_Name_Table_Missing";
+   else if (err==TT_Err_CMap_Table_Missing)
+      errc="TT_Err_CMap_Table_Missing";
+   else if (err==TT_Err_Hmtx_Table_Missing)
+      errc="TT_Err_Hmtx_Table_Missing";
+   else if (err==TT_Err_OS2_Table_Missing)
+      errc="TT_Err_OS2_Table_Missing";
+   else if (err==TT_Err_Post_Table_Missing)
+      errc="TT_Err_Post_Table_Missing";
+   else if (err==TT_Err_Out_Of_Memory)
+      errc="TT_Err_Out_Of_Memory";
+   else if (err==TT_Err_Invalid_File_Offset)
+      errc="TT_Err_Invalid_File_Offset";
+   else if (err==TT_Err_Invalid_File_Read)
+      errc="TT_Err_Invalid_File_Read";
+   else if (err==TT_Err_Invalid_Frame_Access)
+      errc="TT_Err_Invalid_Frame_Access";
+   else if (err==TT_Err_Too_Many_Points)
+      errc="TT_Err_Too_Many_Points";
+   else if (err==TT_Err_Too_Many_Contours)
+      errc="TT_Err_Too_Many_Contours";
+   else if (err==TT_Err_Invalid_Composite)
+      errc="TT_Err_Invalid_Composite";
+   else if (err==TT_Err_Too_Many_Ins)
+      errc="TT_Err_Too_Many_Ins";
+   else if (err==TT_Err_Invalid_Opcode)
+      errc="TT_Err_Invalid_Opcode";
+   else if (err==TT_Err_Too_Few_Arguments)
+      errc="TT_Err_Too_Few_Arguments";
+   else if (err==TT_Err_Stack_Overflow)
+      errc="TT_Err_Stack_Overflow";
+   else if (err==TT_Err_Code_Overflow)
+      errc="TT_Err_Code_Overflow";
+   else if (err==TT_Err_Bad_Argument)
+      errc="TT_Err_Bad_Argument";
+   else if (err==TT_Err_Divide_By_Zero)
+      errc="TT_Err_Divide_By_Zero";
+   else if (err==TT_Err_Storage_Overflow)
+      errc="TT_Err_Storage_Overflow";
+   else if (err==TT_Err_Cvt_Overflow)
+      errc="TT_Err_Cvt_Overflow";
+   else if (err==TT_Err_Invalid_Reference)
+      errc="TT_Err_Invalid_Reference";
+   else if (err==TT_Err_Invalid_Distance)
+      errc="TT_Err_Invalid_Distance";
+   else if (err==TT_Err_Interpolate_Twilight)
+      errc="TT_Err_Interpolate_Twilight";
+   else if (err==TT_Err_Debug_OpCode)
+      errc="TT_Err_Debug_OpCode";
+   else if (err==TT_Err_ENDF_In_Exec_Stream)
+      errc="TT_Err_ENDF_In_Exec_Stream";
+   else if (err==TT_Err_Out_Of_CodeRanges)
+      errc="TT_Err_Out_Of_CodeRanges";
+   else if (err==TT_Err_Nested_DEFS)
+      errc="TT_Err_Nested_DEFS";
+   else if (err==TT_Err_Invalid_CodeRange)
+      errc="TT_Err_Invalid_CodeRange";
+   else if (err==TT_Err_Invalid_Displacement)
+      errc="TT_Err_Invalid_Displacement";
+   else if (err==TT_Err_Nested_Frame_Access)
+      errc="TT_Err_Nested_Frame_Access";
+   else if (err==TT_Err_Invalid_Cache_List)
+      errc="TT_Err_Invalid_Cache_List";
+   else if (err==TT_Err_Could_Not_Find_Context)
+      errc="TT_Err_Could_Not_Find_Context";
+   else if (err==TT_Err_Unlisted_Object)
+      errc="TT_Err_Unlisted_Object";
+   else if (err==TT_Err_Raster_Pool_Overflow)
+      errc="TT_Err_Raster_Pool_Overflow";
+   else if (err==TT_Err_Raster_Negative_Height)
+      errc="TT_Err_Raster_Negative_Height";
+   else if (err==TT_Err_Raster_Invalid_Value)
+      errc="TT_Err_Raster_Invalid_Value";
+   else if (err==TT_Err_Raster_Not_Initialized)
+      errc="TT_Err_Raster_Not_Initialized";
+   error("%s: %sFreeType error 0x%03x (%s)\n",
+	 where,extra,err,errc);
+}
+
+/*
+**! method object `()(string filename)
+**! method object `()(string filename,mapping options)
+**!
+**!	Makes a new TTF Face object.
+**!
+**! returns 0 if failed.
+**!
+**! arg string filename
+**!	The filename of the <tt>TTF</tt> font or the <tt>TTC</tt>
+**!	font collection.
+**!
+**! arg mapping options
+**!	<pre>
+**!	advanced options:
+**!	    "face":int
+**!             If opening a font collection,  '<tt>.TTC</tt>',
+**!		this is used to get other fonts than the first.
+**!	</pre>
+**!
+*/
+
+struct image_ttf_face_struct
+{
+   TT_Face face;
+};
+
+struct image_ttf_faceinstance_struct
+{
+   TT_Instance instance;
+   struct object *faceobj;
+   int load_flags;
+   int baseline,height;  /* "baseline" position */
+   int trans;            /* translation of position (ypos) in 26.6*/
+};
+
+static struct program *image_ttf_face_program=NULL;
+static struct program *image_ttf_faceinstance_program=NULL;
+
+static void image_ttf_make(INT32 args)
+{
+   int col=0;
+   struct object *o;
+   TT_Error res;
+   TT_Face face;
+
+   if (sp[-args].type!=T_STRING)
+      error("Image.TTF(): illegal argument 1\n");
+
+   res=TT_Open_Collection(engine, sp[-args].u.string->str, col, &face);
+   if (res) my_tt_error("Image.TTF()","",res);
+
+   pop_n_elems(args);
+
+   o=clone_object(image_ttf_face_program,0);
+   ((struct image_ttf_face_struct*)o->storage)->face=face;
+
+   push_object(o);
+}
+
+/*
+**! class Face
+**!	
+**!	This represents instances of TTF Faces.
+*/
+
+#define THISOBJ (fp->current_object)
+#define THISf ((struct image_ttf_face_struct*)get_storage(THISOBJ,image_ttf_face_program))
+#define THISi ((struct image_ttf_faceinstance_struct*)get_storage(THISOBJ,image_ttf_faceinstance_program))
+
+static void image_ttf_face_exit()
+{
+   TT_Close_Face(THISf->face);
+}
+
+static void image_ttf_faceinstance_exit()
+{
+   free_object(THISi->faceobj);
+   TT_Done_Instance(THISi->instance);
+}
+
+/*
+**! method mapping properties()
+**!     This gives back a structure of the face's properties.
+**!	Most of this stuff is information you can skip.
+**!
+**!	The most interesting item to look at may be
+**!	<tt>->num_Faces</tt>, which describes the number of faces
+**!	in a <tt>.TTC</tt> font collection.
+**! 
+**! returns a mapping of a lot of properties
+*/
+
+static void image_ttf_face_properties(INT32 args)
+{
+   int res;
+   TT_Face_Properties prop;
+
+   pop_n_elems(args);
+
+   res=TT_Get_Face_Properties(THISf->face,&prop);
+   if (res) my_tt_error("Image.TTF.Face->properties()","",res);
+   
+   push_text("num_Glyphs");   push_int(prop.num_Glyphs);
+   push_text("max_Points");   push_int(prop.max_Points);
+   push_text("max_Contours");   push_int(prop.max_Contours);
+   push_text("num_Faces");   push_int(prop.num_Faces);
+
+   push_text("header");
+   if (prop.header)
+   {
+      push_text("Table_Version"); push_int(prop.header->Table_Version);
+      push_text("Font_Revision"); push_int(prop.header->Font_Revision);
+      push_text("CheckSum_Adjust"); push_int(prop.header->CheckSum_Adjust);
+      push_text("Magic_Number"); push_int(prop.header->Magic_Number);
+      push_text("Flags"); push_int(prop.header->Flags);
+      push_text("Units_Per_EM"); push_int(prop.header->Units_Per_EM);
+      push_text("Created"); 
+      push_int(prop.header->Created[0]);
+      push_int(prop.header->Created[1]);
+      f_aggregate(2);
+      push_text("Modified"); 
+      push_int(prop.header->Modified[0]);
+      push_int(prop.header->Modified[1]);
+      f_aggregate(2);
+      push_text("xMin"); push_int(prop.header->xMin);
+      push_text("yMin"); push_int(prop.header->yMin);
+      push_text("xMax"); push_int(prop.header->xMax);
+      push_text("yMax"); push_int(prop.header->yMax);
+      push_text("Mac_Style"); push_int(prop.header->Mac_Style);
+      push_text("Lowest_Rec_PPEM"); push_int(prop.header->Lowest_Rec_PPEM);
+      push_text("Font_Direction"); push_int(prop.header->Font_Direction);
+      push_text("Index_To_Loc_Format"); 
+      push_int(prop.header->Index_To_Loc_Format);
+      push_text("Glyph_Data_Format"); 
+      push_int(prop.header->Glyph_Data_Format);
+      f_aggregate_mapping(17*2);
+   }
+   else push_int(0);
+
+   push_text("horizontal");
+   if (prop.horizontal)
+   {
+      push_text("Version"); push_int(prop.horizontal->Version);
+      push_text("Ascender"); push_int(prop.horizontal->Ascender);
+      push_text("Descender"); push_int(prop.horizontal->Descender);
+      push_text("Line_Gap"); push_int(prop.horizontal->Line_Gap);
+      push_text("advance_Width_Max"); push_int(prop.horizontal->advance_Width_Max);
+      push_text("min_Left_Side_Bearing"); push_int(prop.horizontal->min_Left_Side_Bearing);
+      push_text("min_Right_Side_Bearing"); push_int(prop.horizontal->min_Right_Side_Bearing);
+      push_text("xMax_Extent"); push_int(prop.horizontal->xMax_Extent);
+      push_text("caret_Slope_Rise"); push_int(prop.horizontal->caret_Slope_Rise);
+      push_text("caret_Slope_Run"); push_int(prop.horizontal->caret_Slope_Run);
+      push_text("Reserved"); 
+      push_int(prop.horizontal->Reserved[0]);
+      push_int(prop.horizontal->Reserved[1]);
+      push_int(prop.horizontal->Reserved[2]);
+      push_int(prop.horizontal->Reserved[3]);
+      push_int(prop.horizontal->Reserved[4]);
+      f_aggregate(5);
+      push_text("metric_Data_Format"); push_int(prop.horizontal->metric_Data_Format);
+      push_text("number_Of_HMetrics"); push_int(prop.horizontal->number_Of_HMetrics);
+      f_aggregate_mapping(13*2);
+   }
+   else push_int(0);
+   
+   push_text("os2");
+   if (prop.os2)
+   {
+      push_text("version"); push_int(prop.os2->version);
+      push_text("xAvgCharWidth"); push_int(prop.os2->xAvgCharWidth);
+      push_text("usWeightClass"); push_int(prop.os2->usWeightClass);
+      push_text("usWidthClass"); push_int(prop.os2->usWidthClass);
+      push_text("fsType"); push_int(prop.os2->fsType);
+      push_text("ySubscriptXSize"); push_int(prop.os2->ySubscriptXSize);
+      push_text("ySubscriptYSize"); push_int(prop.os2->ySubscriptYSize);
+      push_text("ySubscriptXOffset"); push_int(prop.os2->ySubscriptXOffset);
+      push_text("ySubscriptYOffset"); push_int(prop.os2->ySubscriptYOffset);
+      push_text("ySuperscriptXSize"); push_int(prop.os2->ySuperscriptXSize);
+      push_text("ySuperscriptYSize"); push_int(prop.os2->ySuperscriptYSize);
+      push_text("ySuperscriptXOffset"); push_int(prop.os2->ySuperscriptXOffset);
+      push_text("ySuperscriptYOffset"); push_int(prop.os2->ySuperscriptYOffset);
+      push_text("yStrikeoutSize"); push_int(prop.os2->yStrikeoutSize);
+      push_text("yStrikeoutPosition"); push_int(prop.os2->yStrikeoutPosition);
+      push_text("sFamilyClass"); push_int(prop.os2->sFamilyClass);
+
+      push_text("panose"); 
+      push_string(make_shared_binary_string(prop.os2->panose,10));
+
+      push_text("ulUnicodeRange1"); push_int(prop.os2->ulUnicodeRange1);
+      push_text("ulUnicodeRange2"); push_int(prop.os2->ulUnicodeRange2);
+      push_text("ulUnicodeRange3"); push_int(prop.os2->ulUnicodeRange3);
+      push_text("ulUnicodeRange4"); push_int(prop.os2->ulUnicodeRange4);
+
+      push_text("achVendID"); 
+      push_string(make_shared_binary_string(prop.os2->achVendID,4));
+
+      push_text("fsSelection"); push_int(prop.os2->fsSelection);
+      push_text("usFirstCharIndex"); push_int(prop.os2->usFirstCharIndex);
+      push_text("usLastCharIndex"); push_int(prop.os2->usLastCharIndex);
+      push_text("sTypoAscender"); push_int(prop.os2->sTypoAscender);
+      push_text("sTypoDescender"); push_int(prop.os2->sTypoDescender);
+      push_text("sTypoLineGap"); push_int(prop.os2->sTypoLineGap);
+      push_text("usWinAscent"); push_int(prop.os2->usWinAscent);
+      push_text("usWinDescent"); push_int(prop.os2->usWinDescent);
+      push_text("ulCodePageRange1"); push_int(prop.os2->ulCodePageRange1); 
+      push_text("ulCodePageRange2"); push_int(prop.os2->ulCodePageRange2); 
+
+      f_aggregate_mapping(32*2);
+   }
+   else push_int(0);
+
+   push_text("postscript");
+   if (prop.postscript)
+   {
+      push_text("FormatType"); push_int(prop.postscript->FormatType);
+      push_text("italicAngle"); push_int(prop.postscript->italicAngle);
+      push_text("underlinePosition"); push_int(prop.postscript->underlinePosition);
+      push_text("underlineThickness"); push_int(prop.postscript->underlineThickness);
+      push_text("isFixedPitch"); push_int(prop.postscript->isFixedPitch);
+      push_text("minMemType42"); push_int(prop.postscript->minMemType42);
+      push_text("maxMemType42"); push_int(prop.postscript->maxMemType42);
+      push_text("minMemType1"); push_int(prop.postscript->minMemType1);
+      push_text("maxMemType1"); push_int(prop.postscript->maxMemType1);
+      f_aggregate_mapping(9*2);
+   }
+   else push_int(0);
+   
+   push_text("hdmx");
+   if (prop.hdmx)
+   {
+      int i;
+
+      push_text("version"); push_int(prop.hdmx->version);
+      push_text("num_records"); push_int(prop.hdmx->num_records);
+      push_text("records");
+
+      for (i=0; i<prop.hdmx->num_records; i++)
+      {
+	 push_text("ppem"); push_int(prop.hdmx->records[i].ppem);
+	 push_text("max_width"); push_int(prop.hdmx->records[i].max_width);
+	 /*	 push_text("widths"); push_int(prop.hdmx->records[i].widths);*/
+	 f_aggregate_mapping(2*2);
+      }
+      f_aggregate(prop.hdmx->num_records);
+
+      f_aggregate_mapping(3*2);
+   }
+   else push_int(0);
+
+   f_aggregate_mapping(9*2);
+}
+
+/*
+**! method object flush()
+**!     This flushes all cached information.
+**!	Might be used to save memory - the face
+**!	information is read back from disk upon need.
+**! 
+**! returns the called object
+*/
+
+static void image_ttf_face_flush(INT32 args)
+{
+   int res;
+   pop_n_elems(args);
+
+   if ((res=TT_Flush_Face(THISf->face)))
+      my_tt_error("Image.TTF.Face->flush()","",res);
+
+   ref_push_object(THISOBJ);
+}
+
+/*
+**! method mapping names()
+**! method array(array) _names()
+**!     Gives back the names or the complete name-list
+**!	of this face. 
+**!
+**!     The result from <ref>names</ref>() is a mapping,
+**!	which has any or all of these indices:
+**!	<pre>
+**!	"copyright":   ("Copyright the Foo Corporation...bla bla")
+**!	"family":      ("My Little Font")
+**!	"style":       ("Bold")
+**!	"full":	       ("Foo: My Little Font: 1998")
+**!	"expose":      ("My Little Font Bold")
+**!	"version":     ("June 1, 1998; 1.00, ...")
+**!	"postscript":  ("MyLittleFont-Bold")
+**!	"trademark":   ("MyLittleFont is a registered...bla bla")
+**!	</pre>
+**!
+**!	This is extracted from the information from _names(),
+**!	and fit into pike-strings using unicode or iso-8859-1,
+**!	if possible.
+**!
+**!     The result from <ref>_names</ref>() is a 
+**!	matrix, on this form: <pre>
+**!	     ({ ({ int platform, encoding, language, id, string name }),
+**!	        ({ int platform, encoding, language, id, string name }),
+**!             ...
+**!	     })
+**!     </pre>
+**!
+**! returns the name as a mapping to string or the names as a matrix
+**!
+**! note:
+**!	To use the values from <ref>_names</ref>(),
+**!	check the TrueType standard documentation.
+*/
+
+static void image_ttf_face__names(INT32 args)
+{
+   int ns,i,res;
+   TT_Face face=THISf->face;
+   pop_n_elems(args);
+
+   if ((ns=TT_Get_Name_Count(face))==-1)
+      error("Image.TTF.Face->names(): Illegal face handler\n");
+   
+   for (i=0; i<ns; i++)
+   {
+      short platformID,encodingID,languageID,nameID;
+      int length;
+      char *stringPtr;
+      
+      if ((res=TT_Get_Name_ID(face,i,
+			      &platformID,&encodingID,&languageID,&nameID)))
+	 my_tt_error("Image.TTF.Face->names()","TT_Get_Name_ID: ",res);
+
+      push_int(platformID);
+      push_int(encodingID);
+      push_int(languageID);
+      push_int(nameID);
+      
+      if ((res=TT_Get_Name_String(face,i,&stringPtr,&length)))
+	 my_tt_error("Image.TTF.Face->names()","TT_Get_Name_String: ",res);
+
+      push_string(make_shared_binary_string(stringPtr,length));
+      
+      f_aggregate(5);
+   }
+   f_aggregate(ns);
+}
+
+static void image_ttf_face_names(INT32 args)
+{
+   int n,i;
+   int has[8]={0,0,0,0,0,0,0,0}; /* iso8859=20, unicode=30, any=1 */
+   char *hasname[8]={"copyright","family","style","full",
+		     "expose","version","postscript","trademark"};
+   struct array *a,*b;
+
+   image_ttf_face__names(args);
+
+   if (sp[-1].type!=T_ARRAY) 
+      error("Image.TTF.Face->names(): internal error, wierd _names()\n");
+   
+   a=sp[-1].u.array;
+
+   n=0;
+   for (i=0; i<a->size; i++)
+   {
+      int ihas=1;
+      int what;
+      b=a->item[i].u.array;
+
+      what=b->item[3].u.integer;
+      if (what>=8 || what<0) continue; /* wierd */
+      switch (b->item[0].u.integer*100+b->item[1].u.integer)
+      {
+	 case 301: /* M$:  unicode */
+	 case 300: /* M$:  unicode (?) */
+	    ihas=30;
+	    break;
+	 case 202: /* ISO: iso-8859-1 */
+	    ihas=20;
+	    break;
+      }
+      if (ihas<has[what]) continue; /* worse */
+
+      push_text(hasname[what]);
+      
+      if (ihas==30)
+      {
+	 /* replace this unicode magic when pike has WIDESTRINGS */
+	 
+	 struct pike_string *str =
+	    begin_shared_string(b->item[4].u.string->len/2);
+	 int z,y;
+	 for (z=1,y=0; z<b->item[4].u.string->len; z+=2,y++)
+	    str->str[y]=b->item[4].u.string->str[z];
+	 push_string(end_shared_string(str));
+      }
+      else
+	 push_svalue(b->item+4);
+      
+      n++;
+   }
+   f_aggregate_mapping(n*2);
+   stack_swap();
+   pop_stack();
+}
+
+/*
+**! method object `()()
+**!     This instantiates the face for normal usage -
+**!	to convert font data to images.
+**! 
+**! returns a <ref>FaceInstance</ref> object.
+*/
+
+static void image_ttf_face_make(INT32 args)
+{
+   pop_n_elems(args);
+
+   ref_push_object(THISOBJ);
+   push_object(clone_object(image_ttf_faceinstance_program,1));
+}
+
+/*
+**! class FaceInstance
+**!     This is the instance of a face, with geometrics,
+**!	encodings and stuff.
+**! 
+**! method create(object face)
+**!	creates a new Instance from a face.
+*/
+
+static void ttf_instance_setc(struct image_ttf_face_struct *face_s,
+			      struct image_ttf_faceinstance_struct *face_i,
+			      int towhat,
+			      char *where)
+{
+   TT_Face_Properties prop;
+   TT_Instance_Metrics metr;
+   int res;
+   int resol;
+
+   if ((res=TT_Get_Face_Properties(face_s->face,&prop)))
+      my_tt_error(where,"TT_Get_Face_Properties",res);
+
+   resol=50; /* should be 72, but glyphs fit using this value */
+      /*
+         (int)((72*(prop.horizontal->Ascender+
+		    prop.horizontal->Descender)/
+		(float)prop.horizontal->Ascender));
+*/
+
+   if ((res=TT_Set_Instance_Resolutions(face_i->instance,resol,resol)))
+      my_tt_error("Image.TTF.FaceInstance()",
+		  "TT_Set_Instance_Resolutions: ",res);
+
+   if ((res=TT_Set_Instance_CharSize(face_i->instance,towhat)))
+      my_tt_error(where,"TT_Set_Instance_CharSize: ",res);
+
+   if ((res=TT_Get_Instance_Metrics(face_i->instance,&metr)))
+      my_tt_error(where,"TT_Get_Instance_Metrics",res);
+
+#if 1
+   fprintf(stderr,
+	   "prop.horizontal->Ascender: %d\n",
+	   prop.horizontal->Ascender);
+   fprintf(stderr,
+	   "prop.horizontal->Descender: %d\n",
+	   prop.horizontal->Descender);
+   fprintf(stderr,
+	   "prop.horizontal->Line_Gap: %d\n",
+	   prop.horizontal->Line_Gap);
+   fprintf(stderr,
+	   "metrics.scale: %f,%f\n",
+	   metr.x_scale/65536.0,
+	   metr.y_scale/65536.0);
+   fprintf(stderr,
+	   "baseline guess: %f\n",
+	   ((float)metr.y_ppem*prop.horizontal->Ascender)/
+	   (prop.horizontal->Ascender-prop.horizontal->Descender));
+#endif
+   
+   face_i->baseline=
+      (int)(((float)(towhat/64.0)*prop.horizontal->Ascender)/
+	    (prop.horizontal->Ascender-prop.horizontal->Descender));
+   face_i->height=towhat/64;
+   face_i->trans=
+      (32+(int)(64*((towhat/64.0)*prop.horizontal->Ascender)/
+		(prop.horizontal->Ascender-prop.horizontal->Descender)))&~63;
+}
+
+static void image_ttf_faceinstance_create(INT32 args)
+{
+   struct image_ttf_face_struct *face_s;
+   struct image_ttf_faceinstance_struct *face_i=THISi;
+   int res;
+
+   if (!args)
+      error("Image.TTF.FaceInstance(): too few arguments\n");
+
+   if (sp[-args].type!=T_OBJECT ||
+       !(face_s=(struct image_ttf_face_struct*)
+	 get_storage(sp[-args].u.object,image_ttf_face_program)))
+      error("Image.TTF.FaceInstance(): illegal argument 1\n");
+
+   if ((res=TT_New_Instance(face_s->face,&(face_i->instance))))
+      my_tt_error("Image.TTF.FaceInstance()","TT_New_Instance: ",res);
+
+   face_i->load_flags = TTLOAD_SCALE_GLYPH|TTLOAD_HINT_GLYPH;
+   (face_i->faceobj=sp[-args].u.object)->refs++;
+
+   ttf_instance_setc(face_s,face_i,32*64,"Image.TTF.FaceInstance()");
+}
+
+static void image_ttf_faceinstance_set_height(INT32 args)
+{
+   struct image_ttf_face_struct *face_s;
+   struct image_ttf_faceinstance_struct *face_i=THISi;
+   int h=0;
+
+   if (!args)
+      error("Image.TTF.FaceInstance->set_height(): missing arguments\n");
+
+   if (sp[-args].type==T_INT) 
+      h=sp[-args].u.integer*64;
+   else if (sp[-args].type==T_FLOAT) 
+      h=(int)(sp[-args].u.float_number*64);
+   else
+      error("Image.TTF.FaceInstance->set_height(): illegal argument 1\n");
+   if (h<1) h=1;
+
+   if (!(face_s=(struct image_ttf_face_struct*)
+	 get_storage(THISi->faceobj,image_ttf_face_program)))
+      error("Image.TTF.FaceInstance->write(): lost Face\n");
+
+   ttf_instance_setc(face_s,face_i,h,"Image.TTF.FaceInstance->set_height()");
+
+   pop_n_elems(args);
+   ref_push_object(THISOBJ);
+}
+
+static void ttf_translate_8bit(TT_CharMap charMap,
+			       unsigned char *what,
+			       int **dest,
+			       int len,
+			       int base)
+{
+   int i;
+
+   dest[0]=(int*)xalloc(len*sizeof(int));
+
+   THREADS_ALLOW();
+   for (i=0; i<len; i++)
+      dest[0][i]=TT_Char_Index(charMap,what[i]+base);
+   THREADS_DISALLOW();
+}
+
+static void ttf_get_nice_charmap(TT_Face face,
+				 TT_CharMap *charMap,
+				 char *where)
+{
+   int n,i,res,got=-1,best=-1;
+   if (-1==(n=TT_Get_CharMap_Count(face))) 
+      error("%s: illegal face handle\n",where);
+
+   for (i=0; i<n; i++)
+   {
+      int ihas=0;
+      short platformID,encodingID;
+      
+      if ((res=TT_Get_CharMap_ID(face,i,&platformID,&encodingID)))
+	 my_tt_error(where,"TT_Get_CharMap_ID: ",res);
+
+      switch (platformID*100+encodingID)
+      {
+	 case 301: /* M$:  unicode */
+	 case 300: /* M$:  unicode (?) */
+	    ihas=30;
+	    break;
+	 case 202: /* ISO: iso-8859-1 */
+	    ihas=20;
+	    break;
+      }
+      
+      if (ihas>got) 
+      {
+	 best=i;
+	 got=ihas;
+      }
+   }
+   if (got==-1)
+      error("%s: no charmaps at all\n",where);
+
+   if ((res=TT_Get_CharMap(face,best,charMap)))
+      my_tt_error(where,"TT_Get_CharMap: ",res);
+}
+
+static void ttf_please_translate_8bit(TT_Face face,
+				      struct pike_string *s,
+				      int **dest,int *dlen,
+				      int base,
+				      char *where)
+{
+   TT_CharMap charMap;
+   
+   ttf_get_nice_charmap(face,&charMap,where);
+   (*dlen)=s->len;
+   ttf_translate_8bit(charMap,(unsigned char*)(s->str),dest,*dlen,base);
+}
+				      
+	
+static void image_ttf_faceinstance_ponder(INT32 args)
+{
+   int *sstr;
+   int len,i,res,base=0;
+   struct image_ttf_face_struct *face_s;
+   struct image_ttf_faceinstance_struct *face_i=THISi;
+   
+   int xmin=1000,xmax=-1000,pos=0;
+
+   if (!(face_s=(struct image_ttf_face_struct*)
+	 get_storage(THISi->faceobj,image_ttf_face_program)))
+      error("Image.TTF.FaceInstance->ponder(): lost Face\n");
+
+   if (args && sp[-1].type==T_INT)
+   {
+      base=sp[-1].u.integer;
+      args--;
+      pop_stack();
+   }
+
+   if (sp[-args].type!=T_STRING) 
+      error("Image.TTF.FaceInstance->ponder(): illegal argument 1\n");
+
+   ttf_please_translate_8bit(face_s->face,
+			     sp[-args].u.string,&sstr,&len,base,
+			     "Image.TTF.FaceInstance->ponder()");
+
+   pop_n_elems(args);
+
+   for (i=0; i<len; i++)
+   {
+      TT_Glyph glyph;
+      TT_Glyph_Metrics metrics;
+      int ind;
+      
+      ind=sstr[i];
+      fprintf(stderr,"glyph: %d\n",ind);
+      
+      if ((res=TT_New_Glyph(face_s->face,&glyph)))
+	 my_tt_error("Image.TTF.FaceInstance->ponder()","TT_New_Glyph: ",res);
+      
+      if ((res=TT_Load_Glyph(face_i->instance,glyph,ind,face_i->load_flags)))
+	 my_tt_error("Image.TTF.FaceInstance->ponder()","TT_Load_Glyph: ",res);
+
+      if ((res=TT_Get_Glyph_Metrics(glyph,&metrics)))
+	 my_tt_error("Image.TTF.FaceInstance->ponder()",
+		     "TT_Get_Glyph_Metrics: ",res);
+
+      if (pos+metrics.bbox.xMin<xmin) xmin=pos+metrics.bbox.xMin;
+      if (pos+metrics.bbox.xMax>xmax) xmax=pos+metrics.bbox.xMax;
+      pos+=metrics.advance; /* insert kerning stuff here */
+      
+      fprintf(stderr,"bbox: (%f,%f)-(%f,%f)\n",
+	      metrics.bbox.xMin/64.0,
+	      metrics.bbox.yMin/64.0,
+	      metrics.bbox.xMax/64.0,
+	      metrics.bbox.yMax/64.0);
+
+      fprintf(stderr,"BearingX: %f\n",metrics.bearingX/64.0);
+      fprintf(stderr,"BearingY: %f\n",metrics.bearingY/64.0);
+      fprintf(stderr,"advance: %f\n",metrics.advance/64.0);
+
+      fprintf(stderr,"\n");
+   }
+
+   free(sstr);
+
+   fprintf(stderr,"xmin: %f\n",xmin/64.0);
+   fprintf(stderr,"xmax: %f\n",xmax/64.0);
+
+   ref_push_object(THISOBJ);
+}
+
+static void image_ttf_faceinstance_write(INT32 args)
+{
+   int **sstr;
+   int *slen;
+   int i,res=0,base=0,a;
+   struct image_ttf_face_struct *face_s;
+   struct image_ttf_faceinstance_struct *face_i=THISi;
+   TT_CharMap charMap;
+   char *errs=NULL;
+   
+   int xmin=1000,xmax=-1000,pos=0,ypos;
+   int width,height,mod;
+   
+   unsigned char* pixmap;
+
+   if (!(face_s=(struct image_ttf_face_struct*)
+	 get_storage(THISi->faceobj,image_ttf_face_program)))
+      error("Image.TTF.FaceInstance->write(): lost Face\n");
+
+   if (args && sp[-1].type==T_INT)
+   {
+      base=sp[-1].u.integer;
+      args--;
+      pop_stack();
+   }
+
+   if (!args)
+   {
+      push_text("");
+      args=1;
+   }
+
+   ttf_get_nice_charmap(face_s->face,&charMap,
+			"Image.TTF.FaceInstance->write()");
+
+   sstr=alloca(args*sizeof(int*));
+   slen=alloca(args*sizeof(int));
+
+   /* first pass: figure out total bounding box */
+
+   for (a=0; a<args; a++)
+   {
+      char *errs=NULL;
+
+      if (sp[a-args].type!=T_STRING) 
+	 error("Image.TTF.FaceInstance->write(): illegal argument %d\n",a+1);
+
+      ttf_translate_8bit(charMap,(unsigned char*)sp[a-args].u.string->str,
+			 sstr+a,slen[a]=sp[a-args].u.string->len,base);
+
+      pos=0;
+      for (i=0; i<slen[a]; i++)
+      {
+	 TT_Glyph glyph;
+	 TT_Glyph_Metrics metrics;
+	 int ind;
+      
+	 ind=sstr[a][i];
+	 fprintf(stderr,"glyph: %d\n",ind);
+      
+	 if ((res=TT_New_Glyph(face_s->face,&glyph)))
+	    { errs="TT_New_Glyph: "; break; }
+      
+	 if ((res=TT_Load_Glyph(face_i->instance,glyph,
+				ind,face_i->load_flags)))
+	    { errs="TT_Load_Glyph: "; break; }
+
+	 if ((res=TT_Get_Glyph_Metrics(glyph,&metrics)))
+	    { errs="TT_Get_Glyph_Metrics: "; break; }
+
+	 if (pos+metrics.bbox.xMin<xmin) xmin=pos+metrics.bbox.xMin;
+	 if (pos+metrics.bbox.xMax>xmax) xmax=pos+metrics.bbox.xMax;
+	 pos+=metrics.advance; /* insert kerning stuff here */
+
+	 if ((res=TT_Done_Glyph(glyph)))
+	    { errs="TT_Done_Glyph: "; break; }
+      }
+      if (errs)
+      {
+	 for (i=0; i<a; i++) free(sstr[i]);
+	 my_tt_error("Image.TTF.FaceInstance->write()",errs,res);
+      }
+   }
+
+   pop_n_elems(args);
+
+   fprintf(stderr,"xmin=%f xmax=%f\n",xmin/64.0,xmax/64.0);
+
+   xmin&=~63;
+   width=(xmax-xmin+63)>>6;
+   height=face_i->height*args;
+   mod=(4-(width&3))&3;
+   if (width<1) width=1;
+
+   if ((pixmap=malloc((width+mod)*height)))
+   {
+      /* second pass: write the stuff */
+
+      TT_Raster_Map rastermap;
+      struct object *o;
+      struct image *img;
+      unsigned char *s;
+      rgb_group *d;
+      
+      MEMSET(pixmap,0,(width+mod)*height);
+
+      rastermap.rows=args*face_i->height;
+      rastermap.cols=rastermap.width=width+mod;
+      rastermap.flow=TT_Flow_Down;
+      rastermap.bitmap=pixmap;
+      rastermap.size=(width+mod)*height;
+
+      ypos=args*face_i->height*64-face_i->trans;
+
+      fprintf(stderr,"rastermap.rows=%d cols=%d width=%d\n",
+	      rastermap.rows,rastermap.cols,rastermap.width);
+
+      for (a=0; a<args; a++)
+      {
+	 pos=xmin;
+	 for (i=0; i<slen[a]; i++)
+	 {
+	    TT_Glyph glyph;
+	    TT_Glyph_Metrics metrics;
+	    int ind;
+      
+	    ind=sstr[a][i];
+	    fprintf(stderr,"glyph: %d\n",ind);
+      
+      
+	    if ((res=TT_New_Glyph(face_s->face,&glyph)))
+	       { errs="TT_New_Glyph: "; break; }
+      
+	    if ((res=TT_Load_Glyph(face_i->instance,glyph,
+				   ind,face_i->load_flags)))
+	       { errs="TT_Load_Glyph: "; break; }
+
+	    if ((res=TT_Get_Glyph_Metrics(glyph,&metrics)))
+	       { errs="TT_Get_Glyph_Metrics: "; break; }
+
+	    if ((res=TT_Get_Glyph_Pixmap(glyph,
+					 &rastermap,
+					 pos,ypos)))
+	       { errs="TT_Get_Glyph_Pixmap: "; break; }
+
+	    pos+=metrics.advance; /* insert kerning stuff here, too */
+
+
+	    if ((res=TT_Done_Glyph(glyph)))
+	       { errs="TT_Done_Glyph: "; break; }
+	 }
+	 if (errs)
+	 {
+	    for (a=0; a<args; a++) free(sstr[a]);
+	    free(pixmap);
+	    my_tt_error("Image.TTF.FaceInstance->write()",errs,res);
+	 }
+	 ypos-=face_i->height*64;
+      }
+
+      fprintf(stderr,"rastermap.rows=%d cols=%d width=%d\n",
+	      rastermap.rows,rastermap.cols,rastermap.width);
+      
+      push_int(width);
+      push_int(height);
+      o=clone_object(image_program,2);
+      img=(struct image*)get_storage(o,image_program);
+      d=img->img;
+      s=pixmap;
+
+      for (i=0; i<height; i++)
+      {
+#if 1 /* DEBUG */
+	 if (i==face_i->baseline)
+	    for (a=0; a<width; a++)
+	       d->r=255,d->g=0,d->b=*s,d++,s++;
+	 else
+#endif
+	    for (a=0; a<width; a++)
+	       d->r=d->g=d->b=*s,d++,s++;
+	 s+=mod;
+      }
+      
+      push_object(o);
+   }
+   else 
+   {
+      error("Image.TTF.FaceInstance->write(): out of memory\n");
+   }
+
+   for (a=0; a<args; a++)
+      free(sstr[a]);
+}
+
+static void image_ttf_faceinstance_face(INT32 args)
+{
+   pop_n_elems(args);
+   ref_push_object(THISi->faceobj);
+}
+
+#endif /* HAVE_LIBTTF */
+
+/*** module init & exit & stuff *****************************************/
+
+void f_index(INT32 args);
+
+void pike_module_exit(void)
+{
+   free_string(param_baseline);
+   free_string(param_quality);
+
+#ifdef HAVE_LIBTTF
+   TT_Done_FreeType(engine);
+#endif /* HAVE_LIBTTF */
+}
+
+void pike_module_init(void)
+{
+#ifdef HAVE_LIBTTF
+   unsigned char palette[5]={0,64,128,192,255};
+
+#ifdef DYNAMIC_MODULE
+   push_string(make_shared_string("Image"));
+   push_int(0);
+   SAFE_APPLY_MASTER("resolv",2);
+   if (sp[-1].type==T_OBJECT) 
+   {
+      push_string(make_shared_string("image"));
+      f_index(2);
+      image_program=program_from_svalue(sp-1);
+   }
+   pop_n_elems(1);
+#endif /* DYNAMIC_MODULE */
+
+   if (image_program)
+   {
+      add_function("`()",image_ttf_make,
+		   "function(string,void|mapping(string:int):object)",0);
+
+      /* make face program */
+
+      start_new_program();
+      add_storage(sizeof(struct image_ttf_face_struct));
+
+      add_function("properties",image_ttf_face_properties,
+		   "function(:mapping)",0);
+      add_function("flush",image_ttf_face_flush,
+		   "function(:object)",0);
+      add_function("names",image_ttf_face_names,
+		   "function(:mapping(string:string))",0);
+      add_function("_names",image_ttf_face__names,
+		   "function(:array(array))",0);
+
+      add_function("`()",image_ttf_face_make,
+		   "function(:object)",0);
+
+      set_exit_callback(image_ttf_face_exit);
+      image_ttf_face_program=end_program();
+
+      /* make face instance program */
+
+      start_new_program();
+      add_storage(sizeof(struct image_ttf_faceinstance_struct));
+      
+      add_function("create",image_ttf_faceinstance_create,
+		   "function(object:void)",0);
+      add_function("ponder",image_ttf_faceinstance_ponder,
+		   "function(string:object)",0);
+      add_function("write",image_ttf_faceinstance_write,
+		   "function(string...:object)",0);
+      add_function("face",image_ttf_faceinstance_face,
+		   "function(:object)",0);
+      add_function("set_height",image_ttf_faceinstance_set_height,
+		   "function(int:object)",0);
+      
+      set_exit_callback(image_ttf_faceinstance_exit);
+      image_ttf_faceinstance_program=end_program();
+   }
+
+   TT_Init_FreeType(&engine);
+   TT_Set_Raster_Gray_Palette(engine,(char*)palette);
+#endif /* HAVE_LIBTTF */
+
+   param_baseline=make_shared_string("baseline");
+   param_quality=make_shared_string("quality");
+}
diff --git a/src/modules/_Image_TTF/testsuite.in b/src/modules/_Image_TTF/testsuite.in
new file mode 100644
index 0000000000000000000000000000000000000000..235283e0f89ff182723343e17ea0bfc12e002240
--- /dev/null
+++ b/src/modules/_Image_TTF/testsuite.in
@@ -0,0 +1,3 @@
+cond([[ master()->resolv("_Image_JPEG")->encode ]],
+[[
+]])