当前位置: 首页 > news >正文

c:变参函数:汇编解析;va_list;marco 宏:__VA_ARGS__

文章目录

  • 参考
  • gcc 内部的宏定义
  • 代码
  • 汇编
  • 调用
  • 在 SEI CERT C++ Coding Standard 这个标准里
  • 示例
  • 实例
  • 宏里的使用

参考

https://git.sr.ht/~gregkh/presentation-security/blob/3547183843399d693c35b502cf4a313e256d0dd8/security-stuff.pdf

gcc 内部的宏定义

宏定义:
使用的时builtin_va_end 宏定义。

#define va_start(v,l)	__builtin_va_start(v,l)
#define va_end(v)	__builtin_va_end(v)
#define va_arg(v,l)	__builtin_va_arg(v,l)

使用到了下列内置函数来实现va-start、end、arg相关的宏。

/* Expand EXP, a call to __builtin_va_start.  */static rtx
expand_builtin_va_start (tree exp)
{rtx nextarg;tree valist;location_t loc = EXPR_LOCATION (exp);

代码

int add(int first, int second, ... )
{int r=first + second;va_list va;;;va_start(va, second);while(int v= va_arg(va,int)){r+=v;}va_end(va);return r;
}

汇编

   962 0000000000400c56 <add(int, int, ...)>:963   400c56:       55                      push   %rbp964   400c57:       48 89 e5                mov    %rsp,%rbp965   400c5a:       48 83 ec 68             sub    $0x68,%rsp  申请的栈空间是 0x68,但是使用的远远大于这个值。966   400c5e:       89 bd 2c ff ff ff       mov    %edi,-0xd4(%rbp)   六个寄存器参数都用上了967   400c64:       89 b5 28 ff ff ff       mov    %esi,-0xd8(%rbp)    而且将这六个参数放到超远的栈上。968   400c6a:       48 89 95 60 ff ff ff    mov    %rdx,-0xa0(%rbp)969   400c71:       48 89 8d 68 ff ff ff    mov    %rcx,-0x98(%rbp)970   400c78:       4c 89 85 70 ff ff ff    mov    %r8,-0x90(%rbp)971   400c7f:       4c 89 8d 78 ff ff ff    mov    %r9,-0x88(%rbp)972   400c86:       84 c0                   test   %al,%al    看着调用一直时0,是否用到floating 的变量?几个973   400c88:       74 20                   je     400caa <add(int, int, ...)+0x54> 如果又floating的数值,将floating寄存器的值放到 栈974   400c8a:       0f 29 45 80             movaps %xmm0,-0x80(%rbp)  floating的值放到floating 寄存器。975   400c8e:       0f 29 4d 90             movaps %xmm1,-0x70(%rbp)976   400c92:       0f 29 55 a0             movaps %xmm2,-0x60(%rbp)977   400c96:       0f 29 5d b0             movaps %xmm3,-0x50(%rbp)978   400c9a:       0f 29 65 c0             movaps %xmm4,-0x40(%rbp)979   400c9e:       0f 29 6d d0             movaps %xmm5,-0x30(%rbp)980   400ca2:       0f 29 75 e0             movaps %xmm6,-0x20(%rbp)981   400ca6:       0f 29 7d f0             movaps %xmm7,-0x10(%rbp)982   400caa:       8b 95 2c ff ff ff       mov    -0xd4(%rbp),%edx983   400cb0:       8b 85 28 ff ff ff       mov    -0xd8(%rbp),%eax984   400cb6:       01 d0                   add    %edx,%eax   第一个和第二个参数相加放到eax985   400cb8:       89 85 4c ff ff ff       mov    %eax,-0xb4(%rbp)  然后放到 栈986   400cbe:       c7 85 30 ff ff ff 10    movl   $0x10,-0xd0(%rbp)16 放到 栈987   400cc5:       00 00 00988   400cc8:       c7 85 34 ff ff ff 30    movl   $0x30,-0xcc(%rbp)0x30 放到栈989   400ccf:       00 00 00990   400cd2:       48 8d 45 10             lea    0x10(%rbp),%rax   将rbp + 16 的值放到rax,使用到了上一个函数栈。991   400cd6:       48 89 85 38 ff ff ff    mov    %rax,-0xc8(%rbp)  ,放到栈992   400cdd:       48 8d 85 50 ff ff ff    lea    -0xb0(%rbp),%rax   将rbp -0xb0的地址放到rax993   400ce4:       48 89 85 40 ff ff ff    mov    %rax,-0xc0(%rbp)    放到栈994   400ceb:       8b 85 30 ff ff ff       mov    -0xd0(%rbp),%eax    将rbp-0xd0的值放到eax995   400cf1:       83 f8 2f                cmp    $0x2f,%eax     对比0x2f 为什么比对2f996   400cf4:       77 23                   ja     400d19 <add(int, int, ...)+0xc3>997   400cf6:       48 8b 85 40 ff ff ff    mov    -0xc0(%rbp),%rax998   400cfd:       8b 95 30 ff ff ff       mov    -0xd0(%rbp),%edx999   400d03:       89 d2                   mov    %edx,%edx1000   400d05:       48 01 d0                add    %rdx,%rax1001   400d08:       8b 95 30 ff ff ff       mov    -0xd0(%rbp),%edx1002   400d0e:       83 c2 08                add    $0x8,%edx1003   400d11:       89 95 30 ff ff ff       mov    %edx,-0xd0(%rbp)1004   400d17:       eb 12                   jmp    400d2b <add(int, int, ...)+0xd5>1005   400d19:       48 8b 85 38 ff ff ff    mov    -0xc8(%rbp),%rax1006   400d20:       48 8d 50 08             lea    0x8(%rax),%rdx1007   400d24:       48 89 95 38 ff ff ff    mov    %rdx,-0xc8(%rbp)1008   400d2b:       8b 00                   mov    (%rax),%eax1009   400d2d:       89 85 48 ff ff ff       mov    %eax,-0xb8(%rbp)1010   400d33:       83 bd 48 ff ff ff 00    cmpl   $0x0,-0xb8(%rbp)1011   400d3a:       74 0e                   je     400d4a <add(int, int, ...)+0xf4>1012   400d3c:       8b 85 48 ff ff ff       mov    -0xb8(%rbp),%eax1013   400d42:       01 85 4c ff ff ff       add    %eax,-0xb4(%rbp)   将 eax 加到栈里1014   400d48:       eb a1                   jmp    400ceb <add(int, int, ...)+0x95>1015   400d4a:       8b 85 4c ff ff ff       mov    -0xb4(%rbp),%eax  最终的结果放到eax1016   400d50:       c9                      leaveq1017   400d51:       c3                      retq

调用

int main()
{
printf( "abc=%d\n", add(1,3,4,5,8,9,10);
return 1;
}
0000000000400968 <main>:400968:       55                      push   %rbp400969:       48 89 e5                mov    %rsp,%rbp40096c:       48 83 ec 08             sub    $0x8,%rsp  按16 字节对齐。多8个字节占栈,如果多出来一个参数400970:       6a 0a                   pushq  $0xa400972:       41 b9 09 00 00 00       mov    $0x9,%r9d400978:       41 b8 08 00 00 00       mov    $0x8,%r8d40097e:       b9 05 00 00 00          mov    $0x5,%ecx400983:       ba 04 00 00 00          mov    $0x4,%edx400988:       be 03 00 00 00          mov    $0x3,%esi40098d:       bf 01 00 00 00          mov    $0x1,%edi400992:       b8 00 00 00 00          mov    $0x0,%eax  把 eax 清空400997:       e8 aa fe ff ff          callq  400846 <add(int, int, ...)>40099c:       48 83 c4 10             add    $0x10,%rsp  释放栈4009a0:       89 c6                   mov    %eax,%esi4009a2:       bf b2 0a 40 00          mov    $0x400ab2,%edi4009a7:       b8 00 00 00 00          mov    $0x0,%eax4009ac:       e8 3f fd ff ff          callq  4006f0 <printf@plt>4009b1:       b8 01 00 00 00          mov    $0x1,%eax4009b6:       c9                      leaveq4009b7:       c3                      retq

在 SEI CERT C++ Coding Standard 这个标准里

提到了更安全的C++定义方式。 这种方式将编程从运行时变参,转移到了编译时,更安全。

示例

https://en.cppreference.com/w/cpp/language/parameter_pack

#include <iostream>void tprintf(const char* format) // base function
{std::cout << format;
}
、、 这个会产生多少个函数来?
template<typename T, typename... Targs>
void tprintf(const char* format, T value, Targs... Fargs) // recursive variadic function
{for ( ; *format != '\0'; format++ ) {if ( *format == '%' ) {std::cout << value;tprintf(format+1, Fargs...); // recursive callreturn;}std::cout << *format;}
}int main()
{tprintf("% world% %\n","Hello",'!',123);
}

实例

变参传递到另一个函数里:

static inline void abc(int level, const char *format, ...)
{char buff[UMAX_LOG_SIZE];int msgLen;va_list arglist;memset(buff, 0, sizeof(buff));va_start(arglist, format);msgLen = vsnprintf(buff, UMAX_LOG_SIZE, format, arglist);va_end(arglist);

宏里的使用

下面这个宏,只包含有,三个点所代表的参数;不包含三个点以外有名称的参数。

__VA_ARGS__
#define _FUNC1_(tn, constness, ct, Method, ...) \
class mock_##Method { \
public:\RESULT_(tn, __VA_ARGS__) ct Method( \  // 这里__VA_ARGS__, 不包含  tn,constness,ct和Method

相关文章:

  • 数字孪生与智慧城市:开启未来智慧生活
  • 2023CCF中国开源大会 | 麒麟信安作为首批合作伙伴入驻全国信创开源广场
  • 计算机专业毕业设计如何选题、如何规避风险、避免入坑
  • uniapp解决iOS切换语言——原生导航栏buttons文字不生效
  • Elasticsearch聚合----aggregations的简单使用
  • 软考系统架构师知识点集锦二:软件工程
  • Tensorflow2 中模型训练标签顺序和预测结果标签顺序不一致问题解决办法
  • jsp简单实现新闻发布系统中用户注册确认和用户模拟登录功能的开发
  • 设计模式之代理模式
  • 【RabbitMQ 实战】12 镜像队列
  • 跟我学C++中级篇——右值引用和万能引用
  • Positive Technologies 在迪拜宣布与地区网络安全解决方案提供商开展合作
  • 环形链表(C++解法)
  • 京东销量(销额)数据分析:2023年9月京东奶粉行业品牌销售排行榜
  • Linux高性能服务器编程——ch9笔记
  • .pyc 想到的一些问题
  • 〔开发系列〕一次关于小程序开发的深度总结
  • 2017 年终总结 —— 在路上
  • 2017-09-12 前端日报
  • Android Volley源码解析
  • Android组件 - 收藏集 - 掘金
  • Angularjs之国际化
  • bearychat的java client
  • CentOS从零开始部署Nodejs项目
  • ES6系统学习----从Apollo Client看解构赋值
  • GDB 调试 Mysql 实战(三)优先队列排序算法中的行记录长度统计是怎么来的(上)...
  • iOS 系统授权开发
  • JS变量作用域
  • Laravel Mix运行时关于es2015报错解决方案
  • Laravel深入学习6 - 应用体系结构:解耦事件处理器
  • learning koa2.x
  • Promise面试题,控制异步流程
  • Python学习之路16-使用API
  • Twitter赢在开放,三年创造奇迹
  • vue学习系列(二)vue-cli
  • Webpack 4x 之路 ( 四 )
  • 基于Dubbo+ZooKeeper的分布式服务的实现
  • 基于阿里云移动推送的移动应用推送模式最佳实践
  • 判断客户端类型,Android,iOS,PC
  • 如何合理的规划jvm性能调优
  • 手机端车牌号码键盘的vue组件
  • 数据科学 第 3 章 11 字符串处理
  • ​LeetCode解法汇总2696. 删除子串后的字符串最小长度
  • ​人工智能书单(数学基础篇)
  • # Maven错误Error executing Maven
  • #Java第九次作业--输入输出流和文件操作
  • #stm32驱动外设模块总结w5500模块
  • #鸿蒙生态创新中心#揭幕仪式在深圳湾科技生态园举行
  • (ibm)Java 语言的 XPath API
  • (仿QQ聊天消息列表加载)wp7 listbox 列表项逐一加载的一种实现方式,以及加入渐显动画...
  • (附源码)ssm基于jsp高校选课系统 毕业设计 291627
  • (附源码)小程序 交通违法举报系统 毕业设计 242045
  • (九)信息融合方式简介
  • (力扣记录)235. 二叉搜索树的最近公共祖先
  • (利用IDEA+Maven)定制属于自己的jar包