当前位置: 首页 > 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.lqws.cn/news/502705.html

相关文章:

  • GitLab 备份恢复与配置迁移详尽教程(实战版)
  • 创客匠人拆解知识变现从 IP 到商业闭环的关键要素
  • 基于版本控制+WORM的OSS数据保护:防勒索攻击与法规遵从实践
  • OpenCV CUDA模块设备层-----检查 CUDA 错误并输出调试信息内联函数checkCudaError()
  • 【Linux网络编程】多路转接I/O(一)select,poll
  • HTML炫酷烟花
  • ✨从零搭建 Ubuntu22.04 + Python3.11 + PyTorch2.5.1 GPU Docker 镜像并上传 Docker Hub
  • Flask(二) 路由routes
  • 零知开源——STM32F4实现ILI9486显示屏UI界面系列教程(三):记事本功能实现
  • bmc TrueSight 监控mysql配置
  • prometheus+grafana+Linux监控
  • Linux 中的信号处理方式详解
  • 【机器学习深度学习】多层神经网络的构成
  • 在仓颉开发语言中使用数据库
  • TCP/UDP协议深度解析(一):UDP特性与TCP确认应答以及重传机制
  • 计算机网络第九章——数据链路层《介质访问控制》
  • C++(面向对象编程——多态)
  • 曼昆《经济学原理》第九版 宏观经济学 第二十六章货币增长与通货膨胀
  • python中学物理实验模拟:摩檫力
  • BI财务分析 – 反映盈利水平利润占比的指标如何分析(下)
  • iwebsec靶场sqli注入(2)
  • [Linux] Linux用户和组管理
  • GoAdmin代码生成器实践
  • 大模型项目实战:业务场景和解决方案
  • TongWeb替换tomcat
  • Linux Sonic Agent 端部署(详细版)(腾讯云)
  • MySQL:深入总结锁机制
  • 系统架构设计的全方位视角:深入解析4+1视图模型及其应用实践
  • prometheus+grafana+MySQL监控
  • 飞算 JavaAI 插件炸场!一小时搭图书管理系统