_declspec(naked) 使用

最近学习驱动开发,在写绕过inline hook的代码时,有个问题困扰了我一天,最后发现原来是在内嵌汇编时,没有使用_declspec(naked)导致的,看来是偶的基础知识掌握的不牢固啊(得补一下了,磨刀不误砍柴功),在此给记录一下,给自己一个警示。

对于jmp类型的hook, 如果自己的过程没有使用_declspec(naked),那么系统会自动给添加一些额外的代码,控制堆栈平衡,但是这些额外的代码会破坏被hook函数的堆栈。

对于call类型的hook,如果 使用_declspec(naked)修饰的话,要注意自己恢复堆栈平衡

下面是网上对_declspec(naked) 的介绍:
_declspec(naked)
就是告诉编译器,在编译的时候,不要优化代码,通俗的说就是
没代码,完全要自己写
比如
#define NAKED __declspec(naked)

void NAKED code(void)
{
__asm
{
ret
}
}
使用__declspec(naked)关键字定义函数:
1,使用 naked 关键字必须自己构建 EBP 指针 (如果用到了的话);
2,必须自己使用 RET 或 RET n 指令返回 (除非你不返回);

_delcspec(naked)用在驱动编写,C语言内嵌汇编完成一些特定功能。

__declspec(naked)是用来告诉编译器函数代码的汇编语言为自己的所写,不需要编译器添加任何汇编代码。详细的作用和意义,可查找相关资料:

这里主要讲一下可能会发生错误的一个地方。

void __declspec(naked) funname()

{

_asm

{

  ...

}

}

 

注意,这是编译器直接拿来用的汇编函数代码,所以一定要记得在开始的时候保存上下文标志位(压栈),在结束的时候要记得恢复上下文(出栈)。并且在结尾要加上ret命令。

 

对于一般的汇编内嵌代码,不必保存上下文了,保存也不会有事,但是不能再加ret命令,因为编译器也会为其加一个,ret命令不能同时执行两次。会导致越界错误




-------------------------------------------------------------------------------------------------------------------------------
对于用 __declspec(naked) 编写干净函数 
  1. VOID __declspec(naked) MyNakedFunction()
  2. {
  3. strcmp(...);
  4. // __cdecl 函数是调用者清除参数堆栈,对于非内联汇编调用这类函数,编译器将自动平衡堆栈,加入 ADD ESP, 8
  5. }
复制代码
  1. VOID __declspec(naked) MyNakedFunction()
  2. {
  3. //...
  4. __asm CALL strcmp;
  5. __asm ADD ESP, 8; // 内联汇编需要自己平衡堆栈
  6. }
复制代码

__declspec(naked) 是且仅是不产生 prolog 和 epilog 代码 {保存并恢复使用过的寄存器和分配局部变量、平衡堆栈、返回值}。

附上刚翻译的一些 MSDN 可能对 naked 函数有点用的资料(我英语很烂,希望没翻译错误)

Microsoft Specific 
All arguments are widened to 32 bits when they are passed. Return values are also widened to 32 bits and returned in the EAX register, except for 8-byte structures, which are returned in the EDX:EAX register pair. Larger structures are returned in the EAX register as pointers to hidden return structures. Parameters are pushed onto the stack from right to left. Structures that are not PODs will not be returned in registers.
The compiler generates prolog and epilog code to save and restore the ESI, EDI, EBX, and EBP registers, if they are used in the function.
Note  
When a struct, union, or class is returned from a function by value, all definitions of the type need to be the same, else the program may fail at runtime.

For information on how to define your own function prolog and epilog code, see Naked Function Calls.
The following calling conventions are supported by the Visual C/C++ compiler.
Keyword  Stack cleanup  Parameter passing  
__cdecl     Caller               Pushes parameters on the stack, in reverse order (right to left)
__clrcall    n/a                   Load parameters onto CLR expression stack in order (left to right).
__stdcall  Callee               Pushes parameters on the stack, in reverse order (right to left)
__fastcall Callee               Stored in registers, then pushed on stack
__thiscall Callee               Pushed on stack; this pointer stored in ECX

微软细节
所有参数在传递时都被扩展为 32 位。除了 8 字节结构体返回在 EDX:EAX 寄存器一对中,其它返回值同样被扩展为 32 位并返回在 EAX 寄存器中。大的结构体返回在 EAX 寄存器中的是指针来间接返回结构体。参数是自右向左压入堆栈。结构体是非 POD(Plain Old Data)类型将不会返回在寄存器中。
如果 ESI、EDI、EBX 和 EBP 寄存器在函数中使用,编译器将产生 prolog 和 epilog 代码保存并恢复它们。
注释
当一个函数用值返回结构体(struct)、联合体(union)或者类(class)时,所有类型的定义必须相同,否则程序可能在运行过程中失败。
关于如何定义你自身函数 prolog 和 epilog 代码的更多信息,请查阅 Naked Function Calls。
Visual C/C++ 编译器支持以下调用转换:
关键字   堆栈清除 参数传递
__cdecl  调用者  将参数倒序压入堆栈(自右向左)
__clrcall 不适用  将参数顺序加载到 CLR expression 堆栈中(自左向右)
__stdcall 被调用者 将参数倒序压入堆栈(自右向左)
__fastcall 被调用者 保存在寄存器然后压入堆栈
__thiscall 被调用者 压入堆栈,指针保存在 ECX 寄存器中

※ __fastcall:两个以内不大于 4 字节(DWORD)的参数将自左向右分别保存在 ECX、EDX 寄存器内,由于没有将参数压入堆栈,函数返回前无需将参数出栈(这就是它FASTCALL由来);其它参数仍按自右向左压入堆栈,函数返回时需要把这些参数出栈。


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: __declspec(dllimport) 是一个 Microsoft Visual C++ 扩展关键字,用于在动态链接库中导入函数或数据。它可以用于在编译时指定动态链接库中的函数或数据,以便在运行时使用。 ### 回答2: __declspec(dllimport)是用于在C++中声明导入动态链接库中函数和变量的关键字。在Windows平台上,动态链接库(Dynamic Link Library,简称DLL)是一种将可执行代码和数据封装在一起的文件格式,允许多个程序共享这些代码和数据。使用__declspec(dllimport)关键字可以使程序在编译阶段就知道这些函数和变量是从动态链接库中导入的。 在C++程序中,如果我们需要使用动态链接库中的函数或变量,需要先通过__declspec(dllimport)对这些函数或变量进行声明,然后再进行调用或使用。声明方式如下: __declspec(dllimport) returnType functionName( parameter list ); 其中,returnType表示函数返回值的类型,functionName表示函数名,parameter list表示函数的参数列表。 对于变量的声明,格式如下: __declspec(dllimport) dataType variableName; dataType表示变量的类型,variableName表示变量名。 通过这种方式,程序在编译时就会知道这些函数和变量是从外部导入的,可以正确引用和使用。 总结来说,__declspec(dllimport)关键字主要用于在C++中声明导入动态链接库中函数和变量。它在程序编译阶段就告诉编译器这些函数和变量是从其他地方导入的,使得程序可以正确引用和使用这些导入的函数和变量。 ### 回答3: __declspec(dllimport) 是一个用于 C/C++ 编程语言的 Microsoft 扩展,用于指定在编译时导入 DLL 中定义的函数或变量。该关键字的作用是告诉编译器和链接器,在当前的源文件中,所使用的某个函数或变量是在外部的 DLL 文件中定义的,需要从 DLL 文件中导入。 在使用 __declspec(dllimport) 关键字时,需要先使用 __declspec(dllexport) 关键字在 DLL 文件中导出相应的函数或变量。然后在其他的源文件中,通过 __declspec(dllimport) 关键字来声明相应的函数或变量,告诉编译器和链接器需要从 DLL 文件中导入该函数或变量的定义。 __declspec(dllimport) 关键字可以在函数或变量的声明前使用,以指定导入的函数或变量的属性。它告诉编译器该函数或变量是在其他 DLL 文件中定义的,并且编译器在编译时会将其处理为外部引用。这样,程序在运行时就可以动态地加载和使用 DLL 文件中的函数或变量。 总结来说,__declspec(dllimport) 是一个 Microsoft 扩展,用于在 C/C++ 编程语言中指定从 DLL 文件中导入函数或变量的关键字。通过使用这个关键字,可以告诉编译器和链接器在编译时将引用的函数或变量作为外部引用,并在运行时动态地加载和使用 DLL 文件中的相应函数或变量。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值