Ok, so I\'m fairly new to assembly, infact, I\'m very new to assembly. I wrote a piece of code which is simply meant to take numerical input from the user, multiply it by 10, and have the result expressed to the user via the programs exit status (by typing echo $? in terminal)
Problem is, it is not giving the correct number, 4x10 showed as 144. So then I figured the input would probably be as a character, rather than an integer. My question here is, how do I convert the character input to an integer so that it can be used in arithmetic calculations?
It would be great if someone could answer keeping in mind that I\'m a beginner :)
Also, how can I convert said integer back to a character?
section .data
section .bss
input resb 4
section .text
global _start
_start:
mov eax, 3
mov ebx, 0
mov ecx, input
mov edx, 4
int 0x80
mov ebx, 10
imul ebx, ecx
mov eax, 1
int 0x80
Here\'s a couple of functions for converting strings to integers, and vice versa:
; Input:
; ESI = pointer to the string to convert
; ECX = number of digits in the string (must be > 0)
; Output:
; EAX = integer value
string_to_int:
xor ebx,ebx ; clear ebx
.next_digit:
movzx eax,byte[esi]
inc esi
sub al,\'0\' ; convert from ASCII to number
imul ebx,10
add ebx,eax ; ebx = ebx*10 + eax
loop .next_digit ; while (--ecx)
mov eax,ebx
ret
; Input:
; EAX = integer value to convert
; ESI = pointer to buffer to store the string in (must have room for at least 10 bytes)
; Output:
; EAX = pointer to the first character of the generated string
int_to_string:
add esi,9
mov byte [esi],STRING_TERMINATOR
mov ebx,10
.next_digit:
xor edx,edx ; Clear edx prior to dividing edx:eax by ebx
div ebx ; eax /= 10
add dl,\'0\' ; Convert the remainder to ASCII
dec esi ; store characters in reverse order
mov [esi],dl
test eax,eax
jnz .next_digit ; Repeat until eax==0
mov eax,esi
ret
And this is how you\'d use them:
STRING_TERMINATOR equ 0
lea esi,[thestring]
mov ecx,4
call string_to_int
; EAX now contains 1234
; Convert it back to a string
lea esi,[buffer]
call int_to_string
; You now have a string pointer in EAX, which
; you can use with the sys_write system call
thestring: db \"1234\",0
buffer: resb 10
Note that I don\'t do much error checking in these routines (like checking if there are characters outside of the range \'0\' - \'9\'
). Nor do the routines handle signed numbers. So if you need those things you\'ll have to add them yourself.
The basic algorith for string->digit is: total = total*10 + digit
, starting from the MSD. So the left-most / most-significant / first digit (in memory, and in reading order) gets multiplied by 10 N times, where N is the total number of digits after it.
Doing it this way is generally more efficient than multiplying each digit by the right power of 10 before adding. That would need 2 multiplies; one to grow a power of 10, and another to apply it to the digit. (Or a table look-up with ascending powers of 10). Of course, for efficiency you can use SSSE3 pmaddubsw
and SSE2 pmaddwd
to multiply digits by their place-value in parallel: see How to implement atoi using SIMD?. That probably isn\'t a win when numbers are typically short, though.
A loop would only run a couple times.
Adding on to @Michael\'s answer, it may be useful to have the int->string function stop at the first non-digit, instead of at a fixed length. This will catch problems like your string including a newline from when the user pressed return, as well as not turning 12xy34
into a very large number. (Treat it as 12
, like C\'s atoi
function). The stop character can also be the terminating 0
in a C implicit-length string.
I\'ve also made some improvements:
Don\'t use the slow loop
instruction unless you\'re optimizing for code-size. Just forget it exists and use dec
/ jnz
in cases where counting down to zero is still what you want to do, instead of comparing a pointer or something else.
2 LEA instructions are significantly better than imul
+ add
: lower latency.
sub al,\'0\'
saves 1 byte over sub eax,\'0\'
, but causes a partial-register stall on Nehalem/Core2 and even worse on PIII. Fine on all other CPU families, even Sandybridge: it\'s a RMW of AL, so it doesn\'t rename the partial reg separately from EAX.
accumulate the result in EAX where we want to return it anyway. (If you inline this instead of calling it, use whatever register you want the result in.)
I changed the registers so it follows the x86-64 System V ABI (First arg in RDI, return in EAX). Use whatever registers you want, and use 32-bit registers to make it work in 32-bit mode; I wasn\'t using 64-bit operand-size.
; args: pointer in RDI to ASCII decimal digits, terminated by a non-digit
; clobbers: ECX, EDX
; returns: EAX = atoi(RDI) (base 10 unsigned)
; RDI = pointer to first non-digit
global string_to_int
string_to_int:
movzx eax, byte [rdi] ; start with the first digit
sub eax, \'0\' ; convert from ASCII to number
cmp al, 9 ; check that it\'s a decimal digit [0..9]
jbe .loop_entry ; too low -> wraps to high value, fails unsigned compare check
; else: bad first digit: return 0
xor eax,eax
ret
; skew the loop so we can put the JCC at the bottom where it belongs
; but still check the digit before messing up our total
.next_digit: ; do {
lea eax, [rax*4 + rax] ; total *= 5
lea eax, [rax*2 + rcx] ; total = (total*5)*2 + digit
.loop_entry:
inc rdi
movzx ecx, byte [rdi]
sub ecx, \'0\'
cmp ecx, 9 ; cl,9 is equivalent, but doesn\'t save a byte; only the al,imm8 case is special.
jbe .next_digit ; } while( digit <= 9 )
ret ; return with total in eax
Making the first iteration special and handling it before jumping into the main part of the loop is called loop peeling. Peeling the first iteration allows us to optimize it specially, because we know total=0 so there\'s no need to multiply anything by 10. It\'s like starting with sum = array[0]; i=1
instead of sum=0, i=0;
.
To get nice loop structure (with the conditional branch at the bottom), I used the trick of jumping into the middle of the loop for the first iteration. This didn\'t even take an extra jmp
because I was already branching in the peeled first iteration.
The simple way to solve the problem of exiting the loop on a non-digit would be to have a jcc
in the loop body, like an if() break;
statement in C before the total = total*10 + digit
. But then I\'d need a jmp
and have 2 total branch instructions in the loop, meaning more overhead.
If I didn\'t need the sub ecx, \'0\'
result for the loop condition, I could have used lea eax, [rax*2 + rcx - \'0\']
to do it as part of the LEA as well. But that would have made the LEA latency 3 cycles instead of 1, on Sandybridge-family CPUs. (3-component LEA vs. 2 or less.) The two LEAs form a loop-carried dependency chain on eax
(total
), so (especially for large numbers) it would not be worth it on Intel. On CPUs where base + scaled-index
is no faster than base + scaled-index + disp8
(Bulldozer-family / Ryzen), then sure, if you have an explicit length as your loop condition and don\'t want to check the digits at all.
For more optimization stuff, see http://agner.org/optmize, and other links in the x86 tag wiki.
The tag wiki also has beginner links, including an FAQ section with links to integer->string functions, and other common beginner questions. How do I print an integer in Assembly Level Programming without printf from the c library?.