summaryrefslogtreecommitdiff
path: root/printf.s
blob: 3bb24ca2383a28dc4b9319a3e8be35fb1751e296 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
.include "lib.s"

.macro bufpush
	cmpq $bufend, %rdi
	jle 9f
	stosb
	9:
.endm

.globl printf
printf:
	prologue

	backup %rax, %rbx, %rcx, %rdx
	backup %rsi, %rdi

	movq 16(%rbp), %rsi
	leaq 24(%rbp), %rbx
	movq $buf, %rdi

.loop:
	lodsb
	testb %al, %al
	jz .loopout

	cmpb $'%, %al
	jne .append

	lodsb
	testb %al, %al
	jz .loopout

	cmpb $'%, %al
	je .append

	cmpb $'s, %al
	je .print_string

	cmpb $'d, %al
	je .print_signed

	cmpb $'u, %al
	je .print_unsigned

	jmp .loop

.append:
	bufpush
	jmp .loop

.print_string:
	backup %rsi

	movq (%rbx), %rsi
	addq $8, %rbx
	jmp 2f

	1: bufpush 
	2: lodsb
	testb %al, %al
	jnz 1b

	restore %rsi
	jmp .loop

.print_signed:
	movq (%rbx), %rax
	addq $8, %rbx

	cmpq $0, %rax
	jge 1f

	pushq %rax
	movb $'-, %al
	bufpush
	popq %rax
	negq %rax

	1:
	movq $10, %rcx
	call .print_number
	jmp .loop

.print_unsigned:
	movq (%rbx), %rax
	addq $8, %rbx

	movq $10, %rcx
	call .print_number
	jmp .loop

.print_number:
	xorq %rdx, %rdx
	divq %rcx
	pushq %rdx
	testq %rax, %rax
	jz 1f
	call .print_number
	1:
	popq %rax
	addq $'0, %rax
	bufpush
	ret

.loopout:
	movq %rdi, %rdx
	subq $buf, %rdx
	movq $buf, %rsi
	movq $STDOUT, %rdi
	sys write

	restore %rsi, %rdi
	restore %rax, %rbx, %rcx, %rdx

	leave
	ret

.bss
.comm buf, 4096
.comm bufend, 2