2013-03-24 60 views
0

我今天正在做一些编码,并在其他主题的帮助下,我试图做我自己的程序。它的想法是将我在开始时的值(ascii)改为正常值。所以,我想什么控制台看到:32位Linux x86汇编代码和垃圾在输出

  1. 号[A-1,...,Z-26]

输出,当我进入一个是1, B.将2等 我得到的输出是吨垃圾..

这里是我的代码:

SYSCALL = 0X80 
SYSEXIT = 1 
SYSREAD = 3 
SYSWRITE = 4 
STDIN = 0 
STDOUT = 1 

.data 
.align 32 

NUMBER_MAXLEN = 2 
NUMBER: .space NUMBER_MAXLEN 
NUMBER_LEN: .long 0 

MSG_NUMBER: .ascii "Number [A-1, .., Z-26] " 
MSG_NUMBER_LEN = . - MSG_NUMBER 

.text 
.global _start 

_start: 

#show 
mov $SYSWRITE, %eax 
mov $STDOUT, %ebx 
mov $MSG_NUMBER, %ecx 
mov $MSG_NUMBER_LEN, %edx 
int $SYSCALL 

#read 
mov $SYSREAD, %eax 
mov $STDIN, %ebx 
mov $NUMBER, %ecx 
mov $NUMBER_MAXLEN, %edx 
int $SYSCALL 

#length 
sub $1, %eax 
mov %eax, NUMBER_LEN 

#Change to normal value 
xor %eax, %eax 
movb NUMBER, %al 
sub $'A', %al 
add $1, %al 
movb %al, NUMBER 

#Print 
mov $SYSWRITE, %eax 
mov $STDOUT, %ebx 
mov $NUMBER, %ecx 
mov $NUMBER_LEN, %edx 
int $SYSCALL  

END: 
mov $SYSEXIT, %eax 
int $SYSCALL 

我干什么什么克错了?怎么可能修好?

+0

我下NUMBER检查(使用gdb)值,和后(子$ 'A',%人 加$ 1%人 MOVB%人,NUMBER),它像68000,而不是1 ..为什么? – 2013-03-24 15:18:17

回答

0

系统调用(4)简单地打印字符串,它不作为printf函数。因此,在打印之前,您需要将数字转换为其字符串表示形式。

这里的说明一些pseduo代码怎么可以做:

char buffer[10]; 
char *p = &buffer[9]; 

*p = 0; // NULL terminator 
do { 
    p--; 
    *p = (number % 10) + '0'; 
    number /= 10; 
} while (number != 0); 

print(buffer); 
+0

此代码不能帮助我..我不明白这一点。你能解释一下吗,还是只有转换的方法? – 2013-03-24 15:36:24

+0

它所做的只是以倒数的顺序将'0'..'9'中的字符重复添加到字符串缓冲区中。如果数字是25,它会向缓冲区写入'5',然后将25除以10得到2.因此,在下一次迭代时,它将'2'写入缓冲区,将2除以10得到0,停止。此时的缓冲区包含'2','5'(以及在开始处添加的NULL终止符),即字符串“25”。 – Michael 2013-03-24 16:38:02

+0

除了itoa问题,'movl $ NUMBER_LEN,%edx'将'NUMBER_LEN'的地址移动到'$ edx'中 - 打印太多字符。失去'$'。这就是为什么你得到“吨”的垃圾,而不是一个或两个字符...... – 2013-03-24 19:01:08

0

如果用户输入“Z”,要打印“26”?你可以使用查找表,或者自己做一些转换。 INTEL/NASM格式:

%define SYSCALL 0X80 
%define SYSEXIT 1 
%define SYSREAD 3 
%define SYSWRITE 4 
%define STDIN 0 
%define STDOUT 1 

%define NUMBER_MAXLEN 2 
section .data 

MSG_NUMBER: db "Number [A-1, .., Z-26] ", 0 
MSG_NUMBER_LEN equ $ - MSG_NUMBER 

section .bss 

NUMBER:   resb NUMBER_MAXLEN 
NumToPrint  resb 3 
NUMBER_LEN:  resd 1 

section .text 
global main 

main: 

;show 
    mov  eax, SYSWRITE 
    mov  ebx, STDOUT 
    mov  ecx, MSG_NUMBER 
    mov  edx, MSG_NUMBER_LEN 
    int  SYSCALL 

;~ #read 
    mov  eax, SYSREAD 
    mov  ebx, STDIN 
    mov  ecx, NUMBER 
    mov  edx, NUMBER_MAXLEN 
    int  SYSCALL 

;~ #length 
    sub  eax, 1 
    mov  NUMBER_LEN, eax 

;~ #Change to normal value 
    movzx eax, byte [NUMBER] 
    push eax 
    call IsValidChar 
    test eax, eax 
    js  main       ; not valid input, repeat prompt 

    push NumToPrint 
    push eax 
    call dwtoa       ; convert to ASCII 

;~ #Print 
    mov  edx, eax 
    mov  eax, SYSWRITE 
    mov  ebx, STDOUT 
    mov  ecx, NumToPrint   
    int  SYSCALL  

End: 
    mov  eax, SYSEXIT 
    int  SYSCALL 

;~ #########################################  
IsValidChar: 
    mov  eax, [esp + 4] 

.0to9: 
    cmp  eax, "0" 
    jb  .NoGood 
    cmp  eax, "9" 
    ja  .AtoZ 
    jmp  .NoGood 

.AtoZ: 
    cmp  eax, "A" 
    jb  .NoGood 
    cmp  eax, "Z" 
    ja  .atoz 
    sub  eax, 64       ; convert letter index to number equiv. 
    jmp  .Done 

.atoz: 
    cmp  eax, "a" 
    jb  .NoGood 
    cmp  eax, "z" 
    ja  .NoGood 
    sub  eax, 96       ; convert letter index to number equiv. 
    jmp  .Done 

.NoGood: 
    xor  eax, eax 
    dec  eax 
.Done: 
    ret  4 

您需要创建自己的dwtoa(DWORD到ASCII)。

enter image description here