汇编:结构体

在32位汇编中,结构体(structures)用于组织和管理复杂的数据类型,结构体可以包含多个不同类型的数据项(成员);在MASM(Microsoft Macro Assembler)中,使用结构体可以让我们更有条理地处理数据。下面是一个定义和使用结构体的示例。

结构体定义格式
复制代码
STRUCT_NAME STRUCT
    MEMBER_NAME1 TYPE1 [NUMBER_OF_ELEMENTS1] [INITIAL_VALUE1]
    MEMBER_NAME2 TYPE2 [NUMBER_OF_ELEMENTS2] [INITIAL_VALUE2]
    ; 更多成员...
STRUCT_NAME ENDS

STRUCT_NAME: 结构体的名称。

MEMBER_NAME: 结构体成员的名称。

TYPE: 成员的数据类型,如 BYTE, WORD, DWORD, DB, DW, DD 等。

NUMBER_OF_ELEMENTS: 可选,成员的元素个数,用于定义数组。

INITIAL_VALUE: 可选,成员的初始值。

示例:假设我们要定义一个表示人的结构体,其中包含姓名(字符串)和年龄(整数)。
复制代码
Person struct 
    szName db 32 dup(?)
    nAge dd ?
Person ends

在这个示例中,Person结构体包含两个成员:

  • name: 一个32字节的字符数组,用于存储人的姓名。

  • age: 一个32位的整数,用于存储人的年龄。

声明和初始化结构体变量
复制代码
.data
wolven Person <"wolven Chan",24>

这行代码声明了一个名为wolvenPerson结构体变量,并初始化其szName"wolven Chan"nAge24

访问结构体成员
复制代码
.code
main proc
    mov eax,wolven.nAge
    mov ebx,offset wolven.szName
main endp

wolvennAge成员的值移动到eax寄存器;将wolvenszName成员的地址偏移值移动到ebx寄存器。

程序执行完后的EAXEBX寄存器分别为上述结构体变量初始化时的值。

示例代码:

代码展示了如何定义一个结构体,并通过自定义过程printfS输出结构体的成员。

复制代码
.586
.model flat,stdcall
option casemap:none
​
;结构体定义
Person struct 
    szName db 32 dup(?)
    nAge dd ?
Person ends
​
;数据段
.data
szFormatS db '%s',0
wolven Person <"wolven Chan",24>
​
;代码段
.code
;①自定义过程,打印字符串
printfS proc szFormat:DWORD,szValue:DWORD   
    xor eax,eax
    mov eax,szValue
    push eax
    mov eax,szFormat
    push szFormat
    call printf
    add esp,8
    ret
printfS endp
​
;②程序入口
main proc
    mov eax,wolven.nAge
    mov ebx,offset wolven.szName
    invoke printfS,offset szFormatS,ebx
main endp
end 
结构体定义:

Person struct: 定义了一个名为Person的结构体。

szName db 32 dup(?): 一个32字节的字符数组,用于存储姓名。

nAge dd ?: 一个32位的整数,用于存储年龄。

Person ends: 结束结构体定义。

数据段:

szFormatS db '%s',0: 定义了一个以零结尾的字符串,表示格式化字符串%s,用于打印字符串。

wolven Person <"wolven Chan",24>: 定义了一个名为wolvenPerson结构体变量,并初始化szName"wolven Chan"nAge24

代码段
①自定义过程

printfS proc szFormat:DWORD,szValue:DWORD: 定义一个名为printfS的过程,接受两个参数,szFormatszValue

xor eax,eax: 将eax寄存器清零。

mov eax,szValue: 将第二个参数szValue的值移动到eax寄存器。

push eax: 将eax的值压入堆栈。

mov eax,szFormat: 将第一个参数szFormat的值移动到eax寄存器。

push szFormat: 将szFormat的值压入堆栈。

call printf: 调用C运行时库的printf函数。

add esp,8: 调整堆栈指针,移除之前压入的两个参数(每个参数占4个字节)。

ret: 返回到调用printfS的地方。

为什么自定义过程要先将第二个参数压入栈中
复制代码
在汇编中,自定义过程(函数)使用的参数传递方式与调用约定有关;展示代码采用了stdcall调用约定,在这种约定中,参数是按从右到左的顺序压入堆栈的,这样函数就可以按照从左到右的顺序来访问参数。
②程序入口(main)

mov eax,wolven.nAge: 将结构体变量wolvennAge成员的值移动到eax寄存器。

mov ebx,offset wolven.szName: 将结构体变量wolvenszName成员的地址移动到ebx寄存器。

invoke printfS,offset szFormatS,ebx: 调用自定义过程printfS,传入szFormatSwolven.szName的地址。

最后结构体变量wolven的nAge成员值被放入eax中,且黑窗口中打印szName成员内容:

相关推荐
0xCC说逆向7 小时前
Windows逆向工程提升之IMAGE_IMPORT_DESCRIPTOR
c语言·汇编·windows·安全·逆向·pe结构
眸生21 小时前
自制操作系统day7(获取按键编码、FIFO缓冲区、鼠标、键盘控制器(Keyboard Controller, KBC)、PS/2协议)
c语言·汇编·计算机外设·操作系统·计算机组成原理·寄存器
眸生1 天前
自制操作系统day8 (鼠标数据取得、通往32位模式之路、A20GATE、切换到保护模式、控制寄存器cr0-cr4以及cr8、ALIGNB)
c语言·汇编·计算机外设·操作系统·i/o外设
0xCC说逆向1 天前
Windows逆向工程提升之IMAGE_SECTION_HEADER
汇编·windows·单片机·嵌入式硬件·安全·逆向工程·pe结果
0xCC说逆向2 天前
Windows逆向工程提升之IMAGE_OPTIONAL_HEADER
汇编·windows·安全·架构·逆向·pe结构·pe文件
眸生3 天前
至此(day1-day4)代码详解(ai辅助整理)
c语言·汇编·操作系统·计算机组成原理·显存·i/o外设
MaoXian_n4 天前
[IMX] 05.串口 - UART
汇编·arm开发·驱动开发·单片机·嵌入式硬件
0xCC说逆向4 天前
Windows逆向工程提升之二进制分析工具:HEX查看与对比技术
汇编·windows·单片机·嵌入式硬件·安全·pe结构·pe文件
慢一点会很快4 天前
【C++】汇编角度分析栈攻击
开发语言·汇编·c++
Epiphanywh5 天前
编译原理--期末复习
汇编