Skip to content
NSDate.py 6.88 KiB
Newer Older
Enrico Granata's avatar
Enrico Granata committed
# summary provider for NSDate
import lldb
import ctypes
import objc_runtime
import metrics
import struct
import time
import datetime
import CFString
Enrico Granata's avatar
Enrico Granata committed

statistics = metrics.Metrics()
statistics.add_metric('invalid_isa')
statistics.add_metric('invalid_pointer')
statistics.add_metric('unknown_class')
statistics.add_metric('code_notrun')

# Python promises to start counting time at midnight on Jan 1st on the epoch year
# hence, all we need to know is the epoch year
python_epoch = time.gmtime(0).tm_year

osx_epoch = datetime.date(2001,1,1).timetuple()

def mkgmtime(t):
    return time.mktime(t)-time.timezone

osx_epoch = mkgmtime(osx_epoch)

def osx_to_python_time(osx):
Enrico Granata's avatar
Enrico Granata committed
		return osx + osx_epoch
	else:
		return osx - osx_epoch


# despite the similary to synthetic children providers, these classes are not
# trying to provide anything but the summary for NSDate, so they need not
Enrico Granata's avatar
Enrico Granata committed
# obey the interface specification for synthetic children providers
class NSTaggedDate_SummaryProvider:
	def adjust_for_architecture(self):
Enrico Granata's avatar
Enrico Granata committed

	def __init__(self, valobj, info_bits, data, params):
Enrico Granata's avatar
Enrico Granata committed
		self.valobj = valobj;
Enrico Granata's avatar
Enrico Granata committed
		self.update();
		# NSDate is not using its info_bits for info like NSNumber is
		# so we need to regroup info_bits and data
		self.data = ((data << 8) | (info_bits << 4))
Enrico Granata's avatar
Enrico Granata committed

	def update(self):
		self.adjust_for_architecture();

	def value(self):
		# the value of the date-time object is wrapped into the pointer value
		# unfortunately, it is made as a time-delta after Jan 1 2001 midnight GMT
Enrico Granata's avatar
Enrico Granata committed
		# while all Python knows about is the "epoch", which is a platform-dependent
		# year (1970 of *nix) whose Jan 1 at midnight is taken as reference
		value_double = struct.unpack('d', struct.pack('Q', self.data))[0]
		return time.ctime(osx_to_python_time(value_double))
Enrico Granata's avatar
Enrico Granata committed


class NSUntaggedDate_SummaryProvider:
	def adjust_for_architecture(self):
Enrico Granata's avatar
Enrico Granata committed

Enrico Granata's avatar
Enrico Granata committed
		self.valobj = valobj;
		self.sys_params = params
		if not (self.sys_params.types_cache.double):
			self.sys_params.types_cache.double = self.valobj.GetType().GetBasicType(lldb.eBasicTypeDouble)
Enrico Granata's avatar
Enrico Granata committed
		self.update()

	def update(self):
		self.adjust_for_architecture();

	def offset(self):
Enrico Granata's avatar
Enrico Granata committed

	def value(self):
		value = self.valobj.CreateChildAtOffset("value",
							self.offset(),
Enrico Granata's avatar
Enrico Granata committed
		value_double = struct.unpack('d', struct.pack('Q', value.GetValueAsUnsigned(0)))[0]
		return time.ctime(osx_to_python_time(value_double))

class NSCalendarDate_SummaryProvider:
	def adjust_for_architecture(self):
		self.sys_params = params
		if not (self.sys_params.types_cache.double):
			self.sys_params.types_cache.double = self.valobj.GetType().GetBasicType(lldb.eBasicTypeDouble)
		self.update()

	def update(self):
		self.adjust_for_architecture();

	def offset(self):
		return 2*self.sys_params.pointer_size

	def value(self):
		value = self.valobj.CreateChildAtOffset("value",
							self.offset(),
		value_double = struct.unpack('d', struct.pack('Q', value.GetValueAsUnsigned(0)))[0]
		return time.ctime(osx_to_python_time(value_double))

class NSTimeZoneClass_SummaryProvider:
	def adjust_for_architecture(self):
		self.valobj = valobj;
		self.sys_params = params
		if not (self.sys_params.types_cache.voidptr):
			self.sys_params.types_cache.voidptr = self.valobj.GetType().GetBasicType(lldb.eBasicTypeVoid).GetPointerType()
		self.update()

	def update(self):
		self.adjust_for_architecture();

	def offset(self):

	def timezone(self):
		tz_string = self.valobj.CreateChildAtOffset("tz_name",
							self.offset(),
		return CFString.CFString_SummaryProvider(tz_string,None)
Enrico Granata's avatar
Enrico Granata committed
class NSUnknownDate_SummaryProvider:
	def adjust_for_architecture(self):
Enrico Granata's avatar
Enrico Granata committed

	def __init__(self, valobj):
		self.valobj = valobj;
		self.update()

	def update(self):
		self.adjust_for_architecture();

	def value(self):
		stream = lldb.SBStream()
		self.valobj.GetExpressionPath(stream)
		expr = "(NSString*)[" + stream.GetData() + " description]"
		num_children_vo = self.valobj.CreateValueFromExpression("str",expr);
		return num_children_vo.GetSummary()

def GetSummary_Impl(valobj):
	global statistics
	class_data = objc_runtime.ObjCRuntime(valobj)
	if class_data.is_valid() == False:
		statistics.metric_hit('invalid_pointer',valobj)
		wrapper = None
		return
	class_data = class_data.read_class_data()
	if class_data.is_valid() == False:
		statistics.metric_hit('invalid_isa',valobj)
		wrapper = None
		return
	if class_data.is_kvo():
		class_data = class_data.get_superclass()
	if class_data.is_valid() == False:
		statistics.metric_hit('invalid_isa',valobj)
		wrapper = None
		return
	
	name_string = class_data.class_name()
	if name_string == 'NSDate' or name_string == '__NSDate' or name_string == '__NSTaggedDate':
		if class_data.is_tagged():
			wrapper = NSTaggedDate_SummaryProvider(valobj,class_data.info_bits(),class_data.value(), class_data.sys_params)
Enrico Granata's avatar
Enrico Granata committed
			statistics.metric_hit('code_notrun',valobj)
		else:
			wrapper = NSUntaggedDate_SummaryProvider(valobj, class_data.sys_params)
Enrico Granata's avatar
Enrico Granata committed
			statistics.metric_hit('code_notrun',valobj)
	elif name_string == 'NSCalendarDate':
		wrapper = NSCalendarDate_SummaryProvider(valobj, class_data.sys_params)
		statistics.metric_hit('code_notrun',valobj)
	elif name_string == '__NSTimeZone':
		wrapper = NSTimeZoneClass_SummaryProvider(valobj, class_data.sys_params)
		statistics.metric_hit('code_notrun',valobj)
Enrico Granata's avatar
Enrico Granata committed
	else:
		wrapper = NSUnknownDate_SummaryProvider(valobj, class_data.sys_params)
Enrico Granata's avatar
Enrico Granata committed
		statistics.metric_hit('unknown_class',str(valobj) + " seen as " + name_string)
	return wrapper;


def NSDate_SummaryProvider (valobj,dict):
	provider = GetSummary_Impl(valobj);
	if provider != None:
	    #try:
	    summary = provider.value();
	    #except:
	    #    summary = None
	    if summary == None:
	        summary = 'no valid date here'
Enrico Granata's avatar
Enrico Granata committed
	    return str(summary)
	return ''

def NSTimeZone_SummaryProvider (valobj,dict):
	provider = GetSummary_Impl(valobj);
	if provider != None:
	    try:
	        summary = provider.timezone();
	    except:
	        summary = None
	    if summary == None:
	        summary = 'no valid timezone here'
	    return str(summary)
	return ''


def CFAbsoluteTime_SummaryProvider (valobj,dict):
	try:
		value_double = struct.unpack('d', struct.pack('Q', valobj.GetValueAsUnsigned(0)))[0]
		return time.ctime(osx_to_python_time(value_double))
	except:
		return 'unable to provide a summary'

Enrico Granata's avatar
Enrico Granata committed

def __lldb_init_module(debugger,dict):
	debugger.HandleCommand("type summary add -F NSDate.NSDate_SummaryProvider NSDate")
	debugger.HandleCommand("type summary add -F NSDate.CFAbsoluteTime_SummaryProvider CFAbsoluteTime")
	debugger.HandleCommand("type summary add -F NSDate.NSTimeZone_SummaryProvider NSTimeZone CFTimeZoneRef")
Enrico Granata's avatar
Enrico Granata committed