键盘扫描及显示实验(含代码解释)
2020-12-09 知乎
实验题目
将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 STACK DW 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,07H DB 7FH,6FH,77H,7CH,39H,5EH,79H,71H DATA ENDS CODE SEGMENT ASSUME CS:CODE,DS:DATA START: MOV AX,DATA MOV DS,AX ; 把3000H--3005H中的值全部初始化为00H ; 说明初始偏移量全为0(3000H--3005H) MOV SI,3000H MOV AL,00H MOV [SI],AL ;清显示缓冲 MOV [SI+1],AL MOV [SI+2],AL MOV [SI+3],AL MOV [SI+4],AL MOV [SI+5],AL MOV DI,3005H MOV DX,MY8255_CON ;8255控制字初始化 MOV AL,81H ;1000 0001 A、B口输出 C口输入 OUT DX,AL BEGIN: ; 调用显示子程序 CALL DIS ;清屏 CALL CLEAR ;扫描 看是否有键按下 CALL CCSCAN ;有键按下 跳INK1 JNZ INK1 JMP BEGIN INK1: CALL DIS CALL DALLY CALL DALLY CALL CLEAR CALL 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,CH MOV DX,MY8255_A OUT DX,AL MOV DX,MY8255_C IN AL,DX L1: TEST AL,01H ;is L1? JNZ L2 MOV AL,00H ;L1 JMP KCODE L2: TEST AL,02H ;is L2? JNZ L3 MOV AL,04H ;L2 JMP KCODE L3: TEST AL,04H ;is L3? JNZ L4 MOV AL,08H ;L3 JMP KCODE L4: TEST AL,08H ;is L4? JNZ NEXT MOV AL,0CH ;L4 ; 找到按键后 此时AL存的的第一列每一行的初始值 0 4 8 C ; CL 存的是对应行的偏移量 ; 假设 AL为08H CL为2 则表示的总偏移量为 8+2=10H ; 说明在table中该数字的偏移量为10H ; 输出该数字 利用偏移量就行 因为数字其实是存在table中的 KCODE: ADD AL,CL CALL PUTBUF PUSH AX KON: CALL DIS CALL CLEAR CALL CCSCAN JNZ KON POP AX NEXT: INC CL ; CL相当于 行偏移量 MOV AL,CH TEST AL,08H ; 08H=0000 1000 当AL为1111 0111 && 0000 1000 结果为0 ; ZF=1 说明行偏移量达到最大值 3 JZ KERR ; 4次列循环结束 跳KERR ROL AL,1 MOV CH,AL JMP 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,0FH RET ;清屏子程序 ;就是使得所有的灯熄灭 00H表示全不亮 瞬间 很快 CLEAR: MOV DX,MY8255_B MOV AL,00H OUT DX,AL RET ; 显示子程序 (这里稍微有点绕) 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,0DFH MOV AL,DL AGAIN: PUSH DX ; 把AL送给A口 觉得开放哪个灯 (这里要看电路图 A口也控制灯的开放) MOV DX,MY8255_A OUT DX,AL MOV AL,[SI] ; 把3000H--3005H中存的偏移量(相对)取出 MOV BX,OFFSET DTABLE ; 获取DTABLE的首地址 AND AX,00FFH ;因为后面会有加法运算 先把ah清0 这样ax就是 ; al的值,防止出错 ADD BX,AX ; 获取需要的值的偏移量(这个是绝对偏移量) MOV AL,[BX] ; 获取显示数字需要的值 例 显示0需要3FH MOV DX,MY8255_B ; 送往B口 显示数字 OUT DX,AL CALL DALLY ;延时 INC SI ;移动SI 读取下一个偏移量 POP DX MOV AL,DL ; DL: 控制哪个灯的开放 开始是0DF 1101 1111 ; 取后6位(看电路图 只连了6根线)即01 1111 ; 赋值给AL TEST 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,AL JMP AGAIN ; 跳回 继续显示 需循环6次 OUT1: POP AX RET ; 子程序 延时作用 RET为子程序结束标记 DALLY: PUSH CX MOV CX,0006H T1: MOV AX,009FH T2: DEC AX JNZ T2 LOOP T1 POP CX RET ; 将获得的偏移量存入3000H--30005H中 ; 便于后面的显示 ; 显示其实就是从3000H--3005H中读取偏移量 ; 然后在table中找到真正的值即可 PUTBUF: MOV SI,DI ;存键盘值到相应位的缓冲中 MOV [SI],AL ;先存入地址3005H 再递减 也就是下一个存入偏移量的是3004H DEC DI CMP DI,2FFFH JNZ GOBACK MOV DI,3005H GOBACK: RET CODE ENDS END START