x86汇编利用int 16h中断实现伪多线程输入

x86汇编利用int 16h中断实现伪多线程输入

我们都知道,如果想让一个程序,同时又干这个,又干那个,最好的办法就是多线程。这个在高级语言里面已经用烂了。

但是,DOS是只有单线程的。我如果想让程序在运行的同时,又能接受我键盘的输入,那要怎么办呢?

例如,我有一个DOS彩票开奖软件。屏幕上有十个数字在滚动,当我按下空格键的时候,它会停止跳动,按别的键无效。那么要怎么操作?

首先,接收输入最常用的是int 21h中断里的7号功能调用。但是它是一个中断输入,也就是为了接收这个输入,我的程序就暂停在这儿不动了。但是我彩票开奖数字一直在滚动,那要怎么办啊?不能说我按一下它滚动一下吧。

于是请来了今天的主角,int16h中断。

首先我们看看int 16h中断有哪些功能?

BIOS提供了int 16h中断例程供程序员调用,且功能是从软件层面上实现键盘I/O功能中断调用,功能号为0,1,2,并且必须把功能号放在ah寄存器中。

0号功能调用

mov ah,01h
int 16h

功能:从键盘读入字符送进al寄存器,执行时,等待键盘输入,一旦输入,输入字符的ASCII码放入al中。若al为0,则ah为输入的扩展码。但是,如果键盘缓冲区已有数据,则会读取键盘缓冲区数据,并且将其从缓冲区中删除。

0号功能的工作流程为:

  1. 检测键盘缓冲区是否有数据;
  2. 没有则回到1;
  3. 读取缓冲区第一个字单元中的键盘输入;
  4. 将读取的扫描码送入ah,ASCII码送入al;
  5. 将已读取的键盘输入从缓冲区中删除

1号功能调用

mov ah,01h
int 16h

功能:用于查询键盘缓冲区,对键盘扫描但是不等待(也就是不会中断程序),并设置标志寄存器中的ZF。如果有键盘输入(即键盘缓冲区不空),则ZF=0,AL存放的是当前输入的ASCII码,AH存放的是输入字符的扩展码。若无键盘操作,则标志位ZF=1。

2号功能调用

mov ah,02h
int 16h

功能:检查键盘上各特殊功能键的状态。执行后,各种特殊功能键的状态放入AL寄存器中,这个状态字记录在内存0040H:0017H单元中,若对应位为“1”,表示该键状态为“ON”,处于按下状态;若对应位为“0”,表示该键状态为“OFF”,处于断开状态。

mov ah,02h
int 16h					;从键盘输入
and al,04h				;判断是否按下Ctrl
jnz Ctrl_on

Ctrl_on:
	......

了解了以上功能,我们来看如何实现伪多线程输入。

mainLoop:
	mov ah,01h		;检测是否有输入
	int 16h
	jz offKey		;jz是当zf=1时跳转,也就是无输入
	;如果有输入,没有跳转
	mov ah,00h		;那么键盘缓冲区已不为空
	int 16h
	cmp al,32		;和空格的ASCII码比较
	je spaceProc
	...
spaceProc:
	...	
offKey:
	...

这个框架简单易懂,先从mainLoop中判断是否有输入:

  • 如果没有任何输入,则直接执行offKey中的代码内容;
  • 如果有输入,会调用0号功能对键盘缓冲区进行检索。因为我们已知它有输入(键盘缓冲区不为空),所以不会中断程序。
    • 如果键盘输入为空格,就跳转到spaceProc,否则再加别的代码。
    • 当然,我们可以利用这个做很多事情,不只是判断空格。

这样,利用int 16h中断中0号功能和1号功能的联动,我们实现了在只支持单线程的DOS中,实现了伪多线程的输入判断。

posted @ 2020-04-12 16:29  scyq  阅读(1065)  评论(0编辑  收藏  举报