汇编子程序模块化(near&far)
1: Near 近端使用
C语言实现:
#include <stdio.h>
#include <stdlib.h>
void print(){
printf("proc");
}
int main(int argc, char *argv[]) {
print();
return ;
}
汇编实现:
datas segment
x db 'proc$';
datas ends
stacks segment stack
dw dup(?)
stacks ends
codes segment
assume cs:codes,ss:stacks,ds:datas
main proc ; 主程序
start:
mov ax,datas;
mov ds,ax;
call max
mov ah,4ch;
int 21h
main endp
print proc near ;子程序
push bp;
mov bp,sp;
mov dx,offset x;
mov ah,;
int 21h
pop bp
ret
print endp
codes ends
end start

near 调用没有段地址:使用的就是偏移地址 call 000C 地址就是子程序的入口地址
2: Far 远端使用
print 函数我们实现在另一个文件, C语言默认函数的extern的 也就是全局的
// A文件
#include<stdio.h>
void print(){
printf("proc");
}
// 主调用文件
#include <stdio.h>
#include <stdlib.h>
extern void print();
int main(int argc, char *argv[]) {
print();
return ;
}
汇编实现:
;A文件
public printx '声明为远端函数
datas segment
x db 'proc$';
datas ends
assume cs:codes,ds:datas
codes segment printx proc far
push dx
push ax
push bp;
mov bp,sp;
mov ax,datas;这里我们直接在本段进行处理 ,如果不在本段处理 需要将main的段内存的偏低地址,push 到堆栈中操作内容
mov ds,ax
mov dx,offset x;取得偏移地址
mov ah,;
int 21h;
pop dx
pop ax
pop bp
retf
printx endp
codes ends
end
;Main 文件
stacks segment stack
dw dup()
stacks ends; extrn printx:far ;标识远端程序 不写 将会汇编错误 , codes segment
assume cs:codes,ss:stacks
main proc
start:
call far ptr printx
mov ah,4ch
int 21h;
hlt
main endp
codes ends
end start
注意: 以上两个文件编译没有问题,但是链接 如果按照我们过去的思路 将会出现下面的错误: 表示我们调用的函数需要声明 否则无法链接

注意: 调用了多少个far子程序 那么链接时候需要
# 使用方式1
link main.obj+pro1.obj+pro2.obj+''' 使用+链接依次类推 # 使用方式2
link main.obj+pro1.obj pro2.obj+''' 使用空格链接依次类推
debug:
主程序为: 调用地址 0779:0000

看一下调用地址的子程序地址:

3:Far使用过程的问题
如果声明为跨段调用,因为段限制为64k, 需要使用call far ptr进行调用
下面的声明形式是错误的: (并不是错误的,学校上机的时候在32位系统 能够正确使用,但是同样的程序在msbox就不行)
stacks segment stack
dw dup()
stacks ends; extrn printx:far
codes segment
assume cs:codes,ss:stacks ;一般不这样写 主程写成模块也好
start:
call far ptr printx
mov ah,4ch
int 21h;
hlt
codes ends
end start
正确的声明形式:
stacks segment stack
dw dup()
stacks ends; extrn printx:far codes segment
assume cs:codes,ss:stacks
main proc ; 主程序也必须声明为 过程 这里 near调用还是有很大的区别的
start:
call far ptr printx
mov ah,4ch
int 21h;
hlt
main endp
codes ends
end start
4: 参数传递的问题:
4.1 寄存器传递参数
datas segment
x db 'proc$';
datas ends
stacks segment stack
dw dup(?)
stacks ends
codes segment
assume cs:codes,ss:stacks,ds:datas
main proc ; 主程序
start:
mov ax,datas;
mov ds,ax;
mov dx,offset x; 直接使用 传递给寄存器dx
call far ptr print
mov ah,4ch;
int 21h
main endp print proc
mov ah,; 使用寄存器dx
int 21h
retf
print endp codes ends
end start
4.2 内存传递参数
datas segment
x db 'proc$';
datas ends
stacks segment stack
dw dup(?)
stacks ends
codes segment
assume cs:codes,ss:stacks,ds:datas
main proc ; 主程序
start:
mov ax,datas;
mov ds,ax;
call far ptr print
mov ah,4ch;
int 21h
main endp print proc
mov dx,offset x; 取得数据段首地址
mov ah,; 输出dx的数据值
int 21h
retf
print endp codes ends
end start
4.3 内存传递参数的改进(栈恢复)
如果用到ax bx cx dx si di 需要进行恢复
为什么需要恢复:
如果在主程序中用到了一个xx寄存器,然后在子程序中也用到了这个xx寄存器,那么当子程序返回到主程序的时,主程序中存放参数的内存地址已经没有记录了,程序出错
datas segment
x db 'proc$';
datas ends
stacks segment stack
dw dup(?)
stacks ends
codes segment
assume cs:codes,ss:stacks,ds:datas
main proc ; 主程序
start:
mov ax,stacks;初始化栈
mov ss,ax;
mov ax,datas;初始数据段
mov ds,ax;
xor ax,ax
mov ax,offset x;
push ax;ax偏移地址入栈
call print
mov ah,4ch;
int 21h
main endp print proc
push bp;
mov bp,sp
mov dx,[bp+];寻地址
mov ah,;
int 21h
pop bp
ret ;保持恢复
print endp codes ends
end start
提高资料:https://wenku.baidu.com/view/3109f194690203d8ce2f0066f5335a8103d2665a.html


汇编子程序模块化(near&far)的更多相关文章
- 学 Win32 汇编[33] - 探讨 Win32 汇编的模块化编程
我觉得所谓的模块化有两种: "假模块化" 和 "真模块化". 所谓 "假模块化" 就是通过 include 指令把 *.inc 或 *.as ...
- 关于C转汇编(转自网上)
②在KILE软件的菜单中,选择Project-->Options for Target 'Target 1',-->Listing选择Assembly code就能生产*.LST文件.在 ...
- Cortex-M3中C与汇编的交互
以下内容摘自<ARM Cortex-M3权威指南> 概览 在CM3 上编程,既可以使用C 也可以使用汇编.可能还有其它语言的编译器,但是大多数人还是 ...
- 在C中嵌入汇编
早前公布了C和汇编混编的温度控制器程序,收到一些朋友的询问,他们无法在自己程序中使用我的18B20的汇编子程序或无法正常通过混编后的程序编译. 其实在KEIL中嵌入汇编的方法很简单.如图一,在C文件中 ...
- ARM汇编编程基础之一 —— 寄存器
ARM的汇编编程,本质上就是针对CPU寄存器的编程,所以我们首先要弄清楚ARM有哪些寄存器?这些寄存器都是如何使用的? ARM寄存器分为2类,普通寄存器和状态寄存器 寄存器类别 寄存器在汇编中的名称 ...
- 常用 ARM 指令集及汇编
ARM7TDMI(-S)指令集及汇编 ARM 处理器是基于精简指令集计算机(RISC)原理设计的,指令集和相关译码机制 较为简单,ARM7TDMI(-S)具有 32 位 ARM 指令集和 16 位 T ...
- Win64 驱动内核编程-24.64位驱动里内嵌汇编
64位驱动里内嵌汇编 讲道理64位驱动是不能直接内链汇编的,遇到这种问题,可以考虑直接把机器码拷贝到内存里,然后直接执行. 获得机器码的方式,可以写好代码之后,直接通过vs看反汇编,然后根据地址在看内 ...
- 嵌入式系统Linux内核开发工程师必须掌握的三十道题(转)
嵌入式系统Linux内核开发工程师必须掌握的三十道题 如果你能正确回答以下问题并理解相关知识点原理,那么你就可以算得上是基本合格的Linux内核开发工程师,试试看! 1) Linux中主要有哪几种内核 ...
- QBASIC教程
Qbasic 程序设计入门 BASIC(Beginner’s All-purpose Symbolic Instruction Code 的缩写,意为初学者通用符号指令代码)语言是在1964年由美国的 ...
随机推荐
- Servlet还有学习的必要吗?(手工搭建Servlet)
前言 在初学Java web的时候,就曾听到过这样一种说法: java Web的演变过程大概可以分为4个阶段: jsp + Servlet + jdbc spring + struts2+ hiber ...
- Kubernetes基本概念和术语之《Master和Node》
Kubernetes中的大部分概念如Node.Pod.Replication Controller.Service等都可以看作一种“资源对象”,几乎所有的资源对象都可以通过Kubernetes提供的k ...
- LeetCode 1290. 二进制链表转整数
地址 https://www.acwing.com/solution/LeetCode/content/7132/ 题目描述给你一个单链表的引用结点 head.链表中每个结点的值不是 0 就是 1.已 ...
- mysql取消严格模式
配置文件my.ini sql-mode="STRICT_TRANS_TABLES,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION" 修改为 s ...
- c++之指针
一.指针的基本概念 指针的作用:可以通过指针间接访问内存. 内存编号是从0开始记录的,一般用十六进制数字表示. 可以利用指针变量保存地址. 二.指针变量的定义和使用 指针变量定义语法:数据类型 *变量 ...
- Python基础-day01-3
PyCharm 的初始设置(知道) 目标 恢复 PyCharm 的初始设置 第一次启动 PyCharm 新建一个 Python 项目 设置 PyCharm 的字体显示 PyCharm 的升级以及其他 ...
- javascript对url进行编码和解码
这里总结下JavaScript对URL进行编码和解码的三个方法. 为什么要对URL进行编码和解码 只有[0-9[a-Z] $ - _ . + ! * ' ( ) ,]以及某些保留字,才能不经过编码直接 ...
- 虚拟化和Docker
1.硬件层的虚拟化具有高性能和隔离性,因为hypervisor直接在硬件上运行,有利于控制VM的OS访问硬件资源,使用这种解决方案的产品有VMware ESXI和Xen server. 2.hyper ...
- centos7 nginx 配置
1.下载nginx 官方下载1.6.2 2.编译安装 [root@bogon nginx-1.6.2]# ./configure --prefix=/usr/local/webserver/nginx ...
- 让终端更好看--Ubuntu OhMyZsh配置指南
查看shell列表 cat /etc/shells 如果发现没有zsh就安装 安装zsh sudo apt install zsh 设置默认shell chsh -s $(which zsh) 重启主 ...