当你想在你的代码中找到一个错误时,这很难;当你认为你的代码是不会有错误时,这就更难了。

位运算符-英特尔8085程序集中的位移操作做什么?

admin 7℃
我正试图向自己解释下面的8085汇编代码

我的代码要求输入两个数字(来自IDE中的虚拟键盘),并在LED 7和8上显示它们:

.ORG 0000

CALL DATA
MOV C,A
CALL PRNT

CALL DATA
MOV E,A
CALL PRNT

MVI D,00H
MOV L,E
MVI H,00H

MOV A,C
DCR A
JUMP:
DAD D
DCR A
JNZ JUMP

MOV A,L
DAA
JMP IMPR

RET

DATA:
MVI A,00000000B
OUT 00H
IN 00H
RLC
RLC
RLC
RLC
ANI F0H
MOV B,A
MVI A,00000000B
OUT 00H
IN 00H
ANI 0FH
ORA B
RET

IMPR:
MOV B,A

ANI F0H
RLC
RLC
RLC
RLC
CALL NUMZ
OUT 06H
MOV A,B
ANI 0FH
CALL NUMZ
OUT 07H
RET

NUMZ:
CPI 00H
JNZ ONE
MVI A,01110111B
JMP EXIT

ONE:
CPI 01H
JNZ TWO
MVI A,01000100B
JMP EXIT

TWO:
CPI 02H
JNZ THREE
MVI A,00111110B
JMP EXIT

THREE:
CPI 03H
JNZ FOUR
MVI A,01101110B
JMP EXIT

FOUR:
CPI 04H
JNZ FIVE
MVI A,01001101B
JMP EXIT

FIVE:
CPI 05H
JNZ SIX
MVI A,01101011B
JMP EXIT

SIX:
CPI 06H
JNZ SEVEN
MVI A,01111011B
JMP EXIT

SEVEN:
CPI 07H
JNZ EIGHT
MVI A,01000110B
JMP EXIT

EIGHT:
CPI 08H
JNZ NINE
MVI A,01111111B
JMP EXIT

NINE:
CPI 09H
JNZ SAL
MVI A,01001111B
JMP EXIT

EXIT:
RET

我不包括 PRNT 因为这对我的问题不重要

我明白 .ORG 0000 这是程序的开始-就像 BEGIN 在帕斯卡

CALL DATA 是一个子例程,它用二进制零填充累加器并显示它们(?)在端口0(十六进制为00H),然后它(从虚拟键盘)得到一个数字,然后在一个位移位操作中向左旋转

我的问题是为什么?这么做有什么意义?有什么好处?我在维基百科上读过,但还是不明白。它在这段代码中做什么?为什么需要它?

那个 DATA 子程序加载两个ASCII十进制字符,并将它们形成一个两位数的BCD值。它将第一个字符左移4位,只保留LS 4位,然后将第二个字符的LS 4位放入结果的LS 4位

在C中,这大致相当于:

char c = getchar();             // get first ASCII decimal character
char result = (c << 4) & 0xf0;  // shift into MS nybble of result and mask
c = getchar();                  // get second ASCII decimal characters
result = result | (c & 0x0f);   // mask and inset into LS nybble of result
return result;

请注意,屏蔽除LS nyble之外的所有ASCII十进制字符都会给出其十进制等效值,例如ASCII'4'=0x34=>0x04


为了清楚地说明这一点,我绘制了一个图表,逐步显示当用户输入数字'69'时发生的情况,即ASCII'6'后跟ASCII'9',因为这两个字符被屏蔽并组合在一起以给出数字69的BCD表示:

enter image description here

转载请注明:我的代码 » 位运算符-英特尔8085程序集中的位移操作做什么?