续:将基于Nasm汇编的打字小游戏,移植到DOSBox

news/2024/5/18 16:36:53 标签: 汇编, 游戏程序, Nasm, DOSBox

NasmDOSBox_0">续:将基于Nasm汇编的打字小游戏,移植到DOSBox

文章目录

  • 续:将基于Nasm汇编的打字小游戏,移植到DOSBox
  • 前情提要
  • 细说
    • 1 编译
    • 2 程序入口
    • 3 定位段
  • 运行体验

前情提要

上一篇:【编程实践】黑框框里的打字小游戏,但是汇编语言-CSDN博客

在上一篇文章中我对这个小程序进行了介绍,但由于运行环境的安装比较复杂,估计没有谁会将我的代码跑起来,可那样实在是太遗憾了。学习过汇编语言的你大概率是使用过DOSBox的,为此我献祭了2小时的宝贵生命,成功地将代码在DOS上跑起来了。

如果对具体移植过程不感兴趣,可以直接跳到运行体验部分。

细说

上文的虚拟机环境和DOSBox都支持8086汇编,就换个地方运行能遇到什么麻烦?麻烦就是多了DOS操作系统的存在,现在我不需要自己编程从磁盘加载用户程序的引导程序了,这个事情换由DOS操作系统来做,可是我不太明白DOS需要我提供什么。

1 编译

首先我尝试着将曾经编译的typing.bin直接在DOSBox中运行,期待它能识别出我代码里的头部信息,正常地加载我的代码并将cpu的控制权交出来。然而它并不认.bin后缀的程序:Illegal command: typing.bin。于是我搜索到了一篇博客,并欣喜地发现它的代码是可以正常运行的。

参考:DOSBox下编译和运行NASM代码_nasm dosbox-CSDN博客

原来是需要编译成.com文件

> nasm -f bin typing.asm -o typing.com # 在本机命令行窗口中
> typing.com						 # 在DOSBox

2 程序入口

DOS系统如何知道我的程序从哪里开始执行呢?毕竟我源代码中起始是一个头部段而不是代码段。答案是它什么也不知道,只会从源代码的第一行开始执行,所以我在源码的开头添加了一条跳转指令:

org 0x0100                      ;DOS系统会创建一个PSP数据区
jmp section.code.start

org 0x0100是什么?可以参考下面另一个博主的文章。它是在编译阶段发挥作用的伪指令,简单点说就是让标号能够获取到正确的偏移地址。引用标号时本来是获取标号所在处的汇编地址(与程序开头的相对地址),程序第一条指令的汇编地址是0,但被DOS加载后程序第一条指令的偏移地址就变成了0x0100,因为前面256个字节被PSP占据。而org 0x0100会让你在引用标号时自动加上0x0100

补充:如果在段中指定了vstart=0,那么段内的标号被引用时仍然是段内偏移地址,而不会加上0x0100

此时逻辑地址cs:0000指向的是PSP,cs:0100指向的才是程序第一条指令。不过org 0x0100是为后面定位代码段和数据段服务的,写与不写都对依据位移的jmp指令没有影响。

参考:NASM的ORG 0100h的实际含义_org 0100h指令是什么意思-CSDN博客

下面这张图可能更加直观一点:

在这里插入图片描述

3 定位段

3.1 定位数据段

这件事情在Nasm中要比Masm麻烦,因为Masm可以直接获取一个段的段地址,而Nasm中的section.段名.start只能获取到段的汇编地址。我想让数据段寄存器ds指向数据段,还需要将获取的汇编地址除以4,才得到段地址。这里的除操作可以使用右移位指令shr替代。

;1 将ds指向数据段;
mov ax, section.data.start
shr ax, 4

mov bx, cs 
add ax, bx 
mov ds, ax 

3.2 定位代码段

因为在程序开头使用了jmp指令,即便不将cs定位到代码段,大部分代码也可以正常执行,除了键盘中断例程。中断例程的代码逻辑当然没有什么问题,毕竟之前都能正常运行,只是中断向量表的填写出问题了,问题代码片段如下:

cli
mov word [9*4], int9_new
mov word [9*4+2], cs
sti  

回想一下,标号int9_new是个代码段的段内偏移地址,可cs指向的都不是代码段,这能从中断向量表找到中断例程才怪。定位代码段的操作与前面定位数据段类似,但必须同时更改cs和ip寄存器,因为cs:ip永远指向运行的下一条指令的地址。

;2 将cs指向代码段, 并将ip指向程序的正式开始:start标号
mov ax, section.code.start 
shr ax, 4

mov bx, cs 
add ax, bx 
mov free[2], ax
mov word free[0], start 
jmp far [free]

运行体验

Gitee获取代码:清风莫追/Typing_asm (gitee.com)

我在代码仓库的dos文件夹中,已经提供了编译好的程序typing.com,如果你已经安装了DOSBox的话,那么可以直接运行它了!

在这里插入图片描述



http://www.niftyadmin.cn/n/5175582.html

相关文章

centos7下载python3离线安装包

下载离线安装工具 yum install yum-utils -y下载python3离线安装包 yumdownloader --destdir/root/python3 --resolve python3将python3下面的离线安装文件拷贝到其他服务器上就可以安装 离线安装 先进入到离线程序rpm文件存储路径 # 离线安装 rpm -Uvh --force --nodep…

GoLong的学习之路,进阶,语法之并发(并发错误处理)补充并发三部曲

这篇文章主要讲的是如何去处理并发的错误。 在Go语言中十分便捷地开启goroutine去并发地执行任务,但是如何有效的处理并发过程中的错误则是一个很棘手的问题。 文章目录 recovererrgroup recover 哦对,似乎没写错误处理的文章。后面补上。 首先&…

C与汇编深入分析

汇编怎么调用C函数 直接调用 BL main传参数 在arm中有个ATPCS规则(ARM-THUMB procedure call standard)(ARM-Thumb过程调用标准)。 约定r0-r15寄存器的用途: r0-r3:调用者和被调用者之间传递参数r4-r11…

我的创作纪念日-我在csdn的三周年

文章目录 机缘收获日常成就憧憬 机缘 2020 年 11 月 09 日,撰写了第 1 篇技术博客,到现在不知不觉三周年了。 慢慢的也会将自己的感受和知识梳理成专栏,记录日常的学习以及通过文章进行技术交流,和大家分享一个我认为比较好的成…

Spring IOC - BeanDefinition解析

1. BeanDefinition的属性 BeanDefinition作为接口定义了属性的get、set方法。这些属性基本定义在其直接实现类AbstractBeanDefinition中,各属性的含义如下表所示: 类型 名称 含义 常量 SCOPE_DEFAULT 默认作用域:单例模式 AUT…

python实现图片和base64转换

目录 引用base64转图片图片转base64调用源码下载 工作中很多时候使用图片和base64相互转换的地方,下面介绍转换代码 引用 import base64 from PIL import Image from io import BytesIObase64转图片 方法, 入参base64串,及图片路径 转换后把图片存储在…

论文笔记--Baichuan 2: Open Large-scale Language Models

论文笔记--Baichuan 2: Open Large-scale Language Models 1. 文章简介2. 文章概括3 文章重点技术3.1 预训练3.1.1 预训练数据3.1.2 模型架构 3.2 对齐3.2.1 SFT3.2.2 Reward Model(RM)3.2.3 PPO 3.3 安全性 4. 文章亮点5. 原文传送门 1. 文章简介 标题:Baichuan 2…

如何成功创建百度百科词条?教你从零开始创建自己的百度百科【建议收藏】

百度百科是一个开放的网络百科全书,用户可以自由编辑和贡献内容。如果你想创建一个百度百科页面,需要做好以下准备: 1.确定主题:选择一个你熟悉或者感兴趣的主题,确保该主题在百度百科上还没有相关的页面。 2.收集资…