<p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">这篇文章讨论了一个我们如何主动使用它的例子，包括Azure固件组件的安全审计。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">这是Azure服务深度安全审查的更广泛防御的一部分，从假设对手的角度探索攻击向量，该对手已经渗透了至少一个安全边界，现在位于服务后端的操作环境中（在下图中用*标记）。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;"><a href="https://p2.ssl.qhimg.com/t01c8eb749a200939bc.png" class="highslide-img" target="_blank" style="box-sizing: border-box; color: rgb(29, 173, 167); text-decoration: none; background-color: transparent;"><img class="aligncenter" alt="" src="https://p2.ssl.qhimg.com/t01c8eb749a200939bc.png" style="box-sizing: border-box; vertical-align: middle; border-style: none; max-width: 100%; display: block; margin: 12px auto; height: auto !important;"/></a></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">本次审查的目标之一是基于Linux的嵌入式设备，它与服务后端和管理后端相连接，在两者之间传递操作数据。该设备的主要攻击面是两个接口上使用的管理协议。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">对其固件的初步手动审查表明，该管理协议是基于消息的，有400多种不同的消息类型，每种类型都有自己的处理程序功能。手动审计每一个函数都是乏味且容易出错的，所以使用Semmle来扩展我们的代码审查能力是一个简单的选择。我们使用本文中讨论的静态分析技术，总共发现了33个易受攻击的消息处理函数。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">&nbsp;</p><h2 id="h2-1" style="box-sizing: border-box; margin-top: 12px; margin-bottom: 0.5rem; font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-weight: 500; line-height: 2; color: rgba(0, 0, 0, 0.85); font-size: 22px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">定义攻击面</h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">我们的第一步是编写一些QL来模拟来自攻击者的数据。管理协议以请求-响应为基础工作，其中每个消息请求类型都用类别号和命令号来标识。这是在源代码中使用如下结构数组定义的，例如：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">MessageCategoryTable&nbsp;g_MessageCategoryTable[]&nbsp;=
{
&nbsp;&nbsp;&nbsp;&nbsp;{&nbsp;CMD_CATEGORY_BASE,&nbsp;&nbsp;g_CommandHandlers_Base&nbsp;},
&nbsp;&nbsp;&nbsp;&nbsp;{&nbsp;CMD_CATEGORY_APP0,&nbsp;&nbsp;g_CommandHandlers_App0&nbsp;},
&nbsp;&nbsp;&nbsp;&nbsp;…
&nbsp;&nbsp;&nbsp;&nbsp;{&nbsp;NULL,&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;NULL&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}
};
CommandHandlerTable&nbsp;g_CommandHandlers_Base&nbsp;[]&nbsp;=
{
&nbsp;&nbsp;&nbsp;&nbsp;{&nbsp;CMD_GET_COMPONENT_VER,&nbsp;&nbsp;sizeof(ComponentVerReq),&nbsp;&nbsp;GetComponentVer,&nbsp;&nbsp;…&nbsp;},
&nbsp;&nbsp;&nbsp;&nbsp;{&nbsp;CMD_GET_GLOBAL_CONFIG,&nbsp;&nbsp;-1,&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;GetGlobalConfig,&nbsp;&nbsp;…&nbsp;},&nbsp;&nbsp;&nbsp;&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;…
&nbsp;&nbsp;&nbsp;&nbsp;{&nbsp;NULL,&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;NULL,&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;NULL,&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;…&nbsp;}
};</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">在上面的示例中，类别类型为<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">CMD_CATEGORY_BASE</code>、命令类型为<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">CMD_GET_COMPONENT_VER</code>的消息将被路由到<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">GetComponentVer</code>函数。命令处理程序表具有关于请求消息的预期大小的信息，该信息在调用处理函数之前在消息调度例程中得到验证。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">我们使用以下QL定义了消息处理程序表：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">class&nbsp;CommandHandlerTable&nbsp;extends&nbsp;Variable&nbsp;{&nbsp;
&nbsp;&nbsp;CommandHandlerTable()&nbsp;{&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;exists(Variable&nbsp;v&nbsp;|&nbsp;v.hasName("g_MessageCategoryTable")&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;and&nbsp;this.getAnAccess()&nbsp;=&nbsp;v.getInitializer().getExpr().getAChild().getChild(1)
&nbsp;&nbsp;&nbsp;&nbsp;)&nbsp;
&nbsp;&nbsp;}&nbsp;
}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">这需要一个名为<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">g_MessageCategoryTable</code>的变量，找到它的初始化表达式，并匹配该表达式的所有子表达式——每个子表达式对应于消息类别表的一行。对于每一行，它采用第二列(这是<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">getChild(1)</code>，因为getChild的参数是零索引的)，每一列都是对命令处理程序表的引用，并匹配引用的变量。在上面的例子中，它们是<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">g_CommandHandlers_Base</code>和<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">g_CommandHandlers_App0</code>。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">我们使用类似的方法定义了消息处理函数集：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">class&nbsp;MessageHandlerFunction&nbsp;extends&nbsp;Function&nbsp;{&nbsp;
&nbsp;&nbsp;Expr&nbsp;tableEntry;&nbsp;

&nbsp;&nbsp;MessageHandlerFunction()&nbsp;{&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;exists(CommandHandlerTable&nbsp;table&nbsp;|
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;tableEntry&nbsp;=&nbsp;table.getInitializer().getExpr().getAChild()
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;)
&nbsp;&nbsp;&nbsp;&nbsp;and&nbsp;this&nbsp;=&nbsp;tableEntry.getChild(2).(FunctionAccess).getTarget()
&nbsp;&nbsp;}&nbsp;&nbsp;int&nbsp;getExpectedRequestLength()&nbsp;{&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;result&nbsp;=&nbsp;tableEntry.getChild(1).getValue().toInt()&nbsp;
&nbsp;&nbsp;}&nbsp;
&nbsp;&nbsp;…
}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">这个QL类使用成员变量<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">tableEntry</code>来保存所有命令处理程序表中所有行的集合。这就是为什么它可以在（<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">messageHandlerffunction ( ) {…}</code><span>&nbsp;</span>)和<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">getExpectedRequestLength ( )</code>中引用，而无需重复定义。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">所有这些都映射到上面的代码结构，如下所示：</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;"><a href="https://p5.ssl.qhimg.com/t012ce341a4a56916fd.png" class="highslide-img" target="_blank" style="box-sizing: border-box; color: rgb(29, 173, 167); text-decoration: none; background-color: transparent;"><img class="aligncenter" alt="" src="https://p5.ssl.qhimg.com/t012ce341a4a56916fd.png" style="box-sizing: border-box; vertical-align: middle; border-style: none; max-width: 100%; display: block; margin: 12px auto; height: auto !important;"/></a></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">每个消息处理函数都有相同的签名：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">typedef&nbsp;unsigned&nbsp;char&nbsp;UINT8;int&nbsp;ExampleMessageHandler(UINT8&nbsp;*pRequest,&nbsp;int&nbsp;RequestLength,&nbsp;UINT8&nbsp;*pResponse);</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">并遵循一种通用模式，其中请求数据被转换为表示消息布局的结构类型，并通过其字段进行访问：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">int&nbsp;ExampleMessageHandler(UINT8&nbsp;*pRequest,&nbsp;int&nbsp;RequestLength,&nbsp;UINT8&nbsp;*pResponse)
{
&nbsp;&nbsp;&nbsp;&nbsp;ExampleMessageRequest*&nbsp;pMsgReq&nbsp;=&nbsp;(ExampleMessageRequest&nbsp;*)pRequest;
&nbsp;&nbsp;&nbsp;&nbsp;…
&nbsp;&nbsp;&nbsp;&nbsp;someFunction(pMsgReq->aaa.bbb)
&nbsp;&nbsp;&nbsp;&nbsp;…
}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">在这个分析中，我们只对请求数据感兴趣。我们在MessageHandlerFunction QL类中定义了两个额外的谓词来对请求数据及其长度进行建模：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">class&nbsp;MessageHandlerFunction&nbsp;extends&nbsp;Function&nbsp;{
&nbsp;&nbsp;Expr&nbsp;tableEntry;
&nbsp;&nbsp;…&nbsp;&nbsp;Parameter&nbsp;getRequestDataPointer()&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;result&nbsp;=&nbsp;this.getParameter(0)
&nbsp;&nbsp;}&nbsp;&nbsp;Parameter&nbsp;getRequestLength()&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;result&nbsp;=&nbsp;this.getParameter(1)
&nbsp;&nbsp;}
}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">抽象出消息处理函数的定义，它可以像任何其他QL类一样使用。例如，该查询按照<a href="https://en.wikipedia.org/wiki/Cyclomatic_complexity" style="box-sizing: border-box; color: rgb(29, 173, 167); text-decoration: none; background-color: transparent;">循环复杂性</a>的降序列出了所有消息处理程序函数：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">from&nbsp;MessageHandlerFunction&nbsp;mhfselect
&nbsp;&nbsp;mhf,&nbsp;
&nbsp;&nbsp;mhf.getADeclarationEntry().getCyclomaticComplexity()&nbsp;as&nbsp;ccorder&nbsp;by&nbsp;cc&nbsp;desc</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">&nbsp;</p><h2 id="h2-2" style="box-sizing: border-box; margin-top: 12px; margin-bottom: 0.5rem; font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-weight: 500; line-height: 2; color: rgba(0, 0, 0, 0.85); font-size: 22px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">分析数据流</h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">既然我们已经为不可信数据定义了一组入口点，下一步就是找到它可能以不安全的方式被使用的地方。为此，我们需要通过代码库跟踪这些数据流。QL提供了一个强大的全局数据流库，它抽象出了大部分复杂的语言细节。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">DataFlow</code><span>&nbsp;</span>库通过以下方式被纳入查询范围：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">import&nbsp;semmle.code.cpp.dataflow.DataFlow</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">它通过子类化<span>&nbsp;</span><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">DataFlow::Configuration</code><span>&nbsp;</span>和覆盖谓词来定义数据流，就像它应用于<span>&nbsp;</span><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">DataFlow::Node</code>一样，这是一个QL类，表示数据可以流过的任何程序假象：</p><table style="box-sizing: border-box; border-collapse: collapse; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;"><thead style="box-sizing: border-box;"><tr style="box-sizing: border-box;"><th style="box-sizing: border-box; text-align: left;"><strong style="box-sizing: border-box; font-weight: bolder;">配置谓词</strong></th><th style="box-sizing: border-box; text-align: left;">描述</th></tr></thead><tbody style="box-sizing: border-box;"><tr style="box-sizing: border-box;"><td style="box-sizing: border-box; text-align: left;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">isSource(source)</code></td><td style="box-sizing: border-box; text-align: left;">数据必须来自<span>&nbsp;</span><em style="box-sizing: border-box;">source</em></td></tr><tr style="box-sizing: border-box;"><td style="box-sizing: border-box; text-align: left;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">isSink(sink)</code></td><td style="box-sizing: border-box; text-align: left;">数据必须流入<span>&nbsp;</span><em style="box-sizing: border-box;">sink</em></td></tr><tr style="box-sizing: border-box;"><td style="box-sizing: border-box; text-align: left;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">isAdditionalFlowStep(node1, node2)</code></td><td style="box-sizing: border-box; text-align: left;">数据也可以在<em style="box-sizing: border-box;">node1</em>和<em style="box-sizing: border-box;">node2</em>之间流动</td></tr><tr style="box-sizing: border-box;"><td style="box-sizing: border-box; text-align: left;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">isBarrier(node)</code></td><td style="box-sizing: border-box; text-align: left;">数据无法通过<span>&nbsp;</span><em style="box-sizing: border-box;">node</em>流动</td></tr></tbody></table><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">大多数数据流查询将如下所示：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">class&nbsp;RequestDataFlowConfiguration&nbsp;extends&nbsp;DataFlow::Configuration&nbsp;{&nbsp;
&nbsp;&nbsp;RequestDataFlowConfiguration()&nbsp;{&nbsp;this&nbsp;=&nbsp;"RequestDataFlowConfiguration"&nbsp;}&nbsp;

&nbsp;&nbsp;override&nbsp;predicate&nbsp;isSource(DataFlow::Node&nbsp;source)&nbsp;{&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;…
&nbsp;&nbsp;}

&nbsp;&nbsp;override&nbsp;predicate&nbsp;isSink(DataFlow::Node&nbsp;sink)&nbsp;{&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;…
&nbsp;&nbsp;}

&nbsp;&nbsp;override&nbsp;predicate&nbsp;isAdditionalFlowStep(DataFlow::Node&nbsp;node1,&nbsp;DataFlow::Node&nbsp;node2)&nbsp;{&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;…
&nbsp;&nbsp;}

&nbsp;&nbsp;override&nbsp;predicate&nbsp;isBarrier(DataFlow::Node&nbsp;node)&nbsp;{&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;…
&nbsp;&nbsp;}

}
from&nbsp;DataFlow::Node&nbsp;source,&nbsp;DataFlow::Node&nbsp;sink&nbsp;
where&nbsp;any(RequestDataFlowConfiguration&nbsp;c).hasFlow(source,&nbsp;sink)&nbsp;
select&nbsp;
&nbsp;&nbsp;"Data&nbsp;flow&nbsp;from&nbsp;$@&nbsp;to&nbsp;$@",&nbsp;
&nbsp;&nbsp;source,&nbsp;sink</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">请注意，QL数据流库执行过程间分析——除了检查函数本地的数据流，它还将包括通过函数调用参数的数据。这是我们安全审查的一个基本特性，因为尽管下面讨论的易受攻击的代码模式在简单的示例函数中显示以便于演示，但是在我们目标的实际源代码中，大多数结果都有跨越多个复杂函数的数据流。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">&nbsp;</p><h2 id="h2-3" style="box-sizing: border-box; margin-top: 12px; margin-bottom: 0.5rem; font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-weight: 500; line-height: 2; color: rgba(0, 0, 0, 0.85); font-size: 22px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">发现内存安全漏洞</h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">由于这个固件组件是纯C代码库，我们首先决定搜索与内存安全相关的代码模式。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">这种错误的一个常见来源是不执行边界检查的数组索引。单独搜索这种模式将提供很大一部分结果，这些结果很可能不是安全漏洞，因为我们真正感兴趣的是攻击者对索引值的控制。因此，在这种情况下，我们在寻找数据流，其中接收器是数组索引表达式，源是消息处理函数的请求数据，并且任何数据流节点都有一个由相关边界检查保护的屏障。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">例如，我们想要找到匹配代码的数据流，如下所示：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">int&nbsp;ExampleMessageHandler(UINT8&nbsp;*pRequest(1:source),&nbsp;int&nbsp;RequestLength,&nbsp;UINT8&nbsp;*pResponse)
{
&nbsp;&nbsp;&nbsp;&nbsp;ExampleMessageRequest*&nbsp;pMsgReq(3)&nbsp;=&nbsp;(ExampleMessageRequest&nbsp;*)&nbsp;pRequest(2);&nbsp;&nbsp;&nbsp;&nbsp;int&nbsp;index1(6)&nbsp;=&nbsp;pMsgReq(4)->index1(5);
&nbsp;&nbsp;&nbsp;&nbsp;pTable1[index1(7:sink)].field1&nbsp;=&nbsp;pMsgReq->value1;
}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">但我们也希望排除代码的数据流，如下所示：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">int&nbsp;ExampleMessageHandler(UINT8&nbsp;*pRequest(1:source),&nbsp;int&nbsp;RequestLength,&nbsp;UINT8&nbsp;*pResponse)
{
&nbsp;&nbsp;&nbsp;&nbsp;ExampleMessageRequest*&nbsp;pMsgReq(3)&nbsp;=&nbsp;(ExampleMessageRequest&nbsp;*)&nbsp;pRequest(2);&nbsp;&nbsp;&nbsp;&nbsp;int&nbsp;index2(6)&nbsp;=&nbsp;pMsgReq(4)->index2(5);&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(index2&nbsp;>=&nbsp;0&nbsp;&&&nbsp;index2&nbsp;<&nbsp;PTABLE_SIZE)
&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;pTable2[index2].field1&nbsp;=&nbsp;pMsgReq->value2;
&nbsp;&nbsp;&nbsp;&nbsp;}
}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">源是使用前面讨论的<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">MessageHandlerFunction</code>类定义的，我们可以使用<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">ArrayExpr</code>的<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">getArrayOffset</code>谓词来定义合适的接收器：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">&nbsp;&nbsp;override&nbsp;predicate&nbsp;isSource(DataFlow::Node&nbsp;source)&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;any(MessageHandlerFunction&nbsp;mhf).getRequestDataPointer()&nbsp;=&nbsp;source.asParameter()
&nbsp;&nbsp;}&nbsp;&nbsp;override&nbsp;predicate&nbsp;isSink(DataFlow::Node&nbsp;sink)&nbsp;{&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;exists(ArrayExpr&nbsp;ae&nbsp;|&nbsp;ae.getArrayOffset()&nbsp;=&nbsp;sink.asExpr())&nbsp;&nbsp;
&nbsp;&nbsp;}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">默认情况下，数据流库只包括在每个节点保留值的流，如函数调用参数、赋值表达式等。但是我们也需要数据从请求数据指针流向它被转换到的结构的字段。我们将这样做：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">override&nbsp;predicate&nbsp;isAdditionalFlowStep(DataFlow::Node&nbsp;node1,&nbsp;DataFlow::Node&nbsp;node2)
&nbsp;&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;any&nbsp;terminal&nbsp;field&nbsp;access&nbsp;on&nbsp;request&nbsp;packet
&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;&nbsp;&nbsp;e.g.&nbsp;in&nbsp;expression&nbsp;a->b.c&nbsp;the&nbsp;data&nbsp;flows&nbsp;from&nbsp;a&nbsp;to&nbsp;c
&nbsp;&nbsp;&nbsp;&nbsp;exists(Expr&nbsp;e,&nbsp;FieldAccess&nbsp;fa&nbsp;|&nbsp;&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;node1.asExpr()&nbsp;=&nbsp;e&nbsp;and&nbsp;node2.asExpr()&nbsp;=&nbsp;fa&nbsp;|&nbsp;&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;fa.getQualifier*()&nbsp;=&nbsp;e&nbsp;and&nbsp;not&nbsp;(fa.getParent()&nbsp;instanceof&nbsp;FieldAccess)
&nbsp;&nbsp;&nbsp;&nbsp;)
&nbsp;&nbsp;}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">要使用边界检查排除流，我们在控制流图中较早的某些条件语句中使用变量或字段的任何节点上放置一个屏障（目前，我们假设任何这样的边界检查都是正确完成的）：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">&nbsp;&nbsp;override&nbsp;predicate&nbsp;isBarrier(DataFlow::Node&nbsp;node)&nbsp;{&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;exists(ConditionalStmt&nbsp;condstmt&nbsp;|&nbsp;&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;dataflow&nbsp;node&nbsp;variable&nbsp;is&nbsp;used&nbsp;in&nbsp;expression&nbsp;of&nbsp;conditional&nbsp;statement&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;&nbsp;&nbsp;this&nbsp;includes&nbsp;fields&nbsp;(because&nbsp;FieldAccess&nbsp;extends&nbsp;VariableAccess)
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;node.asExpr().(VariableAccess).getTarget().getAnAccess()
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;=&nbsp;condstmt.getControllingExpr().getAChild*()&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;and&nbsp;that&nbsp;statement&nbsp;precedes&nbsp;the&nbsp;dataflow&nbsp;node&nbsp;in&nbsp;the&nbsp;control&nbsp;flow&nbsp;graph&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;and&nbsp;condstmt.getASuccessor+()&nbsp;=&nbsp;node.asExpr()&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;and&nbsp;the&nbsp;dataflow&nbsp;node&nbsp;itself&nbsp;not&nbsp;part&nbsp;of&nbsp;the&nbsp;conditional&nbsp;statement&nbsp;expression&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;and&nbsp;not&nbsp;(node.asExpr()&nbsp;=&nbsp;cs.getControllingExpr().getAChild*())
&nbsp;&nbsp;&nbsp;&nbsp;)&nbsp;
&nbsp;&nbsp;}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">将此应用于以上两个示例，通过每个节点的数据流将是：</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;"><a href="https://p1.ssl.qhimg.com/t01a5ff14f547503c16.png" class="highslide-img" target="_blank" style="box-sizing: border-box; color: rgb(29, 173, 167); text-decoration: none; background-color: transparent;"><img class="aligncenter" alt="" src="https://p1.ssl.qhimg.com/t01a5ff14f547503c16.png" style="box-sizing: border-box; vertical-align: middle; border-style: none; max-width: 100%; display: block; margin: 12px auto; height: auto !important;"/></a></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">在我们的固件代码库中，此查询在15个消息处理程序函数中共定位了18个漏洞，这是攻击者控制的越界读写的混合。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">我们应用了类似的分析来查找函数调用的参数，而不首先验证消息请求数据。首先，我们定义了一个QL类来定义感兴趣的函数调用和参数，包括调用<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">memcpy</code>的<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">大小</code>参数和类似的函数<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">_fmemcpy</code>，以及<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">CalculateChecksum</code>的<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">长度</code>参数。<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">alculateChecksum</code>是一个特定于此代码库的函数，它将返回缓冲区的CRC32，并且可能被用作信息公开原语，消息处理函数将这个值复制到它的响应缓冲区中。</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">class&nbsp;ArgumentMustBeCheckedFunctionCall&nbsp;extends&nbsp;FunctionCall&nbsp;{
&nbsp;&nbsp;int&nbsp;argToCheck;

&nbsp;&nbsp;ArgumentMustBeCheckedFunctionCall()&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;(&nbsp;this.getTarget().hasName("memcpy")&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;and&nbsp;argToCheck&nbsp;=&nbsp;2&nbsp;)&nbsp;or
&nbsp;&nbsp;&nbsp;&nbsp;(&nbsp;this.getTarget().hasName("_fmemcpy")&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;and&nbsp;argToCheck&nbsp;=&nbsp;2&nbsp;)&nbsp;or
&nbsp;&nbsp;&nbsp;&nbsp;(&nbsp;this.getTarget().hasName("CalculateChecksum")&nbsp;and&nbsp;argToCheck&nbsp;=&nbsp;1&nbsp;)
&nbsp;&nbsp;}
&nbsp;&nbsp;Expr&nbsp;getArgumentToCheck()&nbsp;{&nbsp;result&nbsp;=&nbsp;this.getArgument(argToCheck)&nbsp;}
}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">接下来，我们修改了上一个查询的接收器，以匹配<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">argumentMustBecheckedFunctioncall</code>而不是数组索引：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">&nbsp;&nbsp;override&nbsp;predicate&nbsp;isSink(DataFlow::Node&nbsp;sink)&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;sink&nbsp;node&nbsp;is&nbsp;an&nbsp;argument&nbsp;to&nbsp;a&nbsp;function&nbsp;call&nbsp;that&nbsp;must&nbsp;be&nbsp;checked&nbsp;first
&nbsp;&nbsp;&nbsp;&nbsp;exists&nbsp;(ArgumentMustBeCheckedFunctionCall&nbsp;fc&nbsp;|&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;fc.getArgumentToCheck()&nbsp;=&nbsp;sink.asExpr())
&nbsp;&nbsp;}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">该查询揭示了13个消息处理程序中的另外17个漏洞，大部分是攻击者控制的越界读取（我们后来确认在响应消息中披露了这一点），其中一个是越界写入。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">&nbsp;</p><h2 id="h2-4" style="box-sizing: border-box; margin-top: 12px; margin-bottom: 0.5rem; font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-weight: 500; line-height: 2; color: rgba(0, 0, 0, 0.85); font-size: 22px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">污点跟踪</h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">在上面的查询中，我们重写了<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">DataFlow</code>库的<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">isAdditionalFlowStep</code>谓词，以确保当数据流向一个结构指针时，该结构的字段将作为节点添加到数据流图中。我们这样做是因为默认情况下，数据流分析仅包括数据值未经修改的路径，但我们希望跟踪它可能也受影响的特定表达式集。也就是说，我们定义了一组被不受信任的数据污染的特定表达式。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">QL包含一个内置库，可以应用更通用的方法来进行污点跟踪。它在<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">DataFlow</code>库之上开发，会覆盖<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">isAdditionalFlowStep，</code>并为值修改表达式提供更丰富的规则集。这是<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">TaintTracking</code>库，它以类似于<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">DataFlow的</code>方式导入：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">import&nbsp;semmle.code.cpp.dataflow.TaintTracking</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">它的使用方式与数据流库几乎相同，只是要扩展的QL类是<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">TaintTracking :: Configuration</code>，配置谓词如下：</p><table style="box-sizing: border-box; border-collapse: collapse; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;"><thead style="box-sizing: border-box;"><tr style="box-sizing: border-box;"><th style="box-sizing: border-box; text-align: left;">配置谓词</th><th style="box-sizing: border-box; text-align: left;">描述</th></tr></thead><tbody style="box-sizing: border-box;"><tr style="box-sizing: border-box;"><td style="box-sizing: border-box; text-align: left;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">isSource(source)</code></td><td style="box-sizing: border-box; text-align: left;">数据必须来自<em style="box-sizing: border-box;">source</em></td></tr><tr style="box-sizing: border-box;"><td style="box-sizing: border-box; text-align: left;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">isSink(sink)</code></td><td style="box-sizing: border-box; text-align: left;">数据必须流入<em style="box-sizing: border-box;">sink</em></td></tr><tr style="box-sizing: border-box;"><td style="box-sizing: border-box; text-align: left;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">isAdditionalTaintStep(node1, node2)</code></td><td style="box-sizing: border-box; text-align: left;"><em style="box-sizing: border-box;">node1上的</em>数据也会污染<em style="box-sizing: border-box;">node2</em></td></tr><tr style="box-sizing: border-box;"><td style="box-sizing: border-box; text-align: left;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">isSanitizer(node)</code></td><td style="box-sizing: border-box; text-align: left;">数据无法通过<em style="box-sizing: border-box;">node</em>流动</td></tr></tbody></table><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">我们重新运行了先前的查询，删除了<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">isAdditionalFlowStep</code>（因为我们不再需要定义它）并且将<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">isBarrier</code>重命名为<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">isSanitizer</code>。正如预期的那样，它返回了上面提到的所有结果，但也在数组索引中发现了一些额外的整数下溢缺陷。例如：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">int&nbsp;&nbsp;ExampleMessageHandler（UINT8&nbsp;*&nbsp;pRequest&nbsp;（1：source），&nbsp;&nbsp;int&nbsp;&nbsp;RequestLength，UINT8&nbsp;*&nbsp;pResponse）
{&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;ExampleMessageRequest&nbsp;*&nbsp;pMsgReq&nbsp;（3）&nbsp;&nbsp;=（ExampleMessageRequest&nbsp;*）pRequest&nbsp;（2）&nbsp;;&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;int&nbsp;&nbsp;index1&nbsp;（6）&nbsp;&nbsp;=&nbsp;pMsgReq&nbsp;（4）&nbsp;-&nbsp;>&nbsp;index1&nbsp;（5）&nbsp;;
&nbsp;&nbsp;&nbsp;&nbsp;pTable1&nbsp;[&nbsp;（&nbsp;index1&nbsp;（7）&nbsp;&nbsp;-&nbsp;2&nbsp;）（8：sink）&nbsp;]。field1&nbsp;=&nbsp;pMsgReq->&nbsp;value1;&nbsp;
}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">对于每种漏洞类型的内部报告，我们希望将它们与之前的查询结果分开进行分类。这包括使用<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; color: rgb(232, 62, 140); word-break: break-word;">SubExpr</code><span>&nbsp;</span>QL类对接收器进行简单修改：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">&nbsp;&nbsp;override&nbsp;predicate&nbsp;isSink(DataFlow::Node&nbsp;sink)&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;this&nbsp;sink&nbsp;is&nbsp;the&nbsp;left&nbsp;operand&nbsp;of&nbsp;a&nbsp;subtraction&nbsp;expression,
&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;是数组偏移表达式的一部分，例如[x&nbsp;-&nbsp;1]中的x&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;exists(ArrayExpr&nbsp;ae,&nbsp;SubExpr&nbsp;s&nbsp;|&nbsp;sink.asExpr()&nbsp;instanceof&nbsp;FieldAccess&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;and&nbsp;ae.getArrayOffset().getAChild*()&nbsp;=&nbsp;s&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;and&nbsp;s.getLeftOperand().getAChild*()&nbsp;=&nbsp;sink.asExpr())
&nbsp;&nbsp;}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">使我们在2个消息处理函数中增加了3个漏洞。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">&nbsp;</p><h2 id="h2-5" style="box-sizing: border-box; margin-top: 12px; margin-bottom: 0.5rem; font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-weight: 500; line-height: 2; color: rgba(0, 0, 0, 0.85); font-size: 22px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">查找路径遍历漏洞</h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">为了找到潜在的路径遍历漏洞，我们使用QL来尝试识别在文件打开函数中使用攻击者控制的文件名的消息处理函数。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">这次，我们使用了一种稍微不同的方法来跟踪污点，定义了一些额外的污点步骤，这些步骤将流经各种字符串处理的C库函数：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">predicate&nbsp;isTaintedString(Expr&nbsp;expSrc,&nbsp;Expr&nbsp;expDest)&nbsp;{
&nbsp;&nbsp;exists(FunctionCall&nbsp;fc,&nbsp;Function&nbsp;f&nbsp;|&nbsp;&nbsp;&nbsp;&nbsp;expSrc&nbsp;=&nbsp;fc.getArgument(1)&nbsp;and&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;expDest&nbsp;=&nbsp;fc.getArgument(0)&nbsp;and
&nbsp;&nbsp;&nbsp;&nbsp;f&nbsp;=&nbsp;fc.getTarget()&nbsp;and&nbsp;(
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;f.hasName("memcpy")&nbsp;or&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;f.hasName("_fmemcpy")&nbsp;or&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;f.hasName("memmove")&nbsp;or&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;f.hasName("strcpy")&nbsp;or&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;f.hasName("strncpy")&nbsp;or
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;f.hasName("strcat")&nbsp;or
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;f.hasName("strncat")
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;)
&nbsp;&nbsp;)&nbsp;&nbsp;or&nbsp;exists(FunctionCall&nbsp;fc,&nbsp;Function&nbsp;f,&nbsp;int&nbsp;n&nbsp;|
&nbsp;&nbsp;&nbsp;&nbsp;expSrc&nbsp;=&nbsp;fc.getArgument(n)&nbsp;and&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;expDest&nbsp;=&nbsp;fc.getArgument(0)&nbsp;and
&nbsp;&nbsp;&nbsp;&nbsp;f&nbsp;=&nbsp;fc.getTarget()&nbsp;and&nbsp;(&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;(f.hasName("sprintf")&nbsp;and&nbsp;n&nbsp;>=&nbsp;1)&nbsp;or&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;(f.hasName("snprintf")&nbsp;and&nbsp;n&nbsp;>=&nbsp;2)
&nbsp;&nbsp;&nbsp;&nbsp;)
&nbsp;&nbsp;)}
…&nbsp;&nbsp;override&nbsp;predicate&nbsp;isAdditionalTaintStep(DataFlow::Node&nbsp;node1,&nbsp;DataFlow::Node&nbsp;node2)&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;isTaintedString(node1.asExpr(),&nbsp;node2.asExpr())
&nbsp;&nbsp;}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">并将接收器定义为文件打开函数的路径参数：</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">class&nbsp;FileOpenFunction&nbsp;extends&nbsp;Function&nbsp;{
&nbsp;&nbsp;FileOpenFunction()&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;this.hasName("fopen")&nbsp;or&nbsp;this.hasName("open")
&nbsp;&nbsp;}&nbsp;&nbsp;int&nbsp;getPathParameter()&nbsp;{&nbsp;result&nbsp;=&nbsp;0&nbsp;}&nbsp;//&nbsp;filename&nbsp;parameter&nbsp;index}

…&nbsp;&nbsp;override&nbsp;predicate&nbsp;isSink(DataFlow::Node&nbsp;sink)&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;exists(FunctionCall&nbsp;fc,&nbsp;FileOpenFunction&nbsp;fof&nbsp;|
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;fc.getTarget()&nbsp;=&nbsp;fof&nbsp;and&nbsp;fc.getArgument(fof.getPathParameter())&nbsp;=&nbsp;sink.asExpr())
&nbsp;&nbsp;}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">通过对我们的目标设备如何工作的一些预先了解，从最初的回顾中观察到，在我们解决下一个排除数据验证流程的问题之前，我们预计至少会有一些结果，就像之前的查询一样。但是，查询根本没有返回任何内容。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">由于没有要检查的数据流路径，我们回过头来查询函数调用图，以搜索消息处理函数和调用文件打开函数之间的任何路径，不包括path参数为常量的调用:</p><pre style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; font-size: 14px; margin-top: 0px; margin-bottom: 1rem; overflow: auto; display: block; color: rgb(33, 37, 41); font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">//&nbsp;this&nbsp;recursive&nbsp;predicate&nbsp;defines&nbsp;a&nbsp;function&nbsp;call&nbsp;graph
predicate&nbsp;mayCallFunction(Function&nbsp;caller,&nbsp;FunctionCall&nbsp;fc)&nbsp;{
&nbsp;&nbsp;fc.getEnclosingFunction()&nbsp;=&nbsp;caller&nbsp;or&nbsp;mayCallFunction(fc.getTarget(),&nbsp;fc)
}

from&nbsp;MessageHandlerFunction&nbsp;mhf,&nbsp;FunctionCall&nbsp;fc,&nbsp;FileOpenFunction&nbsp;fofwhere&nbsp;mayCallFunction(mhf,&nbsp;fc)
&nbsp;&nbsp;and&nbsp;fc.getTarget()&nbsp;=&nbsp;fof
&nbsp;&nbsp;and&nbsp;not&nbsp;fc.getArgument(fof.getPathParameter()).isConstant()
select&nbsp;
&nbsp;&nbsp;mhf,&nbsp;"$@&nbsp;may&nbsp;have&nbsp;a&nbsp;path&nbsp;to&nbsp;$@",
&nbsp;&nbsp;mhf,&nbsp;mhf.toString(),&nbsp;&nbsp;fc,&nbsp;fc.toString()</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">该查询提供了5个结果——足够少，可以手动检查——由此我们发现了2个路径遍历漏洞，一个写入文件，另一个读取文件，这两个漏洞都是攻击者提供的路径。事实证明，污点跟踪没有标记这些，因为它需要发送两种不同的消息类型：第一种是设置文件名，第二个读取或写入具有该名称的文件的数据。幸运的是，QL足够灵活，可以提供另一种探索途径。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">&nbsp;</p><h2 id="h2-6" style="box-sizing: border-box; margin-top: 12px; margin-bottom: 0.5rem; font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-weight: 500; line-height: 2; color: rgba(0, 0, 0, 0.85); font-size: 22px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">结论</h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 1rem; color: rgb(88, 88, 88); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial;">在微软，我们采取深度防御的方法来保护云和客户数据的安全。其中一个重要部分是对Azure内部攻击面进行全面的安全审查。在这个嵌入式设备的源代码回顾中，我们应用了Semmle QL的高级静态分析技术来发现基于消息的管理协议中的漏洞。这在各种bug类中发现了总共33个易受攻击的消息处理程序。使用QL使我们能够自动执行完全手动代码审查的重复部分，同时仍然应用探索性方法。</p><p><span style="color: rgb(153, 153, 153); font-family: Lato, "Helvetica Neue For Number", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 13px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: left; text-indent: 0px; text-transform: none; white-space: normal; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; background-color: rgb(255, 255, 255); text-decoration-style: initial; text-decoration-color: initial; display: inline !important; float: none;">本文翻译自 blogs.technet.microsoft.com 转载自安全客<br/></span></p>