Сборка Windows/DOS — простая математика
Вопрос
В настоящее время я изучаю сборку Windows/DOS.Я просто делаю небольшую программу, которая складывает два целых числа по основанию 10 и выводит решение на стандартный вывод.Вот мой текущий код:
org 100h
MOV al,5
ADD al,3
mov dx,al
mov ah,9
int 21h
ret
Я не понимаю, почему при компиляции я получаю ошибку:
ошибка:недопустимая комбинация кода операции и операндов
Потому что теоретически все, что я делаю, это помещаю 5 в регистр AL, добавляю к нему три, беру содержимое регистра AL и помещаю его в регистр DX для вывода, а затем отображаю его.
Любая помощь будет оценена по достоинству, спасибо!
Решение
DX
это 16-битный регистр, но AL
является 8-битным.
Нагрузка AL
в DL
, и установите DH
до 0.
Но это не даст того, чего вы хотите;функция 9 [отображает строку, завершающуюся нулем].Вы просите его отобразить строку, которая начинается со смещения 9 сегмента данных, что, вероятно, будет мусором.
Сначала вам нужно будет преобразовать ответ в последовательность цифр, а затем вызвать функцию 9.
Есть пример кода для преобразование содержимого регистра в строку доступный.Скопировано сюда для справки, написано пользователем с псевдонимом Битдог.
; ------ WDDD = Write a Decimal Digit Dword at the cursor & a CRLF ------
;
; Call with, DX:AX = Dword value to print
; Returns, CF set on error, if DX:AX > 655359 max#
; (see WDECEAX.inc for larger number prints)
align 16
WDDD: CMP DX,10
JB WDDDok
STC ;CF=set
RET ;error DX:AX exceeded max value
WDDDok: PUSH AX
PUSH BX
PUSH CX
PUSH DX
XOR CX,CX ; clear count register for push count
MOV BX,10
WDDDnz: DIV BX ; divide DX:AX by BX=10
PUSH DX ; put least siginificant number (remainder) on stack
XOR DX,DX ; clear remainder reciever for next divide
OR AX,AX ; check to see if AX=number is divided to 0 yet
LOOPNE WDDDnz ; get another digit? count the pushes
MOV AH,2 ; function 2 for interupt 21h write digit
NEG CX ; two's compliment, reverse CX
MOV BL,48 ; '0'
WDDDwr: POP DX ; get digit to print, last pushed=most significant
ADD DL,BL ; convert remainder to ASCII character
INT 21h ; print ascii interupt 21h ( function 2 in AH )
LOOP WDDDwr ; deincrement CX, write another, if CX=0 we done
MOV DL,13 ; CR carriage return (AH=2 still)
INT 21h
MOV DL,10 ; LF line feed
INT 21h
POP DX
POP CX
POP BX
POP AX
CLC ;CF=clear, sucess
RET
; A divide error occurs if DX has any value
; when DIV trys to put the remainder into it
; after the DIVide is completed.
; So, DX:AX can be a larger number if the divisor is a larger number.