0

在进行前 2 次调用后,没有任何不好的事情发生,但只要您按 Enter 键选择菜单,它就会输出两个提示,要求用户输入,而不是在每个提示输入操作数时暂停。

为什么中断会跳过?

我现在真的很困惑。

SECTION .data

        AskForCalculationPrompt: db "Choose which operation you want", 0xA, "1. Addition", 0xA, "2.Subtraction", 0xA, "3. Multiplication", 0xA, "4. Division", 0x3
        AskForCalculationPromptln: equ $-AskForCalculationPrompt

        FirstOperandPrompt: db "Enter the first operand:", 0xA
        FirstOperandPromptln: equ $-FirstOperandPrompt

        SecondOperandPrompt: db "Enter the second operand:", 0xA
        SecondOperandPromptln: equ $-SecondOperandPrompt

        AnswerPrompt: db "The answer is: "
        AnswerPromptln: equ $-AnswerPrompt

        ErrorMsg: db "Wrong Choice made...insert correct choice"
        ErrorMsgln: equ $-ErrorMsg


SECTION .bss

        Choice: resb 1
        FirstOperand: resb 1
        SecondOperand: resb 1
        Answer: resb 1

SECTION .text

        ;Make interrupt to ask for a prompt ask for calculation prompt
        global _start

        _start:

        mov eax, 4 ;Specify sys_write call
        mov ebx, 1 ;Standard output
        mov ecx, AskForCalculationPrompt
        mov edx, AskForCalculationPromptln
        int 80h

        ;Make interrupt to read textfrom keyboard
        READ:

        mov eax, 3 ;Sys_read call
        mov ebx, 0 ;Standard input file descriptor 0
        mov ecx, Choice
        mov edx, 1
        int 80h

        ;Determine what we inserted

        ;Prompt for first operand
        mov eax, 4
        mov ebx, 1
        mov ecx, FirstOperandPrompt
        mov edx, FirstOperandPromptln
        int 80h

        ;Retrieve first operand input
        mov eax, 3
        mov ebx, 0
        mov ecx, FirstOperand
        mov ebx, 1
        int 80h

        ;Prompt for second operand
        mov eax, 4
        mov ebx, 1
        mov ecx, SecondOperandPrompt
        mov edx, SecondOperandPromptln
        int 80h

        ;Retrieve second operand input
        mov eax, 3
        mov ebx, 0
        mov ecx, SecondOperand
        mov edx, 1
        int 80h

        ;Load values retrieved into registers eax, ebx, ecx, for comparison and operation
        mov al, byte [Choice]
        mov bl, byte [FirstOperand]
        mov cl, byte [SecondOperand]


        JMP SWITCH
;*******************************************************************************
;SWITCH*************************************************************************
;*******************************************************************************
        SWITCH:

                cmp al, 0x31
                je ADDLABEL

                cmp al, 0x32
                je SUBTRACTLABEL

                cmp al, 0x33
                je MULTIPLICATIONLABEL

                cmp al, 0x34
                je DIVISIONLABEL

                JMP DEFAULTLABEL

        DEFAULTLABEL:

                mov eax, 4
                mov ebx, 1
                mov ecx, ErrorMsg
                mov edx, ErrorMsgln
                int 80h
                JMP READ


;*****************************************************************************
;OPERATIONS*******************************************************************
;*****************************************************************************

        ADDLABEL:
                mov al, cl
                add al, bl
                JMP DISPLAYOPERATION

        SUBTRACTLABEL:
                mov al, cl
                sub al, bl
                JMP DISPLAYOPERATION

        MULTIPLICATIONLABEL:
                mov al, cl
                mul cl
                JMP DISPLAYOPERATION

        DIVISIONLABEL:
                mov al, cl
                div cl
                JMP DISPLAYOPERATION

;*****************************************************************************
;DISPLAYOPERATION*************************************************************
;*****************************************************************************

        DISPLAYOPERATION:

                mov [Answer], eax
                mov eax, 4
                mov ebx, 1
                mov ecx, Answer
                mov edx, 1
                int 80h

                mov eax, 1
                mov ebx, 0
                int 80h
4

1 回答 1

1

我认为您遇到的事实是sys_read(来自标准输入)在看到换行符(“输入”键)之前不会返回。只有您要查找的一个字符进入您的缓冲区(选择),换行符保留在操作系统的缓冲区中(称为“键盘缓冲区”)。当下一个sys_read出现时,它会将该换行符读入您的下一个缓冲区(FirstOperand)......并继续......随之而来的是欢闹!

修复它的简单方法是制作这些缓冲区resb 2并制作edx2,并相信用户只输入一个密钥然后“输入”......每次......然后故意忽略每个缓冲区中的第二个字节。

更好的方法可能是“刷新缓冲区”——操作系统的键盘缓冲区,而不是你刚刚读入的缓冲区。返回时sys_read,读取的数字在eax. 如果它小于edx,你应该很好(在这种情况下不会)。如果它相等(它不应该更大),请检查最后一个字符是否是换行符。如果是,那你很好。如果没有,操作系统的缓冲区中有更多的垃圾(这会搞砸你的下一个sys_read!)。一次读入一个“虚拟”缓冲区(在堆栈中.bss或堆栈上),直到找到换行符 -cmp byte [dummybuf], 0Ah左右。然后你准备好继续...

这就是原始 sys_call 编程的乐趣所在。:)

你不问,但mov [Answer], eax可能是自找麻烦。这将写入您的一字节缓冲区......以及之后的三个字节!它之后没有任何东西,它不会与你不“拥有”的内存对接,所以它现在不会引起问题,但它是一个“等待发生的错误”。要么使用al(应该在这里工作),要么让 Answerresd适合所有eax

这不会给你你期望的答案。在对它们进行算术运算之前,您需要从每个操作数中减去“0”(或 30h 或 48),然后在打印之前将“0”加回答案。不超过一位,就更惨了!首先让中断按预期工作。

于 2013-06-20T08:37:37.933 回答