实验题目
将8255单元与键盘及数码管显示单元连接,编写实验程序,扫描键盘输入,并将扫描结果送数码管显示。键盘采用4X4键盘,每个数码管显示值可为0-F共16个数。
实验具体内容如下:将键盘进行编号,记作0~F,当按下其中一一个按键时,将该按键对应的编号在一一个数码管上显示出来,当再按下一个按键时,便将这个按键的编号在下-一个数码管上显示出来,数码管上可以显示最近4次按下的按键编号。
电路图
实验代码
;=======================================================
; 文件名: Keyscan.asm
; 功能描述: 键盘及数码管显示实验,通过8255控制。
; 8255的 B口控制数码管的段显示,
; A口控制键盘列扫描及数码管的位驱动,
; C口控制键盘的行扫描。
; 按下按键,该按键对应的位置将按顺序显示在数码管上。
;=======================================================IOY0 EQU 0600H ;片选IOY0对应的端口始地址
MY8255_A EQU IOY0+00H*2 ;8255的A口地址
MY8255_B EQU IOY0+01H*2 ;8255的B口地址
MY8255_C EQU IOY0+02H*2 ;8255的C口地址
MY8255_CON EQU IOY0+03H*2 ;8255的控制寄存器地址SSTACK SEGMENT STACKDW 16 DUP(?)
SSTACK ENDS DATA SEGMENT; DATBLE是 将需要输入按键的值对应需要给的显示器的值
; 比如按键1表示的值是1 但是我们送给显示器的是06H
; 该程序是通过判断按键按下 获取其代表的偏移量(相对于DTABLE)
; 比如按键1的偏移量是1 我们扫描按键 得出一个值 1
; 然后利用该值在DTABLE中找到需要输出值的对应显示代码值
; 从B口送出去即可DTABLE DB 3FH,06H,5BH,4FH,66H,6DH,7DH,07HDB 7FH,6FH,77H,7CH,39H,5EH,79H,71H
DATA ENDSCODE SEGMENTASSUME CS:CODE,DS:DATA
START:MOV AX,DATAMOV DS,AX; 把3000H--3005H中的值全部初始化为00H; 说明初始偏移量全为0(3000H--3005H)MOV SI,3000HMOV AL,00HMOV [SI],AL ;清显示缓冲MOV [SI+1],ALMOV [SI+2],ALMOV [SI+3],ALMOV [SI+4],ALMOV [SI+5],ALMOV DI,3005HMOV DX,MY8255_CON ;8255控制字初始化MOV AL,81H ;1000 0001 A、B口输出 C口输入OUT DX,ALBEGIN: ; 调用显示子程序 CALL DIS ;清屏 CALL CLEAR ;扫描 看是否有键按下CALL CCSCAN ;有键按下 跳INK1JNZ INK1 JMP BEGININK1: CALL DISCALL DALLYCALL DALLYCALL CLEARCALL CCSCAN; 若结果不为0 ZF=0 则说明一定有键按下 则跳转 判断哪个键按下JNZ INK2 JMP BEGIN;确定按下键的位置
INK2: MOV CH,0FEH ; FEH=1111 1110(对应关系:PA7 PA6..PA1 PA0 ) ; PA5-PA0=1111 10 (这里对应关系要弄明白) ;PA0对应的按键则是 从左到右第一列(这里不会晕哦) MOV CL,00H ; 初始对于行的偏移量 为0 ;列循环 即扫描列 从第一列开始
COLUM: MOV AL,CHMOV DX,MY8255_A OUT DX,AL MOV DX,MY8255_C IN AL,DX
L1: TEST AL,01H ;is L1?JNZ L2MOV AL,00H ;L1JMP KCODE
L2: TEST AL,02H ;is L2?JNZ L3MOV AL,04H ;L2JMP KCODE
L3: TEST AL,04H ;is L3?JNZ L4MOV AL,08H ;L3JMP KCODE
L4: TEST AL,08H ;is L4?JNZ NEXTMOV AL,0CH ;L4; 找到按键后 此时AL存的的第一列每一行的初始值 0 4 8 C
; CL 存的是对应行的偏移量
; 假设 AL为08H CL为2 则表示的总偏移量为 8+2=10H
; 说明在table中该数字的偏移量为10H
; 输出该数字 利用偏移量就行 因为数字其实是存在table中的
KCODE: ADD AL,CLCALL PUTBUFPUSH AX
KON: CALL DISCALL CLEARCALL CCSCANJNZ KONPOP AX
NEXT: INC CL ; CL相当于 行偏移量MOV AL,CHTEST AL,08H ; 08H=0000 1000 当AL为1111 0111 && 0000 1000 结果为0 ; ZF=1 说明行偏移量达到最大值 3JZ KERR ; 4次列循环结束 跳KERRROL AL,1MOV CH,ALJMP COLUM
KERR: JMP BEGIN; 键盘扫描子程序
; 原理是 先向全部列输出低电平
; 然后从C口读入 行电平
; 如果没有按键按下 所有行应该均为高电平
; 反之 若有按键按下 则开始仔细判断出到底是哪个按键按下 具体判断方法是:
; 先向第一列输出低电平(从左到右)
; 然后从C口读入行电平 利用 AND
; 判断哪一行是否为低电平即可(后面为了计算方便取反了行电平)
; 若行全为高 为开始向下一列输出低电平 循环4次即可
CCSCAN: MOV AL,00H MOV DX,MY8255_A OUT DX,AL ; 向所有列输出 低电平MOV DX,MY8255_C IN AL,DX ;读所有行电平;原来没有任何键按下 4行全为1;这里取反 变成 0000 便于后面的判断NOT AL; 假设没有按键按下 ; 0000&1111=0; 结果为0 ZF=1 AND AL,0FHRET;清屏子程序
;就是使得所有的灯熄灭 00H表示全不亮 瞬间 很快
CLEAR: MOV DX,MY8255_B MOV AL,00HOUT DX,ALRET; 显示子程序 (这里稍微有点绕)
DIS: PUSH AX MOV SI,3000H; 0DFH=1101 1111 对应PA7 PA6 PA5...PA1 PA0; 由电路图 得出 X1-PA0 X2-PA1.....; 6个显示器 从左到右依次是 X1 X2 X3... X5 X6; 所以 对应的PA: PA0 PA1 PA2...PA4 PA5; 这里初始是0DFH 代表 1 1 1 1 1 0 ; 意思是 第六个显示 开始显示数字; 哈哈 这里其实是从X6到X1依次显示的; 每个数字显示间隔很快 我们会认为是6个数字一起显示 其实是逐个显示MOV DL,0DFHMOV AL,DLAGAIN: PUSH DX; 把AL送给A口 觉得开放哪个灯 (这里要看电路图 A口也控制灯的开放)MOV DX,MY8255_A OUT DX,ALMOV AL,[SI] ; 把3000H--3005H中存的偏移量(相对)取出MOV BX,OFFSET DTABLE ; 获取DTABLE的首地址AND AX,00FFH ;因为后面会有加法运算 先把ah清0 这样ax就是 ; al的值,防止出错ADD BX,AX ; 获取需要的值的偏移量(这个是绝对偏移量)MOV AL,[BX] ; 获取显示数字需要的值 例 显示0需要3FHMOV DX,MY8255_B ; 送往B口 显示数字OUT DX,ALCALL DALLY ;延时INC SI ;移动SI 读取下一个偏移量POP DXMOV AL,DL ; DL: 控制哪个灯的开放 开始是0DF 1101 1111; 取后6位(看电路图 只连了6根线)即01 1111; 赋值给ALTEST AL,01H ; 测试AL 看是否为11 1110 ; 6个灯 一次显示需要循环6次; 这里第六次结束是 AL=11 1110; 对于灯 就是x1灯显示完(灯:X6->X1)JZ OUT1 ; 6次循环完成后 跳出ROR AL,1 ; 循环右移; 例 第一个灯亮 AL=01 1111 ; 则 第二个灯亮 为 10 1111; 所以需要循环右移; 反映在灯上 则是左移(不要绕进去了哦)MOV DL,ALJMP AGAIN ; 跳回 继续显示 需循环6次
OUT1: POP AXRET; 子程序 延时作用 RET为子程序结束标记
DALLY: PUSH CX MOV CX,0006H
T1: MOV AX,009FH
T2: DEC AXJNZ T2LOOP T1POP CXRET; 将获得的偏移量存入3000H--30005H中
; 便于后面的显示
; 显示其实就是从3000H--3005H中读取偏移量
; 然后在table中找到真正的值即可
PUTBUF: MOV SI,DI ;存键盘值到相应位的缓冲中MOV [SI],AL ;先存入地址3005H 再递减 也就是下一个存入偏移量的是3004HDEC DICMP DI,2FFFHJNZ GOBACKMOV DI,3005H
GOBACK: RETCODE ENDSEND START