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

逆向入门(7)汇编篇-mul指令的学习

在做一个题的时候发现了mul指令,自己写的kengenOD里面断下来的状态不同,最后定位出了是mul指令的原因。简单记录下,提高印象。

0x01 mul基础概念

32位模式下整数乘法可以实现32168位的操作,64位下还可以使用64位操作数。
MUL执行无符号乘法,IMUL执行有符号乘法
MUL:无符号数乘法
32 位模式下,MUL(无符号数乘法)指令有三种类型:

  • 执行 8 位操作数与 AL 寄存器的乘法;
  • 执行 16 位操作数与 AX 寄存器的乘法;
  • 执行 32 位操作数与 EAX 寄存器的乘法

MUL 指令中的单操作数是乘数。下表按照乘数的大小,列出了默认的被乘数和乘积。由于目的操作数是被乘数和乘数大小的两倍,因此不会发生溢岀,换句话说,两个8位二进制数的乘积不会超过16
在这里插入图片描述

0x02 实际体验

#include <iostream>
#include <string>
#include <cstdint>int main()
{std::string username;// 获取用户名输入std::cout << "用户名: ";std::getline(std::cin, username);int userLen = username.length();if (userLen < 4) {printf("用户名长度过短");return 0;}int result = 1;for (int i = 0; i < userLen;i++) {result *= username[i];printf("0x%x\n", result);}return 1;
}

上面的代码是根据汇编自己写的算法,但是最后结果和预期结果不同,预期比下列结果要大2
在这里插入图片描述
最终在汇编中发现多了一段加上edx的操作,之前因为edx一直为0,就忽略了一段。然后下面通过ai修改过的代码,主要是使用了无符号整数

#include <iostream>
#include <string>
#include <cstdint>  // 包含 uint32_t 和 uint64_tint main() {std::string username;std::cout << "用户名: ";std::getline(std::cin, username);if (username.length() < 4) {std::cout << "用户名长度过短";return 0;}uint32_t eax = 1;  // 模拟EAX寄存器uint32_t edx = 0;  // 模拟EDX寄存器(高位)for (char c : username) {uint8_t ecx = static_cast<uint8_t>(c);  // 当前字符的ASCII值if (ecx == 0) break;  // 遇到空字符终止(如汇编中的OR ECX,ECX/JE)// 模拟 MUL ECX 指令(32位乘法)uint64_t product = static_cast<uint64_t>(eax) * ecx;// 分离高低32位(EDX:EAX)edx = static_cast<uint32_t>(product >> 32);  // 高32位eax = static_cast<uint32_t>(product);        // 低32位// 模拟 ADD EAX, EDXeax += edx;// 输出当前状态(与调试器对比)printf("字符 '%c' (0x%02X): EAX=0x%08X, EDX=0x%08X\n", c, ecx, eax, edx);}std::cout << "最终结果: EAX = 0x" << std::hex << eax << ", EDX = 0x" << edx << std::endl;return 0;
}

在这里插入图片描述
这一次就是汇编正确的效果了,结果相差为2

0x03 分析原因

根本原因:在计算到字符n 时,累乘值eax已经非常大110,996,500,乘以 nASCII110后,结果12,209,615,000超过了 32 位无符号整数的最大值(2³² - 1 = 4,294,967,295)。这导致 mul 指令产生溢出,结果的高 32 位存储在 edx 中。

数值计算:12,209,615,000 ÷ 4,294,967,296 ≈ 2.842,因此商(高位部分)为 2,余数(低位部分)为 3,619,680,408。这就是 edx = 2 的直接原因。

循环行为:在之前的迭代中,乘积结果均小于 2³²,因此 edx 保持为 0。只有在最后一次乘法时,数值足够大,才触发 edx 非零。

http://www.lryc.cn/news/574809.html

相关文章:

  • DAY 41 简单CNN
  • 防御OSS Bucket泄露:RAM权限策略+日志审计+敏感数据扫描三重防护
  • DeepSeek智能总结 | 邓紫棋音乐版权纠纷核心梳理
  • 软件工程:从理论到实践,构建可靠软件的艺术与科学
  • 智慧家政数字化小程序开发:重构行业服务生态的创新引擎
  • 代码随想录|图论|01图论基础
  • 医药企业CMO研发管线管理专项介绍
  • 50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | MovieApp(电影卡片组件)
  • ArkTS与仓颉开发语言:鸿蒙编程的双子星
  • day41
  • 深入理解 BOM:浏览器对象模型详解
  • IoTDB的基本概念及常用命令
  • 【css】增强 CSS 的复用性与灵活性的Mixins
  • ArkUI-X通过Stage模型开发Android端应用指南(二)
  • 【软考高级系统架构论文】### 论软件系统架构评估
  • linux grep的一些坑
  • 接口自动化测试之 pytest 接口关联框架封装
  • Unity_导航操作(鼠标控制人物移动)_运动动画
  • matplotilb实现对MACD的实战
  • SQL关键字三分钟入门:UPDATE —— 修改数据
  • Camera Sensor接口协议全解析(五)SLVS-EC接口深度解析
  • Stable Diffusion 项目实战落地:打造完美海报的秘密武器 第二篇:边缘柔化、蒙版处理与图生图技术大揭秘!
  • 如何通过nvm切换本地node环境详情教程(已装过node.js更改成nvm)
  • 2025.6.24总结
  • useState为异步,测试一下编码时候是否考虑?
  • Unity反射机制
  • mongoose解析http字段值
  • Spring Boot 的Banner的介绍和设置
  • 中科米堆3D扫描逆向建模方案:汽车轮毂三维扫描抄数建模
  • elk+filebeat收集springboot项目日志