| /* | 
 |  * FreeType font engine interface | 
 |  * | 
 |  * Copyright 2001 Huw D M Davies for CodeWeavers. | 
 |  * | 
 |  * This file contains the WineEng* functions. | 
 |  * | 
 |  * This library is free software; you can redistribute it and/or | 
 |  * modify it under the terms of the GNU Lesser General Public | 
 |  * License as published by the Free Software Foundation; either | 
 |  * version 2.1 of the License, or (at your option) any later version. | 
 |  * | 
 |  * This library is distributed in the hope that it will be useful, | 
 |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
 |  * Lesser General Public License for more details. | 
 |  * | 
 |  * You should have received a copy of the GNU Lesser General Public | 
 |  * License along with this library; if not, write to the Free Software | 
 |  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | 
 |  */ | 
 |  | 
 | #include "config.h" | 
 | #include "wine/port.h" | 
 |  | 
 | #include <stdarg.h> | 
 | #include <stdlib.h> | 
 | #include <sys/stat.h> | 
 | #include <string.h> | 
 | #include <dirent.h> | 
 | #include <stdio.h> | 
 | #include <assert.h> | 
 |  | 
 | #include "windef.h" | 
 | #include "winbase.h" | 
 | #include "winerror.h" | 
 | #include "winreg.h" | 
 | #include "wingdi.h" | 
 | #include "gdi.h" | 
 | #include "gdi_private.h" | 
 | #include "wine/unicode.h" | 
 | #include "wine/debug.h" | 
 | #include "wine/list.h" | 
 |  | 
 | WINE_DEFAULT_DEBUG_CHANNEL(font); | 
 |  | 
 | #ifdef HAVE_FREETYPE | 
 |  | 
 | #ifdef HAVE_FT2BUILD_H | 
 | #include <ft2build.h> | 
 | #endif | 
 | #ifdef HAVE_FREETYPE_FREETYPE_H | 
 | #include <freetype/freetype.h> | 
 | #endif | 
 | #ifdef HAVE_FREETYPE_FTGLYPH_H | 
 | #include <freetype/ftglyph.h> | 
 | #endif | 
 | #ifdef HAVE_FREETYPE_TTTABLES_H | 
 | #include <freetype/tttables.h> | 
 | #endif | 
 | #ifdef HAVE_FREETYPE_FTSNAMES_H | 
 | #include <freetype/ftsnames.h> | 
 | #else | 
 | # ifdef HAVE_FREETYPE_FTNAMES_H | 
 | # include <freetype/ftnames.h> | 
 | # endif | 
 | #endif | 
 | #ifdef HAVE_FREETYPE_TTNAMEID_H | 
 | #include <freetype/ttnameid.h> | 
 | #endif | 
 | #ifdef HAVE_FREETYPE_FTOUTLN_H | 
 | #include <freetype/ftoutln.h> | 
 | #endif | 
 | #ifdef HAVE_FREETYPE_INTERNAL_SFNT_H | 
 | #include <freetype/internal/sfnt.h> | 
 | #endif | 
 | #ifdef HAVE_FREETYPE_FTTRIGON_H | 
 | #include <freetype/fttrigon.h> | 
 | #endif | 
 | #ifdef HAVE_FREETYPE_FTWINFNT_H | 
 | #include <freetype/ftwinfnt.h> | 
 | #endif | 
 |  | 
 | #ifndef SONAME_LIBFREETYPE | 
 | #define SONAME_LIBFREETYPE "libfreetype.so" | 
 | #endif | 
 |  | 
 | static FT_Library library = 0; | 
 | typedef struct | 
 | { | 
 |     FT_Int major; | 
 |     FT_Int minor; | 
 |     FT_Int patch; | 
 | } FT_Version_t; | 
 | static FT_Version_t FT_Version; | 
 | static DWORD FT_SimpleVersion; | 
 |  | 
 | static void *ft_handle = NULL; | 
 |  | 
 | #define MAKE_FUNCPTR(f) static typeof(f) * p##f = NULL | 
 | MAKE_FUNCPTR(FT_Vector_Unit); | 
 | MAKE_FUNCPTR(FT_Done_Face); | 
 | MAKE_FUNCPTR(FT_Get_Char_Index); | 
 | MAKE_FUNCPTR(FT_Get_Sfnt_Table); | 
 | MAKE_FUNCPTR(FT_Init_FreeType); | 
 | MAKE_FUNCPTR(FT_Load_Glyph); | 
 | MAKE_FUNCPTR(FT_Matrix_Multiply); | 
 | MAKE_FUNCPTR(FT_MulFix); | 
 | MAKE_FUNCPTR(FT_New_Face); | 
 | MAKE_FUNCPTR(FT_Outline_Get_Bitmap); | 
 | MAKE_FUNCPTR(FT_Outline_Transform); | 
 | MAKE_FUNCPTR(FT_Outline_Translate); | 
 | MAKE_FUNCPTR(FT_Select_Charmap); | 
 | MAKE_FUNCPTR(FT_Set_Pixel_Sizes); | 
 | MAKE_FUNCPTR(FT_Vector_Transform); | 
 | static void (*pFT_Library_Version)(FT_Library,FT_Int*,FT_Int*,FT_Int*); | 
 | static FT_Error (*pFT_Load_Sfnt_Table)(FT_Face,FT_ULong,FT_Long,FT_Byte*,FT_ULong*); | 
 | static FT_ULong (*pFT_Get_First_Char)(FT_Face,FT_UInt*); | 
 | #ifdef HAVE_FREETYPE_FTWINFNT_H | 
 | MAKE_FUNCPTR(FT_Get_WinFNT_Header); | 
 | #endif | 
 |  | 
 | #ifdef HAVE_FONTCONFIG_FONTCONFIG_H | 
 | #include <fontconfig/fontconfig.h> | 
 | MAKE_FUNCPTR(FcConfigGetCurrent); | 
 | MAKE_FUNCPTR(FcFontList); | 
 | MAKE_FUNCPTR(FcFontSetDestroy); | 
 | MAKE_FUNCPTR(FcInit); | 
 | MAKE_FUNCPTR(FcObjectSetAdd); | 
 | MAKE_FUNCPTR(FcObjectSetCreate); | 
 | MAKE_FUNCPTR(FcObjectSetDestroy); | 
 | MAKE_FUNCPTR(FcPatternCreate); | 
 | MAKE_FUNCPTR(FcPatternDestroy); | 
 | MAKE_FUNCPTR(FcPatternGet); | 
 | #ifndef SONAME_LIBFONTCONFIG | 
 | #define SONAME_LIBFONTCONFIG "libfontconfig.so" | 
 | #endif | 
 | #endif | 
 |  | 
 | #undef MAKE_FUNCPTR | 
 |  | 
 | #ifndef ft_encoding_none | 
 | #define FT_ENCODING_NONE ft_encoding_none | 
 | #endif | 
 | #ifndef ft_encoding_ms_symbol | 
 | #define FT_ENCODING_MS_SYMBOL ft_encoding_symbol | 
 | #endif | 
 | #ifndef ft_encoding_unicode | 
 | #define FT_ENCODING_UNICODE ft_encoding_unicode | 
 | #endif | 
 | #ifndef ft_encoding_apple_roman | 
 | #define FT_ENCODING_APPLE_ROMAN ft_encoding_apple_roman | 
 | #endif | 
 |  | 
 | #define GET_BE_WORD(ptr) MAKEWORD( ((BYTE *)(ptr))[1], ((BYTE *)(ptr))[0] ) | 
 |  | 
 | /* This is bascially a copy of FT_Bitmap_Size with an extra element added */ | 
 | typedef struct { | 
 |     FT_Short height; | 
 |     FT_Short width; | 
 |     FT_Pos size; | 
 |     FT_Pos x_ppem; | 
 |     FT_Pos y_ppem; | 
 |     FT_Short internal_leading; | 
 | } Bitmap_Size; | 
 |  | 
 | /* FT_Bitmap_Size gained 3 new elements between FreeType 2.1.4 and 2.1.5 | 
 |    So to let this compile on older versions of FreeType we'll define the | 
 |    new structure here. */ | 
 | typedef struct { | 
 |     FT_Short height, width; | 
 |     FT_Pos size, x_ppem, y_ppem; | 
 | } My_FT_Bitmap_Size; | 
 |  | 
 | typedef struct tagFace { | 
 |     struct list entry; | 
 |     WCHAR *StyleName; | 
 |     char *file; | 
 |     FT_Long face_index; | 
 |     BOOL Italic; | 
 |     BOOL Bold; | 
 |     FONTSIGNATURE fs; | 
 |     FT_Fixed font_version; | 
 |     BOOL scalable; | 
 |     Bitmap_Size size;     /* set if face is a bitmap */ | 
 |     BOOL external; /* TRUE if we should manually add this font to the registry */ | 
 |     struct tagFamily *family; | 
 | } Face; | 
 |  | 
 | typedef struct tagFamily { | 
 |     struct list entry; | 
 |     WCHAR *FamilyName; | 
 |     struct list faces; | 
 | } Family; | 
 |  | 
 | typedef struct { | 
 |     GLYPHMETRICS gm; | 
 |     INT adv; /* These three hold to widths of the unrotated chars */ | 
 |     INT lsb; | 
 |     INT bbx; | 
 |     BOOL init; | 
 | } GM; | 
 |  | 
 | typedef struct { | 
 |     FLOAT eM11, eM12; | 
 |     FLOAT eM21, eM22; | 
 | } FMAT2; | 
 |  | 
 | typedef struct { | 
 |     DWORD hash; | 
 |     LOGFONTW lf; | 
 |     FMAT2 matrix; | 
 | } FONT_DESC; | 
 |  | 
 | typedef struct tagHFONTLIST { | 
 |     struct list entry; | 
 |     HFONT hfont; | 
 | } HFONTLIST; | 
 |  | 
 | struct tagGdiFont { | 
 |     struct list entry; | 
 |     FT_Face ft_face; | 
 |     LPWSTR name; | 
 |     int charset; | 
 |     int codepage; | 
 |     BOOL fake_italic; | 
 |     BOOL fake_bold; | 
 |     BYTE underline; | 
 |     BYTE strikeout; | 
 |     INT orientation; | 
 |     GM *gm; | 
 |     DWORD gmsize; | 
 |     struct list hfontlist; | 
 |     FONT_DESC font_desc; | 
 |     LONG aveWidth; | 
 |     SHORT yMax; | 
 |     SHORT yMin; | 
 |     OUTLINETEXTMETRICW *potm; | 
 |     FONTSIGNATURE fs; | 
 | }; | 
 |  | 
 | #define INIT_GM_SIZE 128 | 
 |  | 
 | static struct list gdi_font_list = LIST_INIT(gdi_font_list); | 
 | static struct list unused_gdi_font_list = LIST_INIT(unused_gdi_font_list); | 
 | #define UNUSED_CACHE_SIZE 10 | 
 |  | 
 | static struct list font_list = LIST_INIT(font_list); | 
 |  | 
 | static const WCHAR defSerif[] = {'T','i','m','e','s',' ','N','e','w',' ', | 
 | 			   'R','o','m','a','n','\0'}; | 
 | static const WCHAR defSans[] = {'A','r','i','a','l','\0'}; | 
 | static const WCHAR defFixed[] = {'C','o','u','r','i','e','r',' ','N','e','w','\0'}; | 
 |  | 
 | static const WCHAR defSystem[] = {'A','r','i','a','l','\0'}; | 
 | static const WCHAR SystemW[] = {'S','y','s','t','e','m','\0'}; | 
 | static const WCHAR MSSansSerifW[] = {'M','S',' ','S','a','n','s',' ', | 
 | 			       'S','e','r','i','f','\0'}; | 
 | static const WCHAR HelvW[] = {'H','e','l','v','\0'}; | 
 | static const WCHAR RegularW[] = {'R','e','g','u','l','a','r','\0'}; | 
 |  | 
 | static const WCHAR fontsW[] = {'\\','F','o','n','t','s','\0'}; | 
 | static const WCHAR win9x_font_reg_key[] = {'S','o','f','t','w','a','r','e','\\','M','i','c','r','o','s','o','f','t','\\', | 
 |                                            'W','i','n','d','o','w','s','\\', | 
 |                                            'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\', | 
 |                                            'F','o','n','t','s','\0'}; | 
 |  | 
 | static const WCHAR winnt_font_reg_key[] = {'S','o','f','t','w','a','r','e','\\','M','i','c','r','o','s','o','f','t','\\', | 
 |                                            'W','i','n','d','o','w','s',' ','N','T','\\', | 
 |                                            'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\', | 
 |                                            'F','o','n','t','s','\0'}; | 
 |  | 
 | static const WCHAR system_fonts_reg_key[] = {'S','o','f','t','w','a','r','e','\\','F','o','n','t','s','\0'}; | 
 | static const WCHAR FixedSys_Value[] = {'F','I','X','E','D','F','O','N','.','F','O','N','\0'}; | 
 | static const WCHAR System_Value[] = {'F','O','N','T','S','.','F','O','N','\0'}; | 
 | static const WCHAR OEMFont_Value[] = {'O','E','M','F','O','N','T','.','F','O','N','\0'}; | 
 |  | 
 | static const WCHAR *SystemFontValues[4] = { | 
 |     System_Value, | 
 |     OEMFont_Value, | 
 |     FixedSys_Value, | 
 |     NULL | 
 | }; | 
 |  | 
 | static const WCHAR external_fonts_reg_key[] = {'S','o','f','t','w','a','r','e','\\','W','i','n','e','\\','W','i','n','e','\\', | 
 |                                                'F','o','n','t','s','\\','E','x','t','e','r','n','a','l',' ','F','o','n','t','s','\0'}; | 
 |  | 
 | static const WCHAR ArabicW[] = {'A','r','a','b','i','c','\0'}; | 
 | static const WCHAR BalticW[] = {'B','a','l','t','i','c','\0'}; | 
 | static const WCHAR CHINESE_BIG5W[] = {'C','H','I','N','E','S','E','_','B','I','G','5','\0'}; | 
 | static const WCHAR CHINESE_GB2312W[] = {'C','H','I','N','E','S','E','_','G','B','2','3','1','2','\0'}; | 
 | static const WCHAR Central_EuropeanW[] = {'C','e','n','t','r','a','l',' ', | 
 | 				    'E','u','r','o','p','e','a','n','\0'}; | 
 | static const WCHAR CyrillicW[] = {'C','y','r','i','l','l','i','c','\0'}; | 
 | static const WCHAR GreekW[] = {'G','r','e','e','k','\0'}; | 
 | static const WCHAR HangulW[] = {'H','a','n','g','u','l','\0'}; | 
 | static const WCHAR Hangul_Johab_W[] = {'H','a','n','g','u','l','(','J','o','h','a','b',')','\0'}; | 
 | static const WCHAR HebrewW[] = {'H','e','b','r','e','w','\0'}; | 
 | static const WCHAR JapaneseW[] = {'J','a','p','a','n','e','s','e','\0'}; | 
 | static const WCHAR SymbolW[] = {'S','y','m','b','o','l','\0'}; | 
 | static const WCHAR ThaiW[] = {'T','h','a','i','\0'}; | 
 | static const WCHAR TurkishW[] = {'T','u','r','k','i','s','h','\0'}; | 
 | static const WCHAR VietnameseW[] = {'V','i','e','t','n','a','m','e','s','e','\0'}; | 
 | static const WCHAR WesternW[] = {'W','e','s','t','e','r','n','\0'}; | 
 | static const WCHAR OEM_DOSW[] = {'O','E','M','/','D','O','S','\0'}; | 
 |  | 
 | static const WCHAR *ElfScriptsW[32] = { /* these are in the order of the fsCsb[0] bits */ | 
 |     WesternW, /*00*/ | 
 |     Central_EuropeanW, | 
 |     CyrillicW, | 
 |     GreekW, | 
 |     TurkishW, | 
 |     HebrewW, | 
 |     ArabicW, | 
 |     BalticW, | 
 |     VietnameseW, /*08*/ | 
 |     NULL, NULL, NULL, NULL, NULL, NULL, NULL, /*15*/ | 
 |     ThaiW, | 
 |     JapaneseW, | 
 |     CHINESE_GB2312W, | 
 |     HangulW, | 
 |     CHINESE_BIG5W, | 
 |     Hangul_Johab_W, | 
 |     NULL, NULL, /*23*/ | 
 |     NULL, NULL, NULL, NULL, NULL, NULL, NULL, | 
 |     SymbolW /*31*/ | 
 | }; | 
 |  | 
 | typedef struct { | 
 |   WCHAR *name; | 
 |   INT charset; | 
 | } NameCs; | 
 |  | 
 | typedef struct tagFontSubst { | 
 |   NameCs from; | 
 |   NameCs to; | 
 |   struct tagFontSubst *next; | 
 | } FontSubst; | 
 |  | 
 | static FontSubst *substlist = NULL; | 
 | static BOOL have_installed_roman_font = FALSE; /* CreateFontInstance will fail if this is still FALSE */ | 
 |  | 
 | static const WCHAR font_mutex_nameW[] = {'_','_','W','I','N','E','_','F','O','N','T','_','M','U','T','E','X','_','_','\0'}; | 
 |  | 
 |  | 
 | /**************************************** | 
 |  *   Notes on .fon files | 
 |  * | 
 |  * The fonts System, FixedSys and Terminal are special.  There are typically multiple | 
 |  * versions installed for different resolutions and codepages.  Windows stores which one to use | 
 |  * in HKEY_CURRENT_CONFIG\\Software\\Fonts. | 
 |  *    Key            Meaning | 
 |  *  FIXEDFON.FON    FixedSys | 
 |  *  FONTS.FON       System | 
 |  *  OEMFONT.FON     Termial | 
 |  *  LogPixels       Current dpi set by the display control panel applet | 
 |  *                  (HKLM\\Software\\Microsft\\Windows NT\\CurrentVersion\\FontDPI | 
 |  *                  also has a LogPixels value that appears to mirror this) | 
 |  * | 
 |  * On my system these values have data: vgafix.fon, vgasys.fon, vga850.fon and 96 respectively | 
 |  * (vgaoem.fon would be your oemfont.fon if you have a US setup). | 
 |  * If the resolution is changed to be >= 109dpi then the fonts goto 8514fix, 8514sys and 8514oem | 
 |  * (not sure what's happening to the oem codepage here). 109 is nicely halfway between 96 and 120dpi, | 
 |  * so that makes sense. | 
 |  * | 
 |  * Additionally Windows also loads the fonts listed in the [386enh] section of system.ini (this doesn't appear | 
 |  * to be mapped into the registry on Windows 2000 at least). | 
 |  * I have | 
 |  * woafont=app850.fon | 
 |  * ega80woa.fon=ega80850.fon | 
 |  * ega40woa.fon=ega40850.fon | 
 |  * cga80woa.fon=cga80850.fon | 
 |  * cga40woa.fon=cga40850.fon | 
 |  */ | 
 |  | 
 |  | 
 | static inline BOOL is_win9x(void) | 
 | { | 
 |     return GetVersion() & 0x80000000; | 
 | } | 
 | /*  | 
 |    This function builds an FT_Fixed from a float. It puts the integer part | 
 |    in the highest 16 bits and the decimal part in the lowest 16 bits of the FT_Fixed. | 
 |    It fails if the integer part of the float number is greater than SHORT_MAX. | 
 | */ | 
 | static inline FT_Fixed FT_FixedFromFloat(float f) | 
 | { | 
 | 	short value = f; | 
 | 	unsigned short fract = (f - value) * 0xFFFF; | 
 | 	return (FT_Fixed)((long)value << 16 | (unsigned long)fract); | 
 | } | 
 |  | 
 | /*  | 
 |    This function builds an FT_Fixed from a FIXED. It simply put f.value  | 
 |    in the highest 16 bits and f.fract in the lowest 16 bits of the FT_Fixed. | 
 | */ | 
 | static inline FT_Fixed FT_FixedFromFIXED(FIXED f) | 
 | { | 
 | 	return (FT_Fixed)((long)f.value << 16 | (unsigned long)f.fract); | 
 | } | 
 |  | 
 | #define ADDFONT_EXTERNAL_FONT 0x01 | 
 | #define ADDFONT_FORCE_BITMAP  0x02 | 
 | static BOOL AddFontFileToList(const char *file, char *fake_family, DWORD flags) | 
 | { | 
 |     FT_Face ft_face; | 
 |     TT_OS2 *pOS2; | 
 |     TT_Header *pHeader = NULL; | 
 |     WCHAR *FamilyW, *StyleW; | 
 |     DWORD len; | 
 |     Family *family; | 
 |     Face *face; | 
 |     struct list *family_elem_ptr, *face_elem_ptr; | 
 |     FT_Error err; | 
 |     FT_Long face_index = 0, num_faces; | 
 | #ifdef HAVE_FREETYPE_FTWINFNT_H | 
 |     FT_WinFNT_HeaderRec winfnt_header; | 
 | #endif | 
 |     int i, bitmap_num; | 
 |  | 
 |     do { | 
 |         char *family_name = fake_family; | 
 |  | 
 |         TRACE("Loading font file %s index %ld\n", debugstr_a(file), face_index); | 
 | 	if((err = pFT_New_Face(library, file, face_index, &ft_face)) != 0) { | 
 | 	    WARN("Unable to load font file %s err = %x\n", debugstr_a(file), err); | 
 | 	    return FALSE; | 
 | 	} | 
 |  | 
 | 	if(!FT_IS_SFNT(ft_face) && (FT_IS_SCALABLE(ft_face) || !(flags & ADDFONT_FORCE_BITMAP))) { /* for now we'll accept TT/OT or bitmap fonts*/ | 
 | 	    pFT_Done_Face(ft_face); | 
 | 	    return FALSE; | 
 | 	} | 
 |  | 
 |         /* There are too many bugs in FreeType < 2.1.9 for bitmap font support */ | 
 |         if(!FT_IS_SCALABLE(ft_face) && FT_SimpleVersion < ((2 << 16) | (1 << 8) | (9 << 0))) { | 
 | 	    pFT_Done_Face(ft_face); | 
 | 	    return FALSE; | 
 | 	} | 
 |  | 
 | 	if(FT_IS_SFNT(ft_face) && (!pFT_Get_Sfnt_Table(ft_face, ft_sfnt_os2) || | 
 | 	   !pFT_Get_Sfnt_Table(ft_face, ft_sfnt_hhea) || | 
 |            !(pHeader = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_head)))) { | 
 | 	    TRACE("Font file %s lacks either an OS2, HHEA or HEAD table.\n" | 
 | 		  "Skipping this font.\n", debugstr_a(file)); | 
 | 	    pFT_Done_Face(ft_face); | 
 | 	    return FALSE; | 
 | 	} | 
 |  | 
 |         if(!ft_face->family_name || !ft_face->style_name) { | 
 |             TRACE("Font file %s lacks either a family or style name\n", debugstr_a(file)); | 
 |             pFT_Done_Face(ft_face); | 
 |             return FALSE; | 
 |         } | 
 |  | 
 |         if(!family_name) | 
 |             family_name = ft_face->family_name; | 
 |  | 
 |         bitmap_num = 0; | 
 |         do { | 
 |             My_FT_Bitmap_Size *size = NULL; | 
 |  | 
 |             if(!FT_IS_SCALABLE(ft_face)) | 
 |                 size = (My_FT_Bitmap_Size *)ft_face->available_sizes + bitmap_num; | 
 |  | 
 |             len = MultiByteToWideChar(CP_ACP, 0, family_name, -1, NULL, 0); | 
 |             FamilyW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); | 
 |             MultiByteToWideChar(CP_ACP, 0, family_name, -1, FamilyW, len); | 
 |  | 
 |             family = NULL; | 
 |             LIST_FOR_EACH(family_elem_ptr, &font_list) { | 
 |                 family = LIST_ENTRY(family_elem_ptr, Family, entry); | 
 |                 if(!strcmpW(family->FamilyName, FamilyW)) | 
 |                     break; | 
 |                 family = NULL; | 
 |             } | 
 |             if(!family) { | 
 |                 family = HeapAlloc(GetProcessHeap(), 0, sizeof(*family)); | 
 |                 family->FamilyName = FamilyW; | 
 |                 list_init(&family->faces); | 
 |                 list_add_tail(&font_list, &family->entry); | 
 |             } else { | 
 |                 HeapFree(GetProcessHeap(), 0, FamilyW); | 
 |             } | 
 |  | 
 |             len = MultiByteToWideChar(CP_ACP, 0, ft_face->style_name, -1, NULL, 0); | 
 |             StyleW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); | 
 |             MultiByteToWideChar(CP_ACP, 0, ft_face->style_name, -1, StyleW, len); | 
 |  | 
 |             face_elem_ptr = list_head(&family->faces); | 
 |             while(face_elem_ptr) { | 
 |                 face = LIST_ENTRY(face_elem_ptr, Face, entry); | 
 |                 face_elem_ptr = list_next(&family->faces, face_elem_ptr); | 
 |                 if(!strcmpW(face->StyleName, StyleW) && | 
 |                    (FT_IS_SCALABLE(ft_face) || (size->y_ppem == face->size.y_ppem))) { | 
 |                     TRACE("Already loaded font %s %s original version is %lx, this version is %lx\n", | 
 |                           debugstr_w(family->FamilyName), debugstr_w(StyleW), | 
 |                           face->font_version,  pHeader ? pHeader->Font_Revision : 0); | 
 |  | 
 |                     if(fake_family) { | 
 |                         TRACE("This font is a replacement but the original really exists, so we'll skip the replacement\n"); | 
 |                         HeapFree(GetProcessHeap(), 0, StyleW); | 
 |                         pFT_Done_Face(ft_face); | 
 |                         return FALSE; | 
 |                     } | 
 |                     if(!pHeader || pHeader->Font_Revision <= face->font_version) { | 
 |                         TRACE("Original font is newer so skipping this one\n"); | 
 |                         HeapFree(GetProcessHeap(), 0, StyleW); | 
 |                         pFT_Done_Face(ft_face); | 
 |                         return FALSE; | 
 |                     } else { | 
 |                         TRACE("Replacing original with this one\n"); | 
 |                         list_remove(&face->entry); | 
 |                         HeapFree(GetProcessHeap(), 0, face->file); | 
 |                         HeapFree(GetProcessHeap(), 0, face->StyleName); | 
 |                         HeapFree(GetProcessHeap(), 0, face); | 
 |                         break; | 
 |                     } | 
 |                 } | 
 |             } | 
 |             face = HeapAlloc(GetProcessHeap(), 0, sizeof(*face)); | 
 |             list_add_tail(&family->faces, &face->entry); | 
 |             face->StyleName = StyleW; | 
 |             face->file = HeapAlloc(GetProcessHeap(),0,strlen(file)+1); | 
 |             strcpy(face->file, file); | 
 |             face->face_index = face_index; | 
 |             face->Italic = (ft_face->style_flags & FT_STYLE_FLAG_ITALIC) ? 1 : 0; | 
 |             face->Bold = (ft_face->style_flags & FT_STYLE_FLAG_BOLD) ? 1 : 0; | 
 |             face->font_version = pHeader ? pHeader->Font_Revision : 0; | 
 |             face->family = family; | 
 |             face->external = (flags & ADDFONT_EXTERNAL_FONT) ? TRUE : FALSE; | 
 |  | 
 |             if(FT_IS_SCALABLE(ft_face)) { | 
 |                 memset(&face->size, 0, sizeof(face->size)); | 
 |                 face->scalable = TRUE; | 
 |             } else { | 
 |                 TRACE("Adding bitmap size h %d w %d size %ld x_ppem %ld y_ppem %ld\n", | 
 |                       size->height, size->width, size->size >> 6, | 
 |                       size->x_ppem >> 6, size->y_ppem >> 6); | 
 |                 face->size.height = size->height; | 
 |                 face->size.width = size->width; | 
 |                 face->size.size = size->size; | 
 |                 face->size.x_ppem = size->x_ppem; | 
 |                 face->size.y_ppem = size->y_ppem; | 
 |                 face->size.internal_leading = 0; | 
 |                 face->scalable = FALSE; | 
 |             } | 
 |  | 
 |             memset(&face->fs, 0, sizeof(face->fs)); | 
 |  | 
 |             pOS2 = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_os2); | 
 |             if(pOS2) { | 
 |                 face->fs.fsCsb[0] = pOS2->ulCodePageRange1; | 
 |                 face->fs.fsCsb[1] = pOS2->ulCodePageRange2; | 
 |                 face->fs.fsUsb[0] = pOS2->ulUnicodeRange1; | 
 |                 face->fs.fsUsb[1] = pOS2->ulUnicodeRange2; | 
 |                 face->fs.fsUsb[2] = pOS2->ulUnicodeRange3; | 
 |                 face->fs.fsUsb[3] = pOS2->ulUnicodeRange4; | 
 |                 if(pOS2->version == 0) { | 
 |                     FT_UInt dummy; | 
 |  | 
 |                     if(!pFT_Get_First_Char || (pFT_Get_First_Char( ft_face, &dummy ) < 0x100)) | 
 |                         face->fs.fsCsb[0] |= 1; | 
 |                     else | 
 |                         face->fs.fsCsb[0] |= 1L << 31; | 
 |                 } | 
 |             } | 
 | #ifdef HAVE_FREETYPE_FTWINFNT_H | 
 |             else if(pFT_Get_WinFNT_Header && !pFT_Get_WinFNT_Header(ft_face, &winfnt_header)) { | 
 |                 CHARSETINFO csi; | 
 |                 TRACE("pix_h %d charset %d dpi %dx%d pt %d\n", winfnt_header.pixel_height, winfnt_header.charset, | 
 |                       winfnt_header.vertical_resolution,winfnt_header.horizontal_resolution, winfnt_header.nominal_point_size); | 
 |                 if(TranslateCharsetInfo((DWORD*)(UINT)winfnt_header.charset, &csi, TCI_SRCCHARSET)) | 
 |                     memcpy(&face->fs, &csi.fs, sizeof(csi.fs)); | 
 |                 face->size.internal_leading = winfnt_header.internal_leading; | 
 |             } | 
 | #endif | 
 |             TRACE("fsCsb = %08lx %08lx/%08lx %08lx %08lx %08lx\n", | 
 |                   face->fs.fsCsb[0], face->fs.fsCsb[1], | 
 |                   face->fs.fsUsb[0], face->fs.fsUsb[1], | 
 |                   face->fs.fsUsb[2], face->fs.fsUsb[3]); | 
 |  | 
 |  | 
 |             if(face->fs.fsCsb[0] == 0) { /* let's see if we can find any interesting cmaps */ | 
 |                 for(i = 0; i < ft_face->num_charmaps; i++) { | 
 |                     switch(ft_face->charmaps[i]->encoding) { | 
 |                     case FT_ENCODING_UNICODE: | 
 |                     case FT_ENCODING_APPLE_ROMAN: | 
 | 			face->fs.fsCsb[0] |= 1; | 
 |                         break; | 
 |                     case FT_ENCODING_MS_SYMBOL: | 
 |                         face->fs.fsCsb[0] |= 1L << 31; | 
 |                         break; | 
 |                     default: | 
 |                         break; | 
 |                     } | 
 |                 } | 
 |             } | 
 |  | 
 |             if(face->fs.fsCsb[0] & ~(1L << 31)) | 
 |                 have_installed_roman_font = TRUE; | 
 |         } while(!FT_IS_SCALABLE(ft_face) && ++bitmap_num < ft_face->num_fixed_sizes); | 
 |  | 
 | 	num_faces = ft_face->num_faces; | 
 | 	pFT_Done_Face(ft_face); | 
 | 	TRACE("Added font %s %s\n", debugstr_w(family->FamilyName), | 
 | 	      debugstr_w(StyleW)); | 
 |     } while(num_faces > ++face_index); | 
 |     return TRUE; | 
 | } | 
 |  | 
 | static void DumpFontList(void) | 
 | { | 
 |     Family *family; | 
 |     Face *face; | 
 |     struct list *family_elem_ptr, *face_elem_ptr; | 
 |  | 
 |     LIST_FOR_EACH(family_elem_ptr, &font_list) { | 
 |         family = LIST_ENTRY(family_elem_ptr, Family, entry);  | 
 |         TRACE("Family: %s\n", debugstr_w(family->FamilyName)); | 
 |         LIST_FOR_EACH(face_elem_ptr, &family->faces) { | 
 |             face = LIST_ENTRY(face_elem_ptr, Face, entry); | 
 | 	    TRACE("\t%s", debugstr_w(face->StyleName)); | 
 |             if(!face->scalable) | 
 |                 TRACE(" %ld", face->size.y_ppem >> 6); | 
 |             TRACE("\n"); | 
 | 	} | 
 |     } | 
 |     return; | 
 | } | 
 |  | 
 | static void DumpSubstList(void) | 
 | { | 
 |     FontSubst *psub; | 
 |  | 
 |     for(psub = substlist; psub; psub = psub->next) | 
 |         if(psub->from.charset != -1 || psub->to.charset != -1) | 
 | 	    TRACE("%s:%d -> %s:%d\n", debugstr_w(psub->from.name), | 
 | 	      psub->from.charset, debugstr_w(psub->to.name), psub->to.charset); | 
 | 	else | 
 | 	    TRACE("%s -> %s\n", debugstr_w(psub->from.name), | 
 | 		  debugstr_w(psub->to.name)); | 
 |     return; | 
 | } | 
 |  | 
 | static LPWSTR strdupW(LPWSTR p) | 
 | { | 
 |     LPWSTR ret; | 
 |     DWORD len = (strlenW(p) + 1) * sizeof(WCHAR); | 
 |     ret = HeapAlloc(GetProcessHeap(), 0, len); | 
 |     memcpy(ret, p, len); | 
 |     return ret; | 
 | } | 
 |  | 
 | static void split_subst_info(NameCs *nc, LPSTR str) | 
 | { | 
 |     CHAR *p = strrchr(str, ','); | 
 |     DWORD len; | 
 |  | 
 |     nc->charset = -1; | 
 |     if(p && *(p+1)) { | 
 |         nc->charset = strtol(p+1, NULL, 10); | 
 | 	*p = '\0'; | 
 |     } | 
 |     len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0); | 
 |     nc->name = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); | 
 |     MultiByteToWideChar(CP_ACP, 0, str, -1, nc->name, len); | 
 | } | 
 |  | 
 | static void LoadSubstList(void) | 
 | { | 
 |     FontSubst *psub, **ppsub; | 
 |     HKEY hkey; | 
 |     DWORD valuelen, datalen, i = 0, type, dlen, vlen; | 
 |     LPSTR value; | 
 |     LPVOID data; | 
 |  | 
 |     if(substlist) { | 
 |         for(psub = substlist; psub;) { | 
 | 	    FontSubst *ptmp; | 
 | 	    HeapFree(GetProcessHeap(), 0, psub->to.name); | 
 | 	    HeapFree(GetProcessHeap(), 0, psub->from.name); | 
 | 	    ptmp = psub; | 
 | 	    psub = psub->next; | 
 | 	    HeapFree(GetProcessHeap(), 0, ptmp); | 
 | 	} | 
 | 	substlist = NULL; | 
 |     } | 
 |  | 
 |     if(RegOpenKeyA(HKEY_LOCAL_MACHINE, | 
 | 		   "Software\\Microsoft\\Windows NT\\CurrentVersion\\FontSubstitutes", | 
 | 		   &hkey) == ERROR_SUCCESS) { | 
 |  | 
 |         RegQueryInfoKeyA(hkey, NULL, NULL, NULL, NULL, NULL, NULL, NULL, | 
 | 			 &valuelen, &datalen, NULL, NULL); | 
 |  | 
 | 	valuelen++; /* returned value doesn't include room for '\0' */ | 
 | 	value = HeapAlloc(GetProcessHeap(), 0, valuelen * sizeof(CHAR)); | 
 | 	data = HeapAlloc(GetProcessHeap(), 0, datalen); | 
 |  | 
 | 	dlen = datalen; | 
 | 	vlen = valuelen; | 
 | 	ppsub = &substlist; | 
 | 	while(RegEnumValueA(hkey, i++, value, &vlen, NULL, &type, data, | 
 | 			    &dlen) == ERROR_SUCCESS) { | 
 | 	    TRACE("Got %s=%s\n", debugstr_a(value), debugstr_a(data)); | 
 |  | 
 | 	    *ppsub = HeapAlloc(GetProcessHeap(), 0, sizeof(**ppsub)); | 
 | 	    (*ppsub)->next = NULL; | 
 | 	    split_subst_info(&((*ppsub)->from), value); | 
 | 	    split_subst_info(&((*ppsub)->to), data); | 
 |  | 
 | 	    /* Win 2000 doesn't allow mapping between different charsets | 
 | 	       or mapping of DEFAULT_CHARSET */ | 
 | 	    if(((*ppsub)->to.charset != (*ppsub)->from.charset) || | 
 | 	       (*ppsub)->to.charset == DEFAULT_CHARSET) { | 
 | 	        HeapFree(GetProcessHeap(), 0, (*ppsub)->to.name); | 
 | 		HeapFree(GetProcessHeap(), 0, (*ppsub)->from.name); | 
 | 		HeapFree(GetProcessHeap(), 0, *ppsub); | 
 |                 *ppsub = NULL; | 
 | 	    } else { | 
 | 	        ppsub = &((*ppsub)->next); | 
 | 	    } | 
 | 	    /* reset dlen and vlen */ | 
 | 	    dlen = datalen; | 
 | 	    vlen = valuelen; | 
 | 	} | 
 | 	HeapFree(GetProcessHeap(), 0, data); | 
 | 	HeapFree(GetProcessHeap(), 0, value); | 
 | 	RegCloseKey(hkey); | 
 |     } | 
 | } | 
 |  | 
 | /*********************************************************** | 
 |  * The replacement list is a way to map an entire font | 
 |  * family onto another family.  For example adding | 
 |  * | 
 |  * [HKLM\Software\Wine\Wine\FontReplacements] | 
 |  * "Wingdings"="Winedings" | 
 |  * | 
 |  * would enumerate the Winedings font both as Winedings and | 
 |  * Wingdings.  However if a real Wingdings font is present the | 
 |  * replacement does not take place. | 
 |  *  | 
 |  */ | 
 | static void LoadReplaceList(void) | 
 | { | 
 |     HKEY hkey; | 
 |     DWORD valuelen, datalen, i = 0, type, dlen, vlen; | 
 |     LPSTR value; | 
 |     LPVOID data; | 
 |     Family *family; | 
 |     Face *face; | 
 |     struct list *family_elem_ptr, *face_elem_ptr; | 
 |     WCHAR old_nameW[200]; | 
 |  | 
 |     if(RegOpenKeyA(HKEY_LOCAL_MACHINE, | 
 | 		   "Software\\Wine\\Wine\\FontReplacements", | 
 | 		   &hkey) == ERROR_SUCCESS) { | 
 |  | 
 |         RegQueryInfoKeyA(hkey, NULL, NULL, NULL, NULL, NULL, NULL, NULL, | 
 | 			 &valuelen, &datalen, NULL, NULL); | 
 |  | 
 | 	valuelen++; /* returned value doesn't include room for '\0' */ | 
 | 	value = HeapAlloc(GetProcessHeap(), 0, valuelen * sizeof(CHAR)); | 
 | 	data = HeapAlloc(GetProcessHeap(), 0, datalen); | 
 |  | 
 | 	dlen = datalen; | 
 | 	vlen = valuelen; | 
 | 	while(RegEnumValueA(hkey, i++, value, &vlen, NULL, &type, data, | 
 | 			    &dlen) == ERROR_SUCCESS) { | 
 | 	    TRACE("Got %s=%s\n", debugstr_a(value), debugstr_a(data)); | 
 |             /* "NewName"="Oldname" */ | 
 |             if(!MultiByteToWideChar(CP_ACP, 0, data, -1, old_nameW, sizeof(old_nameW))) | 
 |                 break; | 
 |  | 
 |             /* Find the old family and hence all of the font files | 
 |                in that family */ | 
 |             LIST_FOR_EACH(family_elem_ptr, &font_list) { | 
 |                 family = LIST_ENTRY(family_elem_ptr, Family, entry);  | 
 |                 if(!strcmpiW(family->FamilyName, old_nameW)) {                 | 
 |                     LIST_FOR_EACH(face_elem_ptr, &family->faces) { | 
 |                         face = LIST_ENTRY(face_elem_ptr, Face, entry); | 
 |                         TRACE("mapping %s %s to %s\n", debugstr_w(family->FamilyName), | 
 |                               debugstr_w(face->StyleName), value); | 
 |                         /* Now add a new entry with the new family name */ | 
 |                         AddFontFileToList(face->file, value, ADDFONT_FORCE_BITMAP | (face->external ? ADDFONT_EXTERNAL_FONT : 0)); | 
 |                     } | 
 |                     break; | 
 |                 } | 
 |             } | 
 | 	    /* reset dlen and vlen */ | 
 | 	    dlen = datalen; | 
 | 	    vlen = valuelen; | 
 | 	} | 
 | 	HeapFree(GetProcessHeap(), 0, data); | 
 | 	HeapFree(GetProcessHeap(), 0, value); | 
 | 	RegCloseKey(hkey); | 
 |     } | 
 | } | 
 |  | 
 |  | 
 | static BOOL ReadFontDir(const char *dirname, BOOL external_fonts) | 
 | { | 
 |     DIR *dir; | 
 |     struct dirent *dent; | 
 |     char path[MAX_PATH]; | 
 |  | 
 |     TRACE("Loading fonts from %s\n", debugstr_a(dirname)); | 
 |  | 
 |     dir = opendir(dirname); | 
 |     if(!dir) { | 
 |         ERR("Can't open directory %s\n", debugstr_a(dirname)); | 
 | 	return FALSE; | 
 |     } | 
 |     while((dent = readdir(dir)) != NULL) { | 
 | 	struct stat statbuf; | 
 |  | 
 |         if(!strcmp(dent->d_name, ".") || !strcmp(dent->d_name, "..")) | 
 | 	    continue; | 
 |  | 
 | 	TRACE("Found %s in %s\n", debugstr_a(dent->d_name), debugstr_a(dirname)); | 
 |  | 
 | 	sprintf(path, "%s/%s", dirname, dent->d_name); | 
 |  | 
 | 	if(stat(path, &statbuf) == -1) | 
 | 	{ | 
 | 	    WARN("Can't stat %s\n", debugstr_a(path)); | 
 | 	    continue; | 
 | 	} | 
 | 	if(S_ISDIR(statbuf.st_mode)) | 
 | 	    ReadFontDir(path, external_fonts); | 
 | 	else | 
 | 	    AddFontFileToList(path, NULL, external_fonts ? ADDFONT_EXTERNAL_FONT : 0); | 
 |     } | 
 |     closedir(dir); | 
 |     return TRUE; | 
 | } | 
 |  | 
 | static void load_fontconfig_fonts(void) | 
 | { | 
 | #ifdef HAVE_FONTCONFIG_FONTCONFIG_H | 
 |     void *fc_handle = NULL; | 
 |     FcConfig *config; | 
 |     FcPattern *pat; | 
 |     FcObjectSet *os; | 
 |     FcFontSet *fontset; | 
 |     FcValue v; | 
 |     int i, len; | 
 |     const char *ext; | 
 |  | 
 |     fc_handle = wine_dlopen(SONAME_LIBFONTCONFIG, RTLD_NOW, NULL, 0); | 
 |     if(!fc_handle) { | 
 |         TRACE("Wine cannot find the fontconfig library (%s).\n", | 
 |               SONAME_LIBFONTCONFIG); | 
 | 	return; | 
 |     } | 
 | #define LOAD_FUNCPTR(f) if((p##f = wine_dlsym(fc_handle, #f, NULL, 0)) == NULL){WARN("Can't find symbol %s\n", #f); goto sym_not_found;} | 
 | LOAD_FUNCPTR(FcConfigGetCurrent); | 
 | LOAD_FUNCPTR(FcFontList); | 
 | LOAD_FUNCPTR(FcFontSetDestroy); | 
 | LOAD_FUNCPTR(FcInit); | 
 | LOAD_FUNCPTR(FcObjectSetAdd); | 
 | LOAD_FUNCPTR(FcObjectSetCreate); | 
 | LOAD_FUNCPTR(FcObjectSetDestroy); | 
 | LOAD_FUNCPTR(FcPatternCreate); | 
 | LOAD_FUNCPTR(FcPatternDestroy); | 
 | LOAD_FUNCPTR(FcPatternGet); | 
 | #undef LOAD_FUNCPTR | 
 |  | 
 |     if(!pFcInit()) return; | 
 |      | 
 |     config = pFcConfigGetCurrent(); | 
 |     pat = pFcPatternCreate(); | 
 |     os = pFcObjectSetCreate(); | 
 |     pFcObjectSetAdd(os, FC_FILE); | 
 |     fontset = pFcFontList(config, pat, os); | 
 |     if(!fontset) return; | 
 |     for(i = 0; i < fontset->nfont; i++) { | 
 |         if(pFcPatternGet(fontset->fonts[i], FC_FILE, 0, &v) != FcResultMatch) | 
 |             continue; | 
 |         if(v.type != FcTypeString) continue; | 
 |         TRACE("fontconfig: %s\n", v.u.s); | 
 |  | 
 |         /* We're just interested in OT/TT fonts for now, so this hack just | 
 |            picks up the standard extensions to save time loading every other | 
 |            font */ | 
 |         len = strlen(v.u.s); | 
 |         if(len < 4) continue; | 
 |         ext = v.u.s + len - 3; | 
 |         if(!strcasecmp(ext, "ttf") || !strcasecmp(ext, "ttc") || !strcasecmp(ext, "otf")) | 
 |             AddFontFileToList(v.u.s, NULL, ADDFONT_EXTERNAL_FONT); | 
 |     } | 
 |     pFcFontSetDestroy(fontset); | 
 |     pFcObjectSetDestroy(os); | 
 |     pFcPatternDestroy(pat); | 
 |  sym_not_found: | 
 | #endif | 
 |     return; | 
 | } | 
 |  | 
 |  | 
 | void load_system_fonts(void) | 
 | { | 
 |     HKEY hkey; | 
 |     WCHAR data[MAX_PATH], windowsdir[MAX_PATH], pathW[MAX_PATH]; | 
 |     const WCHAR **value; | 
 |     DWORD dlen, type; | 
 |     static const WCHAR fmtW[] = {'%','s','\\','%','s','\0'}; | 
 |     char *unixname; | 
 |  | 
 |     if(RegOpenKeyW(HKEY_CURRENT_CONFIG, system_fonts_reg_key, &hkey) == ERROR_SUCCESS) { | 
 |         GetWindowsDirectoryW(windowsdir, sizeof(windowsdir) / sizeof(WCHAR)); | 
 |         strcatW(windowsdir, fontsW); | 
 |         for(value = SystemFontValues; *value; value++) {  | 
 |             dlen = sizeof(data); | 
 |             if(RegQueryValueExW(hkey, *value, 0, &type, (void*)data, &dlen) == ERROR_SUCCESS && | 
 |                type == REG_SZ) { | 
 |                 sprintfW(pathW, fmtW, windowsdir, data); | 
 |                 if((unixname = wine_get_unix_file_name(pathW))) { | 
 |                     AddFontFileToList(unixname, NULL, ADDFONT_FORCE_BITMAP); | 
 |                     HeapFree(GetProcessHeap(), 0, unixname); | 
 |                 } | 
 |             } | 
 |         } | 
 |         RegCloseKey(hkey); | 
 |     } | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  * | 
 |  * This adds registry entries for any externally loaded fonts | 
 |  * (fonts from fontconfig or FontDirs).  It also deletes entries | 
 |  * of no longer existing fonts. | 
 |  * | 
 |  */ | 
 | void update_reg_entries(void) | 
 | { | 
 |     HKEY winkey = 0, externalkey = 0; | 
 |     LPWSTR valueW; | 
 |     LPVOID data; | 
 |     DWORD dlen, vlen, datalen, valuelen, i, type, len, len_fam; | 
 |     Family *family; | 
 |     Face *face; | 
 |     struct list *family_elem_ptr, *face_elem_ptr; | 
 |     WCHAR *file; | 
 |     static const WCHAR TrueType[] = {' ','(','T','r','u','e','T','y','p','e',')','\0'}; | 
 |     static const WCHAR spaceW[] = {' ', '\0'}; | 
 |     char *path; | 
 |  | 
 |     if(RegCreateKeyExW(HKEY_LOCAL_MACHINE, is_win9x() ? win9x_font_reg_key : winnt_font_reg_key, | 
 |                        0, NULL, 0, KEY_ALL_ACCESS, NULL, &winkey, NULL) != ERROR_SUCCESS) { | 
 |         ERR("Can't create Windows font reg key\n"); | 
 |         goto end; | 
 |     } | 
 |     if(RegCreateKeyExW(HKEY_LOCAL_MACHINE, external_fonts_reg_key, | 
 |                        0, NULL, 0, KEY_ALL_ACCESS, NULL, &externalkey, NULL) != ERROR_SUCCESS) { | 
 |         ERR("Can't create external font reg key\n"); | 
 |         goto end; | 
 |     } | 
 |  | 
 |     /* Delete all external fonts added last time */ | 
 |  | 
 |     RegQueryInfoKeyW(externalkey, NULL, NULL, NULL, NULL, NULL, NULL, NULL, | 
 |                      &valuelen, &datalen, NULL, NULL); | 
 |     valuelen++; /* returned value doesn't include room for '\0' */ | 
 |     valueW = HeapAlloc(GetProcessHeap(), 0, valuelen * sizeof(WCHAR)); | 
 |     data = HeapAlloc(GetProcessHeap(), 0, datalen * sizeof(WCHAR)); | 
 |  | 
 |     dlen = datalen * sizeof(WCHAR); | 
 |     vlen = valuelen; | 
 |     i = 0; | 
 |     while(RegEnumValueW(externalkey, i++, valueW, &vlen, NULL, &type, data, | 
 |                         &dlen) == ERROR_SUCCESS) { | 
 |  | 
 |         RegDeleteValueW(winkey, valueW); | 
 |         /* reset dlen and vlen */ | 
 |         dlen = datalen; | 
 |         vlen = valuelen; | 
 |     } | 
 |     HeapFree(GetProcessHeap(), 0, data); | 
 |     HeapFree(GetProcessHeap(), 0, valueW); | 
 |  | 
 |     /* Delete the old external fonts key */ | 
 |     RegCloseKey(externalkey); | 
 |     externalkey = 0; | 
 |     RegDeleteKeyW(HKEY_LOCAL_MACHINE, external_fonts_reg_key); | 
 |  | 
 |     if(RegCreateKeyExW(HKEY_LOCAL_MACHINE, external_fonts_reg_key, | 
 |                        0, NULL, 0, KEY_ALL_ACCESS, NULL, &externalkey, NULL) != ERROR_SUCCESS) { | 
 |         ERR("Can't create external font reg key\n"); | 
 |         goto end; | 
 |     } | 
 |  | 
 |     /* enumerate the fonts and add external ones to the two keys */ | 
 |  | 
 |     LIST_FOR_EACH(family_elem_ptr, &font_list) { | 
 |         family = LIST_ENTRY(family_elem_ptr, Family, entry);  | 
 |         len_fam = strlenW(family->FamilyName) + sizeof(TrueType) / sizeof(WCHAR) + 1; | 
 |         LIST_FOR_EACH(face_elem_ptr, &family->faces) { | 
 |             face = LIST_ENTRY(face_elem_ptr, Face, entry); | 
 |             if(!face->external) continue; | 
 |             len = len_fam; | 
 |             if(strcmpiW(face->StyleName, RegularW)) | 
 |                 len = len_fam + strlenW(face->StyleName) + 1; | 
 |             valueW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); | 
 |             strcpyW(valueW, family->FamilyName); | 
 |             if(len != len_fam) { | 
 |                 strcatW(valueW, spaceW); | 
 |                 strcatW(valueW, face->StyleName); | 
 |             } | 
 |             strcatW(valueW, TrueType); | 
 |             if((path = strrchr(face->file, '/')) == NULL) | 
 |                 path = face->file; | 
 |             else | 
 |                 path++; | 
 |             len = MultiByteToWideChar(CP_ACP, 0, path, -1, NULL, 0); | 
 |  | 
 |             file = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); | 
 |             MultiByteToWideChar(CP_ACP, 0, path, -1, file, len); | 
 |             RegSetValueExW(winkey, valueW, 0, REG_SZ, (BYTE*)file, len * sizeof(WCHAR)); | 
 |             RegSetValueExW(externalkey, valueW, 0, REG_SZ, (BYTE*)file, len * sizeof(WCHAR)); | 
 |  | 
 |             HeapFree(GetProcessHeap(), 0, file); | 
 |             HeapFree(GetProcessHeap(), 0, valueW); | 
 |         } | 
 |     } | 
 |  end: | 
 |     if(externalkey) | 
 |         RegCloseKey(externalkey); | 
 |     if(winkey) | 
 |         RegCloseKey(winkey); | 
 |     return; | 
 | } | 
 |  | 
 |  | 
 | /************************************************************* | 
 |  *    WineEngAddFontResourceEx | 
 |  * | 
 |  */ | 
 | INT WineEngAddFontResourceEx(LPCWSTR file, DWORD flags, PVOID pdv) | 
 | { | 
 |     if (ft_handle)  /* do it only if we have freetype up and running */ | 
 |     { | 
 |         char *unixname; | 
 |  | 
 |         if(flags) | 
 |             FIXME("Ignoring flags %lx\n", flags); | 
 |  | 
 |         if((unixname = wine_get_unix_file_name(file))) | 
 |         { | 
 |             AddFontFileToList(unixname, NULL, ADDFONT_FORCE_BITMAP); | 
 |             HeapFree(GetProcessHeap(), 0, unixname); | 
 |         } | 
 |     } | 
 |     return 1; | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  *    WineEngRemoveFontResourceEx | 
 |  * | 
 |  */ | 
 | BOOL WineEngRemoveFontResourceEx(LPCWSTR file, DWORD flags, PVOID pdv) | 
 | { | 
 |     FIXME(":stub\n"); | 
 |     return TRUE; | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  *    WineEngInit | 
 |  * | 
 |  * Initialize FreeType library and create a list of available faces | 
 |  */ | 
 | BOOL WineEngInit(void) | 
 | { | 
 |     static const WCHAR dot_fonW[] = {'.','f','o','n','\0'}; | 
 |     HKEY hkey; | 
 |     DWORD valuelen, datalen, i = 0, type, dlen, vlen; | 
 |     LPVOID data; | 
 |     WCHAR windowsdir[MAX_PATH]; | 
 |     char *unixname; | 
 |     HANDLE font_mutex; | 
 |  | 
 |     TRACE("\n"); | 
 |  | 
 |     ft_handle = wine_dlopen(SONAME_LIBFREETYPE, RTLD_NOW, NULL, 0); | 
 |     if(!ft_handle) { | 
 |         WINE_MESSAGE( | 
 |       "Wine cannot find the FreeType font library.  To enable Wine to\n" | 
 |       "use TrueType fonts please install a version of FreeType greater than\n" | 
 |       "or equal to 2.0.5.\n" | 
 |       "http://www.freetype.org\n"); | 
 | 	return FALSE; | 
 |     } | 
 |  | 
 | #define LOAD_FUNCPTR(f) if((p##f = wine_dlsym(ft_handle, #f, NULL, 0)) == NULL){WARN("Can't find symbol %s\n", #f); goto sym_not_found;} | 
 |  | 
 |     LOAD_FUNCPTR(FT_Vector_Unit) | 
 |     LOAD_FUNCPTR(FT_Done_Face) | 
 |     LOAD_FUNCPTR(FT_Get_Char_Index) | 
 |     LOAD_FUNCPTR(FT_Get_Sfnt_Table) | 
 |     LOAD_FUNCPTR(FT_Init_FreeType) | 
 |     LOAD_FUNCPTR(FT_Load_Glyph) | 
 |     LOAD_FUNCPTR(FT_Matrix_Multiply) | 
 |     LOAD_FUNCPTR(FT_MulFix) | 
 |     LOAD_FUNCPTR(FT_New_Face) | 
 |     LOAD_FUNCPTR(FT_Outline_Get_Bitmap) | 
 |     LOAD_FUNCPTR(FT_Outline_Transform) | 
 |     LOAD_FUNCPTR(FT_Outline_Translate) | 
 |     LOAD_FUNCPTR(FT_Select_Charmap) | 
 |     LOAD_FUNCPTR(FT_Set_Pixel_Sizes) | 
 |     LOAD_FUNCPTR(FT_Vector_Transform) | 
 |  | 
 | #undef LOAD_FUNCPTR | 
 |     /* Don't warn if this one is missing */ | 
 |     pFT_Library_Version = wine_dlsym(ft_handle, "FT_Library_Version", NULL, 0); | 
 |     pFT_Load_Sfnt_Table = wine_dlsym(ft_handle, "FT_Load_Sfnt_Table", NULL, 0); | 
 |     pFT_Get_First_Char = wine_dlsym(ft_handle, "FT_Get_First_Char", NULL, 0); | 
 | #ifdef HAVE_FREETYPE_FTWINFNT_H | 
 |     pFT_Get_WinFNT_Header = wine_dlsym(ft_handle, "FT_Get_WinFNT_Header", NULL, 0); | 
 | #endif | 
 |       if(!wine_dlsym(ft_handle, "FT_Get_Postscript_Name", NULL, 0) && | 
 | 	 !wine_dlsym(ft_handle, "FT_Sqrt64", NULL, 0)) { | 
 | 	/* try to avoid 2.0.4: >= 2.0.5 has FT_Get_Postscript_Name and | 
 | 	   <= 2.0.3 has FT_Sqrt64 */ | 
 | 	  goto sym_not_found; | 
 |       } | 
 |  | 
 |     if(pFT_Init_FreeType(&library) != 0) { | 
 |         ERR("Can't init FreeType library\n"); | 
 | 	wine_dlclose(ft_handle, NULL, 0); | 
 |         ft_handle = NULL; | 
 | 	return FALSE; | 
 |     } | 
 |     FT_Version.major=FT_Version.minor=FT_Version.patch=-1; | 
 |     if (pFT_Library_Version) | 
 |     { | 
 |         pFT_Library_Version(library,&FT_Version.major,&FT_Version.minor,&FT_Version.patch); | 
 |     } | 
 |     if (FT_Version.major<=0) | 
 |     { | 
 |         FT_Version.major=2; | 
 |         FT_Version.minor=0; | 
 |         FT_Version.patch=5; | 
 |     } | 
 |     TRACE("FreeType version is %d.%d.%d\n",FT_Version.major,FT_Version.minor,FT_Version.patch); | 
 |     FT_SimpleVersion = ((FT_Version.major << 16) & 0xff0000) | | 
 |                        ((FT_Version.minor <<  8) & 0x00ff00) | | 
 |                        ((FT_Version.patch      ) & 0x0000ff); | 
 |  | 
 |     if((font_mutex = CreateMutexW(NULL, FALSE, font_mutex_nameW)) == NULL) { | 
 |         ERR("Failed to create font mutex\n"); | 
 |         return FALSE; | 
 |     } | 
 |     WaitForSingleObject(font_mutex, INFINITE); | 
 |  | 
 |     /* load the system fonts */ | 
 |     load_system_fonts(); | 
 |  | 
 |     /* load in the fonts from %WINDOWSDIR%\\Fonts first of all */ | 
 |     GetWindowsDirectoryW(windowsdir, sizeof(windowsdir) / sizeof(WCHAR)); | 
 |     strcatW(windowsdir, fontsW); | 
 |     if((unixname = wine_get_unix_file_name(windowsdir))) | 
 |     { | 
 |         ReadFontDir(unixname, FALSE); | 
 |         HeapFree(GetProcessHeap(), 0, unixname); | 
 |     } | 
 |  | 
 |     /* now look under HKLM\Software\Microsoft\Windows[ NT]\CurrentVersion\Fonts | 
 |        for any fonts not installed in %WINDOWSDIR%\Fonts.  They will have their | 
 |        full path as the entry.  Also look for any .fon fonts, since ReadFontDir | 
 |        will skip these. */ | 
 |     if(RegOpenKeyW(HKEY_LOCAL_MACHINE, | 
 |                    is_win9x() ? win9x_font_reg_key : winnt_font_reg_key, | 
 | 		   &hkey) == ERROR_SUCCESS) { | 
 |         LPWSTR valueW; | 
 |         RegQueryInfoKeyW(hkey, NULL, NULL, NULL, NULL, NULL, NULL, NULL, | 
 | 			 &valuelen, &datalen, NULL, NULL); | 
 |  | 
 | 	valuelen++; /* returned value doesn't include room for '\0' */ | 
 | 	valueW = HeapAlloc(GetProcessHeap(), 0, valuelen * sizeof(WCHAR)); | 
 | 	data = HeapAlloc(GetProcessHeap(), 0, datalen * sizeof(WCHAR)); | 
 |         if (valueW && data) | 
 |         { | 
 |             dlen = datalen * sizeof(WCHAR); | 
 |             vlen = valuelen; | 
 |             while(RegEnumValueW(hkey, i++, valueW, &vlen, NULL, &type, data, | 
 |                                 &dlen) == ERROR_SUCCESS) { | 
 |                 if(((LPWSTR)data)[0] && ((LPWSTR)data)[1] == ':') | 
 |                 { | 
 |                     if((unixname = wine_get_unix_file_name((LPWSTR)data))) | 
 |                     { | 
 |                         AddFontFileToList(unixname, NULL, ADDFONT_FORCE_BITMAP); | 
 |                         HeapFree(GetProcessHeap(), 0, unixname); | 
 |                     } | 
 |                 } | 
 |                 else if(dlen / 2 >= 6 && !strcmpiW(((LPWSTR)data) + dlen / 2 - 5, dot_fonW)) | 
 |                 { | 
 |                     WCHAR pathW[MAX_PATH]; | 
 |                     static const WCHAR fmtW[] = {'%','s','\\','%','s','\0'}; | 
 |                     sprintfW(pathW, fmtW, windowsdir, data); | 
 |                     if((unixname = wine_get_unix_file_name(pathW))) | 
 |                     { | 
 |                         AddFontFileToList(unixname, NULL, ADDFONT_FORCE_BITMAP); | 
 |                         HeapFree(GetProcessHeap(), 0, unixname); | 
 |                     } | 
 |                 } | 
 |                 /* reset dlen and vlen */ | 
 |                 dlen = datalen; | 
 |                 vlen = valuelen; | 
 |             } | 
 |         } | 
 | 	if (data) HeapFree(GetProcessHeap(), 0, data); | 
 | 	if (valueW) HeapFree(GetProcessHeap(), 0, valueW); | 
 | 	RegCloseKey(hkey); | 
 |     } | 
 |  | 
 |     load_fontconfig_fonts(); | 
 |  | 
 |     /* then look in any directories that we've specified in the config file */ | 
 |     if(RegOpenKeyA(HKEY_LOCAL_MACHINE, | 
 | 		   "Software\\Wine\\Wine\\Config\\FontDirs", | 
 | 		   &hkey) == ERROR_SUCCESS) { | 
 |         LPSTR value; | 
 |         RegQueryInfoKeyA(hkey, NULL, NULL, NULL, NULL, NULL, NULL, NULL, | 
 | 			 &valuelen, &datalen, NULL, NULL); | 
 |  | 
 | 	valuelen++; /* returned value doesn't include room for '\0' */ | 
 | 	value = HeapAlloc(GetProcessHeap(), 0, valuelen); | 
 | 	data = HeapAlloc(GetProcessHeap(), 0, datalen); | 
 |  | 
 | 	dlen = datalen; | 
 | 	vlen = valuelen; | 
 | 	i = 0; | 
 | 	while(RegEnumValueA(hkey, i++, value, &vlen, NULL, &type, data, | 
 | 			    &dlen) == ERROR_SUCCESS) { | 
 | 	    TRACE("Got %s=%s\n", value, (LPSTR)data); | 
 | 	    ReadFontDir((LPSTR)data, TRUE); | 
 | 	    /* reset dlen and vlen */ | 
 | 	    dlen = datalen; | 
 | 	    vlen = valuelen; | 
 | 	} | 
 | 	HeapFree(GetProcessHeap(), 0, data); | 
 | 	HeapFree(GetProcessHeap(), 0, value); | 
 | 	RegCloseKey(hkey); | 
 |     } | 
 |  | 
 |     DumpFontList(); | 
 |     LoadSubstList(); | 
 |     DumpSubstList(); | 
 |     LoadReplaceList(); | 
 |     update_reg_entries(); | 
 |  | 
 |     ReleaseMutex(font_mutex); | 
 |     return TRUE; | 
 | sym_not_found: | 
 |     WINE_MESSAGE( | 
 |       "Wine cannot find certain functions that it needs inside the FreeType\n" | 
 |       "font library.  To enable Wine to use TrueType fonts please upgrade\n" | 
 |       "FreeType to at least version 2.0.5.\n" | 
 |       "http://www.freetype.org\n"); | 
 |     wine_dlclose(ft_handle, NULL, 0); | 
 |     ft_handle = NULL; | 
 |     return FALSE; | 
 | } | 
 |  | 
 |  | 
 | static LONG calc_ppem_for_height(FT_Face ft_face, LONG height) | 
 | { | 
 |     TT_OS2 *pOS2; | 
 |     TT_HoriHeader *pHori; | 
 |  | 
 |     LONG ppem; | 
 |  | 
 |     pOS2 = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_os2); | 
 |     pHori = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_hhea); | 
 |  | 
 |     if(height == 0) height = 16; | 
 |  | 
 |     /* Calc. height of EM square: | 
 |      * | 
 |      * For +ve lfHeight we have | 
 |      * lfHeight = (winAscent + winDescent) * ppem / units_per_em | 
 |      * Re-arranging gives: | 
 |      * ppem = units_per_em * lfheight / (winAscent + winDescent) | 
 |      * | 
 |      * For -ve lfHeight we have | 
 |      * |lfHeight| = ppem | 
 |      * [i.e. |lfHeight| = (winAscent + winDescent - il) * ppem / units_per_em | 
 |      * with il = winAscent + winDescent - units_per_em] | 
 |      * | 
 |      */ | 
 |  | 
 |     if(height > 0) { | 
 |         if(pOS2->usWinAscent + pOS2->usWinDescent == 0) | 
 |             ppem = ft_face->units_per_EM * height / | 
 |                 (pHori->Ascender - pHori->Descender); | 
 |         else | 
 |             ppem = ft_face->units_per_EM * height / | 
 |                 (pOS2->usWinAscent + pOS2->usWinDescent); | 
 |     } | 
 |     else | 
 |         ppem = -height; | 
 |  | 
 |     return ppem; | 
 | } | 
 |  | 
 | static LONG load_VDMX(GdiFont, LONG); | 
 |  | 
 | static FT_Face OpenFontFile(GdiFont font, char *file, FT_Long face_index, LONG width, LONG height) | 
 | { | 
 |     FT_Error err; | 
 |     FT_Face ft_face; | 
 |     LONG ppem; | 
 |  | 
 |     err = pFT_New_Face(library, file, face_index, &ft_face); | 
 |     if(err) { | 
 |         ERR("FT_New_Face rets %d\n", err); | 
 | 	return 0; | 
 |     } | 
 |  | 
 |     /* set it here, as load_VDMX needs it */ | 
 |     font->ft_face = ft_face; | 
 |  | 
 |     if(FT_IS_SCALABLE(ft_face)) { | 
 |         /* load the VDMX table if we have one */ | 
 |         ppem = load_VDMX(font, height); | 
 |         if(ppem == 0) | 
 |             ppem = calc_ppem_for_height(ft_face, height); | 
 |  | 
 |         if((err = pFT_Set_Pixel_Sizes(ft_face, 0, ppem)) != 0) | 
 |             WARN("FT_Set_Pixel_Sizes %d, %ld rets %x\n", 0, ppem, err); | 
 |     } else { | 
 |         if((err = pFT_Set_Pixel_Sizes(ft_face, width, height)) != 0) | 
 |             WARN("FT_Set_Pixel_Sizes %ld, %ld rets %x\n", width, height, err); | 
 |     } | 
 |     return ft_face; | 
 | } | 
 |  | 
 |  | 
 | static int get_nearest_charset(Face *face, int *cp) | 
 | { | 
 |   /* Only get here if lfCharSet == DEFAULT_CHARSET or we couldn't find | 
 |      a single face with the requested charset.  The idea is to check if | 
 |      the selected font supports the current ANSI codepage, if it does | 
 |      return the corresponding charset, else return the first charset */ | 
 |  | 
 |     CHARSETINFO csi; | 
 |     int acp = GetACP(), i; | 
 |     DWORD fs0; | 
 |  | 
 |     *cp = acp; | 
 |     if(TranslateCharsetInfo((DWORD*)acp, &csi, TCI_SRCCODEPAGE)) | 
 |         if(csi.fs.fsCsb[0] & face->fs.fsCsb[0]) | 
 | 	    return csi.ciCharset; | 
 |  | 
 |     for(i = 0; i < 32; i++) { | 
 |         fs0 = 1L << i; | 
 |         if(face->fs.fsCsb[0] & fs0) { | 
 | 	    if(TranslateCharsetInfo(&fs0, &csi, TCI_SRCFONTSIG)) { | 
 |                 *cp = csi.ciACP; | 
 | 	        return csi.ciCharset; | 
 |             } | 
 | 	    else | 
 | 	        FIXME("TCI failing on %lx\n", fs0); | 
 | 	} | 
 |     } | 
 |  | 
 |     FIXME("returning DEFAULT_CHARSET face->fs.fsCsb[0] = %08lx file = %s\n", | 
 | 	  face->fs.fsCsb[0], face->file); | 
 |     *cp = acp; | 
 |     return DEFAULT_CHARSET; | 
 | } | 
 |  | 
 | static GdiFont alloc_font(void) | 
 | { | 
 |     GdiFont ret = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*ret)); | 
 |     ret->gmsize = INIT_GM_SIZE; | 
 |     ret->gm = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, | 
 | 			ret->gmsize * sizeof(*ret->gm)); | 
 |     ret->potm = NULL; | 
 |     ret->font_desc.matrix.eM11 = ret->font_desc.matrix.eM22 = 1.0; | 
 |     list_init(&ret->hfontlist); | 
 |     return ret; | 
 | } | 
 |  | 
 | static void free_font(GdiFont font) | 
 | { | 
 |     if (font->ft_face) pFT_Done_Face(font->ft_face); | 
 |     if (font->potm) HeapFree(GetProcessHeap(), 0, font->potm); | 
 |     if (font->name) HeapFree(GetProcessHeap(), 0, font->name); | 
 |     HeapFree(GetProcessHeap(), 0, font->gm); | 
 |     HeapFree(GetProcessHeap(), 0, font); | 
 | } | 
 |  | 
 |  | 
 | /************************************************************* | 
 |  * load_VDMX | 
 |  * | 
 |  * load the vdmx entry for the specified height | 
 |  */ | 
 |  | 
 | #define MS_MAKE_TAG( _x1, _x2, _x3, _x4 ) \ | 
 |           ( ( (FT_ULong)_x4 << 24 ) |     \ | 
 |             ( (FT_ULong)_x3 << 16 ) |     \ | 
 |             ( (FT_ULong)_x2 <<  8 ) |     \ | 
 |               (FT_ULong)_x1         ) | 
 |  | 
 | #define MS_VDMX_TAG MS_MAKE_TAG('V', 'D', 'M', 'X') | 
 |  | 
 | typedef struct { | 
 |     BYTE bCharSet; | 
 |     BYTE xRatio; | 
 |     BYTE yStartRatio; | 
 |     BYTE yEndRatio; | 
 | } Ratios; | 
 |  | 
 |  | 
 | static LONG load_VDMX(GdiFont font, LONG height) | 
 | { | 
 |     BYTE hdr[6], tmp[2], group[4]; | 
 |     BYTE devXRatio, devYRatio; | 
 |     USHORT numRecs, numRatios; | 
 |     DWORD result, offset = -1; | 
 |     LONG ppem = 0; | 
 |     int i; | 
 |  | 
 |     result = WineEngGetFontData(font, MS_VDMX_TAG, 0, hdr, 6); | 
 |  | 
 |     if(result == GDI_ERROR) /* no vdmx table present, use linear scaling */ | 
 | 	return ppem; | 
 |  | 
 |     /* FIXME: need the real device aspect ratio */ | 
 |     devXRatio = 1; | 
 |     devYRatio = 1; | 
 |  | 
 |     numRecs = GET_BE_WORD(&hdr[2]); | 
 |     numRatios = GET_BE_WORD(&hdr[4]); | 
 |  | 
 |     TRACE("numRecs = %d numRatios = %d\n", numRecs, numRatios); | 
 |     for(i = 0; i < numRatios; i++) { | 
 | 	Ratios ratio; | 
 |  | 
 | 	offset = (3 * 2) + (i * sizeof(Ratios)); | 
 | 	WineEngGetFontData(font, MS_VDMX_TAG, offset, &ratio, sizeof(Ratios)); | 
 | 	offset = -1; | 
 |  | 
 | 	TRACE("Ratios[%d] %d  %d : %d -> %d\n", i, ratio.bCharSet, ratio.xRatio, ratio.yStartRatio, ratio.yEndRatio); | 
 |  | 
 | 	if(ratio.bCharSet != 1) | 
 | 	    continue; | 
 |  | 
 | 	if((ratio.xRatio == 0 && | 
 | 	    ratio.yStartRatio == 0 && | 
 | 	    ratio.yEndRatio == 0) || | 
 | 	   (devXRatio == ratio.xRatio && | 
 | 	    devYRatio >= ratio.yStartRatio && | 
 | 	    devYRatio <= ratio.yEndRatio)) | 
 | 	    { | 
 | 		offset = (3 * 2) + (numRatios * 4) + (i * 2); | 
 | 		WineEngGetFontData(font, MS_VDMX_TAG, offset, tmp, 2); | 
 | 		offset = GET_BE_WORD(tmp); | 
 | 		break; | 
 | 	    } | 
 |     } | 
 |  | 
 |     if(offset == -1) { | 
 | 	FIXME("No suitable ratio found\n"); | 
 | 	return ppem; | 
 |     } | 
 |  | 
 |     if(WineEngGetFontData(font, MS_VDMX_TAG, offset, group, 4) != GDI_ERROR) { | 
 | 	USHORT recs; | 
 | 	BYTE startsz, endsz; | 
 | 	BYTE *vTable; | 
 |  | 
 | 	recs = GET_BE_WORD(group); | 
 | 	startsz = group[2]; | 
 | 	endsz = group[3]; | 
 |  | 
 | 	TRACE("recs=%d  startsz=%d  endsz=%d\n", recs, startsz, endsz); | 
 |  | 
 | 	vTable = HeapAlloc(GetProcessHeap(), 0, recs * 6); | 
 | 	result = WineEngGetFontData(font, MS_VDMX_TAG, offset + 4, vTable, recs * 6); | 
 | 	if(result == GDI_ERROR) { | 
 | 	    FIXME("Failed to retrieve vTable\n"); | 
 | 	    goto end; | 
 | 	} | 
 |  | 
 | 	if(height > 0) { | 
 | 	    for(i = 0; i < recs; i++) { | 
 | 		SHORT yMax = GET_BE_WORD(&vTable[(i * 6) + 2]); | 
 |                 SHORT yMin = GET_BE_WORD(&vTable[(i * 6) + 4]); | 
 | 		ppem = GET_BE_WORD(&vTable[i * 6]); | 
 |  | 
 | 		if(yMax + -yMin == height) { | 
 | 		    font->yMax = yMax; | 
 | 		    font->yMin = yMin; | 
 | 		    TRACE("ppem %ld found; height=%ld  yMax=%d  yMin=%d\n", ppem, height, font->yMax, font->yMin); | 
 | 		    break; | 
 | 		} | 
 | 		if(yMax + -yMin > height) { | 
 | 		    if(--i < 0) { | 
 | 			ppem = 0; | 
 | 			goto end; /* failed */ | 
 | 		    } | 
 | 		    font->yMax = GET_BE_WORD(&vTable[(i * 6) + 2]); | 
 | 		    font->yMin = GET_BE_WORD(&vTable[(i * 6) + 4]); | 
 |                     TRACE("ppem %ld found; height=%ld  yMax=%d  yMin=%d\n", ppem, height, font->yMax, font->yMin); | 
 | 		    break; | 
 | 		} | 
 | 	    } | 
 | 	    if(!font->yMax) { | 
 | 		ppem = 0; | 
 | 		TRACE("ppem not found for height %ld\n", height); | 
 | 	    } | 
 | 	} else { | 
 | 	    ppem = -height; | 
 | 	    if(ppem < startsz || ppem > endsz) | 
 | 		goto end; | 
 |  | 
 | 	    for(i = 0; i < recs; i++) { | 
 | 		USHORT yPelHeight; | 
 | 		yPelHeight = GET_BE_WORD(&vTable[i * 6]); | 
 |  | 
 | 		if(yPelHeight > ppem) | 
 | 		    break; /* failed */ | 
 |  | 
 | 		if(yPelHeight == ppem) { | 
 | 		    font->yMax = GET_BE_WORD(&vTable[(i * 6) + 2]); | 
 | 		    font->yMin = GET_BE_WORD(&vTable[(i * 6) + 4]); | 
 | 		    TRACE("ppem %ld found; yMax=%d  yMin=%d\n", ppem, font->yMax, font->yMin); | 
 | 		    break; | 
 | 		} | 
 | 	    } | 
 | 	} | 
 | 	end: | 
 | 	HeapFree(GetProcessHeap(), 0, vTable); | 
 |     } | 
 |  | 
 |     return ppem; | 
 | } | 
 |  | 
 | static BOOL fontcmp(GdiFont font, FONT_DESC *fd) | 
 | { | 
 |     if(font->font_desc.hash != fd->hash) return TRUE; | 
 |     if(memcmp(&font->font_desc.matrix, &fd->matrix, sizeof(fd->matrix))) return TRUE; | 
 |     if(memcmp(&font->font_desc.lf, &fd->lf, offsetof(LOGFONTW, lfFaceName))) return TRUE; | 
 |     return strcmpiW(font->font_desc.lf.lfFaceName, fd->lf.lfFaceName); | 
 | } | 
 |  | 
 | static void calc_hash(FONT_DESC *pfd) | 
 | { | 
 |     DWORD hash = 0, *ptr, two_chars; | 
 |     WORD *pwc; | 
 |     unsigned int i; | 
 |  | 
 |     for(i = 0, ptr = (DWORD*)&pfd->matrix; i < sizeof(FMAT2)/sizeof(DWORD); i++, ptr++) | 
 |         hash ^= *ptr; | 
 |     for(i = 0, ptr = (DWORD*)&pfd->lf; i < 7; i++, ptr++) | 
 |         hash ^= *ptr; | 
 |     for(i = 0, ptr = (DWORD*)&pfd->lf.lfFaceName; i < LF_FACESIZE/2; i++, ptr++) { | 
 |         two_chars = *ptr; | 
 |         pwc = (WCHAR *)&two_chars; | 
 |         if(!*pwc) break; | 
 |         *pwc = toupperW(*pwc); | 
 |         pwc++; | 
 |         *pwc = toupperW(*pwc); | 
 |         hash ^= two_chars; | 
 |         if(!*pwc) break; | 
 |     } | 
 |     pfd->hash = hash; | 
 |     return; | 
 | } | 
 |  | 
 | static GdiFont find_in_cache(HFONT hfont, LOGFONTW *plf, XFORM *pxf, BOOL can_use_bitmap) | 
 | { | 
 |     GdiFont ret; | 
 |     FONT_DESC fd; | 
 |     HFONTLIST *hflist; | 
 |     struct list *font_elem_ptr, *hfontlist_elem_ptr; | 
 |  | 
 |     memcpy(&fd.lf, plf, sizeof(LOGFONTW)); | 
 |     memcpy(&fd.matrix, pxf, sizeof(FMAT2)); | 
 |     calc_hash(&fd); | 
 |  | 
 |     /* try the in-use list */ | 
 |     LIST_FOR_EACH(font_elem_ptr, &gdi_font_list) { | 
 |         ret = LIST_ENTRY(font_elem_ptr, struct tagGdiFont, entry); | 
 |         if(!fontcmp(ret, &fd)) { | 
 |             if(!can_use_bitmap && !FT_IS_SCALABLE(ret->ft_face)) continue; | 
 |             LIST_FOR_EACH(hfontlist_elem_ptr, &ret->hfontlist) { | 
 |                 hflist = LIST_ENTRY(hfontlist_elem_ptr, struct tagHFONTLIST, entry); | 
 |                 if(hflist->hfont == hfont) | 
 |                     return ret; | 
 |             } | 
 |             hflist = HeapAlloc(GetProcessHeap(), 0, sizeof(*hflist)); | 
 |             hflist->hfont = hfont; | 
 |             list_add_head(&ret->hfontlist, &hflist->entry); | 
 |             return ret; | 
 |         } | 
 |     } | 
 |   | 
 |     /* then the unused list */ | 
 |     font_elem_ptr = list_head(&unused_gdi_font_list); | 
 |     while(font_elem_ptr) { | 
 |         ret = LIST_ENTRY(font_elem_ptr, struct tagGdiFont, entry); | 
 |         font_elem_ptr = list_next(&unused_gdi_font_list, font_elem_ptr); | 
 |         if(!fontcmp(ret, &fd)) { | 
 |             if(!can_use_bitmap && !FT_IS_SCALABLE(ret->ft_face)) continue; | 
 |             assert(list_empty(&ret->hfontlist)); | 
 |             TRACE("Found %p in unused list\n", ret); | 
 |             list_remove(&ret->entry); | 
 |             list_add_head(&gdi_font_list, &ret->entry); | 
 |             hflist = HeapAlloc(GetProcessHeap(), 0, sizeof(*hflist)); | 
 |             hflist->hfont = hfont; | 
 |             list_add_head(&ret->hfontlist, &hflist->entry); | 
 |             return ret; | 
 |         } | 
 |     } | 
 |     return NULL; | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  * WineEngCreateFontInstance | 
 |  * | 
 |  */ | 
 | GdiFont WineEngCreateFontInstance(DC *dc, HFONT hfont) | 
 | { | 
 |     GdiFont ret; | 
 |     Face *face, *best; | 
 |     Family *family; | 
 |     struct list *family_elem_ptr, *face_elem_ptr; | 
 |     INT height, width = 0; | 
 |     signed int diff = 0, newdiff; | 
 |     BOOL bd, it, can_use_bitmap; | 
 |     LOGFONTW lf; | 
 |     CHARSETINFO csi; | 
 |     HFONTLIST *hflist; | 
 |  | 
 |     if (!GetObjectW( hfont, sizeof(lf), &lf )) return NULL; | 
 |     can_use_bitmap = GetDeviceCaps(dc->hSelf, TEXTCAPS) & TC_RA_ABLE; | 
 |  | 
 |     TRACE("%s, h=%ld, it=%d, weight=%ld, PandF=%02x, charset=%d orient %ld escapement %ld\n", | 
 | 	  debugstr_w(lf.lfFaceName), lf.lfHeight, lf.lfItalic, | 
 | 	  lf.lfWeight, lf.lfPitchAndFamily, lf.lfCharSet, lf.lfOrientation, | 
 | 	  lf.lfEscapement); | 
 |  | 
 |     /* check the cache first */ | 
 |     if((ret = find_in_cache(hfont, &lf, &dc->xformWorld2Vport, can_use_bitmap)) != NULL) { | 
 |         TRACE("returning cached gdiFont(%p) for hFont %p\n", ret, hfont); | 
 |         return ret; | 
 |     } | 
 |  | 
 |     TRACE("not in cache\n"); | 
 |     if(list_empty(&font_list) || !have_installed_roman_font) /* No fonts installed */ | 
 |     { | 
 | 	TRACE("No fonts installed\n"); | 
 | 	return NULL; | 
 |     } | 
 |  | 
 |     ret = alloc_font(); | 
 |  | 
 |      memcpy(&ret->font_desc.matrix, &dc->xformWorld2Vport, sizeof(FMAT2)); | 
 |      memcpy(&ret->font_desc.lf, &lf, sizeof(LOGFONTW)); | 
 |      calc_hash(&ret->font_desc); | 
 |      hflist = HeapAlloc(GetProcessHeap(), 0, sizeof(*hflist)); | 
 |      hflist->hfont = hfont; | 
 |      list_add_head(&ret->hfontlist, &hflist->entry); | 
 |  | 
 |  | 
 |     /* If lfFaceName is "Symbol" then Windows fixes up lfCharSet to | 
 |        SYMBOL_CHARSET so that Symbol gets picked irrespective of the | 
 |        original value lfCharSet.  Note this is a special case for | 
 |        Symbol and doesn't happen at least for "Wingdings*" */ | 
 |  | 
 |     if(!strcmpiW(lf.lfFaceName, SymbolW)) | 
 |         lf.lfCharSet = SYMBOL_CHARSET; | 
 |  | 
 |     if(!TranslateCharsetInfo((DWORD*)(INT)lf.lfCharSet, &csi, TCI_SRCCHARSET)) { | 
 |         switch(lf.lfCharSet) { | 
 | 	case DEFAULT_CHARSET: | 
 | 	    csi.fs.fsCsb[0] = 0; | 
 | 	    break; | 
 | 	default: | 
 | 	    FIXME("Untranslated charset %d\n", lf.lfCharSet); | 
 | 	    csi.fs.fsCsb[0] = 0; | 
 | 	    break; | 
 | 	} | 
 |     } | 
 |  | 
 |     family = NULL; | 
 |     if(lf.lfFaceName[0] != '\0') { | 
 |         FontSubst *psub; | 
 | 	for(psub = substlist; psub; psub = psub->next) | 
 | 	    if(!strcmpiW(lf.lfFaceName, psub->from.name) && | 
 | 	       (psub->from.charset == -1 || | 
 | 		psub->from.charset == lf.lfCharSet)) | 
 | 	      break; | 
 | 	if(psub) { | 
 | 	    TRACE("substituting %s -> %s\n", debugstr_w(lf.lfFaceName), | 
 | 		  debugstr_w(psub->to.name)); | 
 | 	    strcpyW(lf.lfFaceName, psub->to.name); | 
 | 	} | 
 |  | 
 | 	/* We want a match on name and charset or just name if | 
 | 	   charset was DEFAULT_CHARSET.  If the latter then | 
 | 	   we fixup the returned charset later in get_nearest_charset | 
 | 	   where we'll either use the charset of the current ansi codepage | 
 | 	   or if that's unavailable the first charset that the font supports. | 
 | 	*/ | 
 |         LIST_FOR_EACH(family_elem_ptr, &font_list) { | 
 |             family = LIST_ENTRY(family_elem_ptr, Family, entry); | 
 | 	    if(!strcmpiW(family->FamilyName, lf.lfFaceName)) { | 
 |                 face_elem_ptr = list_head(&family->faces);  | 
 |                 face = LIST_ENTRY(face_elem_ptr, Face, entry); | 
 | 	        if((csi.fs.fsCsb[0] & face->fs.fsCsb[0]) || !csi.fs.fsCsb[0]) | 
 |                     if(face->scalable || can_use_bitmap) | 
 |                         break; | 
 |             } | 
 |             family = NULL; | 
 | 	} | 
 |     } | 
 |  | 
 |     if(!family) { | 
 |       /* If requested charset was DEFAULT_CHARSET then try using charset | 
 | 	 corresponding to the current ansi codepage */ | 
 |         if(!csi.fs.fsCsb[0]) { | 
 | 	    INT acp = GetACP(); | 
 | 	    if(!TranslateCharsetInfo((DWORD*)acp, &csi, TCI_SRCCODEPAGE)) { | 
 | 	        FIXME("TCI failed on codepage %d\n", acp); | 
 | 		csi.fs.fsCsb[0] = 0; | 
 | 	    } else | 
 | 	        lf.lfCharSet = csi.ciCharset; | 
 | 	} | 
 |  | 
 | 		/* Face families are in the top 4 bits of lfPitchAndFamily, | 
 | 		   so mask with 0xF0 before testing */ | 
 |  | 
 | 	if((lf.lfPitchAndFamily & FIXED_PITCH) || | 
 | 	   (lf.lfPitchAndFamily & 0xF0) == FF_MODERN) | 
 | 	  strcpyW(lf.lfFaceName, defFixed); | 
 | 	else if((lf.lfPitchAndFamily & 0xF0) == FF_ROMAN) | 
 | 	  strcpyW(lf.lfFaceName, defSerif); | 
 | 	else if((lf.lfPitchAndFamily & 0xF0) == FF_SWISS) | 
 | 	  strcpyW(lf.lfFaceName, defSans); | 
 | 	else | 
 | 	  strcpyW(lf.lfFaceName, defSans); | 
 |         LIST_FOR_EACH(family_elem_ptr, &font_list) { | 
 |             family = LIST_ENTRY(family_elem_ptr, Family, entry); | 
 | 	    if(!strcmpiW(family->FamilyName, lf.lfFaceName)) { | 
 |                 face_elem_ptr = list_head(&family->faces);  | 
 |                 face = LIST_ENTRY(face_elem_ptr, Face, entry); | 
 |                 if(csi.fs.fsCsb[0] & face->fs.fsCsb[0]) | 
 |                     if(face->scalable || can_use_bitmap) | 
 |                         break; | 
 |             } | 
 |             family = NULL; | 
 | 	} | 
 |     } | 
 |  | 
 |     if(!family) { | 
 |         LIST_FOR_EACH(family_elem_ptr, &font_list) { | 
 |             family = LIST_ENTRY(family_elem_ptr, Family, entry); | 
 |             face_elem_ptr = list_head(&family->faces);  | 
 |             face = LIST_ENTRY(face_elem_ptr, Face, entry); | 
 | 	    if(csi.fs.fsCsb[0] & face->fs.fsCsb[0]) | 
 |                 if(face->scalable || can_use_bitmap) | 
 |                     break; | 
 |             family = NULL; | 
 | 	} | 
 |     } | 
 |  | 
 |     if(!family) { | 
 |         LIST_FOR_EACH(family_elem_ptr, &font_list) { | 
 |             family = LIST_ENTRY(family_elem_ptr, Family, entry); | 
 |             face_elem_ptr = list_head(&family->faces);  | 
 |             face = LIST_ENTRY(face_elem_ptr, Face, entry); | 
 |             if(face->scalable || can_use_bitmap) { | 
 |                 csi.fs.fsCsb[0] = 0; | 
 |                 FIXME("just using first face for now\n"); | 
 |                 break; | 
 |             } | 
 |             family = NULL; | 
 |         } | 
 |         if(!family) { | 
 |             FIXME("can't find a single appropriate font - bailing\n"); | 
 |             free_font(ret); | 
 |             return NULL; | 
 |         } | 
 |     } | 
 |  | 
 |     it = lf.lfItalic ? 1 : 0; | 
 |     bd = lf.lfWeight > 550 ? 1 : 0; | 
 |  | 
 |     height = GDI_ROUND( (FLOAT)lf.lfHeight * dc->xformWorld2Vport.eM22 ); | 
 |     height = lf.lfHeight < 0 ? -abs(height) : abs(height); | 
 |  | 
 |     face = best = NULL; | 
 |     LIST_FOR_EACH(face_elem_ptr, &family->faces) { | 
 |         face = LIST_ENTRY(face_elem_ptr, Face, entry); | 
 |         if(!(face->Italic ^ it) && !(face->Bold ^ bd)) { | 
 |             if(face->scalable) | 
 |                 break; | 
 |             if(height > 0) | 
 |                 newdiff = height - (signed int)(face->size.y_ppem >> 6); | 
 |             else | 
 |                 newdiff = -height - ((signed int)(face->size.y_ppem >> 6) - face->size.internal_leading); | 
 |             if(!best || (diff > 0 && newdiff < diff && newdiff >= 0) || | 
 |                (diff < 0 && newdiff > diff)) { | 
 |                 TRACE("%ld is better for %d diff was %d\n", face->size.y_ppem >> 6, height, diff); | 
 |                 diff = newdiff; | 
 |                 best = face; | 
 |                 if(diff == 0) | 
 |                     break; | 
 |             } | 
 |         } | 
 |         face = NULL; | 
 |     } | 
 |     if(!face && best) | 
 |         face = best; | 
 |     else if(!face) { | 
 |         best = NULL; | 
 |         LIST_FOR_EACH(face_elem_ptr, &family->faces) { | 
 |             face = LIST_ENTRY(face_elem_ptr, Face, entry); | 
 |             if(face->scalable) | 
 |                 break; | 
 |             if(height > 0) | 
 |                 newdiff = height - (signed int)(face->size.y_ppem >> 6); | 
 |             else | 
 |                 newdiff = -height - ((signed int)(face->size.y_ppem >> 6) - face->size.internal_leading); | 
 |             if(!best || (diff > 0 && newdiff < diff && newdiff >= 0) || | 
 |                (diff < 0 && newdiff > diff)) { | 
 |                 TRACE("%ld is better for %d diff was %d\n", face->size.y_ppem >> 6, height, diff); | 
 |                 diff = newdiff; | 
 |                 best = face; | 
 |                 if(diff == 0) | 
 |                     break; | 
 |             } | 
 |             face = NULL; | 
 |         } | 
 |         if(!face && best) | 
 |             face = best; | 
 | 	if(it && !face->Italic) ret->fake_italic = TRUE; | 
 | 	if(bd && !face->Bold) ret->fake_bold = TRUE; | 
 |     } | 
 |  | 
 |     memcpy(&ret->fs, &face->fs, sizeof(FONTSIGNATURE)); | 
 |  | 
 |     if(csi.fs.fsCsb[0]) { | 
 |         ret->charset = lf.lfCharSet; | 
 |         ret->codepage = csi.ciACP; | 
 |     } | 
 |     else | 
 |         ret->charset = get_nearest_charset(face, &ret->codepage); | 
 |  | 
 |     TRACE("Chosen: %s %s\n", debugstr_w(family->FamilyName), | 
 | 	  debugstr_w(face->StyleName)); | 
 |  | 
 |     if(!face->scalable) { | 
 |         width = face->size.x_ppem >> 6; | 
 |         height = face->size.y_ppem >> 6; | 
 |     } | 
 |     ret->ft_face = OpenFontFile(ret, face->file, face->face_index, width, height); | 
 |  | 
 |     if (!ret->ft_face) | 
 |     { | 
 |         free_font( ret ); | 
 |         return 0; | 
 |     } | 
 |  | 
 |     if (ret->charset == SYMBOL_CHARSET &&  | 
 |         !pFT_Select_Charmap(ret->ft_face, FT_ENCODING_MS_SYMBOL)) { | 
 |         /* No ops */ | 
 |     } | 
 |     else if (!pFT_Select_Charmap(ret->ft_face, FT_ENCODING_UNICODE)) { | 
 |         /* No ops */ | 
 |     } | 
 |     else { | 
 |         pFT_Select_Charmap(ret->ft_face, FT_ENCODING_APPLE_ROMAN); | 
 |     } | 
 |  | 
 |     ret->orientation = FT_IS_SCALABLE(ret->ft_face) ? lf.lfOrientation : 0; | 
 |     ret->name = strdupW(family->FamilyName); | 
 |     ret->underline = lf.lfUnderline ? 0xff : 0; | 
 |     ret->strikeout = lf.lfStrikeOut ? 0xff : 0; | 
 |  | 
 |     TRACE("caching: gdiFont=%p  hfont=%p\n", ret, hfont); | 
 |  | 
 |     ret->aveWidth = FT_IS_SCALABLE(ret->ft_face) ? lf.lfWidth : 0; | 
 |     list_add_head(&gdi_font_list, &ret->entry); | 
 |     return ret; | 
 | } | 
 |  | 
 | static void dump_gdi_font_list(void) | 
 | { | 
 |     GdiFont gdiFont; | 
 |     struct list *elem_ptr; | 
 |  | 
 |     TRACE("---------- gdiFont Cache ----------\n"); | 
 |     LIST_FOR_EACH(elem_ptr, &gdi_font_list) { | 
 |         gdiFont = LIST_ENTRY(elem_ptr, struct tagGdiFont, entry); | 
 |         TRACE("gdiFont=%p %s %ld\n", | 
 |               gdiFont, debugstr_w(gdiFont->font_desc.lf.lfFaceName), gdiFont->font_desc.lf.lfHeight); | 
 |     } | 
 |  | 
 |     TRACE("---------- Unused gdiFont Cache ----------\n"); | 
 |     LIST_FOR_EACH(elem_ptr, &unused_gdi_font_list) { | 
 |         gdiFont = LIST_ENTRY(elem_ptr, struct tagGdiFont, entry); | 
 |         TRACE("gdiFont=%p %s %ld\n", | 
 |               gdiFont, debugstr_w(gdiFont->font_desc.lf.lfFaceName), gdiFont->font_desc.lf.lfHeight); | 
 |     } | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  * WineEngDestroyFontInstance | 
 |  * | 
 |  * free the gdiFont associated with this handle | 
 |  * | 
 |  */ | 
 | BOOL WineEngDestroyFontInstance(HFONT handle) | 
 | { | 
 |     GdiFont gdiFont; | 
 |     HFONTLIST *hflist; | 
 |     BOOL ret = FALSE; | 
 |     struct list *font_elem_ptr, *hfontlist_elem_ptr; | 
 |     int i = 0; | 
 |  | 
 |     TRACE("destroying hfont=%p\n", handle); | 
 |     if(TRACE_ON(font)) | 
 | 	dump_gdi_font_list(); | 
 |  | 
 |     font_elem_ptr = list_head(&gdi_font_list); | 
 |     while(font_elem_ptr) { | 
 |         gdiFont = LIST_ENTRY(font_elem_ptr, struct tagGdiFont, entry); | 
 |         font_elem_ptr = list_next(&gdi_font_list, font_elem_ptr); | 
 |  | 
 |         hfontlist_elem_ptr = list_head(&gdiFont->hfontlist); | 
 |         while(hfontlist_elem_ptr) { | 
 |             hflist = LIST_ENTRY(hfontlist_elem_ptr, struct tagHFONTLIST, entry); | 
 |             hfontlist_elem_ptr = list_next(&gdiFont->hfontlist, hfontlist_elem_ptr); | 
 |             if(hflist->hfont == handle) { | 
 |                 list_remove(&hflist->entry); | 
 |                 HeapFree(GetProcessHeap(), 0, hflist); | 
 |                 ret = TRUE; | 
 |             } | 
 |         } | 
 |         if(list_empty(&gdiFont->hfontlist)) { | 
 |             TRACE("Moving to Unused list\n"); | 
 |             list_remove(&gdiFont->entry); | 
 |             list_add_head(&unused_gdi_font_list, &gdiFont->entry); | 
 |         } | 
 |     } | 
 |  | 
 |  | 
 |     font_elem_ptr = list_head(&unused_gdi_font_list); | 
 |     while(font_elem_ptr && i++ < UNUSED_CACHE_SIZE) | 
 |         font_elem_ptr = list_next(&unused_gdi_font_list, font_elem_ptr); | 
 |     while(font_elem_ptr) { | 
 |         gdiFont = LIST_ENTRY(font_elem_ptr, struct tagGdiFont, entry); | 
 |         font_elem_ptr = list_next(&unused_gdi_font_list, font_elem_ptr); | 
 |         TRACE("freeing %p\n", gdiFont); | 
 |         list_remove(&gdiFont->entry); | 
 |         free_font(gdiFont); | 
 |     } | 
 |     return ret; | 
 | } | 
 |  | 
 | static void GetEnumStructs(Face *face, LPENUMLOGFONTEXW pelf, | 
 | 			   NEWTEXTMETRICEXW *pntm, LPDWORD ptype) | 
 | { | 
 |     OUTLINETEXTMETRICW *potm = NULL; | 
 |     UINT size; | 
 |     TEXTMETRICW tm, *ptm; | 
 |     GdiFont font = alloc_font(); | 
 |     LONG width, height; | 
 |  | 
 |     if(face->scalable) { | 
 |         height = 100; | 
 |         width = 0; | 
 |     } else { | 
 |         height = face->size.y_ppem >> 6; | 
 |         width = face->size.x_ppem >> 6; | 
 |     } | 
 |      | 
 |     if (!(font->ft_face = OpenFontFile(font, face->file, face->face_index, width, height))) | 
 |     { | 
 |         free_font(font); | 
 |         return; | 
 |     } | 
 |  | 
 |     font->name = strdupW(face->family->FamilyName); | 
 |  | 
 |     memset(&pelf->elfLogFont, 0, sizeof(LOGFONTW)); | 
 |  | 
 |     size = WineEngGetOutlineTextMetrics(font, 0, NULL); | 
 |     if(size) { | 
 |         potm = HeapAlloc(GetProcessHeap(), 0, size); | 
 |         WineEngGetOutlineTextMetrics(font, size, potm); | 
 |         ptm = (TEXTMETRICW*)&potm->otmTextMetrics; | 
 |     } else { | 
 |         WineEngGetTextMetrics(font, &tm); | 
 |         ptm = &tm; | 
 |     } | 
 |          | 
 |     pntm->ntmTm.tmHeight = pelf->elfLogFont.lfHeight = ptm->tmHeight; | 
 |     pntm->ntmTm.tmAscent = ptm->tmAscent; | 
 |     pntm->ntmTm.tmDescent = ptm->tmDescent; | 
 |     pntm->ntmTm.tmInternalLeading = ptm->tmInternalLeading; | 
 |     pntm->ntmTm.tmExternalLeading = ptm->tmExternalLeading; | 
 |     pntm->ntmTm.tmAveCharWidth = pelf->elfLogFont.lfWidth = ptm->tmAveCharWidth; | 
 |     pntm->ntmTm.tmMaxCharWidth = ptm->tmMaxCharWidth; | 
 |     pntm->ntmTm.tmWeight = pelf->elfLogFont.lfWeight = ptm->tmWeight; | 
 |     pntm->ntmTm.tmOverhang = ptm->tmOverhang; | 
 |     pntm->ntmTm.tmDigitizedAspectX = ptm->tmDigitizedAspectX; | 
 |     pntm->ntmTm.tmDigitizedAspectY = ptm->tmDigitizedAspectY; | 
 |     pntm->ntmTm.tmFirstChar = ptm->tmFirstChar; | 
 |     pntm->ntmTm.tmLastChar = ptm->tmLastChar; | 
 |     pntm->ntmTm.tmDefaultChar = ptm->tmDefaultChar; | 
 |     pntm->ntmTm.tmBreakChar = ptm->tmBreakChar; | 
 |     pntm->ntmTm.tmItalic = pelf->elfLogFont.lfItalic = ptm->tmItalic; | 
 |     pntm->ntmTm.tmUnderlined = pelf->elfLogFont.lfUnderline = ptm->tmUnderlined; | 
 |     pntm->ntmTm.tmStruckOut = pelf->elfLogFont.lfStrikeOut = ptm->tmStruckOut; | 
 |     pntm->ntmTm.tmPitchAndFamily = ptm->tmPitchAndFamily; | 
 |     pelf->elfLogFont.lfPitchAndFamily = (ptm->tmPitchAndFamily & 0xf1) + 1; | 
 |     pntm->ntmTm.tmCharSet = pelf->elfLogFont.lfCharSet = ptm->tmCharSet; | 
 |     pelf->elfLogFont.lfOutPrecision = OUT_STROKE_PRECIS; | 
 |     pelf->elfLogFont.lfClipPrecision = CLIP_STROKE_PRECIS; | 
 |     pelf->elfLogFont.lfQuality = DRAFT_QUALITY; | 
 |  | 
 |     *ptype = ptm->tmPitchAndFamily & TMPF_TRUETYPE ? TRUETYPE_FONTTYPE : 0; | 
 |     if(!(ptm->tmPitchAndFamily & TMPF_VECTOR)) | 
 |         *ptype |= RASTER_FONTTYPE; | 
 |  | 
 |     pntm->ntmTm.ntmFlags = ptm->tmItalic ? NTM_ITALIC : 0; | 
 |     if(ptm->tmWeight > 550) pntm->ntmTm.ntmFlags |= NTM_BOLD; | 
 |     if(pntm->ntmTm.ntmFlags == 0) pntm->ntmTm.ntmFlags = NTM_REGULAR; | 
 |  | 
 |     pntm->ntmTm.ntmCellHeight = pntm->ntmTm.tmHeight; | 
 |     pntm->ntmTm.ntmAvgWidth = pntm->ntmTm.tmAveCharWidth; | 
 |     memset(&pntm->ntmFontSig, 0, sizeof(FONTSIGNATURE)); | 
 |  | 
 |     if(potm) { | 
 |         pntm->ntmTm.ntmSizeEM = potm->otmEMSquare; | 
 |  | 
 |         strncpyW(pelf->elfLogFont.lfFaceName, | 
 |                  (WCHAR*)((char*)potm + (ptrdiff_t)potm->otmpFamilyName), | 
 |                  LF_FACESIZE); | 
 |         strncpyW(pelf->elfFullName, | 
 |                  (WCHAR*)((char*)potm + (ptrdiff_t)potm->otmpFaceName), | 
 |                  LF_FULLFACESIZE); | 
 |         strncpyW(pelf->elfStyle, | 
 |                  (WCHAR*)((char*)potm + (ptrdiff_t)potm->otmpStyleName), | 
 |                  LF_FACESIZE); | 
 |  | 
 |         HeapFree(GetProcessHeap(), 0, potm); | 
 |     } else { | 
 |         pntm->ntmTm.ntmSizeEM = pntm->ntmTm.tmHeight - pntm->ntmTm.tmInternalLeading; | 
 |  | 
 |         strncpyW(pelf->elfLogFont.lfFaceName, face->family->FamilyName, LF_FACESIZE); | 
 |         strncpyW(pelf->elfFullName, face->family->FamilyName, LF_FACESIZE); | 
 |         pelf->elfStyle[0] = '\0'; | 
 |     } | 
 |  | 
 |     pelf->elfScript[0] = '\0'; /* This will get set in WineEngEnumFonts */ | 
 |  | 
 |     free_font(font); | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  * WineEngEnumFonts | 
 |  * | 
 |  */ | 
 | DWORD WineEngEnumFonts(LPLOGFONTW plf, FONTENUMPROCW proc, LPARAM lparam) | 
 | { | 
 |     Family *family; | 
 |     Face *face; | 
 |     struct list *family_elem_ptr, *face_elem_ptr; | 
 |     ENUMLOGFONTEXW elf; | 
 |     NEWTEXTMETRICEXW ntm; | 
 |     DWORD type, ret = 1; | 
 |     FONTSIGNATURE fs; | 
 |     CHARSETINFO csi; | 
 |     LOGFONTW lf; | 
 |     int i; | 
 |  | 
 |     TRACE("facename = %s charset %d\n", debugstr_w(plf->lfFaceName), plf->lfCharSet); | 
 |  | 
 |     if(plf->lfFaceName[0]) { | 
 |         FontSubst *psub; | 
 |         for(psub = substlist; psub; psub = psub->next) | 
 |             if(!strcmpiW(plf->lfFaceName, psub->from.name) && | 
 |                (psub->from.charset == -1 || | 
 |                 psub->from.charset == plf->lfCharSet)) | 
 |                 break; | 
 |         if(psub) { | 
 |             TRACE("substituting %s -> %s\n", debugstr_w(plf->lfFaceName), | 
 |                   debugstr_w(psub->to.name)); | 
 |             memcpy(&lf, plf, sizeof(lf)); | 
 |             strcpyW(lf.lfFaceName, psub->to.name); | 
 |             plf = &lf; | 
 |         } | 
 |  | 
 |         LIST_FOR_EACH(family_elem_ptr, &font_list) { | 
 |             family = LIST_ENTRY(family_elem_ptr, Family, entry); | 
 |             if(!strcmpiW(plf->lfFaceName, family->FamilyName)) { | 
 |                 LIST_FOR_EACH(face_elem_ptr, &family->faces) { | 
 |                     face = LIST_ENTRY(face_elem_ptr, Face, entry); | 
 |                     GetEnumStructs(face, &elf, &ntm, &type); | 
 |                     for(i = 0; i < 32; i++) { | 
 |                         if(!face->scalable && face->fs.fsCsb[0] == 0) { /* OEM bitmap */ | 
 |                             elf.elfLogFont.lfCharSet = ntm.ntmTm.tmCharSet = OEM_CHARSET; | 
 |                             strcpyW(elf.elfScript, OEM_DOSW); | 
 |                             i = 32; /* break out of loop */ | 
 |                         } else if(!(face->fs.fsCsb[0] & (1L << i))) | 
 |                             continue; | 
 |                         else { | 
 |                             fs.fsCsb[0] = 1L << i; | 
 |                             fs.fsCsb[1] = 0; | 
 |                             if(!TranslateCharsetInfo(fs.fsCsb, &csi, | 
 |                                                      TCI_SRCFONTSIG)) | 
 |                                 csi.ciCharset = DEFAULT_CHARSET; | 
 |                             if(i == 31) csi.ciCharset = SYMBOL_CHARSET; | 
 |                             if(csi.ciCharset != DEFAULT_CHARSET) { | 
 |                                 elf.elfLogFont.lfCharSet = | 
 |                                     ntm.ntmTm.tmCharSet = csi.ciCharset; | 
 |                                 if(ElfScriptsW[i]) | 
 |                                     strcpyW(elf.elfScript, ElfScriptsW[i]); | 
 |                                 else | 
 |                                     FIXME("Unknown elfscript for bit %d\n", i); | 
 |                             } | 
 |                         } | 
 |                         TRACE("enuming face %s full %s style %s charset %d type %ld script %s it %d weight %ld ntmflags %08lx\n", | 
 |                               debugstr_w(elf.elfLogFont.lfFaceName), | 
 |                               debugstr_w(elf.elfFullName), debugstr_w(elf.elfStyle), | 
 |                               csi.ciCharset, type, debugstr_w(elf.elfScript), | 
 |                               elf.elfLogFont.lfItalic, elf.elfLogFont.lfWeight, | 
 |                               ntm.ntmTm.ntmFlags); | 
 |                         ret = proc(&elf.elfLogFont, (TEXTMETRICW *)&ntm, type, lparam); | 
 |                         if(!ret) goto end; | 
 | 		    } | 
 | 		} | 
 | 	    } | 
 | 	} | 
 |     } else { | 
 |         LIST_FOR_EACH(family_elem_ptr, &font_list) { | 
 |             family = LIST_ENTRY(family_elem_ptr, Family, entry); | 
 |             face_elem_ptr = list_head(&family->faces); | 
 |             face = LIST_ENTRY(face_elem_ptr, Face, entry); | 
 |             GetEnumStructs(face, &elf, &ntm, &type); | 
 |             for(i = 0; i < 32; i++) { | 
 |                 if(!face->scalable && face->fs.fsCsb[0] == 0) { /* OEM bitmap */ | 
 |                     elf.elfLogFont.lfCharSet = ntm.ntmTm.tmCharSet = OEM_CHARSET; | 
 |                     strcpyW(elf.elfScript, OEM_DOSW); | 
 |                     i = 32; /* break out of loop */ | 
 | 	        } else if(!(face->fs.fsCsb[0] & (1L << i))) | 
 |                     continue; | 
 |                 else { | 
 | 		    fs.fsCsb[0] = 1L << i; | 
 | 		    fs.fsCsb[1] = 0; | 
 | 		    if(!TranslateCharsetInfo(fs.fsCsb, &csi, | 
 | 					     TCI_SRCFONTSIG)) | 
 | 		        csi.ciCharset = DEFAULT_CHARSET; | 
 | 		    if(i == 31) csi.ciCharset = SYMBOL_CHARSET; | 
 | 		    if(csi.ciCharset != DEFAULT_CHARSET) { | 
 | 		        elf.elfLogFont.lfCharSet = ntm.ntmTm.tmCharSet = | 
 | 			  csi.ciCharset; | 
 | 			  if(ElfScriptsW[i]) | 
 | 			      strcpyW(elf.elfScript, ElfScriptsW[i]); | 
 | 			  else | 
 | 			      FIXME("Unknown elfscript for bit %d\n", i); | 
 |                     } | 
 |                 } | 
 |                 TRACE("enuming face %s full %s style %s charset = %d type %ld script %s it %d weight %ld ntmflags %08lx\n", | 
 |                       debugstr_w(elf.elfLogFont.lfFaceName), | 
 |                       debugstr_w(elf.elfFullName), debugstr_w(elf.elfStyle), | 
 |                       csi.ciCharset, type, debugstr_w(elf.elfScript), | 
 |                       elf.elfLogFont.lfItalic, elf.elfLogFont.lfWeight, | 
 |                       ntm.ntmTm.ntmFlags); | 
 |                 ret = proc(&elf.elfLogFont, (TEXTMETRICW *)&ntm, type, lparam); | 
 |                 if(!ret) goto end; | 
 | 	    } | 
 | 	} | 
 |     } | 
 | end: | 
 |     return ret; | 
 | } | 
 |  | 
 | static void FTVectorToPOINTFX(FT_Vector *vec, POINTFX *pt) | 
 | { | 
 |     pt->x.value = vec->x >> 6; | 
 |     pt->x.fract = (vec->x & 0x3f) << 10; | 
 |     pt->x.fract |= ((pt->x.fract >> 6) | (pt->x.fract >> 12)); | 
 |     pt->y.value = vec->y >> 6; | 
 |     pt->y.fract = (vec->y & 0x3f) << 10; | 
 |     pt->y.fract |= ((pt->y.fract >> 6) | (pt->y.fract >> 12)); | 
 |     return; | 
 | } | 
 |  | 
 | static FT_UInt get_glyph_index(GdiFont font, UINT glyph) | 
 | { | 
 |     if(font->ft_face->charmap->encoding == FT_ENCODING_NONE) { | 
 |         WCHAR wc = (WCHAR)glyph; | 
 |         unsigned char buf; | 
 |         WideCharToMultiByte(font->codepage, 0, &wc, 1, &buf, sizeof(buf), 0, 0); | 
 |         return pFT_Get_Char_Index(font->ft_face, buf); | 
 |     } | 
 |  | 
 |     if(font->charset == SYMBOL_CHARSET && glyph < 0x100) | 
 |         glyph = glyph + 0xf000; | 
 |     return pFT_Get_Char_Index(font->ft_face, glyph); | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  * WineEngGetGlyphIndices | 
 |  * | 
 |  * FIXME: add support for GGI_MARK_NONEXISTING_GLYPHS | 
 |  */ | 
 | DWORD WineEngGetGlyphIndices(GdiFont font, LPCWSTR lpstr, INT count, | 
 | 				LPWORD pgi, DWORD flags) | 
 | { | 
 |     INT i; | 
 |  | 
 |     for(i = 0; i < count; i++) | 
 |         pgi[i] = get_glyph_index(font, lpstr[i]); | 
 |  | 
 |     return count; | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  * WineEngGetGlyphOutline | 
 |  * | 
 |  * Behaves in exactly the same way as the win32 api GetGlyphOutline | 
 |  * except that the first parameter is the HWINEENGFONT of the font in | 
 |  * question rather than an HDC. | 
 |  * | 
 |  */ | 
 | DWORD WineEngGetGlyphOutline(GdiFont font, UINT glyph, UINT format, | 
 | 			     LPGLYPHMETRICS lpgm, DWORD buflen, LPVOID buf, | 
 | 			     const MAT2* lpmat) | 
 | { | 
 |     static const FT_Matrix identityMat = {(1 << 16), 0, 0, (1 << 16)}; | 
 |     FT_Face ft_face = font->ft_face; | 
 |     FT_UInt glyph_index; | 
 |     DWORD width, height, pitch, needed = 0; | 
 |     FT_Bitmap ft_bitmap; | 
 |     FT_Error err; | 
 |     INT left, right, top = 0, bottom = 0; | 
 |     FT_Angle angle = 0; | 
 |     FT_Int load_flags = FT_LOAD_DEFAULT | FT_LOAD_IGNORE_GLOBAL_ADVANCE_WIDTH; | 
 |     float widthRatio = 1.0; | 
 |     FT_Matrix transMat = identityMat; | 
 |     BOOL needsTransform = FALSE; | 
 |  | 
 |  | 
 |     TRACE("%p, %04x, %08x, %p, %08lx, %p, %p\n", font, glyph, format, lpgm, | 
 | 	  buflen, buf, lpmat); | 
 |  | 
 |     if(format & GGO_GLYPH_INDEX) { | 
 |         glyph_index = glyph; | 
 | 	format &= ~GGO_GLYPH_INDEX; | 
 |     } else | 
 |         glyph_index = get_glyph_index(font, glyph); | 
 |  | 
 |     if(glyph_index >= font->gmsize) { | 
 |         font->gmsize = (glyph_index / INIT_GM_SIZE + 1) * INIT_GM_SIZE; | 
 | 	font->gm = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, font->gm, | 
 | 			       font->gmsize * sizeof(*font->gm)); | 
 |     } else { | 
 |         if(format == GGO_METRICS && font->gm[glyph_index].init) { | 
 | 	    memcpy(lpgm, &font->gm[glyph_index].gm, sizeof(*lpgm)); | 
 | 	    return 1; /* FIXME */ | 
 | 	} | 
 |     } | 
 |  | 
 |     if(font->orientation || (format != GGO_METRICS && format != GGO_BITMAP) || font->aveWidth || lpmat) | 
 |         load_flags |= FT_LOAD_NO_BITMAP; | 
 |  | 
 |     err = pFT_Load_Glyph(ft_face, glyph_index, load_flags); | 
 |  | 
 |     if(err) { | 
 |         FIXME("FT_Load_Glyph on index %x returns %d\n", glyph_index, err); | 
 | 	return GDI_ERROR; | 
 |     } | 
 | 	 | 
 |     /* Scaling factor */ | 
 |     if (font->aveWidth && font->potm) { | 
 |         widthRatio = (float)font->aveWidth * font->font_desc.matrix.eM11 / (float) font->potm->otmTextMetrics.tmAveCharWidth; | 
 |     } | 
 |  | 
 |     left = (INT)(ft_face->glyph->metrics.horiBearingX * widthRatio) & -64; | 
 |     right = (INT)((ft_face->glyph->metrics.horiBearingX + ft_face->glyph->metrics.width) * widthRatio + 63) & -64; | 
 |  | 
 |     font->gm[glyph_index].adv = (INT)((ft_face->glyph->metrics.horiAdvance * widthRatio) + 63) >> 6; | 
 |     font->gm[glyph_index].lsb = left >> 6; | 
 |     font->gm[glyph_index].bbx = (right - left) >> 6; | 
 |  | 
 |     /* Scaling transform */ | 
 |     if(font->aveWidth) { | 
 |         FT_Matrix scaleMat; | 
 |         scaleMat.xx = FT_FixedFromFloat(widthRatio); | 
 |         scaleMat.xy = 0; | 
 |         scaleMat.yx = 0; | 
 |         scaleMat.yy = (1 << 16); | 
 |  | 
 |         pFT_Matrix_Multiply(&scaleMat, &transMat); | 
 |         needsTransform = TRUE; | 
 |     } | 
 |  | 
 |     /* Rotation transform */ | 
 |     if(font->orientation) { | 
 |         FT_Matrix rotationMat; | 
 |         FT_Vector vecAngle; | 
 |         angle = FT_FixedFromFloat((float)font->orientation / 10.0); | 
 |         pFT_Vector_Unit(&vecAngle, angle); | 
 |         rotationMat.xx = vecAngle.x; | 
 |         rotationMat.xy = -vecAngle.y; | 
 |         rotationMat.yx = -rotationMat.xy; | 
 |         rotationMat.yy = rotationMat.xx; | 
 |          | 
 |         pFT_Matrix_Multiply(&rotationMat, &transMat); | 
 |         needsTransform = TRUE; | 
 |     } | 
 |  | 
 |     /* Extra transformation specified by caller */ | 
 |     if (lpmat) { | 
 |         FT_Matrix extraMat; | 
 |         extraMat.xx = FT_FixedFromFIXED(lpmat->eM11); | 
 |         extraMat.xy = FT_FixedFromFIXED(lpmat->eM21); | 
 |         extraMat.yx = FT_FixedFromFIXED(lpmat->eM12); | 
 |         extraMat.yy = FT_FixedFromFIXED(lpmat->eM22); | 
 |         pFT_Matrix_Multiply(&extraMat, &transMat); | 
 |         needsTransform = TRUE; | 
 |     } | 
 |  | 
 |     if(!needsTransform) { | 
 | 	top = (ft_face->glyph->metrics.horiBearingY + 63) & -64; | 
 | 	bottom = (ft_face->glyph->metrics.horiBearingY - | 
 | 		  ft_face->glyph->metrics.height) & -64; | 
 | 	lpgm->gmCellIncX = font->gm[glyph_index].adv; | 
 | 	lpgm->gmCellIncY = 0; | 
 |     } else { | 
 |         INT xc, yc; | 
 | 	FT_Vector vec; | 
 | 	for(xc = 0; xc < 2; xc++) { | 
 | 	    for(yc = 0; yc < 2; yc++) { | 
 | 	        vec.x = (ft_face->glyph->metrics.horiBearingX + | 
 | 		  xc * ft_face->glyph->metrics.width); | 
 | 		vec.y = ft_face->glyph->metrics.horiBearingY - | 
 | 		  yc * ft_face->glyph->metrics.height; | 
 | 		TRACE("Vec %ld,%ld\n", vec.x, vec.y); | 
 | 		pFT_Vector_Transform(&vec, &transMat); | 
 | 		if(xc == 0 && yc == 0) { | 
 | 		    left = right = vec.x; | 
 | 		    top = bottom = vec.y; | 
 | 		} else { | 
 | 		    if(vec.x < left) left = vec.x; | 
 | 		    else if(vec.x > right) right = vec.x; | 
 | 		    if(vec.y < bottom) bottom = vec.y; | 
 | 		    else if(vec.y > top) top = vec.y; | 
 | 		} | 
 | 	    } | 
 | 	} | 
 | 	left = left & -64; | 
 | 	right = (right + 63) & -64; | 
 | 	bottom = bottom & -64; | 
 | 	top = (top + 63) & -64; | 
 |  | 
 | 	TRACE("transformed box: (%d,%d - %d,%d)\n", left, top, right, bottom); | 
 | 	vec.x = ft_face->glyph->metrics.horiAdvance; | 
 | 	vec.y = 0; | 
 | 	pFT_Vector_Transform(&vec, &transMat); | 
 | 	lpgm->gmCellIncX = (vec.x+63) >> 6; | 
 | 	lpgm->gmCellIncY = -((vec.y+63) >> 6); | 
 |     } | 
 |     lpgm->gmBlackBoxX = (right - left) >> 6; | 
 |     lpgm->gmBlackBoxY = (top - bottom) >> 6; | 
 |     lpgm->gmptGlyphOrigin.x = left >> 6; | 
 |     lpgm->gmptGlyphOrigin.y = top >> 6; | 
 |  | 
 |     memcpy(&font->gm[glyph_index].gm, lpgm, sizeof(*lpgm)); | 
 |     font->gm[glyph_index].init = TRUE; | 
 |  | 
 |     if(format == GGO_METRICS) | 
 |         return 1; /* FIXME */ | 
 |  | 
 |     if (buf && !buflen){ | 
 |         return GDI_ERROR; | 
 |     } | 
 |  | 
 |     if(ft_face->glyph->format != ft_glyph_format_outline && format != GGO_BITMAP) { | 
 |         TRACE("loaded a bitmap\n"); | 
 | 	return GDI_ERROR; | 
 |     } | 
 |  | 
 |     switch(format) { | 
 |     case GGO_BITMAP: | 
 |         width = lpgm->gmBlackBoxX; | 
 | 	height = lpgm->gmBlackBoxY; | 
 | 	pitch = ((width + 31) >> 5) << 2; | 
 |         needed = pitch * height; | 
 |  | 
 | 	if(!buf || !buflen) break; | 
 |  | 
 | 	switch(ft_face->glyph->format) { | 
 | 	case ft_glyph_format_bitmap: | 
 | 	  { | 
 | 	    BYTE *src = ft_face->glyph->bitmap.buffer, *dst = buf; | 
 | 	    INT w = (ft_face->glyph->bitmap.width + 7) >> 3; | 
 | 	    INT h = ft_face->glyph->bitmap.rows; | 
 | 	    while(h--) { | 
 | 	        memcpy(dst, src, w); | 
 | 		src += ft_face->glyph->bitmap.pitch; | 
 | 		dst += pitch; | 
 | 	    } | 
 | 	    break; | 
 | 	  } | 
 |  | 
 | 	case ft_glyph_format_outline: | 
 | 	    ft_bitmap.width = width; | 
 | 	    ft_bitmap.rows = height; | 
 | 	    ft_bitmap.pitch = pitch; | 
 | 	    ft_bitmap.pixel_mode = ft_pixel_mode_mono; | 
 | 	    ft_bitmap.buffer = buf; | 
 |  | 
 | 		if(needsTransform) { | 
 | 			pFT_Outline_Transform(&ft_face->glyph->outline, &transMat); | 
 | 	    } | 
 |  | 
 | 	    pFT_Outline_Translate(&ft_face->glyph->outline, -left, -bottom ); | 
 |  | 
 | 	    /* Note: FreeType will only set 'black' bits for us. */ | 
 | 	    memset(buf, 0, needed); | 
 | 	    pFT_Outline_Get_Bitmap(library, &ft_face->glyph->outline, &ft_bitmap); | 
 | 	    break; | 
 |  | 
 | 	default: | 
 | 	    FIXME("loaded glyph format %x\n", ft_face->glyph->format); | 
 | 	    return GDI_ERROR; | 
 | 	} | 
 | 	break; | 
 |  | 
 |     case GGO_GRAY2_BITMAP: | 
 |     case GGO_GRAY4_BITMAP: | 
 |     case GGO_GRAY8_BITMAP: | 
 |     case WINE_GGO_GRAY16_BITMAP: | 
 |       { | 
 | 	unsigned int mult, row, col; | 
 | 	BYTE *start, *ptr; | 
 |  | 
 |         width = lpgm->gmBlackBoxX; | 
 | 	height = lpgm->gmBlackBoxY; | 
 | 	pitch = (width + 3) / 4 * 4; | 
 | 	needed = pitch * height; | 
 |  | 
 | 	if(!buf || !buflen) break; | 
 | 	ft_bitmap.width = width; | 
 | 	ft_bitmap.rows = height; | 
 | 	ft_bitmap.pitch = pitch; | 
 | 	ft_bitmap.pixel_mode = ft_pixel_mode_grays; | 
 | 	ft_bitmap.buffer = buf; | 
 |  | 
 | 	if(needsTransform) { | 
 | 		pFT_Outline_Transform(&ft_face->glyph->outline, &transMat); | 
 | 	} | 
 |  | 
 | 	pFT_Outline_Translate(&ft_face->glyph->outline, -left, -bottom ); | 
 |  | 
 | 	pFT_Outline_Get_Bitmap(library, &ft_face->glyph->outline, &ft_bitmap); | 
 |  | 
 | 	if(format == GGO_GRAY2_BITMAP) | 
 | 	    mult = 5; | 
 | 	else if(format == GGO_GRAY4_BITMAP) | 
 | 	    mult = 17; | 
 | 	else if(format == GGO_GRAY8_BITMAP) | 
 | 	    mult = 65; | 
 | 	else if(format == WINE_GGO_GRAY16_BITMAP) | 
 | 	    break; | 
 | 	else { | 
 | 	    assert(0); | 
 | 	    break; | 
 | 	} | 
 |  | 
 | 	start = buf; | 
 | 	for(row = 0; row < height; row++) { | 
 | 	    ptr = start; | 
 | 	    for(col = 0; col < width; col++, ptr++) { | 
 | 	        *ptr = (*(unsigned int*)ptr * mult + 128) / 256; | 
 | 	    } | 
 | 	    start += pitch; | 
 | 	} | 
 | 	break; | 
 |       } | 
 |  | 
 |     case GGO_NATIVE: | 
 |       { | 
 | 	int contour, point = 0, first_pt; | 
 | 	FT_Outline *outline = &ft_face->glyph->outline; | 
 | 	TTPOLYGONHEADER *pph; | 
 | 	TTPOLYCURVE *ppc; | 
 | 	DWORD pph_start, cpfx, type; | 
 |  | 
 | 	if(buflen == 0) buf = NULL; | 
 |  | 
 | 	if (needsTransform && buf) { | 
 | 		pFT_Outline_Transform(outline, &transMat); | 
 | 	} | 
 |  | 
 |         for(contour = 0; contour < outline->n_contours; contour++) { | 
 | 	    pph_start = needed; | 
 | 	    pph = (TTPOLYGONHEADER *)((char *)buf + needed); | 
 | 	    first_pt = point; | 
 | 	    if(buf) { | 
 | 	        pph->dwType = TT_POLYGON_TYPE; | 
 | 		FTVectorToPOINTFX(&outline->points[point], &pph->pfxStart); | 
 | 	    } | 
 | 	    needed += sizeof(*pph); | 
 | 	    point++; | 
 | 	    while(point <= outline->contours[contour]) { | 
 | 	        ppc = (TTPOLYCURVE *)((char *)buf + needed); | 
 | 		type = (outline->tags[point] & FT_Curve_Tag_On) ? | 
 | 		  TT_PRIM_LINE : TT_PRIM_QSPLINE; | 
 | 		cpfx = 0; | 
 | 		do { | 
 | 		    if(buf) | 
 | 		        FTVectorToPOINTFX(&outline->points[point], &ppc->apfx[cpfx]); | 
 | 		    cpfx++; | 
 | 		    point++; | 
 | 		} while(point <= outline->contours[contour] && | 
 | 			(outline->tags[point] & FT_Curve_Tag_On) == | 
 | 			(outline->tags[point-1] & FT_Curve_Tag_On)); | 
 | 		/* At the end of a contour Windows adds the start point, but | 
 | 		   only for Beziers */ | 
 | 		if(point > outline->contours[contour] && | 
 | 		   !(outline->tags[point-1] & FT_Curve_Tag_On)) { | 
 | 		    if(buf) | 
 | 		        FTVectorToPOINTFX(&outline->points[first_pt], &ppc->apfx[cpfx]); | 
 | 		    cpfx++; | 
 | 		} else if(point <= outline->contours[contour] && | 
 | 			  outline->tags[point] & FT_Curve_Tag_On) { | 
 | 		  /* add closing pt for bezier */ | 
 | 		    if(buf) | 
 | 		        FTVectorToPOINTFX(&outline->points[point], &ppc->apfx[cpfx]); | 
 | 		    cpfx++; | 
 | 		    point++; | 
 | 		} | 
 | 		if(buf) { | 
 | 		    ppc->wType = type; | 
 | 		    ppc->cpfx = cpfx; | 
 | 		} | 
 | 		needed += sizeof(*ppc) + (cpfx - 1) * sizeof(POINTFX); | 
 | 	    } | 
 | 	    if(buf) | 
 | 	        pph->cb = needed - pph_start; | 
 | 	} | 
 | 	break; | 
 |       } | 
 |     case GGO_BEZIER: | 
 |       { | 
 | 	/* Convert the quadratic Beziers to cubic Beziers. | 
 | 	   The parametric eqn for a cubic Bezier is, from PLRM: | 
 | 	   r(t) = at^3 + bt^2 + ct + r0 | 
 | 	   with the control points: | 
 | 	   r1 = r0 + c/3 | 
 | 	   r2 = r1 + (c + b)/3 | 
 | 	   r3 = r0 + c + b + a | 
 |  | 
 | 	   A quadratic Beizer has the form: | 
 | 	   p(t) = (1-t)^2 p0 + 2(1-t)t p1 + t^2 p2 | 
 |  | 
 | 	   So equating powers of t leads to: | 
 | 	   r1 = 2/3 p1 + 1/3 p0 | 
 | 	   r2 = 2/3 p1 + 1/3 p2 | 
 | 	   and of course r0 = p0, r3 = p2 | 
 | 	*/ | 
 |  | 
 | 	int contour, point = 0, first_pt; | 
 | 	FT_Outline *outline = &ft_face->glyph->outline; | 
 | 	TTPOLYGONHEADER *pph; | 
 | 	TTPOLYCURVE *ppc; | 
 | 	DWORD pph_start, cpfx, type; | 
 | 	FT_Vector cubic_control[4]; | 
 | 	if(buflen == 0) buf = NULL; | 
 |  | 
 | 	if (needsTransform && buf) { | 
 | 		pFT_Outline_Transform(outline, &transMat); | 
 | 	} | 
 |  | 
 |         for(contour = 0; contour < outline->n_contours; contour++) { | 
 | 	    pph_start = needed; | 
 | 	    pph = (TTPOLYGONHEADER *)((char *)buf + needed); | 
 | 	    first_pt = point; | 
 | 	    if(buf) { | 
 | 	        pph->dwType = TT_POLYGON_TYPE; | 
 | 		FTVectorToPOINTFX(&outline->points[point], &pph->pfxStart); | 
 | 	    } | 
 | 	    needed += sizeof(*pph); | 
 | 	    point++; | 
 | 	    while(point <= outline->contours[contour]) { | 
 | 	        ppc = (TTPOLYCURVE *)((char *)buf + needed); | 
 | 		type = (outline->tags[point] & FT_Curve_Tag_On) ? | 
 | 		  TT_PRIM_LINE : TT_PRIM_CSPLINE; | 
 | 		cpfx = 0; | 
 | 		do { | 
 | 		    if(type == TT_PRIM_LINE) { | 
 | 		        if(buf) | 
 | 			    FTVectorToPOINTFX(&outline->points[point], &ppc->apfx[cpfx]); | 
 | 			cpfx++; | 
 | 			point++; | 
 | 		    } else { | 
 | 		      /* Unlike QSPLINEs, CSPLINEs always have their endpoint | 
 | 			 so cpfx = 3n */ | 
 |  | 
 | 		      /* FIXME: Possible optimization in endpoint calculation | 
 | 			 if there are two consecutive curves */ | 
 | 		        cubic_control[0] = outline->points[point-1]; | 
 | 		        if(!(outline->tags[point-1] & FT_Curve_Tag_On)) { | 
 | 			    cubic_control[0].x += outline->points[point].x + 1; | 
 | 			    cubic_control[0].y += outline->points[point].y + 1; | 
 | 			    cubic_control[0].x >>= 1; | 
 | 			    cubic_control[0].y >>= 1; | 
 | 			} | 
 | 			if(point+1 > outline->contours[contour]) | 
 |  			    cubic_control[3] = outline->points[first_pt]; | 
 | 			else { | 
 | 			    cubic_control[3] = outline->points[point+1]; | 
 | 			    if(!(outline->tags[point+1] & FT_Curve_Tag_On)) { | 
 | 			        cubic_control[3].x += outline->points[point].x + 1; | 
 | 				cubic_control[3].y += outline->points[point].y + 1; | 
 | 				cubic_control[3].x >>= 1; | 
 | 				cubic_control[3].y >>= 1; | 
 | 			    } | 
 | 			} | 
 | 			/* r1 = 1/3 p0 + 2/3 p1 | 
 | 			   r2 = 1/3 p2 + 2/3 p1 */ | 
 | 		        cubic_control[1].x = (2 * outline->points[point].x + 1) / 3; | 
 | 			cubic_control[1].y = (2 * outline->points[point].y + 1) / 3; | 
 | 			cubic_control[2] = cubic_control[1]; | 
 | 			cubic_control[1].x += (cubic_control[0].x + 1) / 3; | 
 | 			cubic_control[1].y += (cubic_control[0].y + 1) / 3; | 
 | 			cubic_control[2].x += (cubic_control[3].x + 1) / 3; | 
 | 			cubic_control[2].y += (cubic_control[3].y + 1) / 3; | 
 | 		        if(buf) { | 
 | 			    FTVectorToPOINTFX(&cubic_control[1], &ppc->apfx[cpfx]); | 
 | 			    FTVectorToPOINTFX(&cubic_control[2], &ppc->apfx[cpfx+1]); | 
 | 			    FTVectorToPOINTFX(&cubic_control[3], &ppc->apfx[cpfx+2]); | 
 | 			} | 
 | 			cpfx += 3; | 
 | 			point++; | 
 | 		    } | 
 | 		} while(point <= outline->contours[contour] && | 
 | 			(outline->tags[point] & FT_Curve_Tag_On) == | 
 | 			(outline->tags[point-1] & FT_Curve_Tag_On)); | 
 | 		/* At the end of a contour Windows adds the start point, | 
 | 		   but only for Beziers and we've already done that. | 
 | 		*/ | 
 | 		if(point <= outline->contours[contour] && | 
 | 		   outline->tags[point] & FT_Curve_Tag_On) { | 
 | 		  /* This is the closing pt of a bezier, but we've already | 
 | 		     added it, so just inc point and carry on */ | 
 | 		    point++; | 
 | 		} | 
 | 		if(buf) { | 
 | 		    ppc->wType = type; | 
 | 		    ppc->cpfx = cpfx; | 
 | 		} | 
 | 		needed += sizeof(*ppc) + (cpfx - 1) * sizeof(POINTFX); | 
 | 	    } | 
 | 	    if(buf) | 
 | 	        pph->cb = needed - pph_start; | 
 | 	} | 
 | 	break; | 
 |       } | 
 |  | 
 |     default: | 
 |         FIXME("Unsupported format %d\n", format); | 
 | 	return GDI_ERROR; | 
 |     } | 
 |     return needed; | 
 | } | 
 |  | 
 | static BOOL get_bitmap_text_metrics(GdiFont font) | 
 | { | 
 |     FT_Face ft_face = font->ft_face; | 
 | #ifdef HAVE_FREETYPE_FTWINFNT_H | 
 |     FT_WinFNT_HeaderRec winfnt_header; | 
 | #endif | 
 |     const DWORD size = offsetof(OUTLINETEXTMETRICW, otmFiller);  | 
 |     font->potm = HeapAlloc(GetProcessHeap(), 0, size); | 
 |     font->potm->otmSize = size; | 
 |  | 
 | #define TM font->potm->otmTextMetrics | 
 | #ifdef HAVE_FREETYPE_FTWINFNT_H | 
 |     if(pFT_Get_WinFNT_Header && !pFT_Get_WinFNT_Header(ft_face, &winfnt_header)) | 
 |     { | 
 |         TM.tmHeight = winfnt_header.pixel_height; | 
 |         TM.tmAscent = winfnt_header.ascent; | 
 |         TM.tmDescent = TM.tmHeight - TM.tmAscent; | 
 |         TM.tmInternalLeading = winfnt_header.internal_leading; | 
 |         TM.tmExternalLeading = winfnt_header.external_leading; | 
 |         TM.tmAveCharWidth = winfnt_header.avg_width; | 
 |         TM.tmMaxCharWidth = winfnt_header.max_width; | 
 |         TM.tmWeight = winfnt_header.weight; | 
 |         TM.tmOverhang = 0; | 
 |         TM.tmDigitizedAspectX = winfnt_header.horizontal_resolution; | 
 |         TM.tmDigitizedAspectY = winfnt_header.vertical_resolution; | 
 |         TM.tmFirstChar = winfnt_header.first_char; | 
 |         TM.tmLastChar = winfnt_header.last_char; | 
 |         TM.tmDefaultChar = winfnt_header.default_char + winfnt_header.first_char; | 
 |         TM.tmBreakChar = winfnt_header.break_char + winfnt_header.first_char; | 
 |         TM.tmItalic = winfnt_header.italic; | 
 |         TM.tmUnderlined = font->underline; | 
 |         TM.tmStruckOut = font->strikeout; | 
 |         TM.tmPitchAndFamily = winfnt_header.pitch_and_family; | 
 |         TM.tmCharSet = winfnt_header.charset; | 
 |     } | 
 |     else | 
 | #endif | 
 |     { | 
 |         TM.tmAscent = ft_face->size->metrics.ascender >> 6; | 
 |         TM.tmDescent = -ft_face->size->metrics.descender >> 6; | 
 |         TM.tmHeight = TM.tmAscent + TM.tmDescent; | 
 |         TM.tmInternalLeading = TM.tmHeight - ft_face->size->metrics.y_ppem; | 
 |         TM.tmExternalLeading = (ft_face->size->metrics.height >> 6) - TM.tmHeight; | 
 |         TM.tmMaxCharWidth = ft_face->size->metrics.max_advance >> 6; | 
 |         TM.tmAveCharWidth = TM.tmMaxCharWidth * 2 / 3; /* FIXME */ | 
 |         TM.tmWeight = ft_face->style_flags & FT_STYLE_FLAG_BOLD ? FW_BOLD : FW_NORMAL; | 
 |         TM.tmOverhang = 0; | 
 |         TM.tmDigitizedAspectX = 96; /* FIXME */ | 
 |         TM.tmDigitizedAspectY = 96; /* FIXME */ | 
 |         TM.tmFirstChar = 1; | 
 |         TM.tmLastChar = 255; | 
 |         TM.tmDefaultChar = 32; | 
 |         TM.tmBreakChar = 32; | 
 |         TM.tmItalic = ft_face->style_flags & FT_STYLE_FLAG_ITALIC ? 1 : 0; | 
 |         TM.tmUnderlined = font->underline; | 
 |         TM.tmStruckOut = font->strikeout; | 
 |         /* NB inverted meaning of TMPF_FIXED_PITCH */ | 
 |         TM.tmPitchAndFamily = ft_face->face_flags & FT_FACE_FLAG_FIXED_WIDTH ? 0 : TMPF_FIXED_PITCH; | 
 |         TM.tmCharSet = font->charset; | 
 |     } | 
 | #undef TM | 
 |  | 
 |     return TRUE; | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  * WineEngGetTextMetrics | 
 |  * | 
 |  */ | 
 | BOOL WineEngGetTextMetrics(GdiFont font, LPTEXTMETRICW ptm) | 
 | { | 
 |     if(!font->potm) { | 
 |         if(!WineEngGetOutlineTextMetrics(font, 0, NULL)) | 
 |             if(!get_bitmap_text_metrics(font)) | 
 |                 return FALSE; | 
 |     } | 
 |     if(!font->potm) return FALSE; | 
 |     memcpy(ptm, &font->potm->otmTextMetrics, sizeof(*ptm)); | 
 |  | 
 |     if (font->aveWidth) { | 
 |         ptm->tmAveCharWidth = font->aveWidth * font->font_desc.matrix.eM11; | 
 |     } | 
 |     return TRUE; | 
 | } | 
 |  | 
 |  | 
 | /************************************************************* | 
 |  * WineEngGetOutlineTextMetrics | 
 |  * | 
 |  */ | 
 | UINT WineEngGetOutlineTextMetrics(GdiFont font, UINT cbSize, | 
 | 				  OUTLINETEXTMETRICW *potm) | 
 | { | 
 |     FT_Face ft_face = font->ft_face; | 
 |     UINT needed, lenfam, lensty, ret; | 
 |     TT_OS2 *pOS2; | 
 |     TT_HoriHeader *pHori; | 
 |     TT_Postscript *pPost; | 
 |     FT_Fixed x_scale, y_scale; | 
 |     WCHAR *family_nameW, *style_nameW; | 
 |     static const WCHAR spaceW[] = {' ', '\0'}; | 
 |     char *cp; | 
 |     INT ascent, descent; | 
 |  | 
 |     TRACE("font=%p\n", font); | 
 |  | 
 |     if(!FT_IS_SCALABLE(ft_face)) | 
 |         return 0; | 
 |  | 
 |     if(font->potm) { | 
 |         if(cbSize >= font->potm->otmSize) | 
 | 	    memcpy(potm, font->potm, font->potm->otmSize); | 
 | 	return font->potm->otmSize; | 
 |     } | 
 |  | 
 |  | 
 |     needed = sizeof(*potm); | 
 |  | 
 |     lenfam = (strlenW(font->name) + 1) * sizeof(WCHAR); | 
 |     family_nameW = strdupW(font->name); | 
 |  | 
 |     lensty = MultiByteToWideChar(CP_ACP, 0, ft_face->style_name, -1, NULL, 0) | 
 |       * sizeof(WCHAR); | 
 |     style_nameW = HeapAlloc(GetProcessHeap(), 0, lensty); | 
 |     MultiByteToWideChar(CP_ACP, 0, ft_face->style_name, -1, | 
 | 			style_nameW, lensty); | 
 |  | 
 |     /* These names should be read from the TT name table */ | 
 |  | 
 |     /* length of otmpFamilyName */ | 
 |     needed += lenfam; | 
 |  | 
 |     /* length of otmpFaceName */ | 
 |     if(!strcasecmp(ft_face->style_name, "regular")) { | 
 |       needed += lenfam; /* just the family name */ | 
 |     } else { | 
 |       needed += lenfam + lensty; /* family + " " + style */ | 
 |     } | 
 |  | 
 |     /* length of otmpStyleName */ | 
 |     needed += lensty; | 
 |  | 
 |     /* length of otmpFullName */ | 
 |     needed += lenfam + lensty; | 
 |  | 
 |  | 
 |     x_scale = ft_face->size->metrics.x_scale; | 
 |     y_scale = ft_face->size->metrics.y_scale; | 
 |  | 
 |     pOS2 = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_os2); | 
 |     if(!pOS2) { | 
 |         FIXME("Can't find OS/2 table - not TT font?\n"); | 
 | 	ret = 0; | 
 | 	goto end; | 
 |     } | 
 |  | 
 |     pHori = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_hhea); | 
 |     if(!pHori) { | 
 |         FIXME("Can't find HHEA table - not TT font?\n"); | 
 | 	ret = 0; | 
 | 	goto end; | 
 |     } | 
 |  | 
 |     pPost = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_post); /* we can live with this failing */ | 
 |  | 
 |     TRACE("OS/2 winA = %d winD = %d typoA = %d typoD = %d typoLG = %d FT_Face a = %d, d = %d, h = %d: HORZ a = %d, d = %d lg = %d maxY = %ld minY = %ld\n", | 
 | 	  pOS2->usWinAscent, pOS2->usWinDescent, | 
 | 	  pOS2->sTypoAscender, pOS2->sTypoDescender, pOS2->sTypoLineGap, | 
 | 	  ft_face->ascender, ft_face->descender, ft_face->height, | 
 | 	  pHori->Ascender, pHori->Descender, pHori->Line_Gap, | 
 | 	  ft_face->bbox.yMax, ft_face->bbox.yMin); | 
 |  | 
 |     font->potm = HeapAlloc(GetProcessHeap(), 0, needed); | 
 |     font->potm->otmSize = needed; | 
 |  | 
 | #define TM font->potm->otmTextMetrics | 
 |  | 
 |     if(pOS2->usWinAscent + pOS2->usWinDescent == 0) { | 
 |         ascent = pHori->Ascender; | 
 |         descent = -pHori->Descender; | 
 |     } else { | 
 |         ascent = pOS2->usWinAscent; | 
 |         descent = pOS2->usWinDescent; | 
 |     } | 
 |  | 
 |     if(font->yMax) { | 
 | 	TM.tmAscent = font->yMax; | 
 | 	TM.tmDescent = -font->yMin; | 
 | 	TM.tmInternalLeading = (TM.tmAscent + TM.tmDescent) - ft_face->size->metrics.y_ppem; | 
 |     } else { | 
 | 	TM.tmAscent = (pFT_MulFix(ascent, y_scale) + 32) >> 6; | 
 | 	TM.tmDescent = (pFT_MulFix(descent, y_scale) + 32) >> 6; | 
 | 	TM.tmInternalLeading = (pFT_MulFix(ascent + descent | 
 | 					    - ft_face->units_per_EM, y_scale) + 32) >> 6; | 
 |     } | 
 |  | 
 |     TM.tmHeight = TM.tmAscent + TM.tmDescent; | 
 |  | 
 |     /* MSDN says: | 
 |      el = MAX(0, LineGap - ((WinAscent + WinDescent) - (Ascender - Descender))) | 
 |     */ | 
 |     TM.tmExternalLeading = max(0, (pFT_MulFix(pHori->Line_Gap - | 
 |        		 ((ascent + descent) - | 
 | 		  (pHori->Ascender - pHori->Descender)), y_scale) + 32) >> 6); | 
 |  | 
 |     TM.tmAveCharWidth = (pFT_MulFix(pOS2->xAvgCharWidth, x_scale) + 32) >> 6; | 
 |     if (TM.tmAveCharWidth == 0) { | 
 |         TM.tmAveCharWidth = 1;  | 
 |     } | 
 |     TM.tmMaxCharWidth = (pFT_MulFix(ft_face->bbox.xMax - ft_face->bbox.xMin, x_scale) + 32) >> 6; | 
 |     TM.tmWeight = font->fake_bold ? FW_BOLD : pOS2->usWeightClass; | 
 |     TM.tmOverhang = 0; | 
 |     TM.tmDigitizedAspectX = 300; | 
 |     TM.tmDigitizedAspectY = 300; | 
 |     TM.tmFirstChar = pOS2->usFirstCharIndex; | 
 |     TM.tmLastChar = pOS2->usLastCharIndex; | 
 |     TM.tmDefaultChar = pOS2->usDefaultChar; | 
 |     TM.tmBreakChar = pOS2->usBreakChar ? pOS2->usBreakChar : ' '; | 
 |     TM.tmItalic = font->fake_italic ? 255 : ((ft_face->style_flags & FT_STYLE_FLAG_ITALIC) ? 255 : 0); | 
 |     TM.tmUnderlined = font->underline; | 
 |     TM.tmStruckOut = font->strikeout; | 
 |  | 
 |     /* Yes TPMF_FIXED_PITCH is correct; braindead api */ | 
 |     if(!FT_IS_FIXED_WIDTH(ft_face)) | 
 |         TM.tmPitchAndFamily = TMPF_FIXED_PITCH; | 
 |     else | 
 |         TM.tmPitchAndFamily = 0; | 
 |  | 
 |     switch(pOS2->panose[PAN_FAMILYTYPE_INDEX]) { | 
 |     case PAN_FAMILY_SCRIPT: | 
 |         TM.tmPitchAndFamily |= FF_SCRIPT; | 
 | 	break; | 
 |     case PAN_FAMILY_DECORATIVE: | 
 |     case PAN_FAMILY_PICTORIAL: | 
 |         TM.tmPitchAndFamily |= FF_DECORATIVE; | 
 | 	break; | 
 |     case PAN_FAMILY_TEXT_DISPLAY: | 
 |         if(TM.tmPitchAndFamily == 0) /* fixed */ | 
 | 	    TM.tmPitchAndFamily = FF_MODERN; | 
 | 	else { | 
 | 	    switch(pOS2->panose[PAN_SERIFSTYLE_INDEX]) { | 
 | 	    case PAN_SERIF_NORMAL_SANS: | 
 | 	    case PAN_SERIF_OBTUSE_SANS: | 
 | 	    case PAN_SERIF_PERP_SANS: | 
 | 	        TM.tmPitchAndFamily |= FF_SWISS; | 
 | 		break; | 
 | 	    default: | 
 | 	        TM.tmPitchAndFamily |= FF_ROMAN; | 
 | 	    } | 
 | 	} | 
 | 	break; | 
 |     default: | 
 |         TM.tmPitchAndFamily |= FF_DONTCARE; | 
 |     } | 
 |  | 
 |     if(FT_IS_SCALABLE(ft_face)) | 
 |         TM.tmPitchAndFamily |= TMPF_VECTOR; | 
 |     if(FT_IS_SFNT(ft_face)) | 
 |         TM.tmPitchAndFamily |= TMPF_TRUETYPE; | 
 |  | 
 |     TM.tmCharSet = font->charset; | 
 | #undef TM | 
 |  | 
 |     font->potm->otmFiller = 0; | 
 |     memcpy(&font->potm->otmPanoseNumber, pOS2->panose, PANOSE_COUNT); | 
 |     font->potm->otmfsSelection = pOS2->fsSelection; | 
 |     font->potm->otmfsType = pOS2->fsType; | 
 |     font->potm->otmsCharSlopeRise = pHori->caret_Slope_Rise; | 
 |     font->potm->otmsCharSlopeRun = pHori->caret_Slope_Run; | 
 |     font->potm->otmItalicAngle = 0; /* POST table */ | 
 |     font->potm->otmEMSquare = ft_face->units_per_EM; | 
 |     font->potm->otmAscent = (pFT_MulFix(pOS2->sTypoAscender, y_scale) + 32) >> 6; | 
 |     font->potm->otmDescent = (pFT_MulFix(pOS2->sTypoDescender, y_scale) + 32) >> 6; | 
 |     font->potm->otmLineGap = (pFT_MulFix(pOS2->sTypoLineGap, y_scale) + 32) >> 6; | 
 |     font->potm->otmsCapEmHeight = (pFT_MulFix(pOS2->sCapHeight, y_scale) + 32) >> 6; | 
 |     font->potm->otmsXHeight = (pFT_MulFix(pOS2->sxHeight, y_scale) + 32) >> 6; | 
 |     font->potm->otmrcFontBox.left = (pFT_MulFix(ft_face->bbox.xMin, x_scale) + 32) >> 6; | 
 |     font->potm->otmrcFontBox.right = (pFT_MulFix(ft_face->bbox.xMax, x_scale) + 32) >> 6; | 
 |     font->potm->otmrcFontBox.top = (pFT_MulFix(ft_face->bbox.yMax, y_scale) + 32) >> 6; | 
 |     font->potm->otmrcFontBox.bottom = (pFT_MulFix(ft_face->bbox.yMin, y_scale) + 32) >> 6; | 
 |     font->potm->otmMacAscent = 0; /* where do these come from ? */ | 
 |     font->potm->otmMacDescent = 0; | 
 |     font->potm->otmMacLineGap = 0; | 
 |     font->potm->otmusMinimumPPEM = 0; /* TT Header */ | 
 |     font->potm->otmptSubscriptSize.x = (pFT_MulFix(pOS2->ySubscriptXSize, x_scale) + 32) >> 6; | 
 |     font->potm->otmptSubscriptSize.y = (pFT_MulFix(pOS2->ySubscriptYSize, y_scale) + 32) >> 6; | 
 |     font->potm->otmptSubscriptOffset.x = (pFT_MulFix(pOS2->ySubscriptXOffset, x_scale) + 32) >> 6; | 
 |     font->potm->otmptSubscriptOffset.y = (pFT_MulFix(pOS2->ySubscriptYOffset, y_scale) + 32) >> 6; | 
 |     font->potm->otmptSuperscriptSize.x = (pFT_MulFix(pOS2->ySuperscriptXSize, x_scale) + 32) >> 6; | 
 |     font->potm->otmptSuperscriptSize.y = (pFT_MulFix(pOS2->ySuperscriptYSize, y_scale) + 32) >> 6; | 
 |     font->potm->otmptSuperscriptOffset.x = (pFT_MulFix(pOS2->ySuperscriptXOffset, x_scale) + 32) >> 6; | 
 |     font->potm->otmptSuperscriptOffset.y = (pFT_MulFix(pOS2->ySuperscriptYOffset, y_scale) + 32) >> 6; | 
 |     font->potm->otmsStrikeoutSize = (pFT_MulFix(pOS2->yStrikeoutSize, y_scale) + 32) >> 6; | 
 |     font->potm->otmsStrikeoutPosition = (pFT_MulFix(pOS2->yStrikeoutPosition, y_scale) + 32) >> 6; | 
 |     if(!pPost) { | 
 |         font->potm->otmsUnderscoreSize = 0; | 
 | 	font->potm->otmsUnderscorePosition = 0; | 
 |     } else { | 
 |         font->potm->otmsUnderscoreSize = (pFT_MulFix(pPost->underlineThickness, y_scale) + 32) >> 6; | 
 | 	font->potm->otmsUnderscorePosition = (pFT_MulFix(pPost->underlinePosition, y_scale) + 32) >> 6; | 
 |     } | 
 |  | 
 |     /* otmp* members should clearly have type ptrdiff_t, but M$ knows best */ | 
 |     cp = (char*)font->potm + sizeof(*font->potm); | 
 |     font->potm->otmpFamilyName = (LPSTR)(cp - (char*)font->potm); | 
 |     strcpyW((WCHAR*)cp, family_nameW); | 
 |     cp += lenfam; | 
 |     font->potm->otmpStyleName = (LPSTR)(cp - (char*)font->potm); | 
 |     strcpyW((WCHAR*)cp, style_nameW); | 
 |     cp += lensty; | 
 |     font->potm->otmpFaceName = (LPSTR)(cp - (char*)font->potm); | 
 |     strcpyW((WCHAR*)cp, family_nameW); | 
 |     if(strcasecmp(ft_face->style_name, "regular")) { | 
 |         strcatW((WCHAR*)cp, spaceW); | 
 | 	strcatW((WCHAR*)cp, style_nameW); | 
 | 	cp += lenfam + lensty; | 
 |     } else | 
 |         cp += lenfam; | 
 |     font->potm->otmpFullName = (LPSTR)(cp - (char*)font->potm); | 
 |     strcpyW((WCHAR*)cp, family_nameW); | 
 |     strcatW((WCHAR*)cp, spaceW); | 
 |     strcatW((WCHAR*)cp, style_nameW); | 
 |     ret = needed; | 
 |  | 
 |     if(potm && needed <= cbSize) | 
 |         memcpy(potm, font->potm, font->potm->otmSize); | 
 |  | 
 | end: | 
 |     HeapFree(GetProcessHeap(), 0, style_nameW); | 
 |     HeapFree(GetProcessHeap(), 0, family_nameW); | 
 |  | 
 |     return ret; | 
 | } | 
 |  | 
 |  | 
 | /************************************************************* | 
 |  * WineEngGetCharWidth | 
 |  * | 
 |  */ | 
 | BOOL WineEngGetCharWidth(GdiFont font, UINT firstChar, UINT lastChar, | 
 | 			 LPINT buffer) | 
 | { | 
 |     UINT c; | 
 |     GLYPHMETRICS gm; | 
 |     FT_UInt glyph_index; | 
 |  | 
 |     TRACE("%p, %d, %d, %p\n", font, firstChar, lastChar, buffer); | 
 |  | 
 |     for(c = firstChar; c <= lastChar; c++) { | 
 |         glyph_index = get_glyph_index(font, c); | 
 |         WineEngGetGlyphOutline(font, glyph_index, GGO_METRICS | GGO_GLYPH_INDEX, | 
 |                                &gm, 0, NULL, NULL); | 
 | 	buffer[c - firstChar] = font->gm[glyph_index].adv; | 
 |     } | 
 |     return TRUE; | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  * WineEngGetCharABCWidths | 
 |  * | 
 |  */ | 
 | BOOL WineEngGetCharABCWidths(GdiFont font, UINT firstChar, UINT lastChar, | 
 | 			     LPABC buffer) | 
 | { | 
 |     UINT c; | 
 |     GLYPHMETRICS gm; | 
 |     FT_UInt glyph_index; | 
 |  | 
 |     TRACE("%p, %d, %d, %p\n", font, firstChar, lastChar, buffer); | 
 |  | 
 |     if(!FT_IS_SCALABLE(font->ft_face)) | 
 |         return FALSE; | 
 |  | 
 |     for(c = firstChar; c <= lastChar; c++) { | 
 |         glyph_index = get_glyph_index(font, c); | 
 |         WineEngGetGlyphOutline(font, glyph_index, GGO_METRICS | GGO_GLYPH_INDEX, | 
 |                                &gm, 0, NULL, NULL); | 
 | 	buffer[c - firstChar].abcA = font->gm[glyph_index].lsb; | 
 | 	buffer[c - firstChar].abcB = font->gm[glyph_index].bbx; | 
 | 	buffer[c - firstChar].abcC = font->gm[glyph_index].adv - font->gm[glyph_index].lsb - | 
 | 	  font->gm[glyph_index].bbx; | 
 |     } | 
 |     return TRUE; | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  * WineEngGetTextExtentPoint | 
 |  * | 
 |  */ | 
 | BOOL WineEngGetTextExtentPoint(GdiFont font, LPCWSTR wstr, INT count, | 
 | 			       LPSIZE size) | 
 | { | 
 |     INT idx; | 
 |     GLYPHMETRICS gm; | 
 |     TEXTMETRICW tm; | 
 |     FT_UInt glyph_index; | 
 |  | 
 |     TRACE("%p, %s, %d, %p\n", font, debugstr_wn(wstr, count), count, | 
 | 	  size); | 
 |  | 
 |     size->cx = 0; | 
 |     WineEngGetTextMetrics(font, &tm); | 
 |     size->cy = tm.tmHeight; | 
 |  | 
 |     for(idx = 0; idx < count; idx++) { | 
 | 	glyph_index = get_glyph_index(font, wstr[idx]); | 
 |         WineEngGetGlyphOutline(font, glyph_index, GGO_METRICS | GGO_GLYPH_INDEX, | 
 |                                &gm, 0, NULL, NULL); | 
 | 	size->cx += font->gm[glyph_index].adv; | 
 |     } | 
 |     TRACE("return %ld,%ld\n", size->cx, size->cy); | 
 |     return TRUE; | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  * WineEngGetTextExtentPointI | 
 |  * | 
 |  */ | 
 | BOOL WineEngGetTextExtentPointI(GdiFont font, const WORD *indices, INT count, | 
 | 				LPSIZE size) | 
 | { | 
 |     INT idx; | 
 |     GLYPHMETRICS gm; | 
 |     TEXTMETRICW tm; | 
 |  | 
 |     TRACE("%p, %p, %d, %p\n", font, indices, count, size); | 
 |  | 
 |     size->cx = 0; | 
 |     WineEngGetTextMetrics(font, &tm); | 
 |     size->cy = tm.tmHeight; | 
 |  | 
 |     for(idx = 0; idx < count; idx++) { | 
 |         WineEngGetGlyphOutline(font, indices[idx], | 
 | 			       GGO_METRICS | GGO_GLYPH_INDEX, &gm, 0, NULL, | 
 | 			       NULL); | 
 | 	size->cx += font->gm[indices[idx]].adv; | 
 |     } | 
 |     TRACE("return %ld,%ld\n", size->cx, size->cy); | 
 |     return TRUE; | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  * WineEngGetFontData | 
 |  * | 
 |  */ | 
 | DWORD WineEngGetFontData(GdiFont font, DWORD table, DWORD offset, LPVOID buf, | 
 | 			 DWORD cbData) | 
 | { | 
 |     FT_Face ft_face = font->ft_face; | 
 |     DWORD len; | 
 |     FT_Error err; | 
 |  | 
 |     TRACE("font=%p, table=%08lx, offset=%08lx, buf=%p, cbData=%lx\n", | 
 | 	font, table, offset, buf, cbData); | 
 |  | 
 |     if(!FT_IS_SFNT(ft_face)) | 
 |         return GDI_ERROR; | 
 |  | 
 |     if(!buf || !cbData) | 
 |         len = 0; | 
 |     else | 
 |         len = cbData; | 
 |  | 
 |     if(table) { /* MS tags differ in endidness from FT ones */ | 
 |         table = table >> 24 | table << 24 | | 
 | 	  (table >> 8 & 0xff00) | (table << 8 & 0xff0000); | 
 |     } | 
 |  | 
 |     /* If the FT_Load_Sfnt_Table function is there we'll use it */ | 
 |     if(pFT_Load_Sfnt_Table) | 
 |         err = pFT_Load_Sfnt_Table(ft_face, table, offset, buf, &len); | 
 |     else { /* Do it the hard way */ | 
 |         TT_Face tt_face = (TT_Face) ft_face; | 
 |         SFNT_Interface *sfnt; | 
 |         if (FT_Version.major==2 && FT_Version.minor==0) | 
 |         { | 
 |             /* 2.0.x */ | 
 |             sfnt = *(SFNT_Interface**)((char*)tt_face + 528); | 
 |         } | 
 |         else | 
 |         { | 
 |             /* A field was added in the middle of the structure in 2.1.x */ | 
 |             sfnt = *(SFNT_Interface**)((char*)tt_face + 532); | 
 |         } | 
 |         err = sfnt->load_any(tt_face, table, offset, buf, &len); | 
 |     } | 
 |     if(err) { | 
 |         TRACE("Can't find table %08lx.\n", table); | 
 | 	return GDI_ERROR; | 
 |     } | 
 |     return len; | 
 | } | 
 |  | 
 | /************************************************************* | 
 |  * WineEngGetTextFace | 
 |  * | 
 |  */ | 
 | INT WineEngGetTextFace(GdiFont font, INT count, LPWSTR str) | 
 | { | 
 |     if(str) { | 
 |         lstrcpynW(str, font->name, count); | 
 | 	return strlenW(font->name); | 
 |     } else | 
 |         return strlenW(font->name) + 1; | 
 | } | 
 |  | 
 | UINT WineEngGetTextCharsetInfo(GdiFont font, LPFONTSIGNATURE fs, DWORD flags) | 
 | { | 
 |     if (fs) memcpy(fs, &font->fs, sizeof(FONTSIGNATURE)); | 
 |     return font->charset; | 
 | } | 
 |  | 
 | #else /* HAVE_FREETYPE */ | 
 |  | 
 | BOOL WineEngInit(void) | 
 | { | 
 |     return FALSE; | 
 | } | 
 | GdiFont WineEngCreateFontInstance(DC *dc, HFONT hfont) | 
 | { | 
 |     return NULL; | 
 | } | 
 | BOOL WineEngDestroyFontInstance(HFONT hfont) | 
 | { | 
 |     return FALSE; | 
 | } | 
 |  | 
 | DWORD WineEngEnumFonts(LPLOGFONTW plf, FONTENUMPROCW proc, LPARAM lparam) | 
 | { | 
 |     return 1; | 
 | } | 
 |  | 
 | DWORD WineEngGetGlyphIndices(GdiFont font, LPCWSTR lpstr, INT count, | 
 | 				LPWORD pgi, DWORD flags) | 
 | { | 
 |     return GDI_ERROR; | 
 | } | 
 |  | 
 | DWORD WineEngGetGlyphOutline(GdiFont font, UINT glyph, UINT format, | 
 | 			     LPGLYPHMETRICS lpgm, DWORD buflen, LPVOID buf, | 
 | 			     const MAT2* lpmat) | 
 | { | 
 |     ERR("called but we don't have FreeType\n"); | 
 |     return GDI_ERROR; | 
 | } | 
 |  | 
 | BOOL WineEngGetTextMetrics(GdiFont font, LPTEXTMETRICW ptm) | 
 | { | 
 |     ERR("called but we don't have FreeType\n"); | 
 |     return FALSE; | 
 | } | 
 |  | 
 | UINT WineEngGetOutlineTextMetrics(GdiFont font, UINT cbSize, | 
 | 				  OUTLINETEXTMETRICW *potm) | 
 | { | 
 |     ERR("called but we don't have FreeType\n"); | 
 |     return 0; | 
 | } | 
 |  | 
 | BOOL WineEngGetCharWidth(GdiFont font, UINT firstChar, UINT lastChar, | 
 | 			 LPINT buffer) | 
 | { | 
 |     ERR("called but we don't have FreeType\n"); | 
 |     return FALSE; | 
 | } | 
 |  | 
 | BOOL WineEngGetCharABCWidths(GdiFont font, UINT firstChar, UINT lastChar, | 
 | 			     LPABC buffer) | 
 | { | 
 |     ERR("called but we don't have FreeType\n"); | 
 |     return FALSE; | 
 | } | 
 |  | 
 | BOOL WineEngGetTextExtentPoint(GdiFont font, LPCWSTR wstr, INT count, | 
 | 			       LPSIZE size) | 
 | { | 
 |     ERR("called but we don't have FreeType\n"); | 
 |     return FALSE; | 
 | } | 
 |  | 
 | BOOL WineEngGetTextExtentPointI(GdiFont font, const WORD *indices, INT count, | 
 | 				LPSIZE size) | 
 | { | 
 |     ERR("called but we don't have FreeType\n"); | 
 |     return FALSE; | 
 | } | 
 |  | 
 | DWORD WineEngGetFontData(GdiFont font, DWORD table, DWORD offset, LPVOID buf, | 
 | 			 DWORD cbData) | 
 | { | 
 |     ERR("called but we don't have FreeType\n"); | 
 |     return GDI_ERROR; | 
 | } | 
 |  | 
 | INT WineEngGetTextFace(GdiFont font, INT count, LPWSTR str) | 
 | { | 
 |     ERR("called but we don't have FreeType\n"); | 
 |     return 0; | 
 | } | 
 |  | 
 | INT WineEngAddFontResourceEx(LPCWSTR file, DWORD flags, PVOID pdv) | 
 | { | 
 |     FIXME(":stub\n"); | 
 |     return 1; | 
 | } | 
 |  | 
 | INT WineEngRemoveFontResourceEx(LPCWSTR file, DWORD flags, PVOID pdv) | 
 | { | 
 |     FIXME(":stub\n"); | 
 |     return TRUE; | 
 | } | 
 |  | 
 | UINT WineEngGetTextCharsetInfo(GdiFont font, LPFONTSIGNATURE fs, DWORD flags) | 
 | { | 
 |     FIXME(":stub\n"); | 
 |     return DEFAULT_CHARSET; | 
 | } | 
 |  | 
 | #endif /* HAVE_FREETYPE */ |