1from __future__ import print_function, division, absolute_import
2from fontTools.misc.py23 import *
3from fontTools.misc import sstruct
4from fontTools.misc.textTools import safeEval, num2binary, binary2num
5from fontTools.misc.timeTools import timestampFromString, timestampToString, timestampNow
6from fontTools.misc.timeTools import epoch_diff as mac_epoch_diff # For backward compat
7from . import DefaultTable
8import logging
9
10
11log = logging.getLogger(__name__)
12
13headFormat = """
14		>	# big endian
15		tableVersion:       16.16F
16		fontRevision:       16.16F
17		checkSumAdjustment: I
18		magicNumber:        I
19		flags:              H
20		unitsPerEm:         H
21		created:            Q
22		modified:           Q
23		xMin:               h
24		yMin:               h
25		xMax:               h
26		yMax:               h
27		macStyle:           H
28		lowestRecPPEM:      H
29		fontDirectionHint:  h
30		indexToLocFormat:   h
31		glyphDataFormat:    h
32"""
33
34class table__h_e_a_d(DefaultTable.DefaultTable):
35
36	dependencies = ['maxp', 'loca', 'CFF ']
37
38	def decompile(self, data, ttFont):
39		dummy, rest = sstruct.unpack2(headFormat, data, self)
40		if rest:
41			# this is quite illegal, but there seem to be fonts out there that do this
42			log.warning("extra bytes at the end of 'head' table")
43			assert rest == "\0\0"
44
45		# For timestamp fields, ignore the top four bytes.  Some fonts have
46		# bogus values there.  Since till 2038 those bytes only can be zero,
47		# ignore them.
48		#
49		# https://github.com/fonttools/fonttools/issues/99#issuecomment-66776810
50		for stamp in 'created', 'modified':
51			value = getattr(self, stamp)
52			if value > 0xFFFFFFFF:
53				log.warning("'%s' timestamp out of range; ignoring top bytes", stamp)
54				value &= 0xFFFFFFFF
55				setattr(self, stamp, value)
56			if value < 0x7C259DC0: # January 1, 1970 00:00:00
57				log.warning("'%s' timestamp seems very low; regarding as unix timestamp", stamp)
58				value += 0x7C259DC0
59				setattr(self, stamp, value)
60
61	def compile(self, ttFont):
62		if ttFont.recalcBBoxes:
63			# For TT-flavored fonts, xMin, yMin, xMax and yMax are set in table__m_a_x_p.recalc().
64			if 'CFF ' in ttFont:
65				topDict = ttFont['CFF '].cff.topDictIndex[0]
66				self.xMin, self.yMin, self.xMax, self.yMax = topDict.FontBBox
67		if ttFont.recalcTimestamp:
68			self.modified = timestampNow()
69		data = sstruct.pack(headFormat, self)
70		return data
71
72	def toXML(self, writer, ttFont):
73		writer.comment("Most of this table will be recalculated by the compiler")
74		writer.newline()
75		formatstring, names, fixes = sstruct.getformat(headFormat)
76		for name in names:
77			value = getattr(self, name)
78			if name in ("created", "modified"):
79				value = timestampToString(value)
80			if name in ("magicNumber", "checkSumAdjustment"):
81				if value < 0:
82					value = value + 0x100000000
83				value = hex(value)
84				if value[-1:] == "L":
85					value = value[:-1]
86			elif name in ("macStyle", "flags"):
87				value = num2binary(value, 16)
88			writer.simpletag(name, value=value)
89			writer.newline()
90
91	def fromXML(self, name, attrs, content, ttFont):
92		value = attrs["value"]
93		if name in ("created", "modified"):
94			value = timestampFromString(value)
95		elif name in ("macStyle", "flags"):
96			value = binary2num(value)
97		else:
98			value = safeEval(value)
99		setattr(self, name, value)
100