漫谈LLDB:SBExpressionOptions简介

806 阅读4分钟

漫谈LLDB:SBExpressionOptions简介

前言

首先要说明的是,这篇文章只是总结我的经验,不保证完全正确。如有错漏还望读者留言指正。
之所以要写这篇文章,是因为我的开源项目HMLLDB有个核心的函数HMLLDBHelpers.evaluateExpressionValue,里面用到了SBExpressionOptions这个选项,如果不说明一下,未来要写的相关文章都不好交代。
另外LLDB自带的expression命令有些选项和SBExpressionOptions是对应关系,读者可以根据名称对比一下。

SBExpressionOptions是什么

如果有用过SB API的话, 肯定对SBExpressionOptions有印象,无论是SBTarget还是SBFrame都有一个EvaluateExpression函数用于执行代码表达式(两者有区别,以后有机会再谈),而这个函数参数支持传入一个SBExpressionOptions对象,看名称就知道是干什么的,附上官方文档对此类的说明:

A container for options to use when evaluating expressions.

SBExpressionOptions方法解释

方法解释
SetCoerceResultToId执行表达式后,是否将结果强制转换为id类型
SetFetchDynamicValue可将结果转换为动态类型。可以设置3个值,依次是eNoDynamicValues(0),eDynamicCanRunTarget(1),eDynamicDontRunTarget(2)。如果一个NSArray *对象,0表示为NSArray *类型,1、2则表示为__NSArrayI *类型
SetUnwindOnError是否在错误时回溯堆栈信息
SetIgnoreBreakpoints执行表达式时,是否忽略断点的命中
SetGenerateDebugInfo官方给出的解释是设置是否为表达式生成调试信息,还控制是否生成SBModule。但我却没测试出区别,还不了解
SetTimeoutInMicroSeconds执行表达式的超时时间,单位为微秒,设置为0则永不超时
SetOneThreadTimeoutInMicroSeconds执行表达式在一个线程上的超时时间,微秒,必须小于GetTimeoutInMicroSeconds的值
SetTryAllThreads如果表达式在一个线程上未完成,是否运行所有的线程
SetStopOthers执行表达式时是否停止其他线程
SetTrapExceptions如果执行表达式引发了异常,是否中止执行
SetPlaygroundTransformEnabled这个函数官网文档没有,源码也没有,意义不明
SetREPLMode这个函数官网文档没有,源码也没有,意义不明
SetLanguage设置表达式所用的语言LanguageType,一般常用eLanguageTypeObjCeLanguageTypeObjC_plus_pluseLanguageTypeSwift。不设置则会根据上下文推断语言类型
SetSuppressPersistentResult设置为True,表达式结果就不会持久化。另外需要注意的是,LLDB的即时编译没有自动插入ARC释放的代码,表达式内alloc的对象,并不会释放,所以未释放对象和此项配置无关
SetPrefix设置表达式前缀。LLDB中经常要使用typedef,可以先配置好,另外也可以配置宏定义
SetAutoApplyFixIts设置为True,如果表达式有简单的拼写错误,会自动修正
SetRetriesWithFixIts尝试修正的次数(Xcode 12.5新增)
SetTopLevel是否忽略上下文,将表达式解析为顶级的实体。比如可以设为True,在表达式里定义一个C语言的外部函数或外部变量。
SetAllowJIT如果不能解析表达式,是否进行即时编译

SBExpressionOptions的默认值

安装HMLLDB,随便运行一个iOS项目,进入LLDB调试模式后,可以通过plldbClassInfo SBExpressionOptions命令查看其默认值

方法默认值
GetCoerceResultToIdFalse
GetFetchDynamicValue0,即eNoDynamicValues
GetUnwindOnErrorTrue
GetIgnoreBreakpointsFalse
GetGenerateDebugInfoFalse
GetTimeoutInMicroSeconds500000
GetOneThreadTimeoutInMicroSeconds0
GetTryAllThreadsTrue
GetStopOthersTrue
GetTrapExceptionsTrue
GetPlaygroundTransformEnabledFalse
GetREPLModeFalse
SetLanguage没有对应的Get方法,不设置则会根据上下文推断语言类型。
GetSuppressPersistentResultFalse
GetPrefixNone
GetAutoApplyFixItsTrue
GetRetriesWithFixIts1
GetTopLevelFalse
GetAllowJITTrue

HMLLDB项目的配置

相关的配置在HMLLDBHelpers.py文件的evaluateExpressionValue函数中能看到。

方法配置
SetCoerceResultToIdFalse,因为返回的是SBValueSBValue可操作性更强
SetFetchDynamicValueeNoDynamicValues(0),因为返回的是SBValue,本身就可以获取动态类型值,这样设置能保证最大的可操作性
SetUnwindOnErrorTrue,LLDB执行不稳定,有堆栈信息可更好debug
SetIgnoreBreakpointsTrue,HMLLDB所有命令都不需要命中断点而中断
SetGenerateDebugInfoFalse,不了解,保持默认值
SetTimeoutInMicroSeconds5000000,即5秒
SetOneThreadTimeoutInMicroSeconds4900000,即4.9秒。当SetTryAllThreads和SetStopOthers都设为True时,这个方法设置的值最好略小于GetTimeoutInMicroSeconds
SetTryAllThreadsTrue,不了解,保持默认值
SetStopOthersTrue,不了解,看字面意思,设置为True能保证执行时的稳定性
SetTrapExceptionsFalse,HMLLDB有Traceback机制,此项配置意义不大。如果大部分情况下都是runtime错误,设置为False可让系统自动捕获
SetPlaygroundTransformEnabledFalse,意义不明,保持默认
SetREPLModeFalse,意义不明,保持默认
SetLanguageeLanguageTypeObjC_plus_plus,OC比Swift稳定,且HMLLDB项目使用了大量的objc_msgSend函数,故不选用Swift。选OC++则是为了增加兼容性
SetSuppressPersistentResultTrue,表达式结果不需要持久化
SetPrefix具体内容可以查看HMExpressionPrefix.py文件。根据表达式实际情况,按需配置
SetAutoApplyFixItsTrue,HMLLDB是固定的代码,此项配置意义不大,保持默认
SetRetriesWithFixIts1,HMLLDB是固定的代码,此项配置意义不大,保持默认
SetTopLevelFalse,目前功能暂时不需要定义外部函数和外部变量,未来可期
SetAllowJITTrue,大部分功能都是必要的。

相关文章