营销型网站功能表,网络分析的应用案例,建站之星网站模版商城,怎么制作网页链接在微信上发研发效能是一个涉及面很广的话题#xff0c;它涵盖了软件交付的整个生命周期#xff0c;涉及产品、架构、开发、测试、运维#xff0c;每个环节都可能影响顺畅、高质量地持续有效交付。在腾讯安全平台部实际研发与测试工作中我们发现#xff0c;代码插桩隔离是单元测试工作… 研发效能是一个涉及面很广的话题它涵盖了软件交付的整个生命周期涉及产品、架构、开发、测试、运维每个环节都可能影响顺畅、高质量地持续有效交付。在腾讯安全平台部实际研发与测试工作中我们发现代码插桩隔离是单元测试工作中的一个强需求然而业界现有 C/C 插桩工具由于使用上的局限性运行效率和体验仍有很大改善空间。本文介绍了团队基于研效优化实践而自研的动态插桩工具旨在实现单元测试的轻量化运行提高代码覆盖率从而助力研发团队的效能提升。问题思路目前存在的 C/C 插桩工具基本上都有各种使用上的局限比如流行的 gmock只能对 C 的虚函数进行插桩替换针对非虚函数则需要先对被测代码进行改造同时对于系统接口C 风格的第三方库代码也无能为力。如果可以绕开编译器直接从底层入手比如做机器指令修改则可以不受语法及编译器的束缚直接达到目的这样在使用中就 几乎不受限制。原理C/C 语言编译后的可执行体其实就是一个个的函数实现每个函数的开头就是它的入口。一个函数 A 调用另一个函数 B就是代码在执行过程中控制流从函数 A 的某处跳到了函数 B 的开头所以如果想用一个新的函数 C 取代函数 B可以在函数 B 的开头用机器码的形式写入如下等价逻辑MOVQ ADDRESS_OF_C %RAX //将函数C的地址放到寄存器RAX
JMPQ *RAX //无条件跳转到RAX所指向的位置这样当控制流从函数 A 进入函数 B 的开始位置的时候即会执行上述代码从而直接跳转到 C 的开头处。其最终效果是所有对函数 B 的调用都如同直接调用了函数 C。基于上述原理被插桩的代码包括第三方库如 MySql、其他同事未完成的模块、甚至是操作系统的 API 接口如 read、select 等同时桩函数不仅可以模拟原函数的返回值实际上它作为一个普通的 C 函数对原函数有完全的操作能力比如可以访问传递给原函数调用真实的参数、C 成员变量针对对成员函数的模拟给定任意的返回值访问全局变量、对调用进行计数等。实际实现中考虑到不同测试用例间的互不干扰除了能执行函数替换还需要在执行完一个测试时还原现场。这些具体细节可以直接参考代码。使用对全局函数插桩原始函数int global(int a, int b) {return a b;
}对应的桩函数int fake_global(int a, int b) {//校验参数正确性确定被测代码传入了正确的值assert(a 3);assert(b 2);//给一个返回值配合被测代码走特定分支return a - b;
}插桩示例assert(global(3, 2) 5);//通过mock调用完成函数动态替换
assert(0 mock(global, fake_global));//调用mock后的函数可以看到返回值变了
assert(global(3, 2) 1);//结束mock
reset();//函数行为恢复
assert(global(3, 2) 5);对普通成员函数插桩被测代码class A {
public:int member(int a) {return a;}static int static_member(int a) {return 200;}virtual int virtual_member() {return 400;}
};桩函数int fake_member(A *pTihs, int a) {//由于是对成员函数插桩,这里需要这个this指针参数return --a;
}插桩示例A a;
assert(a.member(100) 101);mock(A::member, fake_member);
assert(a.member(100) 99);reset();assert(a.member(100) 101);对静态成员函数插桩桩函数int fake_static_member() {//静态函数不需要this指针return 300;
}插桩示例assert(A::static_member(200) 200);mock(A::static_member, fake_static_member);
assert(A::static_member(100) 300);reset();assert(A::static_member(200) 200);对虚函数插桩桩函数int fake_virtual_member(A *pThis) {//虚函数同普通的成员函数由于,同样需要this指针return 500;
}插桩示例A a;
assert(a.virtual_member() 400);//虚函数mock需要多传一个相关类的对象任意一个对象即可跟实际代码中的对象没有关系
A a_obj;
mock(A::virtual_member, fake_virtual_member, a_obj);
assert(a.virtual_member() 500);reset();
assert(a.virtual_member() 400);对系统及第三方库函数插桩桩函数int fake_write(int, char*, int) {return 100;
}插桩示例//直接写入一个无效的文件描述符会失败
assert(write(5, hello, 5) -1);//来一个假的wirte
mock(write, fake_write);
//模拟调用成功
assert(write(5, hello, 5) 100);reset();assert(write(5, hello, 5) -1);可以看到对系统函数的 mock,其实跟普通的全局函数并无两样第三方库函数也是同理。使用限制注意事项目前支持 X86_64 平台上的 Linux、MacOS 系统如有需求Windows 和其它硬件平台如 X86_32、ARM也可在短期内支持。MacOS 下需要在执行前对单测可执行文件做以下修改printf \x07 | dd of bs1 seek160 count1 convnotrunc显然这种方法对内联函数无效不过对于单元测试来说可以关闭内联同时也建议关闭其它编译器优化。可以使用-fno-access-control 编译你的测试代码可以使 g 关闭 c 成员的访问控制即 protected 及 private 不再生效。项目地址https://github.com/wangyongfeng5/lmock结语持续改进是研效工具平台发展的必经之路欢迎感兴趣的同学与我们交流探讨共同助力测试效能的优化。声明本文于网络整理版权归原作者所有如来源信息有误或侵犯权益请联系我们删除或授权事宜。