diff options
author | Jan Djärv <jan.h.d@swipnet.se> | 2013-09-15 21:36:20 +0200 |
---|---|---|
committer | Jan Djärv <jan.h.d@swipnet.se> | 2013-09-15 21:36:20 +0200 |
commit | d93ab42eb9ee55bccc1c014ee846f8e2bc555020 (patch) | |
tree | a324c3bf0f16dd7065ca2a4e8f824ae709575ae0 /src/macfont.m | |
parent | 3fa2054efdfa3c22456072254e6c67682a595233 (diff) | |
download | emacs-d93ab42eb9ee55bccc1c014ee846f8e2bc555020.tar.gz |
Forgot to add files.
Diffstat (limited to 'src/macfont.m')
-rw-r--r-- | src/macfont.m | 3858 |
1 files changed, 3858 insertions, 0 deletions
diff --git a/src/macfont.m b/src/macfont.m new file mode 100644 index 00000000000..2a6d219d059 --- /dev/null +++ b/src/macfont.m @@ -0,0 +1,3858 @@ +/* Font driver on Mac OSX Core text. + Copyright (C) 2009-2013 Free Software Foundation, Inc. + +This file is part of GNU Emacs. + +GNU Emacs is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 3 of the License, or +(at your option) any later version. + +GNU Emacs is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +You should have received a copy of the GNU General Public License +along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>. + +Original author: YAMAMOTO Mitsuharu +*/ + +#include <config.h> + +#include "lisp.h" +#include "dispextern.h" +#include "frame.h" +#include "blockinput.h" +#include "character.h" +#include "charset.h" +#include "composite.h" +#include "fontset.h" +#include "font.h" +#include "nsgui.h" +#include "nsterm.h" +#include "macfont.h" +#include "macuvs.h" + +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 1050 + +#include <libkern/OSByteOrder.h> + +static struct font_driver macfont_driver; + +/* Core Text, for Mac OS X 10.5 and later. */ +static Lisp_Object Qmac_ct; + +static double mac_ctfont_get_advance_width_for_glyph (CTFontRef, CGGlyph); +static CGRect mac_ctfont_get_bounding_rect_for_glyph (CTFontRef, CGGlyph); +static CFArrayRef mac_ctfont_create_available_families (void); +static Boolean mac_ctfont_equal_in_postscript_name (CTFontRef, CTFontRef); +static CTLineRef mac_ctfont_create_line_with_string_and_font (CFStringRef, + CTFontRef); +static CFComparisonResult mac_font_family_compare (const void *, + const void *, void *); +static Boolean mac_ctfont_descriptor_supports_languages (CTFontDescriptorRef, + CFArrayRef); +static CFStringRef mac_ctfont_create_preferred_family_for_attributes (CFDictionaryRef); +static CFIndex mac_ctfont_shape (CTFontRef, CFStringRef, + struct mac_glyph_layout *, CFIndex); +#if USE_CT_GLYPH_INFO +static CGGlyph mac_ctfont_get_glyph_for_cid (CTFontRef, + CTCharacterCollection, + CGFontIndex); +#endif + +/* The font property key specifying the font design destination. The + value is an unsigned integer code: 0 for WYSIWIG, and 1 for Video + text. (See the documentation of X Logical Font Description + Conventions.) In the Mac font driver, 1 means the screen font is + used for calculating some glyph metrics. You can see the + difference with Monaco 8pt or 9pt, for example. */ +static Lisp_Object QCdestination; + +/* The boolean-valued font property key specifying the use of + leading. */ +static Lisp_Object QCminspace; + +struct macfont_metrics; + +/* The actual structure for Mac font that can be casted to struct font. */ + +struct macfont_info +{ + struct font font; + FontRef macfont; + CGFontRef cgfont; + ScreenFontRef screen_font; + struct macfont_cache *cache; + struct macfont_metrics **metrics; + short metrics_nrows; + unsigned synthetic_italic_p : 1; + unsigned synthetic_bold_p : 1; + unsigned spacing : 2; + unsigned antialias : 2; + unsigned color_bitmap_p : 1; +}; + +/* Values for the `spacing' member in `struct macfont_info'. */ + +enum + { + MACFONT_SPACING_PROPORTIONAL, + MACFONT_SPACING_MONO, + MACFONT_SPACING_SYNTHETIC_MONO, + }; + +/* Values for the `antialias' member in `struct macfont_info'. */ + +enum + { + MACFONT_ANTIALIAS_DEFAULT, + MACFONT_ANTIALIAS_OFF, + MACFONT_ANTIALIAS_ON, + }; + +enum {FONT_SLANT_SYNTHETIC_ITALIC = 200}; /* FC_SLANT_ITALIC + 100 */ +enum {FONT_WEIGHT_SYNTHETIC_BOLD = 200}; /* FC_WEIGHT_BOLD */ +enum {FONT_SPACING_SYNTHETIC_MONO = FONT_SPACING_MONO}; + +static const CGAffineTransform synthetic_italic_atfm = {1, 0, 0.25, 1, 0, 0}; +static const CGFloat synthetic_bold_factor = 0.024; + +static Boolean cfnumber_get_font_symbolic_traits_value (CFNumberRef, + FontSymbolicTraits *); +static void macfont_store_descriptor_attributes (FontDescriptorRef, + Lisp_Object); +static Lisp_Object macfont_descriptor_entity (FontDescriptorRef, + Lisp_Object, + FontSymbolicTraits); +static CFStringRef macfont_create_family_with_symbol (Lisp_Object); +static int macfont_glyph_extents (struct font *, CGGlyph, + struct font_metrics *, CGFloat *, int); +static CFMutableDictionaryRef macfont_create_attributes_with_spec (Lisp_Object); +static Boolean macfont_supports_charset_and_languages_p (FontDescriptorRef, + CFCharacterSetRef, + Lisp_Object, + CFArrayRef); +static CFIndex macfont_closest_traits_index (CFArrayRef, + FontSymbolicTraits); +static CFDataRef mac_font_copy_uvs_table (FontRef); +static void mac_font_get_glyphs_for_variants (CFDataRef, UTF32Char, + const UTF32Char [], + CGGlyph [], CFIndex); + +/* From CFData to a lisp string. Always returns a unibyte string. */ + +static Lisp_Object +cfdata_to_lisp (CFDataRef data) +{ + CFIndex len = CFDataGetLength (data); + Lisp_Object result = make_uninit_string (len); + + CFDataGetBytes (data, CFRangeMake (0, len), SDATA (result)); + + return result; +} + + + +/* From CFString to a lisp string. Returns a unibyte string + containing a UTF-8 byte sequence. */ + +static Lisp_Object +cfstring_to_lisp_nodecode (CFStringRef string) +{ + Lisp_Object result = Qnil; + CFDataRef data; + const char *s = CFStringGetCStringPtr (string, kCFStringEncodingUTF8); + + if (s) + { + CFIndex i, length = CFStringGetLength (string); + + for (i = 0; i < length; i++) + if (CFStringGetCharacterAtIndex (string, i) == 0) + break; + + if (i == length) + return make_unibyte_string (s, strlen (s)); + } + + data = CFStringCreateExternalRepresentation (NULL, string, + kCFStringEncodingUTF8, '?'); + if (data) + { + result = cfdata_to_lisp (data); + CFRelease (data); + } + + return result; +} + +/* Lisp string containing a UTF-8 byte sequence to CFString. Unlike + cfstring_create_with_utf8_cstring, this function preserves NUL + characters. */ + +static CFStringRef +cfstring_create_with_string_noencode (Lisp_Object s) +{ + CFStringRef string = CFStringCreateWithBytes (NULL, SDATA (s), SBYTES (s), + kCFStringEncodingUTF8, false); + + if (string == NULL) + /* Failed to interpret as UTF 8. Fall back on Mac Roman. */ + string = CFStringCreateWithBytes (NULL, SDATA (s), SBYTES (s), + kCFStringEncodingMacRoman, false); + + return string; +} + +static CGFloat +mac_screen_font_get_advance_width_for_glyph (ScreenFontRef font, CGGlyph glyph) +{ + NSSize advancement = [(NSFont *)font advancementForGlyph:glyph]; + + return advancement.width; +} + +static CGGlyph +mac_font_get_glyph_for_cid (FontRef font, CharacterCollection collection, + CGFontIndex cid) +{ +#if USE_CT_GLYPH_INFO + return mac_ctfont_get_glyph_for_cid ((CTFontRef) font, collection, cid); +#else + { + CGGlyph result = kCGFontIndexInvalid; + NSFont *nsFont = (NSFont *) font; + unichar characters[] = {0xfffd}; + NSString *string = + [NSString stringWithCharacters:characters + length:(sizeof (characters) + / sizeof (characters[0]))]; + NSGlyphInfo *glyphInfo = + [NSGlyphInfo glyphInfoWithCharacterIdentifier:cid + collection:collection + baseString:string]; + NSDictionary *attributes = + [NSDictionary dictionaryWithObjectsAndKeys:nsFont,NSFontAttributeName, + glyphInfo,NSGlyphInfoAttributeName,nil]; + NSTextStorage *textStorage = + [[NSTextStorage alloc] initWithString:string + attributes:attributes]; + NSLayoutManager *layoutManager = [[NSLayoutManager alloc] init]; + NSTextContainer *textContainer = [[NSTextContainer alloc] init]; + NSFont *fontInTextStorage; + + [layoutManager addTextContainer:textContainer]; + [textContainer release]; + [textStorage addLayoutManager:layoutManager]; + [layoutManager release]; + + /* Force layout. */ + (void) [layoutManager glyphRangeForTextContainer:textContainer]; + + fontInTextStorage = [textStorage attribute:NSFontAttributeName atIndex:0 + effectiveRange:NULL]; + if (fontInTextStorage == nsFont + || [[fontInTextStorage fontName] isEqualToString:[nsFont fontName]]) + { + NSGlyph glyph = [layoutManager glyphAtIndex:0]; + + if (glyph < [nsFont numberOfGlyphs]) + result = glyph; + } + + [textStorage release]; + + return result; + } +} +#endif + +static ScreenFontRef +mac_screen_font_create_with_name (CFStringRef name, CGFloat size) +{ + NSFont *result, *font; + + font = [NSFont fontWithName:((NSString *) name) size:size]; + result = [font screenFont]; + + return (ScreenFontRef)[result retain]; +} + + +static Boolean +mac_screen_font_get_metrics (ScreenFontRef font, CGFloat *ascent, + CGFloat *descent, CGFloat *leading) +{ + NSFont *nsFont = [(NSFont *)font printerFont]; + NSTextStorage *textStorage; + NSLayoutManager *layoutManager; + NSTextContainer *textContainer; + NSRect usedRect; + NSPoint spaceLocation; + CGFloat descender; + + textStorage = [[NSTextStorage alloc] initWithString:@" "]; + layoutManager = [[NSLayoutManager alloc] init]; + textContainer = [[NSTextContainer alloc] init]; + + [textStorage setFont:nsFont]; + [textContainer setLineFragmentPadding:0]; + [layoutManager setUsesScreenFonts:YES]; + + [layoutManager addTextContainer:textContainer]; + [textContainer release]; + [textStorage addLayoutManager:layoutManager]; + [layoutManager release]; + + if (!(textStorage && layoutManager && textContainer)) + { + [textStorage release]; + + return false; + } + + usedRect = [layoutManager lineFragmentUsedRectForGlyphAtIndex:0 + effectiveRange:NULL]; + spaceLocation = [layoutManager locationForGlyphAtIndex:0]; + [textStorage release]; + + *ascent = spaceLocation.y; + *descent = NSHeight (usedRect) - spaceLocation.y; + *leading = 0; + descender = [nsFont descender]; + if (- descender < *descent) + { + *leading = *descent + descender; + *descent = - descender; + } + + return true; +} + +static CFIndex +mac_font_shape_1 (NSFont *font, NSString *string, + struct mac_glyph_layout *glyph_layouts, CFIndex glyph_len, + BOOL screen_font_p) +{ + NSUInteger i; + CFIndex result = 0; + NSTextStorage *textStorage; + NSLayoutManager *layoutManager; + NSTextContainer *textContainer; + NSUInteger stringLength; + NSPoint spaceLocation; + NSUInteger used, numberOfGlyphs; + + textStorage = [[NSTextStorage alloc] initWithString:string]; + layoutManager = [[NSLayoutManager alloc] init]; + textContainer = [[NSTextContainer alloc] init]; + + /* Append a trailing space to measure baseline position. */ + [textStorage appendAttributedString:([[[NSAttributedString alloc] + initWithString:@" "] autorelease])]; + [textStorage setFont:font]; + [textContainer setLineFragmentPadding:0]; + [layoutManager setUsesScreenFonts:screen_font_p]; + + [layoutManager addTextContainer:textContainer]; + [textContainer release]; + [textStorage addLayoutManager:layoutManager]; + [layoutManager release]; + + if (!(textStorage && layoutManager && textContainer)) + { + [textStorage release]; + + return 0; + } + + stringLength = [string length]; + + /* Force layout. */ + (void) [layoutManager glyphRangeForTextContainer:textContainer]; + + spaceLocation = [layoutManager locationForGlyphAtIndex:stringLength]; + + /* Remove the appended trailing space because otherwise it may + generate a wrong result for a right-to-left text. */ + [textStorage beginEditing]; + [textStorage deleteCharactersInRange:(NSMakeRange (stringLength, 1))]; + [textStorage endEditing]; + (void) [layoutManager glyphRangeForTextContainer:textContainer]; + + i = 0; + while (i < stringLength) + { + NSRange range; + NSFont *fontInTextStorage = + [textStorage attribute:NSFontAttributeName atIndex:i + longestEffectiveRange:&range + inRange:(NSMakeRange (0, stringLength))]; + + if (!(fontInTextStorage == font + || [[fontInTextStorage fontName] isEqualToString:[font fontName]])) + break; + i = NSMaxRange (range); + } + if (i < stringLength) + /* Make the test `used <= glyph_len' below fail if textStorage + contained some fonts other than the specified one. */ + used = glyph_len + 1; + else + { + NSRange range = NSMakeRange (0, stringLength); + + range = [layoutManager glyphRangeForCharacterRange:range + actualCharacterRange:NULL]; + numberOfGlyphs = NSMaxRange (range); + used = numberOfGlyphs; + for (i = 0; i < numberOfGlyphs; i++) + if ([layoutManager notShownAttributeForGlyphAtIndex:i]) + used--; + } + + if (0 < used && used <= glyph_len) + { + NSUInteger glyphIndex, prevGlyphIndex; + unsigned char bidiLevel; + NSUInteger *permutation; + NSRange compRange, range; + CGFloat totalAdvance; + + glyphIndex = 0; + while ([layoutManager notShownAttributeForGlyphAtIndex:glyphIndex]) + glyphIndex++; + + /* For now we assume the direction is not changed within the + string. */ + [layoutManager getGlyphsInRange:(NSMakeRange (glyphIndex, 1)) + glyphs:NULL characterIndexes:NULL + glyphInscriptions:NULL elasticBits:NULL + bidiLevels:&bidiLevel]; + if (bidiLevel & 1) + permutation = xmalloc (sizeof (NSUInteger) * used); + else + permutation = NULL; + +#define RIGHT_TO_LEFT_P permutation + + /* Fill the `comp_range' member of struct mac_glyph_layout, and + setup a permutation for right-to-left text. */ + compRange = NSMakeRange (0, 0); + for (range = NSMakeRange (0, 0); NSMaxRange (range) < used; + range.length++) + { + struct mac_glyph_layout *gl = glyph_layouts + NSMaxRange (range); + NSUInteger characterIndex = + [layoutManager characterIndexForGlyphAtIndex:glyphIndex]; + + gl->string_index = characterIndex; + + if (characterIndex >= NSMaxRange (compRange)) + { + compRange.location = NSMaxRange (compRange); + do + { + NSRange characterRange = + [string + rangeOfComposedCharacterSequenceAtIndex:characterIndex]; + + compRange.length = + NSMaxRange (characterRange) - compRange.location; + [layoutManager glyphRangeForCharacterRange:compRange + actualCharacterRange:&characterRange]; + characterIndex = NSMaxRange (characterRange) - 1; + } + while (characterIndex >= NSMaxRange (compRange)); + + if (RIGHT_TO_LEFT_P) + for (i = 0; i < range.length; i++) + permutation[range.location + i] = NSMaxRange (range) - i - 1; + + range = NSMakeRange (NSMaxRange (range), 0); + } + + gl->comp_range.location = compRange.location; + gl->comp_range.length = compRange.length; + + while (++glyphIndex < numberOfGlyphs) + if (![layoutManager notShownAttributeForGlyphAtIndex:glyphIndex]) + break; + } + if (RIGHT_TO_LEFT_P) + for (i = 0; i < range.length; i++) + permutation[range.location + i] = NSMaxRange (range) - i - 1; + + /* Then fill the remaining members. */ + glyphIndex = prevGlyphIndex = 0; + while ([layoutManager notShownAttributeForGlyphAtIndex:glyphIndex]) + glyphIndex++; + + if (!RIGHT_TO_LEFT_P) + totalAdvance = 0; + else + { + NSUInteger nrects; + NSRect *glyphRects = + [layoutManager + rectArrayForGlyphRange:(NSMakeRange (0, numberOfGlyphs)) + withinSelectedGlyphRange:(NSMakeRange (NSNotFound, 0)) + inTextContainer:textContainer rectCount:&nrects]; + + totalAdvance = NSMaxX (glyphRects[0]); + } + + for (i = 0; i < used; i++) + { + struct mac_glyph_layout *gl; + NSPoint location; + NSUInteger nextGlyphIndex; + NSRange glyphRange; + NSRect *glyphRects; + NSUInteger nrects; + + if (!RIGHT_TO_LEFT_P) + gl = glyph_layouts + i; + else + { + NSUInteger dest = permutation[i]; + + gl = glyph_layouts + dest; + if (i < dest) + { + CFIndex tmp = gl->string_index; + + gl->string_index = glyph_layouts[i].string_index; + glyph_layouts[i].string_index = tmp; + } + } + gl->glyph_id = [layoutManager glyphAtIndex:glyphIndex]; + + location = [layoutManager locationForGlyphAtIndex:glyphIndex]; + gl->baseline_delta = spaceLocation.y - location.y; + + for (nextGlyphIndex = glyphIndex + 1; nextGlyphIndex < numberOfGlyphs; + nextGlyphIndex++) + if (![layoutManager + notShownAttributeForGlyphAtIndex:nextGlyphIndex]) + break; + + if (!RIGHT_TO_LEFT_P) + { + CGFloat maxX; + + if (prevGlyphIndex == 0) + glyphRange = NSMakeRange (0, nextGlyphIndex); + else + glyphRange = NSMakeRange (glyphIndex, + nextGlyphIndex - glyphIndex); + glyphRects = + [layoutManager + rectArrayForGlyphRange:glyphRange + withinSelectedGlyphRange:(NSMakeRange (NSNotFound, 0)) + inTextContainer:textContainer rectCount:&nrects]; + maxX = max (NSMaxX (glyphRects[0]), totalAdvance); + gl->advance_delta = location.x - totalAdvance; + gl->advance = maxX - totalAdvance; + totalAdvance = maxX; + } + else + { + CGFloat minX; + + if (nextGlyphIndex == numberOfGlyphs) + glyphRange = NSMakeRange (prevGlyphIndex, + numberOfGlyphs - prevGlyphIndex); + else + glyphRange = NSMakeRange (prevGlyphIndex, + glyphIndex + 1 - prevGlyphIndex); + glyphRects = + [layoutManager + rectArrayForGlyphRange:glyphRange + withinSelectedGlyphRange:(NSMakeRange (NSNotFound, 0)) + inTextContainer:textContainer rectCount:&nrects]; + minX = min (NSMinX (glyphRects[0]), totalAdvance); + gl->advance = totalAdvance - minX; + totalAdvance = minX; + gl->advance_delta = location.x - totalAdvance; + } + + prevGlyphIndex = glyphIndex + 1; + glyphIndex = nextGlyphIndex; + } + + if (RIGHT_TO_LEFT_P) + xfree (permutation); + +#undef RIGHT_TO_LEFT_P + + result = used; + } + [textStorage release]; + + return result; +} + +static CFIndex +mac_screen_font_shape (ScreenFontRef font, CFStringRef string, + struct mac_glyph_layout *glyph_layouts, + CFIndex glyph_len) +{ + return mac_font_shape_1 ([(NSFont *)font printerFont], + (NSString *) string, + glyph_layouts, glyph_len, YES); +} + +static CGColorRef +get_cgcolor(unsigned long idx, struct frame *f) +{ + NSColor *nsColor = ns_lookup_indexed_color (idx, f); + [nsColor set]; + CGColorSpaceRef colorSpace = [[nsColor colorSpace] CGColorSpace]; + NSInteger noc = [nsColor numberOfComponents]; + CGFloat *components = xmalloc (sizeof(CGFloat)*(1+noc)); + CGColorRef cgColor; + + [nsColor getComponents: components]; + cgColor = CGColorCreate (colorSpace, components); + xfree (components); + return cgColor; +} + +#define CG_SET_FILL_COLOR_WITH_GC_FOREGROUND(context, s) \ + CGContextSetFillColorWithColor (context, \ + get_cgcolor (NS_FACE_FOREGROUND (s->face), \ + s->f)) + +#define CG_SET_FILL_COLOR_WITH_GC_BACKGROUND(context, s) \ + CGContextSetFillColorWithColor (context, \ + get_cgcolor (NS_FACE_BACKGROUND (s->face), \ + s->f)) + +#define CG_SET_STROKE_COLOR_WITH_GC_FOREGROUND(context, s) \ + CGContextSetStrokeColorWithColor (context, \ + get_cgcolor (NS_FACE_FOREGROUND (s->face),\ + s->f)) + + +/* Mac font driver. */ + +static struct +{ + /* registry name */ + const char *name; + /* characters to distinguish the charset from the others */ + int uniquifier[6]; + /* additional constraint by language */ + CFStringRef lang; + /* set on demand */ + CFCharacterSetRef cf_charset; + CFStringRef cf_charset_string; +} cf_charset_table[] = + { { "iso8859-1", { 0x00A0, 0x00A1, 0x00B4, 0x00BC, 0x00D0 } }, + { "iso8859-2", { 0x00A0, 0x010E }}, + { "iso8859-3", { 0x00A0, 0x0108 }}, + { "iso8859-4", { 0x00A0, 0x00AF, 0x0128, 0x0156, 0x02C7 }}, + { "iso8859-5", { 0x00A0, 0x0401 }}, + { "iso8859-6", { 0x00A0, 0x060C }}, + { "iso8859-7", { 0x00A0, 0x0384 }}, + { "iso8859-8", { 0x00A0, 0x05D0 }}, + { "iso8859-9", { 0x00A0, 0x00A1, 0x00BC, 0x011E }}, + { "iso8859-10", { 0x00A0, 0x00D0, 0x0128, 0x2015 }}, + { "iso8859-11", { 0x00A0, 0x0E01 }}, + { "iso8859-13", { 0x00A0, 0x201C }}, + { "iso8859-14", { 0x00A0, 0x0174 }}, + { "iso8859-15", { 0x00A0, 0x00A1, 0x00D0, 0x0152 }}, + { "iso8859-16", { 0x00A0, 0x0218}}, + { "gb2312.1980-0", { 0x4E13 }, CFSTR ("zh-Hans")}, + { "big5-0", { /* 0xF6B1 in ftfont.c */ 0xF7E5 }, CFSTR ("zh-Hant") }, + { "jisx0208.1983-0", { 0x4E55 }, CFSTR ("ja")}, + { "ksc5601.1987-0", { 0xAC00 }, CFSTR ("ko")}, + { "cns11643.1992-1", { 0xFE32 }, CFSTR ("zh-Hant")}, + { "cns11643.1992-2", { 0x4E33, 0x7934 }}, + { "cns11643.1992-3", { 0x201A9 }}, + { "cns11643.1992-4", { 0x20057 }}, + { "cns11643.1992-5", { 0x20000 }}, + { "cns11643.1992-6", { 0x20003 }}, + { "cns11643.1992-7", { 0x20055 }}, + { "gbk-0", { 0x4E06 }, CFSTR ("zh-Hans")}, + { "jisx0212.1990-0", { 0x4E44 }}, + { "jisx0213.2000-1", { 0xFA10 }, CFSTR ("ja")}, + { "jisx0213.2000-2", { 0xFA49 }}, + { "jisx0213.2004-1", { 0x20B9F }}, + { "viscii1.1-1", { 0x1EA0, 0x1EAE, 0x1ED2 }, CFSTR ("vi")}, + { "tis620.2529-1", { 0x0E01 }, CFSTR ("th")}, + { "windows-1251", { 0x0401, 0x0490 }, CFSTR ("ru")}, + { "koi8-r", { 0x0401, 0x2219 }, CFSTR ("ru")}, + { "mulelao-1", { 0x0E81 }, CFSTR ("lo")}, + { "unicode-sip", { 0x20000 }}, + { NULL } + }; + +static CGFloat macfont_antialias_threshold; + +void +macfont_update_antialias_threshold (void) +{ + int threshold; + Boolean valid_p; + + threshold = + CFPreferencesGetAppIntegerValue (CFSTR ("AppleAntiAliasingThreshold"), + kCFPreferencesCurrentApplication, + &valid_p); + if (valid_p) + macfont_antialias_threshold = threshold; +} + +static inline Lisp_Object +macfont_intern_prop_cfstring (CFStringRef cfstring) +{ + Lisp_Object string = cfstring_to_lisp_nodecode (cfstring); + + return font_intern_prop (SSDATA (string), SBYTES (string), 1); +} + +static inline CFIndex +macfont_store_utf32char_to_unichars (UTF32Char c, UniChar *unichars) +{ + if (c < 0x10000) + { + unichars[0] = c; + + return 1; + } + else + { + c -= 0x10000; + unichars[0] = (c >> 10) + 0xD800; + unichars[1] = (c & 0x3FF) + 0xDC00; + + return 2; + } +} + +static Boolean +cfnumber_get_font_symbolic_traits_value (CFNumberRef number, + FontSymbolicTraits *sym_traits) +{ + SInt64 sint64_value; + + /* Getting symbolic traits with kCFNumberSInt32Type is lossy on Mac + OS 10.6 when the value is greater than or equal to 1 << 31. */ + if (CFNumberGetValue (number, kCFNumberSInt64Type, &sint64_value)) + { + *sym_traits = (FontSymbolicTraits) sint64_value; + + return true; + } + + return false; +} + +static void +macfont_store_descriptor_attributes (FontDescriptorRef desc, + Lisp_Object spec_or_entity) +{ + CFStringRef str; + CFDictionaryRef dict; + CFNumberRef num; + CGFloat floatval; + + str = mac_font_descriptor_copy_attribute (desc, + MAC_FONT_FAMILY_NAME_ATTRIBUTE); + if (str) + { + ASET (spec_or_entity, FONT_FAMILY_INDEX, + macfont_intern_prop_cfstring (str)); + CFRelease (str); + } + dict = mac_font_descriptor_copy_attribute (desc, MAC_FONT_TRAITS_ATTRIBUTE); + if (dict) + { + struct { + enum font_property_index index; + CFStringRef trait; + CGPoint points[6]; + } numeric_traits[] = + {{FONT_WEIGHT_INDEX, MAC_FONT_WEIGHT_TRAIT, + {{-0.4, 50}, /* light */ + {-0.24, 87.5}, /* (semi-light + normal) / 2 */ + {0, 100}, /* normal */ + {0.24, 140}, /* (semi-bold + normal) / 2 */ + {0.4, 200}, /* bold */ + {CGFLOAT_MAX, CGFLOAT_MAX}}}, + {FONT_SLANT_INDEX, MAC_FONT_SLANT_TRAIT, + {{0, 100}, {0.1, 200}, {CGFLOAT_MAX, CGFLOAT_MAX}}}, + {FONT_WIDTH_INDEX, MAC_FONT_WIDTH_TRAIT, + {{0, 100}, {1, 200}, {CGFLOAT_MAX, CGFLOAT_MAX}}}}; + int i; + + for (i = 0; i < sizeof (numeric_traits) / sizeof (numeric_traits[0]); i++) + { + num = CFDictionaryGetValue (dict, numeric_traits[i].trait); + if (num && CFNumberGetValue (num, kCFNumberCGFloatType, &floatval)) + { + CGPoint *point = numeric_traits[i].points; + + while (point->x < floatval) + point++; + if (point == numeric_traits[i].points) + point++; + else if (point->x == CGFLOAT_MAX) + point--; + floatval = (point - 1)->y + ((floatval - (point - 1)->x) + * ((point->y - (point - 1)->y) + / (point->x - (point - 1)->x))); + FONT_SET_STYLE (spec_or_entity, numeric_traits[i].index, + make_number (lround (floatval))); + } + } + + num = CFDictionaryGetValue (dict, MAC_FONT_SYMBOLIC_TRAIT); + if (num) + { + FontSymbolicTraits sym_traits; + int spacing; + + cfnumber_get_font_symbolic_traits_value (num, &sym_traits); + spacing = (sym_traits & MAC_FONT_TRAIT_MONO_SPACE + ? FONT_SPACING_MONO : FONT_SPACING_PROPORTIONAL); + ASET (spec_or_entity, FONT_SPACING_INDEX, make_number (spacing)); + } + + CFRelease (dict); + } + num = mac_font_descriptor_copy_attribute (desc, MAC_FONT_SIZE_ATTRIBUTE); + if (num && CFNumberGetValue (num, kCFNumberCGFloatType, &floatval)) + ASET (spec_or_entity, FONT_SIZE_INDEX, make_number (floatval)); + else + ASET (spec_or_entity, FONT_SIZE_INDEX, make_number (0)); + if (num) + CFRelease (num); +} + +static Lisp_Object +macfont_descriptor_entity (FontDescriptorRef desc, Lisp_Object extra, + FontSymbolicTraits synth_sym_traits) +{ + Lisp_Object entity; + CFDictionaryRef dict; + FontSymbolicTraits sym_traits = 0; + CFStringRef name; + + entity = font_make_entity (); + + ASET (entity, FONT_TYPE_INDEX, macfont_driver.type); + ASET (entity, FONT_REGISTRY_INDEX, Qiso10646_1); + + macfont_store_descriptor_attributes (desc, entity); + + dict = mac_font_descriptor_copy_attribute (desc, MAC_FONT_TRAITS_ATTRIBUTE); + if (dict) + { + CFNumberRef num = CFDictionaryGetValue (dict, MAC_FONT_SYMBOLIC_TRAIT); + + if (num) + cfnumber_get_font_symbolic_traits_value (num, &sym_traits); + CFRelease (dict); + } + if (EQ (AREF (entity, FONT_SIZE_INDEX), make_number (0))) + ASET (entity, FONT_AVGWIDTH_INDEX, make_number (0)); + ASET (entity, FONT_EXTRA_INDEX, Fcopy_sequence (extra)); + name = mac_font_descriptor_copy_attribute (desc, MAC_FONT_NAME_ATTRIBUTE); + font_put_extra (entity, QCfont_entity, + make_save_ptr_int ((void *) name, sym_traits)); + if (synth_sym_traits & MAC_FONT_TRAIT_ITALIC) + FONT_SET_STYLE (entity, FONT_SLANT_INDEX, + make_number (FONT_SLANT_SYNTHETIC_ITALIC)); + if (synth_sym_traits & MAC_FONT_TRAIT_BOLD) + FONT_SET_STYLE (entity, FONT_WEIGHT_INDEX, + make_number (FONT_WEIGHT_SYNTHETIC_BOLD)); + if (synth_sym_traits & MAC_FONT_TRAIT_MONO_SPACE) + ASET (entity, FONT_SPACING_INDEX, + make_number (FONT_SPACING_SYNTHETIC_MONO)); + + return entity; +} + +static CFStringRef +macfont_create_family_with_symbol (Lisp_Object symbol) +{ + static CFArrayRef families = NULL; + CFStringRef result = NULL, family_name; + int using_cache_p = 1; + CFComparatorFunction family_name_comparator; + + family_name = cfstring_create_with_string_noencode (SYMBOL_NAME (symbol)); + if (family_name == NULL) + return NULL; + +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 1060 +#if MAC_OS_X_VERSION_MIN_REQUIRED < 1060 + if (CTFontManagerCompareFontFamilyNames != NULL) +#endif + { + family_name_comparator = CTFontManagerCompareFontFamilyNames; + } +#if MAC_OS_X_VERSION_MIN_REQUIRED < 1060 + else /* CTFontManagerCompareFontFamilyNames == NULL */ +#endif +#endif /* MAC_OS_X_VERSION_MAX_ALLOWED >= 1060 */ +#if MAC_OS_X_VERSION_MIN_REQUIRED < 1060 + { + family_name_comparator = mac_font_family_compare; + } +#endif + + if ((*family_name_comparator) (family_name, CFSTR ("LastResort"), NULL) + == kCFCompareEqualTo) + result = CFSTR ("LastResort"); + else + while (1) + { + CFIndex i, count; + + if (families == NULL) + { + families = mac_font_create_available_families (); + using_cache_p = 0; + if (families == NULL) + break; + } + + count = CFArrayGetCount (families); + i = CFArrayBSearchValues (families, CFRangeMake (0, count), + (const void *) family_name, + family_name_comparator, NULL); + if (i < count) + { + CFStringRef name = CFArrayGetValueAtIndex (families, i); + + if ((*family_name_comparator) (name, family_name, NULL) + == kCFCompareEqualTo) + result = CFRetain (name); + } + + if (result || !using_cache_p) + break; + else + { + CFRelease (families); + families = NULL; + } + } + + CFRelease (family_name); + + return result; +} + +#define WIDTH_FRAC_BITS (4) +#define WIDTH_FRAC_SCALE (2 * ((1 << (WIDTH_FRAC_BITS - 1)) - 1)) + +struct macfont_metrics +{ + unsigned char lbearing_low, rbearing_low; + signed lbearing_high : 4, rbearing_high : 4; + unsigned char ascent_low, descent_low; + signed ascent_high : 4, descent_high : 4; + + /* These two members are used for fixed-point representation of + glyph width. The `width_int' member is an integer that is + closest to the width. The `width_frac' member is the fractional + adjustment representing a value in [-.5, .5], multiplied by + WIDTH_FRAC_SCALE. For synthetic monospace fonts, they represent + the advance delta for centering instead of the glyph width. */ + signed width_frac : WIDTH_FRAC_BITS, width_int : 16 - WIDTH_FRAC_BITS; +}; + +#define METRICS_VALUE(metrics, member) \ + (((metrics)->member##_high << 8) | (metrics)->member##_low) +#define METRICS_SET_VALUE(metrics, member, value) \ + do {short tmp = (value); (metrics)->member##_low = tmp & 0xff; \ + (metrics)->member##_high = tmp >> 8;} while (0) + +enum metrics_status + { + METRICS_INVALID = -1, /* metrics entry is invalid */ + METRICS_WIDTH_VALID = -2 /* width is valid but others are invalid */ + }; + +#define METRICS_STATUS(metrics) \ + (METRICS_VALUE (metrics, ascent) + METRICS_VALUE (metrics, descent)) +#define METRICS_SET_STATUS(metrics, status) \ + do {METRICS_SET_VALUE (metrics, ascent, 0); \ + METRICS_SET_VALUE (metrics, descent, status);} while (0) + +#define METRICS_NCOLS_PER_ROW (128) +#define LCD_FONT_SMOOTHING_LEFT_MARGIN (0.396f) +#define LCD_FONT_SMOOTHING_RIGHT_MARGIN (0.396f) + +static int +macfont_glyph_extents (struct font *font, CGGlyph glyph, + struct font_metrics *metrics, CGFloat *advance_delta, + int force_integral_p) +{ + struct macfont_info *macfont_info = (struct macfont_info *) font; + FontRef macfont = macfont_info->macfont; + int row, col; + struct macfont_metrics *cache; + int width; + + row = glyph / METRICS_NCOLS_PER_ROW; + col = glyph % METRICS_NCOLS_PER_ROW; + if (row >= macfont_info->metrics_nrows) + { + macfont_info->metrics = + xrealloc (macfont_info->metrics, + sizeof (struct macfont_metrics *) * (row + 1)); + memset (macfont_info->metrics + macfont_info->metrics_nrows, 0, + (sizeof (struct macfont_metrics *) + * (row + 1 - macfont_info->metrics_nrows))); + macfont_info->metrics_nrows = row + 1; + } + if (macfont_info->metrics[row] == NULL) + { + struct macfont_metrics *new; + int i; + + new = xmalloc (sizeof (struct macfont_metrics) * METRICS_NCOLS_PER_ROW); + for (i = 0; i < METRICS_NCOLS_PER_ROW; i++) + METRICS_SET_STATUS (new + i, METRICS_INVALID); + macfont_info->metrics[row] = new; + } + cache = macfont_info->metrics[row] + col; + + if (METRICS_STATUS (cache) == METRICS_INVALID) + { + CGFloat fwidth; + + if (macfont_info->screen_font) + fwidth = mac_screen_font_get_advance_width_for_glyph (macfont_info->screen_font, glyph); + else + fwidth = mac_font_get_advance_width_for_glyph (macfont, glyph); + + /* For synthetic mono fonts, cache->width_{int,frac} holds the + advance delta value. */ + if (macfont_info->spacing == MACFONT_SPACING_SYNTHETIC_MONO) + fwidth = (font->pixel_size - fwidth) / 2; + cache->width_int = lround (fwidth); + cache->width_frac = lround ((fwidth - cache->width_int) + * WIDTH_FRAC_SCALE); + METRICS_SET_STATUS (cache, METRICS_WIDTH_VALID); + } + if (macfont_info->spacing == MACFONT_SPACING_SYNTHETIC_MONO) + width = font->pixel_size; + else + width = cache->width_int; + + if (metrics) + { + if (METRICS_STATUS (cache) == METRICS_WIDTH_VALID) + { + CGRect bounds = mac_font_get_bounding_rect_for_glyph (macfont, glyph); + + if (macfont_info->synthetic_italic_p) + { + /* We assume the members a, b, c, and d in + synthetic_italic_atfm are non-negative. */ + bounds.origin = + CGPointApplyAffineTransform (bounds.origin, + synthetic_italic_atfm); + bounds.size = + CGSizeApplyAffineTransform (bounds.size, synthetic_italic_atfm); + } + if (macfont_info->synthetic_bold_p) + { + CGFloat d = + - synthetic_bold_factor * mac_font_get_size (macfont) / 2; + + bounds = CGRectInset (bounds, d, d); + } + switch (macfont_info->spacing) + { + case MACFONT_SPACING_PROPORTIONAL: + bounds.origin.x += - (cache->width_frac + / (CGFloat) (WIDTH_FRAC_SCALE * 2)); + break; + case MACFONT_SPACING_MONO: + break; + case MACFONT_SPACING_SYNTHETIC_MONO: + bounds.origin.x += (cache->width_int + + (cache->width_frac + / (CGFloat) WIDTH_FRAC_SCALE)); + break; + } + if (bounds.size.width > 0) + { + bounds.origin.x -= LCD_FONT_SMOOTHING_LEFT_MARGIN; + bounds.size.width += (LCD_FONT_SMOOTHING_LEFT_MARGIN + + LCD_FONT_SMOOTHING_RIGHT_MARGIN); + } + bounds = CGRectIntegral (bounds); + METRICS_SET_VALUE (cache, lbearing, CGRectGetMinX (bounds)); + METRICS_SET_VALUE (cache, rbearing, CGRectGetMaxX (bounds)); + METRICS_SET_VALUE (cache, ascent, CGRectGetMaxY (bounds)); + METRICS_SET_VALUE (cache, descent, -CGRectGetMinY (bounds)); + } + metrics->lbearing = METRICS_VALUE (cache, lbearing); + metrics->rbearing = METRICS_VALUE (cache, rbearing); + metrics->width = width; + metrics->ascent = METRICS_VALUE (cache, ascent); + metrics->descent = METRICS_VALUE (cache, descent); + } + + if (advance_delta) + { + switch (macfont_info->spacing) + { + case MACFONT_SPACING_PROPORTIONAL: + *advance_delta = (force_integral_p ? 0 + : - (cache->width_frac + / (CGFloat) (WIDTH_FRAC_SCALE * 2))); + break; + case MACFONT_SPACING_MONO: + *advance_delta = 0; + break; + case MACFONT_SPACING_SYNTHETIC_MONO: + *advance_delta = (force_integral_p ? cache->width_int + : (cache->width_int + + (cache->width_frac + / (CGFloat) WIDTH_FRAC_SCALE))); + break; + } + } + + return width; +} + +static CFMutableDictionaryRef macfont_cache_dictionary; + +/* Threshold used in row_nkeys_or_perm. This must be less than or + equal to the number of rows that are invalid as BMP (i.e., from + U+D800 to U+DFFF). */ +#define ROW_PERM_OFFSET (8) + +/* The number of glyphs that can be stored in a value for a single + entry of CFDictionary. */ +#define NGLYPHS_IN_VALUE (sizeof (void *) / sizeof (CGGlyph)) + +struct macfont_cache +{ + int reference_count; + CFCharacterSetRef cf_charset; + struct { + /* The cached glyph for a BMP character c is stored in + matrix[row_nkeys_or_perm[c / 256] - ROW_PERM_OFFSET][c % 256] + if row_nkeys_or_perm[c / 256] >= ROW_PERM_OFFSET. */ + unsigned char row_nkeys_or_perm[256]; + CGGlyph **matrix; + + /* Number of rows for which the BMP cache is allocated so far. + I.e., matrix[0] ... matrix[nrows - 1] are non-NULL. */ + int nrows; + + /* The cached glyph for a character c is stored as the (c % + NGLYPHS_IN_VALUE)-th CGGlyph block of a value for the key (c / + NGLYPHS_IN_VALUE). However, the glyph for a BMP characrer c is + not stored here if row_nkeys_or_perm[c / 256] >= + ROW_PERM_OFFSET. */ + CFMutableDictionaryRef dictionary; + } glyph; + + struct { + /* UVS (Unicode Variation Sequence) subtable data, which is of + type CFDataRef if available. NULL means it is not initialized + yet. kCFNull means the subtable is not found and there is no + suitable fallback table for this font. */ + CFTypeRef table; + + /* Character collection specifying the destination of the mapping + provided by `table' above. If `table' is obtained from the UVS + subtable in the font cmap table, then the value of this member + should be MAC_CHARACTER_COLLECTION_IDENTITY_MAPPING. */ + CharacterCollection collection; + } uvs; +}; + +static struct macfont_cache *macfont_lookup_cache (CFStringRef); +static struct macfont_cache *macfont_retain_cache (struct macfont_cache *); +static void macfont_release_cache (struct macfont_cache *); +static CFCharacterSetRef macfont_get_cf_charset (struct font *); +static CFCharacterSetRef macfont_get_cf_charset_for_name (CFStringRef); +static CGGlyph macfont_get_glyph_for_character (struct font *, UTF32Char); +static CGGlyph macfont_get_glyph_for_cid (struct font *font, + CharacterCollection, CGFontIndex); +static CFDataRef macfont_get_uvs_table (struct font *, CharacterCollection *); + +static struct macfont_cache * +macfont_lookup_cache (CFStringRef key) +{ + struct macfont_cache *cache; + + if (macfont_cache_dictionary == NULL) + { + macfont_cache_dictionary = + CFDictionaryCreateMutable (NULL, 0, + &kCFTypeDictionaryKeyCallBacks, NULL); + cache = NULL; + } + else + cache = ((struct macfont_cache *) + CFDictionaryGetValue (macfont_cache_dictionary, key)); + + if (cache == NULL) + { + FontRef macfont = mac_font_create_with_name (key, 0); + + if (macfont) + { + cache = xzalloc (sizeof (struct macfont_cache)); + /* Treat the LastResort font as if it contained glyphs for + all characters. This may look too rough, but neither + CTFontCopyCharacterSet nor -[NSFont coveredCharacterSet] + for this font is correct for non-BMP characters on Mac OS + X 10.5, anyway. */ + if (CFStringCompare (key, CFSTR ("LastResort"), 0) + == kCFCompareEqualTo) + { + CFRange range = CFRangeMake (0, MAX_UNICODE_CHAR + 1); + + cache->cf_charset = + CFCharacterSetCreateWithCharactersInRange (NULL, range); + } + if (cache->cf_charset == NULL) + cache->cf_charset = mac_font_copy_character_set (macfont); + CFDictionaryAddValue (macfont_cache_dictionary, key, + (const void *) cache); + CFRelease (macfont); + } + } + + return cache; +} + +static struct macfont_cache * +macfont_retain_cache (struct macfont_cache *cache) +{ + cache->reference_count++; + + return cache; +} + +static void +macfont_release_cache (struct macfont_cache *cache) +{ + if (--cache->reference_count == 0) + { + int i; + + for (i = 0; i < cache->glyph.nrows; i++) + xfree (cache->glyph.matrix[i]); + xfree (cache->glyph.matrix); + if (cache->glyph.dictionary) + CFRelease (cache->glyph.dictionary); + memset (&cache->glyph, 0, sizeof (cache->glyph)); + if (cache->uvs.table) + CFRelease (cache->uvs.table); + memset (&cache->uvs, 0, sizeof (cache->uvs)); + } +} + +static CFCharacterSetRef +macfont_get_cf_charset (struct font *font) +{ + struct macfont_info *macfont_info = (struct macfont_info *) font; + + return macfont_info->cache->cf_charset; +} + +static CFCharacterSetRef +macfont_get_cf_charset_for_name (CFStringRef name) +{ + struct macfont_cache *cache = macfont_lookup_cache (name); + + return cache->cf_charset; +} + +static CGGlyph +macfont_get_glyph_for_character (struct font *font, UTF32Char c) +{ + struct macfont_info *macfont_info = (struct macfont_info *) font; + FontRef macfont = macfont_info->macfont; + struct macfont_cache *cache = macfont_info->cache; + + if (c < 0xD800 || (c > 0xDFFF && c < 0x10000)) + { + int row = c / 256; + int nkeys_or_perm = cache->glyph.row_nkeys_or_perm[row]; + + if (nkeys_or_perm < ROW_PERM_OFFSET) + { + UniChar unichars[256], ch; + CGGlyph *glyphs; + int i, len; + int nrows; +#if MAC_OS_X_VERSION_MIN_REQUIRED >= 1060 + dispatch_queue_t queue; + dispatch_group_t group = NULL; +#else + int nkeys; +#endif + + if (row != 0) + { + CFMutableDictionaryRef dictionary; + uintptr_t key, value; + int nshifts; + CGGlyph glyph; + + if (cache->glyph.dictionary == NULL) + cache->glyph.dictionary = + CFDictionaryCreateMutable (NULL, 0, NULL, NULL); + dictionary = cache->glyph.dictionary; + key = c / NGLYPHS_IN_VALUE; + nshifts = ((c % NGLYPHS_IN_VALUE) * sizeof (CGGlyph) * 8); + value = ((uintptr_t) + CFDictionaryGetValue (dictionary, (const void *) key)); + glyph = (value >> nshifts); + if (glyph) + return glyph; + + if (nkeys_or_perm + 1 != ROW_PERM_OFFSET) + { + ch = c; + if (!mac_font_get_glyphs_for_characters (macfont, &ch, + &glyph, 1) + || glyph == 0) + glyph = kCGFontIndexInvalid; + + if (value == 0) + cache->glyph.row_nkeys_or_perm[row] = nkeys_or_perm + 1; + value |= ((uintptr_t) glyph << nshifts); + CFDictionarySetValue (dictionary, (const void *) key, + (const void *) value); + + return glyph; + } + +#if MAC_OS_X_VERSION_MIN_REQUIRED >= 1060 + queue = + dispatch_get_global_queue (DISPATCH_QUEUE_PRIORITY_DEFAULT, 0); + group = dispatch_group_create (); + dispatch_group_async (group, queue, ^{ + int nkeys; + uintptr_t key; +#endif + nkeys = nkeys_or_perm; + for (key = row * (256 / NGLYPHS_IN_VALUE); ; key++) + if (CFDictionaryContainsKey (dictionary, + (const void *) key)) + { + CFDictionaryRemoveValue (dictionary, + (const void *) key); + if (--nkeys == 0) + break; + } +#if MAC_OS_X_VERSION_MIN_REQUIRED >= 1060 + }); +#endif + } + + len = 0; + for (i = 0; i < 256; i++) + { + ch = row * 256 + i; + if (CFCharacterSetIsLongCharacterMember (cache->cf_charset, ch)) + unichars[len++] = ch; + } + + glyphs = xmalloc (sizeof (CGGlyph) * 256); + if (len > 0) + { + mac_font_get_glyphs_for_characters (macfont, unichars, + glyphs, len); + while (i > len) + { + int next = unichars[len - 1] % 256; + + while (--i > next) + glyphs[i] = kCGFontIndexInvalid; + + len--; + glyphs[i] = glyphs[len]; + if (len == 0) + break; + } + } + if (i > len) + while (i-- > 0) + glyphs[i] = kCGFontIndexInvalid; + + nrows = cache->glyph.nrows; + nkeys_or_perm = nrows + ROW_PERM_OFFSET; + cache->glyph.row_nkeys_or_perm[row] = nkeys_or_perm; + nrows++; + cache->glyph.matrix = xrealloc (cache->glyph.matrix, + sizeof (CGGlyph *) * nrows); + cache->glyph.matrix[nrows - 1] = glyphs; + cache->glyph.nrows = nrows; + +#if MAC_OS_X_VERSION_MIN_REQUIRED >= 1060 + if (group) + { + dispatch_group_wait (group, DISPATCH_TIME_FOREVER); + dispatch_release (group); + } +#endif + } + + return cache->glyph.matrix[nkeys_or_perm - ROW_PERM_OFFSET][c % 256]; + } + else + { + uintptr_t key, value; + int nshifts; + CGGlyph glyph; + + if (cache->glyph.dictionary == NULL) + cache->glyph.dictionary = + CFDictionaryCreateMutable (NULL, 0, NULL, NULL); + key = c / NGLYPHS_IN_VALUE; + nshifts = ((c % NGLYPHS_IN_VALUE) * sizeof (CGGlyph) * 8); + value = (uintptr_t) CFDictionaryGetValue (cache->glyph.dictionary, + (const void *) key); + glyph = (value >> nshifts); + if (glyph == 0) + { + UniChar unichars[2]; + CGGlyph glyphs[2]; + CFIndex count = macfont_store_utf32char_to_unichars (c, unichars); + + if (mac_font_get_glyphs_for_characters (macfont, unichars, glyphs, + count)) + glyph = glyphs[0]; + if (glyph == 0) + glyph = kCGFontIndexInvalid; + + value |= ((uintptr_t) glyph << nshifts); + CFDictionarySetValue (cache->glyph.dictionary, + (const void *) key, (const void *) value); + } + + return glyph; + } +} + +static CGGlyph +macfont_get_glyph_for_cid (struct font *font, CharacterCollection collection, + CGFontIndex cid) +{ + struct macfont_info *macfont_info = (struct macfont_info *) font; + FontRef macfont = macfont_info->macfont; + + /* Cache it? */ + return mac_font_get_glyph_for_cid (macfont, collection, cid); +} + +static CFDataRef +macfont_get_uvs_table (struct font *font, CharacterCollection *collection) +{ + struct macfont_info *macfont_info = (struct macfont_info *) font; + FontRef macfont = macfont_info->macfont; + struct macfont_cache *cache = macfont_info->cache; + CFDataRef result = NULL; + + if (cache->uvs.table == NULL) + { + CFDataRef uvs_table = mac_font_copy_uvs_table (macfont); + CharacterCollection uvs_collection = + MAC_CHARACTER_COLLECTION_IDENTITY_MAPPING; + + if (uvs_table == NULL + && mac_font_get_glyph_for_cid (macfont, + MAC_CHARACTER_COLLECTION_ADOBE_JAPAN1, + 6480) != kCGFontIndexInvalid) + { + /* If the glyph for U+4E55 is accessible via its CID 6480, + then we use the Adobe-Japan1 UVS table, which maps a + variation sequence to a CID, as a fallback. */ + static CFDataRef mac_uvs_table_adobe_japan1 = NULL; + + if (mac_uvs_table_adobe_japan1 == NULL) + mac_uvs_table_adobe_japan1 = + CFDataCreateWithBytesNoCopy (NULL, + mac_uvs_table_adobe_japan1_bytes, + sizeof (mac_uvs_table_adobe_japan1_bytes), + kCFAllocatorNull); + if (mac_uvs_table_adobe_japan1) + { + uvs_table = CFRetain (mac_uvs_table_adobe_japan1); + uvs_collection = MAC_CHARACTER_COLLECTION_ADOBE_JAPAN1; + } + } + if (uvs_table == NULL) + cache->uvs.table = kCFNull; + else + cache->uvs.table = uvs_table; + cache->uvs.collection = uvs_collection; + } + + if (cache->uvs.table != kCFNull) + { + result = cache->uvs.table; + *collection = cache->uvs.collection; + } + + return result; +} + +static Lisp_Object macfont_get_cache (struct frame *); +static Lisp_Object macfont_list (struct frame *, Lisp_Object); +static Lisp_Object macfont_match (struct frame *, Lisp_Object); +static Lisp_Object macfont_list_family (struct frame *); +static void macfont_free_entity (Lisp_Object); +static Lisp_Object macfont_open (struct frame *, Lisp_Object, int); +static void macfont_close (struct frame *, struct font *); +static int macfont_has_char (Lisp_Object, int); +static unsigned macfont_encode_char (struct font *, int); +static int macfont_text_extents (struct font *, unsigned int *, int, + struct font_metrics *); +static int macfont_draw (struct glyph_string *, int, int, int, int, bool); +static Lisp_Object macfont_shape (Lisp_Object); +static int macfont_variation_glyphs (struct font *, int c, + unsigned variations[256]); +static void macfont_filter_properties (Lisp_Object, Lisp_Object); + +static struct font_driver macfont_driver = + { + LISP_INITIALLY_ZERO, /* Qmac_ct */ + 0, /* case insensitive */ + macfont_get_cache, + macfont_list, + macfont_match, + macfont_list_family, + macfont_free_entity, + macfont_open, + macfont_close, + NULL, /* prepare_face */ + NULL, /* done_face */ + macfont_has_char, + macfont_encode_char, + macfont_text_extents, + macfont_draw, + NULL, /* get_bitmap */ + NULL, /* free_bitmap */ + NULL, /* get_outline */ + NULL, /* free_outline */ + NULL, /* anchor_point */ + NULL, /* otf_capability */ + NULL, /* otf_drive */ + NULL, /* start_for_frame */ + NULL, /* end_for_frame */ + macfont_shape, + NULL, /* check */ + macfont_variation_glyphs, + macfont_filter_properties, + }; + +static Lisp_Object +macfont_get_cache (struct frame * f) +{ + Display_Info *dpyinfo = FRAME_DISPLAY_INFO (f); + + return (dpyinfo->name_list_element); +} + +static int +macfont_get_charset (Lisp_Object registry) +{ + char *str = SSDATA (SYMBOL_NAME (registry)); + char *re = alloca (SBYTES (SYMBOL_NAME (registry)) * 2 + 1); + Lisp_Object regexp; + int i, j; + + for (i = j = 0; i < SBYTES (SYMBOL_NAME (registry)); i++, j++) + { + if (str[i] == '.') + re[j++] = '\\'; + else if (str[i] == '*') + re[j++] = '.'; + re[j] = str[i]; + if (re[j] == '?') + re[j] = '.'; + } + re[j] = '\0'; + regexp = make_unibyte_string (re, j); + for (i = 0; cf_charset_table[i].name; i++) + if (fast_c_string_match_ignore_case + (regexp, cf_charset_table[i].name, + strlen (cf_charset_table[i].name)) >= 0) + break; + if (! cf_charset_table[i].name) + return -1; + if (! cf_charset_table[i].cf_charset) + { + int *uniquifier = cf_charset_table[i].uniquifier; + UniChar *unichars = alloca (sizeof (cf_charset_table[i].uniquifier)); + CFIndex count = 0; + CFStringRef string; + CFMutableCharacterSetRef charset = CFCharacterSetCreateMutable (NULL); + + if (! charset) + return -1; + for (j = 0; uniquifier[j]; j++) + { + count += macfont_store_utf32char_to_unichars (uniquifier[j], + unichars + count); + CFCharacterSetAddCharactersInRange (charset, + CFRangeMake (uniquifier[j], 1)); + } + + string = CFStringCreateWithCharacters (NULL, unichars, count); + if (! string) + { + CFRelease (charset); + return -1; + } + cf_charset_table[i].cf_charset = CFCharacterSetCreateCopy (NULL, + charset); + CFRelease (charset); + /* CFCharacterSetCreateWithCharactersInString does not handle + surrogate pairs properly as of Mac OS X 10.5. */ + cf_charset_table[i].cf_charset_string = string; + } + return i; +} + +struct OpenTypeSpec +{ + Lisp_Object script; + unsigned int script_tag, langsys_tag; + int nfeatures[2]; + unsigned int *features[2]; +}; + +#define OTF_SYM_TAG(SYM, TAG) \ + do { \ + unsigned char *p = SDATA (SYMBOL_NAME (SYM)); \ + TAG = (p[0] << 24) | (p[1] << 16) | (p[2] << 8) | p[3]; \ + } while (0) + +#define OTF_TAG_STR(TAG, P) \ + do { \ + (P)[0] = (char) (TAG >> 24); \ + (P)[1] = (char) ((TAG >> 16) & 0xFF); \ + (P)[2] = (char) ((TAG >> 8) & 0xFF); \ + (P)[3] = (char) (TAG & 0xFF); \ + (P)[4] = '\0'; \ + } while (0) + +static struct OpenTypeSpec * +macfont_get_open_type_spec (Lisp_Object otf_spec) +{ + struct OpenTypeSpec *spec = xmalloc (sizeof *spec); + Lisp_Object val; + int i, j; + bool negative; + + if (! spec) + return NULL; + spec->script = XCAR (otf_spec); + if (! NILP (spec->script)) + { + OTF_SYM_TAG (spec->script, spec->script_tag); + val = assq_no_quit (spec->script, Votf_script_alist); + if (CONSP (val) && SYMBOLP (XCDR (val))) + spec->script = XCDR (val); + else + spec->script = Qnil; + } + else + spec->script_tag = 0x44464C54; /* "DFLT" */ + otf_spec = XCDR (otf_spec); + spec->langsys_tag = 0; + if (! NILP (otf_spec)) + { + val = XCAR (otf_spec); + if (! NILP (val)) + OTF_SYM_TAG (val, spec->langsys_tag); + otf_spec = XCDR (otf_spec); + } + spec->nfeatures[0] = spec->nfeatures[1] = 0; + for (i = 0; i < 2 && ! NILP (otf_spec); i++, otf_spec = XCDR (otf_spec)) + { + Lisp_Object len; + + val = XCAR (otf_spec); + if (NILP (val)) + continue; + len = Flength (val); + spec->features[i] = + (min (PTRDIFF_MAX, SIZE_MAX) / sizeof (int) < XINT (len) + ? 0 + : malloc (XINT (len) * sizeof *spec->features[i])); + if (! spec->features[i]) + { + if (i > 0 && spec->features[0]) + free (spec->features[0]); + free (spec); + return NULL; + } + for (j = 0, negative = 0; CONSP (val); val = XCDR (val)) + { + if (NILP (XCAR (val))) + negative = 1; + else + { + unsigned int tag; + + OTF_SYM_TAG (XCAR (val), tag); + spec->features[i][j++] = negative ? tag & 0x80000000 : tag; + } + } + spec->nfeatures[i] = j; + } + return spec; +} + +static CFMutableDictionaryRef +macfont_create_attributes_with_spec (Lisp_Object spec) +{ + Lisp_Object tmp, extra; + CFMutableArrayRef langarray = NULL; + CFCharacterSetRef charset = NULL; + CFStringRef charset_string = NULL; + CFMutableDictionaryRef attributes = NULL, traits = NULL; + Lisp_Object script = Qnil; + Lisp_Object registry; + int cf_charset_idx, i; + struct OpenTypeSpec *otspec = NULL; + struct { + enum font_property_index index; + CFStringRef trait; + CGPoint points[6]; + } numeric_traits[] = + {{FONT_WEIGHT_INDEX, MAC_FONT_WEIGHT_TRAIT, + {{-0.4, 50}, /* light */ + {-0.24, 87.5}, /* (semi-light + normal) / 2 */ + {0, 100}, /* normal */ + {0.24, 140}, /* (semi-bold + normal) / 2 */ + {0.4, 200}, /* bold */ + {CGFLOAT_MAX, CGFLOAT_MAX}}}, + {FONT_SLANT_INDEX, MAC_FONT_SLANT_TRAIT, + {{0, 100}, {0.1, 200}, {CGFLOAT_MAX, CGFLOAT_MAX}}}, + {FONT_WIDTH_INDEX, MAC_FONT_WIDTH_TRAIT, + {{0, 100}, {1, 200}, {CGFLOAT_MAX, CGFLOAT_MAX}}}}; + + registry = AREF (spec, FONT_REGISTRY_INDEX); + if (NILP (registry) + || EQ (registry, Qascii_0) + || EQ (registry, Qiso10646_1) + || EQ (registry, Qunicode_bmp)) + cf_charset_idx = -1; + else + { + CFStringRef lang; + + cf_charset_idx = macfont_get_charset (registry); + if (cf_charset_idx < 0) + goto err; + charset = cf_charset_table[cf_charset_idx].cf_charset; + charset_string = cf_charset_table[cf_charset_idx].cf_charset_string; + lang = cf_charset_table[cf_charset_idx].lang; + if (lang) + { + langarray = CFArrayCreateMutable (NULL, 0, &kCFTypeArrayCallBacks); + if (! langarray) + goto err; + CFArrayAppendValue (langarray, lang); + } + } + + for (extra = AREF (spec, FONT_EXTRA_INDEX); + CONSP (extra); extra = XCDR (extra)) + { + Lisp_Object key, val; + + tmp = XCAR (extra); + key = XCAR (tmp), val = XCDR (tmp); + if (EQ (key, QClang)) + { + if (! langarray) + langarray = CFArrayCreateMutable (NULL, 0, &kCFTypeArrayCallBacks); + if (! langarray) + goto err; + if (SYMBOLP (val)) + val = list1 (val); + for (; CONSP (val); val = XCDR (val)) + if (SYMBOLP (XCAR (val))) + { + CFStringRef lang = + cfstring_create_with_string_noencode (SYMBOL_NAME + (XCAR (val))); + + if (lang == NULL) + goto err; + CFArrayAppendValue (langarray, lang); + CFRelease (lang); + } + } + else if (EQ (key, QCotf)) + { + otspec = macfont_get_open_type_spec (val); + if (! otspec) + goto err; + script = otspec->script; + } + else if (EQ (key, QCscript)) + script = val; + } + + if (! NILP (script) && ! charset) + { + Lisp_Object chars = assq_no_quit (script, Vscript_representative_chars); + + if (CONSP (chars) && CONSP (CDR (chars))) + { + CFMutableStringRef string = CFStringCreateMutable (NULL, 0); + CFMutableCharacterSetRef cs = CFCharacterSetCreateMutable (NULL); + + if (! string || !cs) + { + if (string) + CFRelease (string); + else if (cs) + CFRelease (cs); + goto err; + } + for (chars = XCDR (chars); CONSP (chars); chars = XCDR (chars)) + if (CHARACTERP (XCAR (chars))) + { + UniChar unichars[2]; + CFIndex count = + macfont_store_utf32char_to_unichars (XFASTINT (XCAR (chars)), + unichars); + CFRange range = CFRangeMake (XFASTINT (XCAR (chars)), 1); + + CFStringAppendCharacters (string, unichars, count); + CFCharacterSetAddCharactersInRange (cs, range); + } + charset = cs; + /* CFCharacterSetCreateWithCharactersInString does not + handle surrogate pairs properly as of Mac OS X 10.5. */ + charset_string = string; + } + } + + attributes = CFDictionaryCreateMutable (NULL, 0, + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + if (! attributes) + goto err; + + tmp = AREF (spec, FONT_FAMILY_INDEX); + if (SYMBOLP (tmp) && ! NILP (tmp)) + { + CFStringRef family = macfont_create_family_with_symbol (tmp); + + if (! family) + goto err; + CFDictionaryAddValue (attributes, MAC_FONT_FAMILY_NAME_ATTRIBUTE, + family); + CFRelease (family); + } + + traits = CFDictionaryCreateMutable (NULL, 4, + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + if (! traits) + goto err; + + for (i = 0; i < sizeof (numeric_traits) / sizeof (numeric_traits[0]); i++) + { + tmp = AREF (spec, numeric_traits[i].index); + if (INTEGERP (tmp)) + { + CGPoint *point = numeric_traits[i].points; + CGFloat floatval = (XINT (tmp) >> 8); // XXX + CFNumberRef num; + + while (point->y < floatval) + point++; + if (point == numeric_traits[i].points) + point++; + else if (point->y == CGFLOAT_MAX) + point--; + floatval = (point - 1)->x + ((floatval - (point - 1)->y) + * ((point->x - (point - 1)->x) + / (point->y - (point - 1)->y))); + if (floatval > 1.0) + floatval = 1.0; + else if (floatval < -1.0) + floatval = -1.0; + num = CFNumberCreate (NULL, kCFNumberCGFloatType, &floatval); + if (! num) + goto err; + CFDictionaryAddValue (traits, numeric_traits[i].trait, num); + CFRelease (num); + } + } + if (CFDictionaryGetCount (traits)) + CFDictionaryAddValue (attributes, MAC_FONT_TRAITS_ATTRIBUTE, traits); + + if (charset) + CFDictionaryAddValue (attributes, MAC_FONT_CHARACTER_SET_ATTRIBUTE, + charset); + if (charset_string) + CFDictionaryAddValue (attributes, MAC_FONT_CHARACTER_SET_STRING_ATTRIBUTE, + charset_string); + if (langarray) + CFDictionaryAddValue (attributes, MAC_FONT_LANGUAGES_ATTRIBUTE, langarray); + + goto finish; + + err: + if (attributes) + { + CFRelease (attributes); + attributes = NULL; + } + + finish: + if (langarray) CFRelease (langarray); + if (charset && cf_charset_idx < 0) CFRelease (charset); + if (charset_string && cf_charset_idx < 0) CFRelease (charset_string); + if (traits) CFRelease (traits); + if (otspec) + { + if (otspec->nfeatures[0] > 0) + free (otspec->features[0]); + if (otspec->nfeatures[1] > 0) + free (otspec->features[1]); + free (otspec); + } + + return attributes; +} + +static Boolean +macfont_supports_charset_and_languages_p (FontDescriptorRef desc, + CFCharacterSetRef charset, + Lisp_Object chars, + CFArrayRef languages) +{ + Boolean result = true; + + if (charset || VECTORP (chars)) + { + CFCharacterSetRef desc_charset = + mac_font_descriptor_copy_attribute (desc, + MAC_FONT_CHARACTER_SET_ATTRIBUTE); + + if (desc_charset == NULL) + result = false; + else + { + if (charset) + result = CFCharacterSetIsSupersetOfSet (desc_charset, charset); + else /* VECTORP (chars) */ + { + ptrdiff_t j; + + for (j = 0; j < ASIZE (chars); j++) + if (TYPE_RANGED_INTEGERP (UTF32Char, AREF (chars, j)) + && CFCharacterSetIsLongCharacterMember (desc_charset, + XFASTINT (AREF (chars, j)))) + break; + if (j == ASIZE (chars)) + result = false; + } + CFRelease (desc_charset); + } + } + if (result && languages) + result = mac_font_descriptor_supports_languages (desc, languages); + + return result; +} + +static CFIndex +macfont_closest_traits_index (CFArrayRef traits_array, + FontSymbolicTraits target) +{ + CFIndex i, result = -1, count = CFArrayGetCount (traits_array); + int min_distance = (1 << 3); + + for (i = 0; i < count; i++) + { + FontSymbolicTraits traits, diff; + int distance = 0; + + traits = ((FontSymbolicTraits) (uintptr_t) + CFArrayGetValueAtIndex (traits_array, i)); + diff = (target ^ traits); + /* We prefer synthetic bold of italic to synthetic italic of + bold when both bold and italic are available but bold-italic + is not available. */ + if (diff & MAC_FONT_TRAIT_BOLD) + distance |= (1 << 0); + if (diff & MAC_FONT_TRAIT_ITALIC) + distance |= (1 << 1); + if (diff & MAC_FONT_TRAIT_MONO_SPACE) + distance |= (1 << 2); + if (distance < min_distance) + { + min_distance = distance; + result = i; + } + } + + return result; +} + +static Lisp_Object +macfont_list (struct frame *f, Lisp_Object spec) +{ + Lisp_Object val = Qnil, family, extra; + int i, n; + CFStringRef family_name = NULL; + CFMutableDictionaryRef attributes = NULL, traits; + Lisp_Object chars = Qnil; + int spacing = -1; + FontSymbolicTraits synth_sym_traits = 0; + CFArrayRef families; + CFIndex families_count; + CFCharacterSetRef charset = NULL; + CFArrayRef languages = NULL; + + block_input (); + + family = AREF (spec, FONT_FAMILY_INDEX); + if (! NILP (family)) + { + family_name = macfont_create_family_with_symbol (family); + if (family_name == NULL) + goto finish; + } + + attributes = macfont_create_attributes_with_spec (spec); + if (! attributes) + goto finish; + + charset = ((CFCharacterSetRef) + CFDictionaryGetValue (attributes, + MAC_FONT_CHARACTER_SET_ATTRIBUTE)); + if (charset) + { + CFRetain (charset); + CFDictionaryRemoveValue (attributes, MAC_FONT_CHARACTER_SET_ATTRIBUTE); + } + else + { + val = assq_no_quit (QCscript, AREF (spec, FONT_EXTRA_INDEX)); + if (! NILP (val)) + { + val = assq_no_quit (XCDR (val), Vscript_representative_chars); + if (CONSP (val) && VECTORP (XCDR (val))) + chars = XCDR (val); + } + val = Qnil; + } + + languages = ((CFArrayRef) + CFDictionaryGetValue (attributes, MAC_FONT_LANGUAGES_ATTRIBUTE)); + if (languages) + { + CFRetain (languages); + CFDictionaryRemoveValue (attributes, MAC_FONT_LANGUAGES_ATTRIBUTE); + } + + if (INTEGERP (AREF (spec, FONT_SPACING_INDEX))) + spacing = XINT (AREF (spec, FONT_SPACING_INDEX)); + + traits = ((CFMutableDictionaryRef) + CFDictionaryGetValue (attributes, MAC_FONT_TRAITS_ATTRIBUTE)); + + n = FONT_SLANT_NUMERIC (spec); + if (n < 0 || n == FONT_SLANT_SYNTHETIC_ITALIC) + { + synth_sym_traits |= MAC_FONT_TRAIT_ITALIC; + if (traits) + CFDictionaryRemoveValue (traits, MAC_FONT_SLANT_TRAIT); + } + + n = FONT_WEIGHT_NUMERIC (spec); + if (n < 0 || n == FONT_WEIGHT_SYNTHETIC_BOLD) + { + synth_sym_traits |= MAC_FONT_TRAIT_BOLD; + if (traits) + CFDictionaryRemoveValue (traits, MAC_FONT_WEIGHT_TRAIT); + } + + if (languages + && (spacing < 0 || spacing == FONT_SPACING_SYNTHETIC_MONO)) + { + CFStringRef language = CFArrayGetValueAtIndex (languages, 0); + + if (CFStringHasPrefix (language, CFSTR ("ja")) + || CFStringHasPrefix (language, CFSTR ("ko")) + || CFStringHasPrefix (language, CFSTR ("zh"))) + synth_sym_traits |= MAC_FONT_TRAIT_MONO_SPACE; + } + + /* Create array of families. */ + if (family_name) + families = CFArrayCreate (NULL, (const void **) &family_name, + 1, &kCFTypeArrayCallBacks); + else + { + CFStringRef pref_family; + CFIndex families_count, pref_family_index = -1; + + families = mac_font_create_available_families (); + if (families == NULL) + goto err; + + families_count = CFArrayGetCount (families); + + /* Move preferred family to the front if exists. */ + pref_family = + mac_font_create_preferred_family_for_attributes (attributes); + if (pref_family) + { + pref_family_index = + CFArrayGetFirstIndexOfValue (families, + CFRangeMake (0, families_count), + pref_family); + CFRelease (pref_family); + } + if (pref_family_index > 0) + { + CFMutableArrayRef mutable_families = + CFArrayCreateMutable (NULL, families_count, &kCFTypeArrayCallBacks); + + if (mutable_families) + { + CFArrayAppendValue (mutable_families, + CFArrayGetValueAtIndex (families, + pref_family_index)); + CFArrayAppendArray (mutable_families, families, + CFRangeMake (0, pref_family_index)); + if (pref_family_index + 1 < families_count) + CFArrayAppendArray (mutable_families, families, + CFRangeMake (pref_family_index + 1, + families_count + - (pref_family_index + 1))); + CFRelease (families); + families = mutable_families; + } + } + } + + val = Qnil; + extra = AREF (spec, FONT_EXTRA_INDEX); + families_count = CFArrayGetCount (families); + for (i = 0; i < families_count; i++) + { + CFStringRef family_name = CFArrayGetValueAtIndex (families, i); + FontDescriptorRef pat_desc; + CFArrayRef descs; + CFIndex descs_count; + CFMutableArrayRef filtered_descs, traits_array; + Lisp_Object entity; + int j; + + CFDictionarySetValue (attributes, MAC_FONT_FAMILY_NAME_ATTRIBUTE, + family_name); + pat_desc = mac_font_descriptor_create_with_attributes (attributes); + if (! pat_desc) + goto err; + + /* CTFontDescriptorCreateMatchingFontDescriptors on Mac OS X + 10.7 returns NULL if pat_desc represents the LastResort font. + So we use CTFontDescriptorCreateMatchingFontDescriptor (no + trailing "s") for such a font. */ + if (CFStringCompare (family_name, CFSTR ("LastResort"), 0) + != kCFCompareEqualTo) + descs = mac_font_descriptor_create_matching_font_descriptors (pat_desc, + NULL); + else + { + FontDescriptorRef lr_desc = + mac_font_descriptor_create_matching_font_descriptor (pat_desc, + NULL); + if (lr_desc) + { + descs = CFArrayCreate (NULL, (const void **) &lr_desc, 1, + &kCFTypeArrayCallBacks); + CFRelease (lr_desc); + } + else + descs = NULL; + } + CFRelease (pat_desc); + if (! descs) + goto err; + + descs_count = CFArrayGetCount (descs); + if (descs_count == 0 + || !macfont_supports_charset_and_languages_p (CFArrayGetValueAtIndex (descs, 0), + charset, chars, + languages)) + { + CFRelease (descs); + continue; + } + + filtered_descs = + CFArrayCreateMutable (NULL, descs_count, &kCFTypeArrayCallBacks); + traits_array = CFArrayCreateMutable (NULL, descs_count, NULL); + for (j = 0; j < descs_count; j++) + { + FontDescriptorRef desc = CFArrayGetValueAtIndex (descs, j); + CFDictionaryRef dict; + CFNumberRef num; + FontSymbolicTraits sym_traits; + + dict = mac_font_descriptor_copy_attribute (desc, + MAC_FONT_TRAITS_ATTRIBUTE); + if (dict == NULL) + continue; + + num = CFDictionaryGetValue (dict, MAC_FONT_SYMBOLIC_TRAIT); + CFRelease (dict); + if (num == NULL + || !cfnumber_get_font_symbolic_traits_value (num, &sym_traits)) + continue; + + if (spacing >= 0 + && !(synth_sym_traits & MAC_FONT_TRAIT_MONO_SPACE) + && (((sym_traits & MAC_FONT_TRAIT_MONO_SPACE) != 0) + != (spacing >= FONT_SPACING_MONO))) + continue; + + /* Don't use a color bitmap font unless its family is + explicitly specified. */ + if ((sym_traits & MAC_FONT_TRAIT_COLOR_GLYPHS) && NILP (family)) + continue; + + if (j > 0 + && !macfont_supports_charset_and_languages_p (desc, charset, + chars, languages)) + continue; + + CFArrayAppendValue (filtered_descs, desc); + CFArrayAppendValue (traits_array, + (const void *) (uintptr_t) sym_traits); + } + + CFRelease (descs); + descs = filtered_descs; + descs_count = CFArrayGetCount (descs); + + for (j = 0; j < descs_count; j++) + { + FontDescriptorRef desc = CFArrayGetValueAtIndex (descs, j); + FontSymbolicTraits sym_traits = + ((FontSymbolicTraits) (uintptr_t) + CFArrayGetValueAtIndex (traits_array, j)); + FontSymbolicTraits mask_min, mask_max, imask, bmask, mmask; + + mask_min = ((synth_sym_traits ^ sym_traits) + & (MAC_FONT_TRAIT_ITALIC | MAC_FONT_TRAIT_BOLD)); + if (FONT_SLANT_NUMERIC (spec) < 0) + mask_min &= ~MAC_FONT_TRAIT_ITALIC; + if (FONT_WEIGHT_NUMERIC (spec) < 0) + mask_min &= ~MAC_FONT_TRAIT_BOLD; + + mask_max = (synth_sym_traits & ~sym_traits); + /* Synthetic bold does not work for bitmap-only fonts on Mac + OS X 10.6. */ + if ((mask_min ^ mask_max) & MAC_FONT_TRAIT_BOLD) + { + CFNumberRef format = + mac_font_descriptor_copy_attribute (desc, + MAC_FONT_FORMAT_ATTRIBUTE); + + if (format) + { + uint32_t format_val; + + if (CFNumberGetValue (format, kCFNumberSInt32Type, + &format_val) + && format_val == MAC_FONT_FORMAT_BITMAP) + mask_max &= ~MAC_FONT_TRAIT_BOLD; + } + } + if (spacing >= 0) + mask_min |= (mask_max & MAC_FONT_TRAIT_MONO_SPACE); + + for (mmask = (mask_min & MAC_FONT_TRAIT_MONO_SPACE); + mmask <= (mask_max & MAC_FONT_TRAIT_MONO_SPACE); + mmask += MAC_FONT_TRAIT_MONO_SPACE) + for (bmask = (mask_min & MAC_FONT_TRAIT_BOLD); + bmask <= (mask_max & MAC_FONT_TRAIT_BOLD); + bmask += MAC_FONT_TRAIT_BOLD) + for (imask = (mask_min & MAC_FONT_TRAIT_ITALIC); + imask <= (mask_max & MAC_FONT_TRAIT_ITALIC); + imask += MAC_FONT_TRAIT_ITALIC) + { + FontSymbolicTraits synth = (imask | bmask | mmask); + + if (synth == 0 + || j == macfont_closest_traits_index (traits_array, + (sym_traits | synth))) + { + entity = macfont_descriptor_entity (desc, extra, synth); + if (! NILP (entity)) + val = Fcons (entity, val); + } + } + } + + CFRelease (traits_array); + CFRelease (descs); + } + + CFRelease (families); + val = Fnreverse (val); + goto finish; + err: + val = Qnil; + + finish: + FONT_ADD_LOG ("macfont-list", spec, val); + if (charset) CFRelease (charset); + if (languages) CFRelease (languages); + if (attributes) CFRelease (attributes); + if (family_name) CFRelease (family_name); + + unblock_input (); + + return val; +} + +static Lisp_Object +macfont_match (struct frame * frame, Lisp_Object spec) +{ + Lisp_Object entity = Qnil; + CFMutableDictionaryRef attributes; + FontDescriptorRef pat_desc = NULL, desc = NULL; + + block_input (); + + attributes = macfont_create_attributes_with_spec (spec); + if (attributes) + { + pat_desc = mac_font_descriptor_create_with_attributes (attributes); + CFRelease (attributes); + } + if (pat_desc) + { + desc = mac_font_descriptor_create_matching_font_descriptor (pat_desc, + NULL); + CFRelease (pat_desc); + } + if (desc) + { + entity = macfont_descriptor_entity (desc, AREF (spec, FONT_EXTRA_INDEX), + 0); + CFRelease (desc); + } + unblock_input (); + + FONT_ADD_LOG ("macfont-match", spec, entity); + return entity; +} + +static Lisp_Object +macfont_list_family (struct frame *frame) +{ + Lisp_Object list = Qnil; + CFArrayRef families; + + block_input (); + + families = mac_font_create_available_families (); + if (families) + { + CFIndex i, count = CFArrayGetCount (families); + + for (i = 0; i < count; i++) + list = Fcons (macfont_intern_prop_cfstring (CFArrayGetValueAtIndex (families, i)), list); + CFRelease (families); + } + + unblock_input (); + + return list; +} + +static void +macfont_free_entity (Lisp_Object entity) +{ + Lisp_Object val = assq_no_quit (QCfont_entity, + AREF (entity, FONT_EXTRA_INDEX)); + CFStringRef name = XSAVE_POINTER (XCDR (val), 0); + + block_input (); + CFRelease (name); + unblock_input (); +} + +static Lisp_Object +macfont_open (struct frame * f, Lisp_Object entity, int pixel_size) +{ + Lisp_Object val, font_object; + CFStringRef font_name; + struct macfont_info *macfont_info = NULL; + struct font *font; + int size; + FontRef macfont; + FontSymbolicTraits sym_traits; + char name[256]; + int len, i, total_width; + CGGlyph glyph; + CGFloat ascent, descent, leading; + + val = assq_no_quit (QCfont_entity, AREF (entity, FONT_EXTRA_INDEX)); + if (! CONSP (val) + || XTYPE (XCDR (val)) != Lisp_Misc + || XMISCTYPE (XCDR (val)) != Lisp_Misc_Save_Value) + return Qnil; + font_name = XSAVE_POINTER (XCDR (val), 0); + sym_traits = XSAVE_INTEGER (XCDR (val), 1); + + size = XINT (AREF (entity, FONT_SIZE_INDEX)); + if (size == 0) + size = pixel_size; + + block_input (); + macfont = mac_font_create_with_name (font_name, size); + if (macfont) + { + int fontsize = (int) [((NSFont *) macfont) pointSize]; + if (fontsize != size) size = fontsize; + } + unblock_input (); + if (! macfont) + return Qnil; + + font_object = font_make_object (VECSIZE (struct macfont_info), entity, size); + ASET (font_object, FONT_TYPE_INDEX, macfont_driver.type); + len = font_unparse_xlfd (entity, size, name, 256); + if (len > 0) + ASET (font_object, FONT_NAME_INDEX, make_string (name, len)); + len = font_unparse_fcname (entity, size, name, 256); + if (len > 0) + ASET (font_object, FONT_FULLNAME_INDEX, make_string (name, len)); + else + ASET (font_object, FONT_FULLNAME_INDEX, + AREF (font_object, FONT_NAME_INDEX)); + font = XFONT_OBJECT (font_object); + font->pixel_size = size; + font->driver = &macfont_driver; + font->encoding_charset = font->repertory_charset = -1; + + block_input (); + + macfont_info = (struct macfont_info *) font; + macfont_info->macfont = macfont; + macfont_info->cgfont = mac_font_copy_graphics_font (macfont); + + val = assq_no_quit (QCdestination, AREF (entity, FONT_EXTRA_INDEX)); + if (CONSP (val) && EQ (XCDR (val), make_number (1))) + macfont_info->screen_font = mac_screen_font_create_with_name (font_name, + size); + else + macfont_info->screen_font = NULL; + macfont_info->cache = macfont_lookup_cache (font_name); + macfont_retain_cache (macfont_info->cache); + macfont_info->metrics = NULL; + macfont_info->metrics_nrows = 0; + macfont_info->synthetic_italic_p = 0; + macfont_info->synthetic_bold_p = 0; + macfont_info->spacing = MACFONT_SPACING_PROPORTIONAL; + macfont_info->antialias = MACFONT_ANTIALIAS_DEFAULT; + if (!(sym_traits & MAC_FONT_TRAIT_ITALIC) + && FONT_SLANT_NUMERIC (entity) == FONT_SLANT_SYNTHETIC_ITALIC) + macfont_info->synthetic_italic_p = 1; + if (!(sym_traits & MAC_FONT_TRAIT_BOLD) + && FONT_WEIGHT_NUMERIC (entity) == FONT_WEIGHT_SYNTHETIC_BOLD) + macfont_info->synthetic_bold_p = 1; + if (sym_traits & MAC_FONT_TRAIT_MONO_SPACE) + macfont_info->spacing = MACFONT_SPACING_MONO; + else if (INTEGERP (AREF (entity, FONT_SPACING_INDEX)) + && (XINT (AREF (entity, FONT_SPACING_INDEX)) + == FONT_SPACING_SYNTHETIC_MONO)) + macfont_info->spacing = MACFONT_SPACING_SYNTHETIC_MONO; + if (macfont_info->synthetic_italic_p || macfont_info->synthetic_bold_p) + macfont_info->antialias = MACFONT_ANTIALIAS_ON; + else + { + val = assq_no_quit (QCantialias, AREF (entity, FONT_EXTRA_INDEX)); + if (CONSP (val)) + macfont_info->antialias = + NILP (XCDR (val)) ? MACFONT_ANTIALIAS_OFF : MACFONT_ANTIALIAS_ON; + } + macfont_info->color_bitmap_p = 0; + if (sym_traits & MAC_FONT_TRAIT_COLOR_GLYPHS) + macfont_info->color_bitmap_p = 1; + + glyph = macfont_get_glyph_for_character (font, ' '); + if (glyph != kCGFontIndexInvalid) + font->space_width = macfont_glyph_extents (font, glyph, NULL, NULL, 0); + else + /* dirty workaround */ + font->space_width = pixel_size; + + total_width = font->space_width; + for (i = 1; i < 95; i++) + { + glyph = macfont_get_glyph_for_character (font, ' ' + i); + if (glyph == kCGFontIndexInvalid) + break; + total_width += macfont_glyph_extents (font, glyph, NULL, NULL, 0); + } + if (i == 95) + font->average_width = total_width / 95; + else + font->average_width = font->space_width; /* XXX */ + + if (!(macfont_info->screen_font + && mac_screen_font_get_metrics (macfont_info->screen_font, + &ascent, &descent, &leading))) + { + CFStringRef family_name; + + ascent = mac_font_get_ascent (macfont); + descent = mac_font_get_descent (macfont); + leading = mac_font_get_leading (macfont); + /* AppKit and WebKit do some adjustment to the heights of + Courier, Helvetica, and Times. */ + family_name = mac_font_copy_family_name (macfont); + if (family_name) + { + if ((CFStringCompare (family_name, CFSTR ("Courier"), 0) + == kCFCompareEqualTo) + || (CFStringCompare (family_name, CFSTR ("Helvetica"), 0) + == kCFCompareEqualTo) + || (CFStringCompare (family_name, CFSTR ("Times"), 0) + == kCFCompareEqualTo)) + ascent += (ascent + descent) * .15f; + else if (CFStringHasPrefix (family_name, CFSTR ("Hiragino"))) + { + leading *= .25f; + ascent += leading; + } + CFRelease (family_name); + } + } + font->ascent = ascent + 0.5f; + val = assq_no_quit (QCminspace, AREF (entity, FONT_EXTRA_INDEX)); + if (CONSP (val) && !NILP (XCDR (val))) + font->descent = descent + 0.5f; + else + font->descent = descent + leading + 0.5f; + font->height = font->ascent + font->descent; + + font->underline_position = - mac_font_get_underline_position (macfont) + 0.5f; + font->underline_thickness = mac_font_get_underline_thickness (macfont) + 0.5f; + + unblock_input (); + + /* Unfortunately Xft doesn't provide a way to get minimum char + width. So, we use space_width instead. */ + font->min_width = font->max_width = font->space_width; /* XXX */ + + font->baseline_offset = 0; + font->relative_compose = 0; + font->default_ascent = 0; + font->vertical_centering = 0; + + return font_object; +} + +static void +macfont_close (struct frame * f, struct font *font) +{ + struct macfont_info *macfont_info = (struct macfont_info *) font; + int i; + + block_input (); + CFRelease (macfont_info->macfont); + CGFontRelease (macfont_info->cgfont); + if (macfont_info->screen_font) + CFRelease (macfont_info->screen_font); + macfont_release_cache (macfont_info->cache); + for (i = 0; i < macfont_info->metrics_nrows; i++) + if (macfont_info->metrics[i]) + xfree (macfont_info->metrics[i]); + if (macfont_info->metrics) + xfree (macfont_info->metrics); + unblock_input (); +} + +static int +macfont_has_char (Lisp_Object font, int c) +{ + int result; + CFCharacterSetRef charset; + + block_input (); + if (FONT_ENTITY_P (font)) + { + Lisp_Object val; + CFStringRef name; + + val = assq_no_quit (QCfont_entity, AREF (font, FONT_EXTRA_INDEX)); + val = XCDR (val); + name = XSAVE_POINTER (val, 0); + charset = macfont_get_cf_charset_for_name (name); + } + else + charset = macfont_get_cf_charset (XFONT_OBJECT (font)); + + result = CFCharacterSetIsLongCharacterMember (charset, c); + unblock_input (); + + return result; +} + +static unsigned +macfont_encode_char (struct font *font, int c) +{ + struct macfont_info *macfont_info = (struct macfont_info *) font; + CGGlyph glyph; + + block_input (); + glyph = macfont_get_glyph_for_character (font, c); + unblock_input (); + + return glyph != kCGFontIndexInvalid ? glyph : FONT_INVALID_CODE; +} + +static int +macfont_text_extents (struct font *font, unsigned int *code, int nglyphs, + struct font_metrics *metrics) +{ + int width, i; + + block_input (); + width = macfont_glyph_extents (font, code[0], metrics, NULL, 0); + for (i = 1; i < nglyphs; i++) + { + struct font_metrics m; + int w = macfont_glyph_extents (font, code[i], metrics ? &m : NULL, + NULL, 0); + + if (metrics) + { + if (width + m.lbearing < metrics->lbearing) + metrics->lbearing = width + m.lbearing; + if (width + m.rbearing > metrics->rbearing) + metrics->rbearing = width + m.rbearing; + if (m.ascent > metrics->ascent) + metrics->ascent = m.ascent; + if (m.descent > metrics->descent) + metrics->descent = m.descent; + } + width += w; + } + unblock_input (); + + if (metrics) + metrics->width = width; + + return width; +} + +static int +macfont_draw (struct glyph_string *s, int from, int to, int x, int y, + bool with_background) +{ + struct frame * f = s->f; + struct macfont_info *macfont_info = (struct macfont_info *) s->font; + FontRef macfont = macfont_info->macfont; + CGContextRef context; + BOOL isComposite = s->first_glyph->type == COMPOSITE_GLYPH; + int end = isComposite ? s->cmp_to : s->nchars; + int len = end - s->cmp_from; + int i; + + block_input (); + + context = [[NSGraphicsContext currentContext] graphicsPort]; + CGContextSaveGState (context); + +#if 0 + if (s->num_clips > 0) + { + CGRect clips[2]; + + for (i = 0; i < s->num_clips; i++) + clips[i] = mac_rect_make (f, s->clip[i].left, s->clip[i].top, + s->clip[i].right - s->clip[i].left, + s->clip[i].bottom - s->clip[i].top); + CGContextClipToRects (context, clips, s->num_clips); + } +#endif + + if (with_background) + { + CG_SET_FILL_COLOR_WITH_GC_BACKGROUND (context, s); + CGContextFillRect (context, + NSMakeRect (x, y, + s->width, FONT_HEIGHT (s->font))); + } + + if (macfont_info->cgfont) + { + CGGlyph *glyphs = alloca (sizeof (CGGlyph) * len); + CGPoint *positions = alloca (sizeof (CGPoint) * len); + CGFloat total_width = 0; + CGFloat font_size = mac_font_get_size (macfont); + CGAffineTransform atfm; + CGFloat advance_delta = 0; + int y_draw = -y-FONT_BASE (s->font); + int no_antialias_p = + (macfont_info->antialias == MACFONT_ANTIALIAS_OFF + || (macfont_info->antialias == MACFONT_ANTIALIAS_DEFAULT + && font_size <= macfont_antialias_threshold)); + + for (i = 0; i < len; i++) + { + int width; + + glyphs[i] = *(s->char2b + s->cmp_from + i); + width = (s->padding_p ? 1 + : macfont_glyph_extents (s->font, glyphs[i], + NULL, &advance_delta, + no_antialias_p)); + positions[i].x = total_width + advance_delta; + positions[i].y = 0; + total_width += width; + } + + CGContextScaleCTM (context, 1, -1); + CG_SET_FILL_COLOR_WITH_GC_FOREGROUND (context, s); + if (macfont_info->synthetic_italic_p) + atfm = synthetic_italic_atfm; + else + atfm = CGAffineTransformIdentity; + if (macfont_info->synthetic_bold_p) + { + CGContextSetTextDrawingMode (context, kCGTextFillStroke); + CGContextSetLineWidth (context, synthetic_bold_factor * font_size); + CG_SET_STROKE_COLOR_WITH_GC_FOREGROUND (context, s); + } + if (no_antialias_p) + CGContextSetShouldAntialias (context, false); + + CGContextSetTextMatrix (context, atfm); + CGContextSetTextPosition (context, x, y_draw); + +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 1070 + if (macfont_info->color_bitmap_p +#if MAC_OS_X_VERSION_MIN_REQUIRED < 1070 + && CTFontDrawGlyphs != NULL +#endif + ) + { + if (len > 0) + { + CTFontDrawGlyphs (macfont, glyphs, positions, len, context); + } + } + else +#endif /* MAC_OS_X_VERSION_MAX_ALLOWED >= 1070 */ + { + CGContextSetFont (context, macfont_info->cgfont); + CGContextSetFontSize (context, font_size); + CGContextShowGlyphsAtPositions (context, glyphs, positions, len); + } + } + + CGContextRestoreGState (context); + + unblock_input (); + + return len; +} + +Lisp_Object +macfont_shape (Lisp_Object lgstring) +{ + struct font *font; + struct macfont_info *macfont_info; + FontRef macfont; + ptrdiff_t glyph_len, len, i, j; + CFIndex nonbmp_len; + UniChar *unichars; + CFIndex *nonbmp_indices; + CFStringRef string; + CFIndex used = 0; + struct mac_glyph_layout *glyph_layouts; + + CHECK_FONT_GET_OBJECT (LGSTRING_FONT (lgstring), font); + macfont_info = (struct macfont_info *) font; + macfont = macfont_info->macfont; + + glyph_len = LGSTRING_GLYPH_LEN (lgstring); + nonbmp_len = 0; + for (i = 0; i < glyph_len; i++) + { + Lisp_Object lglyph = LGSTRING_GLYPH (lgstring, i); + + if (NILP (lglyph)) + break; + if (LGLYPH_CHAR (lglyph) >= 0x10000) + nonbmp_len++; + } + + len = i; + lint_assume (len <= TYPE_MAXIMUM (EMACS_INT) - 2); + + if (INT_MAX / 2 < len) + memory_full (SIZE_MAX); + + unichars = alloca (sizeof (UniChar) * (len + nonbmp_len)); + nonbmp_indices = alloca (sizeof (CFIndex) * (nonbmp_len + 1)); + for (i = j = 0; i < len; i++) + { + UTF32Char c = LGLYPH_CHAR (LGSTRING_GLYPH (lgstring, i)); + + if (macfont_store_utf32char_to_unichars (c, unichars + i + j) > 1) + { + nonbmp_indices[j] = i + j; + j++; + } + } + nonbmp_indices[j] = len + j; /* sentinel */ + + block_input (); + + string = CFStringCreateWithCharactersNoCopy (NULL, unichars, len + nonbmp_len, + kCFAllocatorNull); + if (string) + { + glyph_layouts = alloca (sizeof (struct mac_glyph_layout) * glyph_len); + if (macfont_info->screen_font) + used = mac_screen_font_shape (macfont_info->screen_font, string, + glyph_layouts, glyph_len); + else + used = mac_font_shape (macfont, string, glyph_layouts, glyph_len); + CFRelease (string); + } + + unblock_input (); + + if (used == 0) + return Qnil; + + block_input (); + + for (i = 0; i < used; i++) + { + Lisp_Object lglyph = LGSTRING_GLYPH (lgstring, i); + struct mac_glyph_layout *gl = glyph_layouts + i; + EMACS_INT from, to; + struct font_metrics metrics; + int xoff, yoff, wadjust; + + if (NILP (lglyph)) + { + lglyph = Fmake_vector (make_number (LGLYPH_SIZE), Qnil); + LGSTRING_SET_GLYPH (lgstring, i, lglyph); + } + + from = gl->comp_range.location; + /* Convert UTF-16 index to UTF-32. */ + j = 0; + while (nonbmp_indices[j] < from) + j++; + from -= j; + LGLYPH_SET_FROM (lglyph, from); + + to = gl->comp_range.location + gl->comp_range.length; + /* Convert UTF-16 index to UTF-32. */ + while (nonbmp_indices[j] < to) + j++; + to -= j; + LGLYPH_SET_TO (lglyph, to - 1); + + /* LGLYPH_CHAR is used in `describe-char' for checking whether + the composition is trivial. */ + { + UTF32Char c; + + if (unichars[gl->string_index] >= 0xD800 + && unichars[gl->string_index] < 0xDC00) + c = (((unichars[gl->string_index] - 0xD800) << 10) + + (unichars[gl->string_index + 1] - 0xDC00) + 0x10000); + else + c = unichars[gl->string_index]; + if (macfont_get_glyph_for_character (font, c) != gl->glyph_id) + c = 0; + LGLYPH_SET_CHAR (lglyph, c); + } + + { + unsigned long cc = gl->glyph_id; + LGLYPH_SET_CODE (lglyph, cc); + } + + macfont_glyph_extents (font, gl->glyph_id, &metrics, NULL, 0); + LGLYPH_SET_WIDTH (lglyph, metrics.width); + LGLYPH_SET_LBEARING (lglyph, metrics.lbearing); + LGLYPH_SET_RBEARING (lglyph, metrics.rbearing); + LGLYPH_SET_ASCENT (lglyph, metrics.ascent); + LGLYPH_SET_DESCENT (lglyph, metrics.descent); + + xoff = lround (gl->advance_delta); + yoff = lround (- gl->baseline_delta); + wadjust = lround (gl->advance); + if (xoff != 0 || yoff != 0 || wadjust != metrics.width) + { + Lisp_Object vec; + + vec = Fmake_vector (make_number (3), Qnil); + ASET (vec, 0, make_number (xoff)); + ASET (vec, 1, make_number (yoff)); + ASET (vec, 2, make_number (wadjust)); + LGLYPH_SET_ADJUSTMENT (lglyph, vec); + } + } + + unblock_input (); + + return make_number (used); +} + +/* Structures for the UVS subtable (format 14) in the cmap table. */ +typedef UInt8 UINT24[3]; + +#pragma pack(push, 1) +struct variation_selector_record +{ + UINT24 var_selector; + UInt32 default_uvs_offset, non_default_uvs_offset; +}; +struct uvs_table +{ + UInt16 format; + UInt32 length, num_var_selector_records; + struct variation_selector_record variation_selector_records[1]; +}; +#define SIZEOF_UVS_TABLE_HEADER \ + (sizeof (struct uvs_table) - sizeof (struct variation_selector_record)) + +struct unicode_value_range +{ + UINT24 start_unicode_value; + UInt8 additional_count; +}; +struct default_uvs_table { + UInt32 num_unicode_value_ranges; + struct unicode_value_range unicode_value_ranges[1]; +}; +#define SIZEOF_DEFAULT_UVS_TABLE_HEADER \ + (sizeof (struct default_uvs_table) - sizeof (struct unicode_value_range)) + +struct uvs_mapping +{ + UINT24 unicode_value; + UInt16 glyph_id; +}; +struct non_default_uvs_table +{ + UInt32 num_uvs_mappings; + struct uvs_mapping uvs_mappings[1]; +}; +#define SIZEOF_NON_DEFAULT_UVS_TABLE_HEADER \ + (sizeof (struct non_default_uvs_table) - sizeof (struct uvs_mapping)) +#pragma pack(pop) + +/* Read big endian values. The argument LVAL must be an lvalue. */ +/* I suppose OSReadBigInt* takes care of unaligned data. At least, we + can find "... = OSReadBigInt32(cdb, 2);" followed by "... = + OSReadBigInt16(cdb, 7);" in a sample code by Apple. */ +#define BUINT8_VALUE(lval) (*((UInt8 *) &(lval))) +#define BUINT16_VALUE(lval) OSReadBigInt16 (&(lval), 0) +/* Succeeding one byte should also be accessible. */ +#define BUINT24_VALUE(lval) (OSReadBigInt32 (&(lval), 0) >> 8) +#define BUINT32_VALUE(lval) OSReadBigInt32 (&(lval), 0) + +/* Return UVS subtable for the specified FONT. If the subtable is not + found or ill-formated, then return NULL. */ + +static CFDataRef +mac_font_copy_uvs_table (FontRef font) +{ + CFDataRef cmap_table, uvs_table = NULL; + + cmap_table = mac_font_copy_non_synthetic_table (font, cmapFontTableTag); + if (cmap_table) + { + sfntCMapHeader *cmap = (sfntCMapHeader *) CFDataGetBytePtr (cmap_table); + struct uvs_table *uvs; + struct variation_selector_record *records; + UInt32 cmap_len, ntables, i, uvs_offset, uvs_len, nrecords; + +#if __LP64__ + if (CFDataGetLength (cmap_table) > UINT32_MAX) + goto finish; +#endif + + cmap_len = CFDataGetLength (cmap_table); + if (sizeof_sfntCMapHeader > cmap_len) + goto finish; + + ntables = BUINT16_VALUE (cmap->numTables); + if (ntables > ((cmap_len - sizeof_sfntCMapHeader) + / sizeof_sfntCMapEncoding)) + goto finish; + + for (i = 0; i < ntables; i++) + if ((BUINT16_VALUE (cmap->encoding[i].platformID) + == kFontUnicodePlatform) + && (BUINT16_VALUE (cmap->encoding[i].scriptID) + == 5)) /* kFontUnicodeV4_0VariationSequenceSemantics */ + { + uvs_offset = BUINT32_VALUE (cmap->encoding[i].offset); + break; + } + if (i == ntables + || uvs_offset > cmap_len + || SIZEOF_UVS_TABLE_HEADER > cmap_len - uvs_offset) + goto finish; + + uvs = (struct uvs_table *) ((UInt8 *) cmap + uvs_offset); + uvs_len = BUINT32_VALUE (uvs->length); + if (uvs_len > cmap_len - uvs_offset + || SIZEOF_UVS_TABLE_HEADER > uvs_len) + goto finish; + + if (BUINT16_VALUE (uvs->format) != 14) + goto finish; + + nrecords = BUINT32_VALUE (uvs->num_var_selector_records); + if (nrecords > ((uvs_len - SIZEOF_UVS_TABLE_HEADER) + / sizeof (struct variation_selector_record))) + goto finish; + + records = uvs->variation_selector_records; + for (i = 0; i < nrecords; i++) + { + UInt32 default_uvs_offset, non_default_uvs_offset; + + default_uvs_offset = BUINT32_VALUE (records[i].default_uvs_offset); + if (default_uvs_offset) + { + struct default_uvs_table *default_uvs; + UInt32 nranges; + + if (default_uvs_offset > uvs_len + || (SIZEOF_DEFAULT_UVS_TABLE_HEADER + > uvs_len - default_uvs_offset)) + goto finish; + + default_uvs = ((struct default_uvs_table *) + ((UInt8 *) uvs + default_uvs_offset)); + nranges = BUINT32_VALUE (default_uvs->num_unicode_value_ranges); + if (nranges > ((uvs_len - default_uvs_offset + - SIZEOF_DEFAULT_UVS_TABLE_HEADER) + / sizeof (struct unicode_value_range))) + goto finish; + /* Now 2 * nranges can't overflow, so we can safely use + `(lo + hi) / 2' instead of `lo + (hi - lo) / 2' in + mac_font_get_glyphs_for_variants. */ + } + + non_default_uvs_offset = + BUINT32_VALUE (records[i].non_default_uvs_offset); + if (non_default_uvs_offset) + { + struct non_default_uvs_table *non_default_uvs; + UInt32 nmappings; + + if (non_default_uvs_offset > uvs_len + || (SIZEOF_NON_DEFAULT_UVS_TABLE_HEADER + > uvs_len - non_default_uvs_offset)) + goto finish; + + non_default_uvs = ((struct non_default_uvs_table *) + ((UInt8 *) uvs + non_default_uvs_offset)); + nmappings = BUINT32_VALUE (non_default_uvs->num_uvs_mappings); + if (nmappings > ((uvs_len - non_default_uvs_offset + - SIZEOF_NON_DEFAULT_UVS_TABLE_HEADER) + / sizeof (struct uvs_mapping))) + goto finish; + /* Now 2 * nmappings can't overflow, so we can safely + use `(lo + hi) / 2' instead of `lo + (hi - lo) / 2' + in mac_font_get_glyphs_for_variants. */ + } + } + + uvs_table = CFDataCreate (NULL, (UInt8 *) uvs, uvs_len); + + finish: + CFRelease (cmap_table); + } + + return uvs_table; +} + +/* Find an entry in the given UVS subtable UVS_TABLE for a variation + sequence consisting of the given base character C and each + variation selector SELECTORS[i] for 0 <= i < COUNT, and store the + result (explained below) into the corresponding GLYPHS[i]. If the + entry is found in the Default UVS Table, then the result is 0. If + the entry is found in the Non-Default UVS Table, then the result is + the associated glyph ID. Otherwise, kCGFontIndexInvalid. The + elements in SELECTORS must be sorted in strictly increasing + order. */ + +static void +mac_font_get_glyphs_for_variants (CFDataRef uvs_table, UTF32Char c, + const UTF32Char selectors[], CGGlyph glyphs[], + CFIndex count) +{ + struct uvs_table *uvs = (struct uvs_table *) CFDataGetBytePtr (uvs_table); + struct variation_selector_record *records = uvs->variation_selector_records; + CFIndex i; + UInt32 ir, nrecords; +#if MAC_OS_X_VERSION_MIN_REQUIRED >= 1060 + dispatch_queue_t queue = + dispatch_get_global_queue (DISPATCH_QUEUE_PRIORITY_DEFAULT, 0); + dispatch_group_t group = dispatch_group_create (); +#endif + + nrecords = BUINT32_VALUE (uvs->num_var_selector_records); + i = 0; + ir = 0; + while (i < count && ir < nrecords) + { + UInt32 default_uvs_offset, non_default_uvs_offset; + + if (selectors[i] < BUINT24_VALUE (records[ir].var_selector)) + { + glyphs[i++] = kCGFontIndexInvalid; + continue; + } + else if (selectors[i] > BUINT24_VALUE (records[ir].var_selector)) + { + ir++; + continue; + } + + /* selectors[i] == BUINT24_VALUE (records[ir].var_selector) */ + default_uvs_offset = BUINT32_VALUE (records[ir].default_uvs_offset); + non_default_uvs_offset = + BUINT32_VALUE (records[ir].non_default_uvs_offset); +#if MAC_OS_X_VERSION_MIN_REQUIRED >= 1060 + dispatch_group_async (group, queue, ^{ +#endif + glyphs[i] = kCGFontIndexInvalid; + + if (default_uvs_offset) + { + struct default_uvs_table *default_uvs = + (struct default_uvs_table *) ((UInt8 *) uvs + + default_uvs_offset); + struct unicode_value_range *ranges = + default_uvs->unicode_value_ranges; + UInt32 lo, hi; + + lo = 0; + hi = BUINT32_VALUE (default_uvs->num_unicode_value_ranges); + while (lo < hi) + { + UInt32 mid = (lo + hi) / 2; + + if (c < BUINT24_VALUE (ranges[mid].start_unicode_value)) + hi = mid; + else + lo = mid + 1; + } + if (hi > 0 + && (c <= (BUINT24_VALUE (ranges[hi - 1].start_unicode_value) + + BUINT8_VALUE (ranges[hi - 1].additional_count)))) + glyphs[i] = 0; + } + + if (glyphs[i] == kCGFontIndexInvalid && non_default_uvs_offset) + { + struct non_default_uvs_table *non_default_uvs = + (struct non_default_uvs_table *) ((UInt8 *) uvs + + non_default_uvs_offset); + struct uvs_mapping *mappings = non_default_uvs->uvs_mappings; + UInt32 lo, hi; + + lo = 0; + hi = BUINT32_VALUE (non_default_uvs->num_uvs_mappings); + while (lo < hi) + { + UInt32 mid = (lo + hi) / 2; + + if (c < BUINT24_VALUE (mappings[mid].unicode_value)) + hi = mid; + else + lo = mid + 1; + } + if (hi > 0 && + BUINT24_VALUE (mappings[hi - 1].unicode_value) == c) + glyphs[i] = BUINT16_VALUE (mappings[hi - 1].glyph_id); + } +#if MAC_OS_X_VERSION_MIN_REQUIRED >= 1060 + }); +#endif + i++; + ir++; + } + while (i < count) + glyphs[i++] = kCGFontIndexInvalid; +#if MAC_OS_X_VERSION_MIN_REQUIRED >= 1060 + dispatch_group_wait (group, DISPATCH_TIME_FOREVER); + dispatch_release (group); +#endif +} + +static int +macfont_variation_glyphs (struct font *font, int c, unsigned variations[256]) +{ + CFDataRef uvs_table; + CharacterCollection uvs_collection; + int i, n = 0; + + block_input (); + uvs_table = macfont_get_uvs_table (font, &uvs_collection); + + if (uvs_table) + { + UTF32Char selectors[256]; + CGGlyph glyphs[256]; + + for (i = 0; i < 16; i++) + selectors[i] = 0xFE00 + i; + for (; i < 256; i++) + selectors[i] = 0xE0100 + (i - 16); + mac_font_get_glyphs_for_variants (uvs_table, c, selectors, glyphs, 256); + for (i = 0; i < 256; i++) + { + CGGlyph glyph = glyphs[i]; + + if (uvs_collection != MAC_CHARACTER_COLLECTION_IDENTITY_MAPPING + && glyph != kCGFontIndexInvalid) + glyph = macfont_get_glyph_for_cid (font, uvs_collection, glyph); + if (glyph == kCGFontIndexInvalid) + variations[i] = 0; + else + { + variations[i] = (glyph ? glyph + : macfont_get_glyph_for_character (font, c)); + n++; + } + } + } + unblock_input (); + + return n; +} + +static const char *const macfont_booleans[] = { + ":antialias", + ":minspace", + NULL, +}; + +static const char *const macfont_non_booleans[] = { + ":lang", + ":script", + ":destination", + NULL, +}; + +static void +macfont_filter_properties (Lisp_Object font, Lisp_Object alist) +{ + font_filter_properties (font, alist, macfont_booleans, macfont_non_booleans); +} + +static Boolean +mac_ctfont_descriptor_supports_languages (CTFontDescriptorRef descriptor, + CFArrayRef languages) +{ + Boolean result = true; + CFArrayRef desc_languages = + CTFontDescriptorCopyAttribute (descriptor, kCTFontLanguagesAttribute); + + if (desc_languages == NULL) + result = false; + else + { + CFIndex desc_languages_count, i, languages_count; + + desc_languages_count = CFArrayGetCount (desc_languages); + languages_count = CFArrayGetCount (languages); + for (i = 0; i < languages_count; i++) + if (!CFArrayContainsValue (desc_languages, + CFRangeMake (0, desc_languages_count), + CFArrayGetValueAtIndex (languages, i))) + { + result = false; + break; + } + CFRelease (desc_languages); + } + + return result; +} + +static CFStringRef +mac_ctfont_create_preferred_family_for_attributes (CFDictionaryRef attributes) +{ + CFStringRef result = NULL; + CFStringRef charset_string = + CFDictionaryGetValue (attributes, MAC_FONT_CHARACTER_SET_STRING_ATTRIBUTE); + CFIndex length; + + if (charset_string + && (length = CFStringGetLength (charset_string)) > 0) + { + CFAttributedStringRef attr_string = NULL; + CTLineRef ctline = NULL; + CFDictionaryRef attrs = + CFDictionaryCreate (NULL, NULL, NULL, 0, + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + + if (attrs) + { + attr_string = CFAttributedStringCreate (NULL, charset_string, attrs); + CFRelease (attrs); + } + if (attr_string) + { + ctline = CTLineCreateWithAttributedString (attr_string); + CFRelease (attr_string); + } + if (ctline) + { + CFArrayRef runs = CTLineGetGlyphRuns (ctline); + CFIndex i, nruns = CFArrayGetCount (runs); + CTFontRef font; + + for (i = 0; i < nruns; i++) + { + CTRunRef run = CFArrayGetValueAtIndex (runs, i); + CFDictionaryRef attributes = CTRunGetAttributes (run); + CTFontRef font_in_run; + + if (attributes == NULL) + break; + font_in_run = + CFDictionaryGetValue (attributes, kCTFontAttributeName); + if (font_in_run == NULL) + break; + if (i == 0) + font = font_in_run; + else if (!mac_ctfont_equal_in_postscript_name (font, font_in_run)) + break; + } + if (nruns > 0 && i == nruns) + result = CTFontCopyAttribute (font, kCTFontFamilyNameAttribute); + CFRelease (ctline); + } + } + + return result; +} + +static inline double +mac_ctfont_get_advance_width_for_glyph (CTFontRef font, CGGlyph glyph) +{ + return CTFontGetAdvancesForGlyphs (font, kCTFontDefaultOrientation, + &glyph, NULL, 1); +} + +static inline CGRect +mac_ctfont_get_bounding_rect_for_glyph (CTFontRef font, CGGlyph glyph) +{ + return CTFontGetBoundingRectsForGlyphs (font, kCTFontDefaultOrientation, + &glyph, NULL, 1); +} + +static CFArrayRef +mac_ctfont_create_available_families (void) +{ + CFMutableArrayRef families = NULL; + +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 1060 +#if MAC_OS_X_VERSION_MIN_REQUIRED < 1060 + if (CTFontManagerCopyAvailableFontFamilyNames != NULL) +#endif + { + CFArrayRef orig_families = CTFontManagerCopyAvailableFontFamilyNames (); + + if (orig_families) + { + CFIndex i, count = CFArrayGetCount (orig_families); + + families = CFArrayCreateMutable (NULL, count, &kCFTypeArrayCallBacks); + if (families) + for (i = 0; i < count; i++) + { + CFStringRef family = CFArrayGetValueAtIndex (orig_families, i); + + if (!CFStringHasPrefix (family, CFSTR (".")) + && (CTFontManagerCompareFontFamilyNames (family, + CFSTR ("LastResort"), + NULL) + != kCFCompareEqualTo)) + CFArrayAppendValue (families, family); + } + CFRelease (orig_families); + } + } +#if MAC_OS_X_VERSION_MIN_REQUIRED < 1060 + else /* CTFontManagerCopyAvailableFontFamilyNames == NULL */ +#endif +#endif /* MAC_OS_X_VERSION_MAX_ALLOWED >= 1060 */ +#if MAC_OS_X_VERSION_MIN_REQUIRED < 1060 + { + CTFontCollectionRef collection; + CFArrayRef descs = NULL; + + collection = CTFontCollectionCreateFromAvailableFonts (NULL); + if (collection) + { + descs = CTFontCollectionCreateMatchingFontDescriptors (collection); + CFRelease (collection); + } + if (descs) + { + CFIndex i, count = CFArrayGetCount (descs); + + families = CFArrayCreateMutable (NULL, count, &kCFTypeArrayCallBacks); + if (families) + for (i = 0; i < count; i++) + { + FontDescriptorRef desc = CFArrayGetValueAtIndex (descs, i); + CFStringRef name = + mac_font_descriptor_copy_attribute (desc, + MAC_FONT_FAMILY_NAME_ATTRIBUTE); + + if (name) + { + CFIndex p, limit = CFArrayGetCount (families); + + p = CFArrayBSearchValues (families, CFRangeMake (0, limit), + (const void *) name, + mac_font_family_compare, NULL); + if (p >= limit) + CFArrayAppendValue (families, name); + else if (mac_font_family_compare + (CFArrayGetValueAtIndex (families, p), + name, NULL) != kCFCompareEqualTo) + CFArrayInsertValueAtIndex (families, p, name); + CFRelease (name); + } + } + CFRelease (descs); + } + } +#endif + + return families; +} + +static Boolean +mac_ctfont_equal_in_postscript_name (CTFontRef font1, CTFontRef font2) +{ + Boolean result; + CFStringRef name1, name2; + + if (font1 == font2) + return true; + + result = false; + name1 = CTFontCopyPostScriptName (font1); + if (name1) + { + name2 = CTFontCopyPostScriptName (font2); + if (name2) + { + result = (CFStringCompare (name1, name2, 0) == kCFCompareEqualTo); + CFRelease (name2); + } + CFRelease (name1); + } + + return result; +} + +static CTLineRef +mac_ctfont_create_line_with_string_and_font (CFStringRef string, + CTFontRef macfont) +{ + CFStringRef keys[] = {kCTFontAttributeName, kCTKernAttributeName}; + CFTypeRef values[] = {NULL, NULL}; + CFDictionaryRef attributes = NULL; + CFAttributedStringRef attr_string = NULL; + CTLineRef ctline = NULL; + float float_zero = 0.0f; + + values[0] = macfont; + values[1] = CFNumberCreate (NULL, kCFNumberFloatType, &float_zero); + if (values[1]) + { + attributes = CFDictionaryCreate (NULL, (const void **) keys, + (const void **) values, + sizeof (keys) / sizeof (keys[0]), + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + CFRelease (values[1]); + } + if (attributes) + { + attr_string = CFAttributedStringCreate (NULL, string, attributes); + CFRelease (attributes); + } + if (attr_string) + { + ctline = CTLineCreateWithAttributedString (attr_string); + CFRelease (attr_string); + } + if (ctline) + { + /* Abandon if ctline contains some fonts other than the + specified one. */ + CFArrayRef runs = CTLineGetGlyphRuns (ctline); + CFIndex i, nruns = CFArrayGetCount (runs); + + for (i = 0; i < nruns; i++) + { + CTRunRef run = CFArrayGetValueAtIndex (runs, i); + CFDictionaryRef attributes = CTRunGetAttributes (run); + CTFontRef font_in_run; + + if (attributes == NULL) + break; + font_in_run = + CFDictionaryGetValue (attributes, kCTFontAttributeName); + if (font_in_run == NULL) + break; + if (!mac_ctfont_equal_in_postscript_name (macfont, font_in_run)) + break; + } + if (i < nruns) + { + CFRelease (ctline); + ctline = NULL; + } + } + + return ctline; +} + +CFIndex +mac_ctfont_shape (CTFontRef font, CFStringRef string, + struct mac_glyph_layout *glyph_layouts, CFIndex glyph_len) +{ + CFIndex used, result = 0; + CTLineRef ctline = mac_ctfont_create_line_with_string_and_font (string, font); + + if (ctline == NULL) + return 0; + + used = CTLineGetGlyphCount (ctline); + if (used <= glyph_len) + { + CFArrayRef ctruns = CTLineGetGlyphRuns (ctline); + CFIndex k, ctrun_count = CFArrayGetCount (ctruns); + CGFloat total_advance = 0; + CFIndex total_glyph_count = 0; + + for (k = 0; k < ctrun_count; k++) + { + CTRunRef ctrun = CFArrayGetValueAtIndex (ctruns, k); + CFIndex i, min_location, glyph_count = CTRunGetGlyphCount (ctrun); + struct mac_glyph_layout *glbuf = glyph_layouts + total_glyph_count; + CFRange string_range, comp_range, range; + CFIndex *permutation; + + if (CTRunGetStatus (ctrun) & kCTRunStatusRightToLeft) + permutation = xmalloc (sizeof (CFIndex) * glyph_count); + else + permutation = NULL; + +#define RIGHT_TO_LEFT_P permutation + + /* Now the `comp_range' member of struct mac_glyph_layout is + temporarily used as a work area such that: + glbuf[i].comp_range.location = + min {compRange[i + 1].location, ..., + compRange[glyph_count - 1].location, + maxRange (stringRangeForCTRun)} + glbuf[i].comp_range.length = maxRange (compRange[i]) + where compRange[i] is the range of composed characters + containing i-th glyph. */ + string_range = CTRunGetStringRange (ctrun); + min_location = string_range.location + string_range.length; + for (i = 0; i < glyph_count; i++) + { + struct mac_glyph_layout *gl = glbuf + glyph_count - i - 1; + CFIndex glyph_index; + CFRange rng; + + if (!RIGHT_TO_LEFT_P) + glyph_index = glyph_count - i - 1; + else + glyph_index = i; + CTRunGetStringIndices (ctrun, CFRangeMake (glyph_index, 1), + &gl->string_index); + rng = + CFStringGetRangeOfComposedCharactersAtIndex (string, + gl->string_index); + gl->comp_range.location = min_location; + gl->comp_range.length = rng.location + rng.length; + if (rng.location < min_location) + min_location = rng.location; + } + + /* Fill the `comp_range' member of struct mac_glyph_layout, + and setup a permutation for right-to-left text. */ + comp_range = CFRangeMake (string_range.location, 0); + range = CFRangeMake (0, 0); + while (1) + { + struct mac_glyph_layout *gl = + glbuf + range.location + range.length; + + if (gl->comp_range.length + > comp_range.location + comp_range.length) + comp_range.length = gl->comp_range.length - comp_range.location; + min_location = gl->comp_range.location; + range.length++; + + if (min_location >= comp_range.location + comp_range.length) + { + comp_range.length = min_location - comp_range.location; + for (i = 0; i < range.length; i++) + { + glbuf[range.location + i].comp_range = comp_range; + if (RIGHT_TO_LEFT_P) + permutation[range.location + i] = + range.location + range.length - i - 1; + } + + comp_range = CFRangeMake (min_location, 0); + range.location += range.length; + range.length = 0; + if (range.location == glyph_count) + break; + } + } + + /* Then fill the remaining members. */ + for (range = CFRangeMake (0, 1); range.location < glyph_count; + range.location++) + { + struct mac_glyph_layout *gl; + CGPoint position; + + if (!RIGHT_TO_LEFT_P) + gl = glbuf + range.location; + else + { + CFIndex src, dest; + + src = glyph_count - 1 - range.location; + dest = permutation[src]; + gl = glbuf + dest; + if (src < dest) + { + CFIndex tmp = gl->string_index; + + gl->string_index = glbuf[src].string_index; + glbuf[src].string_index = tmp; + } + } + CTRunGetGlyphs (ctrun, range, &gl->glyph_id); + + CTRunGetPositions (ctrun, range, &position); + gl->advance_delta = position.x - total_advance; + gl->baseline_delta = position.y; + gl->advance = (gl->advance_delta + + CTRunGetTypographicBounds (ctrun, range, + NULL, NULL, NULL)); + total_advance += gl->advance; + } + + if (RIGHT_TO_LEFT_P) + xfree (permutation); + +#undef RIGHT_TO_LEFT_P + + total_glyph_count += glyph_count; + } + + result = used; + } + CFRelease (ctline); + + return result; +} + +/* The function below seems to cause a memory leak for the CFString + created by CFStringCreateWithCharacters as of Mac OS X 10.5.8 and + 10.6.3. For now, we use the NSGlyphInfo version instead. */ +#if USE_CT_GLYPH_INFO +CGGlyph +mac_ctfont_get_glyph_for_cid (CTFontRef font, CTCharacterCollection collection, + CGFontIndex cid) +{ + CGGlyph result = kCGFontIndexInvalid; + UniChar characters[] = {0xfffd}; + CFStringRef string; + CFAttributedStringRef attr_string = NULL; + CTLineRef ctline = NULL; + + string = CFStringCreateWithCharacters (NULL, characters, + sizeof (characters) + / sizeof (characters[0])); + if (string) + { + CTGlyphInfoRef glyph_info = + CTGlyphInfoCreateWithCharacterIdentifier (cid, collection, string); + CFDictionaryRef attributes = NULL; + + if (glyph_info) + { + CFStringRef keys[] = {kCTFontAttributeName, + kCTGlyphInfoAttributeName}; + CFTypeRef values[] = {font, glyph_info}; + + attributes = CFDictionaryCreate (NULL, (const void **) keys, + (const void **) values, + sizeof (keys) / sizeof (keys[0]), + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + CFRelease (glyph_info); + } + if (attributes) + { + attr_string = CFAttributedStringCreate (NULL, string, attributes); + CFRelease (attributes); + } + CFRelease (string); + } + if (attr_string) + { + ctline = CTLineCreateWithAttributedString (attr_string); + CFRelease (attr_string); + } + if (ctline) + { + CFArrayRef runs = CTLineGetGlyphRuns (ctline); + + if (CFArrayGetCount (runs) > 0) + { + CTRunRef run = CFArrayGetValueAtIndex (runs, 0); + CFDictionaryRef attributes = CTRunGetAttributes (run); + + if (attributes) + { + CTFontRef font_in_run = + CFDictionaryGetValue (attributes, kCTFontAttributeName); + + if (font_in_run + && mac_ctfont_equal_in_postscript_name (font_in_run, font)) + { + CTRunGetGlyphs (run, CFRangeMake (0, 1), &result); + if (result >= CTFontGetGlyphCount (font)) + result = kCGFontIndexInvalid; + } + } + } + CFRelease (ctline); + } + + return result; +} +#endif + +#if MAC_OS_X_VERSION_MIN_REQUIRED < 1060 +static inline int +mac_font_family_group (CFStringRef family) +{ + if (CFStringHasPrefix (family, CFSTR ("#"))) + return 2; + else + { + CFRange range; + + range = CFStringFind (family, CFSTR ("Apple"), + kCFCompareCaseInsensitive | kCFCompareAnchored); + if (range.location != kCFNotFound) + return 1; + + return 0; + } +} + +CFComparisonResult +mac_font_family_compare (const void *val1, const void *val2, void *context) +{ + CFStringRef family1 = (CFStringRef) val1, family2 = (CFStringRef) val2; + int group1, group2; + + group1 = mac_font_family_group (family1); + group2 = mac_font_family_group (family2); + if (group1 < group2) + return kCFCompareLessThan; + if (group1 > group2) + return kCFCompareGreaterThan; + return CFStringCompare (family1, family2, kCFCompareCaseInsensitive); +} +#endif /* MAC_OS_X_VERSION_MIN_REQUIRED < 1060 */ + +void * +macfont_get_nsctfont (struct font *font) +{ + struct macfont_info *macfont_info = (struct macfont_info *) font; + FontRef macfont = macfont_info->macfont; + + return (void *) macfont; +} + +void +mac_register_font_driver (struct frame *f) +{ + register_font_driver (&macfont_driver, f); +} + +#endif // MAC_OS_X_VERSION_MAX_ALLOWED >= 1050 + + +void +syms_of_macfont (void) +{ +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 1050 + static struct font_driver mac_font_driver; + + DEFSYM (Qmac_ct, "mac-ct"); + macfont_driver.type = Qmac_ct; + register_font_driver (&macfont_driver, NULL); + + DEFSYM (QCdestination, ":destination"); + DEFSYM (QCminspace, ":minspace"); +#endif +} |