前篇分析过在objc中函数调用传参的一般方式,本篇分析vc在x64体系中的一般传参方式。手头上因为没有64位的vc编译器,只好用windbg看ms自身的函数是怎么样调用的。

首先看两个再熟悉不过的api,memset和CreateHeap。函数原型和使用像呼吸一样了,也不用多解释一看就明白。

调用memset(rcx, rdx, r8):

`76df5f63 488b842480000000 mov     rax,qword ptr [rsp+80h]
`76df5f6b mov qword ptr [rsp+98h],rax
`76df5f73 458bc5 mov r8d,r13d
`76df5f76 33d2 xor edx,edx
`76df5f78 488bc8 mov rcx,rax
`76df5f7b e810cfffff call ntdll!memset (`76df2e90)

调用RtlCreateHeap(rcx, rdx, r8, r9, [rsp+20h], [rsp+28h]):

`76de47a6 4c8b842498000000 mov     r8,qword ptr [rsp+98h]
`76de47ae 488d05ab200000 lea rax,[ntdll!RtlReleaseMemoryStream (`76de6860)]
`76de47b5 448d4e01 lea r9d,[rsi+]
`76de47b9 488bd7 mov rdx,rdi
`76de47bc mov qword ptr [rsp+188h],rax
`76de47c4 488d842440010000 lea rax,[rsp+140h]
`76de47cc b900800000 mov ecx,8000h
`76de47d1 mov qword ptr [rsp+28h],rax
`76de47d6 c784244001000060000000 mov dword ptr [rsp+140h],60h
`76de47e1 48c784247801000000100000 mov qword ptr [rsp+178h],1000h
`76de47ed mov qword ptr [rsp+20h],rsi
`76de47f2 4c89842480010000 mov qword ptr [rsp+180h],r8
`76de47fa e8a18cffff call ntdll!RtlCreateHeap (`76ddd4a0)

再来分析一下有多个参数的api函数。

`74a4b6f9 4c8d5c2450      lea     r11,[rsp+50h]
`74a4b6fe 488d442470 lea rax,[rsp+70h]
`74a4b703 4c895c2448 mov qword ptr [rsp+48h],r11
`74a4b708 mov qword ptr [rsp+40h],rax
`74a4b70d 488d842480000000 lea rax,[rsp+80h]
`74a4b715 mov qword ptr [rsp+38h],rax
`74a4b71a 488d842490000000 lea rax,[rsp+90h]
`74a4b722 4c8d8c24b0000000 lea r9,[rsp+0B0h]
`74a4b72a mov qword ptr [rsp+30h],rax
`74a4b72f 488d8424a0000000 lea rax,[rsp+0A0h]
`74a4b737 4c8d8424c0000000 lea r8,[rsp+0C0h]
`74a4b73f 488d542460 lea rdx,[rsp+60h]
`74a4b744 488d8c2400010000 lea rcx,[rsp+100h]
`74a4b74c 4c89742428 mov qword ptr [rsp+28h],r14
`74a4b751 mov qword ptr [rsp+20h],rax
`74a4b756 ff153c5affff call qword ptr [wow64+0x1198 (`74a41198)]

wow64经一些处理后跳转到ntdll!RtlCreateProcessParameters

ntdll!RtlCreateProcessParameters:
`76dede20 4883ec68 sub rsp,68h
`76dede24 488b8424b8000000 mov rax,qword ptr [rsp+0B8h]
`76dede2c c744245000000000 mov dword ptr [rsp+50h],
`76dede34 mov qword ptr [rsp+48h],rax
`76dede39 488b8424b0000000 mov rax,qword ptr [rsp+0B0h]
`76dede41 mov qword ptr [rsp+40h],rax
`76dede46 488b8424a8000000 mov rax,qword ptr [rsp+0A8h]
`76dede4e mov qword ptr [rsp+38h],rax
`76dede53 488b8424a0000000 mov rax,qword ptr [rsp+0A0h]
`76dede5b mov qword ptr [rsp+30h],rax
`76dede60 488b842498000000 mov rax,qword ptr [rsp+98h]
`76dede68 mov qword ptr [rsp+28h],rax
`76dede6d 488b842490000000 mov rax,qword ptr [rsp+90h]
`76dede75 mov qword ptr [rsp+20h],rax
`76dede7a e851b4feff call ntdll!RtlCreateProcessParametersEx (`76dd92d0)

RtlCreateProcessParameters函数将上一层传入的参数原封不动,并在后面增加一个参数0,然用后调RtlCreateProcessParametersEx。

RtlCreateProcessParameters函数一共有10个参数,RtlCreateProcessParametersEx函数一共有11个参数。
前4个参数使用的寄存器rcx,rdx,r8和r9, 后6个参数用了入栈方式,虽然前4个参数作用了寄存器,但仍然在堆栈中空出了4个qword的位置。第五个参数起是由rsp+20h处开始放入(mov)堆栈(,入栈push是由最后一个开始)。
线程在调用RtlCreateProcessParameters前,准备调用栈过程中并没有往rsp~rsp+18h处写东西,而从寄存器的调配可以看出用了寄存器rcx,rdx,r8,r9。
线程在调用RtlCreateProcessParametersEx前,将上一层调用栈后面6个参数搬运了过来,并增加了一个参数放到rsp+50h处。

通过windbg浏览几处api函数调用,可以看出vc的__stdcall调用的传参方式。前4个参数使用rcx,rdx,r8,r9,之后的参数使用堆栈并且在调用栈中空出4个qword位置,第5个参数不是rsp-8而是rsp-28h在函数入口处时。如果参数数量不超过4个,不使用堆栈。反汇编代码也说明vc在传参数时没有使用到rdi,rsi。

下一篇分析gcc在x64的传参方式。

上一篇分析objc在x64的传参方式。

vc在x64体系的一般传参数方式的更多相关文章

  1. gcc在x64体系中如何传递参数,linux,mac,iOS适用

    上一篇介绍了vc(windows)平台在x64体系当中,c函数的传参方式.本篇将要介绍gcc(类linux,mac)平台在x64中,c函数是如何传参的.为节约时间和篇幅,首先来定义一个有十个参数的函数 ...

  2. GET传参数方式

    controller:/getDetail/{id} /getDetail?id1234567 /getDetail?id=id1234567

  3. 超链接向servlet传参数

    超链接传参数方式如下: <a href=xxxServlet?flag=1  target=“XX”></a> 注意:target=“XX”是用来指定在什么窗体打开.xx为该窗 ...

  4. 传参数应该用哪种形式——值、引用、指针?

    类型:C++ & Qt4,创建时间:十二月 30, 2011, 7:43 p.m. 标题无"转载"即原创文章,版权所有.转载请注明来源:http://hgoldfish.c ...

  5. Mybatis传参数

    1使用@Param注解传参数 mapper接口:public void updateUser(@Param("user")User user)throws Exception; m ...

  6. 如何给main传参数

    main 函数的参数有连个argc argcv[]  argc 是参数个数 argcv是参数的数组指针,且argcv的第一个参数是默认程序路径加程序名 给main传参数,需要在命令行启动程序时设置 如 ...

  7. mvc中多参数URL会很长,首次加载不传参数让url很短,路由规则实现方法[bubuko.com]

    如要实现列表中地址全路径“bubuko-11-2.html”,在首次进入时,使用短路径“bubuko.html”,只有再次href后才显示全路径“bubuko-11-2.html”,下面使用路由规则来 ...

  8. Chrome和Firefox浏览器执行new Date() 函数传参数得到不同结果的陷阱

    某日,同事问到关于new Date() 函数传参数,在火狐浏览器和谷歌浏览器控制台运行,会得到不同的结果,刚开始觉得不可能,后来实际操作才发现此陷阱 var date = new Date('2014 ...

  9. web service上传参数代码实例

    web service上传参数代码实例 这次做的项目用到webservice比较多,最开始在网上看的参考dome,发现都不行,后来发现安卓4.0以后有很大的不同,在做传参时,有些东西需要注意: 第一, ...

随机推荐

  1. 利用python模拟菜刀反弹shell绕过限制

    有的时候我们在获取到目标电脑时候如果对方电脑又python 编译环境时可以利用python 反弹shell 主要用到python os库和sokect库 这里的服务端在目标机上运行 from sock ...

  2. pytest4-单文件使用fixture(Fixtures as Function arguments)

    Fixtures as Function arguments (fixture作为函数参数传入)Test functions can receive fixture objects by naming ...

  3. .NET中国开发者峰会11.9 下午分会场1 内容解析

    China .NET Conf 2019中国 .NET 开发者峰会即将在上海召开,这次大会是一届完全由社区组织举办的中国.NET 开发者盛会,我们筹备大会之初就定下了大会的主题是“开源.共享.创新”. ...

  4. Redis 文章一 之持久化机制的介绍

    我们已经知道对于一个企业级的redis架构来说,持久化是不可减少的 企业级redis集群架构:海量数据.高并发.高可用 持久化主要是做灾难恢复,数据恢复,也可以归类到高可用的一个环节里面去,比如你re ...

  5. typescript 入门教程一

    ##### 从今天开始,持续更新typescript入门教程系列.... 目前ts越来越火,主流的前端框架,好比*angular,vue 3*均是采用ts来编写,所有很多公司的项目都是用**ts**来 ...

  6. java面试官:兄弟简单谈谈Static、final、Static final各种用法吧

    前言 对Static.final.Static final这几个关键词熟悉又陌生?想说却又不知怎么准确说出口?好的,本篇博客文章将简短概要出他们之间的各自的使用,希望各位要是被你的面试官问到了,也能从 ...

  7. 四、pymysql模块、索引和慢查询

    目录 一.pymysql模块 (一)如何使用 (二)sql注入问题 二.索引 (一)主键索引 (二)唯一索引 (三)普通索引 (四)联合索引 (五)不会命中索引的情况 (六)explain (七)索引 ...

  8. Django之多对多表之through第三张表之InlineModelAdmin后台内嵌

    话不多说,来看表结构 这里有两个表,一个是阶段表,一个是老师表,一个老师可以带多个阶段,一个阶段也可以由多个老师带,所以是多对多关系 # 阶段表 class Stage(models.Model): ...

  9. 设计模式(五)Singleton模式

    Singleton模式就是确保只生成一个实例的模式.这里有两个意思,即想确保任何情况下都绝对只有一个实例和想在程序上表现出“只存在一个实例”. 下面通过一个实例来说明这种设计模式. package B ...

  10. 解开HTTPS的神秘面纱

    在说HTTP前,一定要先介绍一下HTTP,这家伙应该不用过多说明了,大家每天都在用,每一次HTTP请求,都是一次TCP连接.遗憾的是,请求的内容在TCP报文中是明文传输的,任何人截取到请求都可以读取其 ...