Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
MorsGames
GitHub Repository: MorsGames/sm64plus
Path: blob/master/lib/src/_Printf.c
7857 views
1
#include "libultra_internal.h"
2
#include <stdarg.h>
3
#include <string.h>
4
#include "printf.h"
5
6
#define ATOI(i, a) \
7
for (i = 0; *a >= '0' && *a <= '9'; a++) \
8
if (i < 999) \
9
i = *a + i * 10 - '0';
10
#define _PROUT(dst, fmt, _size) \
11
if (_size > 0) { \
12
dst = prout(dst, fmt, _size); \
13
if (dst != 0) \
14
sp78.size += _size; \
15
else \
16
return sp78.size; \
17
}
18
#define _PAD(i, m, c, src, extracond) \
19
if (extracond && m > 0) \
20
for (i = m; i > 0; i -= c) { \
21
if ((u32) i > 32) \
22
c = 32; \
23
else \
24
c = i; \
25
_PROUT(dst, src, c); \
26
}
27
28
const char length_str[] = "hlL";
29
const char flags_str[] = " +-#0";
30
const u32 flags_arr[] = { FLAGS_SPACE, FLAGS_PLUS, FLAGS_MINUS, FLAGS_HASH, FLAGS_ZERO, 0 };
31
char _spaces[] = " ";
32
char _zeroes[] = "00000000000000000000000000000000";
33
34
static void _Putfld(printf_struct *, va_list *, u8, u8 *);
35
36
s32 _Printf(char *(*prout)(char *, const char *, size_t), char *dst, const char *fmt, va_list args) {
37
printf_struct sp78;
38
const u8 *fmt_ptr;
39
u8 c;
40
const char *flag_index;
41
u8 sp4c[0x20]; // probably a buffer?
42
s32 sp48, sp44, sp40, sp3c, sp38, sp34, sp30, sp2c, sp28, sp24;
43
sp78.size = 0;
44
while (TRUE) {
45
fmt_ptr = (u8 *) fmt;
46
#ifdef VERSION_SH
47
// new version: don't point fmt_ptr beyond NUL character
48
while ((c = *fmt_ptr) != 0 && c != '%') {
49
fmt_ptr++;
50
}
51
#else
52
while ((c = *fmt_ptr++) > 0) {
53
if (c == '%') {
54
fmt_ptr--;
55
break;
56
}
57
}
58
#endif
59
_PROUT(dst, fmt, fmt_ptr - (u8 *) fmt);
60
if (c == 0) {
61
return sp78.size;
62
}
63
fmt = (char *) ++fmt_ptr;
64
sp78.flags = 0;
65
for (; (flag_index = strchr(flags_str, *fmt_ptr)) != NULL; fmt_ptr++) {
66
sp78.flags |= flags_arr[flag_index - flags_str];
67
}
68
if (*fmt_ptr == '*') {
69
sp78.width = va_arg(args, s32);
70
if (sp78.width < 0) {
71
sp78.width = -sp78.width;
72
sp78.flags |= FLAGS_MINUS;
73
}
74
fmt_ptr++;
75
} else {
76
ATOI(sp78.width, fmt_ptr);
77
}
78
if (*fmt_ptr != '.') {
79
sp78.precision = -1;
80
} else {
81
fmt_ptr++;
82
if (*fmt_ptr == '*') {
83
sp78.precision = va_arg(args, s32);
84
fmt_ptr++;
85
} else {
86
ATOI(sp78.precision, fmt_ptr);
87
}
88
}
89
if (strchr(length_str, *fmt_ptr) != NULL) {
90
sp78.length = *fmt_ptr++;
91
} else {
92
sp78.length = 0;
93
}
94
95
if (sp78.length == 'l' && *fmt_ptr == 'l') {
96
sp78.length = 'L';
97
fmt_ptr++;
98
}
99
_Putfld(&sp78, &args, *fmt_ptr, sp4c);
100
sp78.width -= sp78.part1_len + sp78.num_leading_zeros + sp78.part2_len + sp78.num_mid_zeros
101
+ sp78.part3_len + sp78.num_trailing_zeros;
102
_PAD(sp44, sp78.width, sp48, _spaces, !(sp78.flags & FLAGS_MINUS));
103
_PROUT(dst, (char *) sp4c, sp78.part1_len);
104
_PAD(sp3c, sp78.num_leading_zeros, sp40, _zeroes, 1);
105
_PROUT(dst, sp78.buff, sp78.part2_len);
106
_PAD(sp34, sp78.num_mid_zeros, sp38, _zeroes, 1);
107
_PROUT(dst, (char *) (&sp78.buff[sp78.part2_len]), sp78.part3_len)
108
_PAD(sp2c, sp78.num_trailing_zeros, sp30, _zeroes, 1);
109
_PAD(sp24, sp78.width, sp28, _spaces, sp78.flags & FLAGS_MINUS);
110
fmt = (char *) fmt_ptr + 1;
111
}
112
}
113
114
static void _Putfld(printf_struct *a0, va_list *args, u8 type, u8 *buff) {
115
a0->part1_len = a0->num_leading_zeros = a0->part2_len = a0->num_mid_zeros = a0->part3_len =
116
a0->num_trailing_zeros = 0;
117
118
switch (type) {
119
120
case 'c':
121
buff[a0->part1_len++] = va_arg(*args, u32);
122
break;
123
124
case 'd':
125
case 'i':
126
if (a0->length == 'l') {
127
a0->value.s64 = va_arg(*args, s32);
128
} else if (a0->length == 'L') {
129
a0->value.s64 = va_arg(*args, s64);
130
} else {
131
a0->value.s64 = va_arg(*args, s32);
132
}
133
134
if (a0->length == 'h') {
135
a0->value.s64 = (s16) a0->value.s64;
136
}
137
138
if (a0->value.s64 < 0) {
139
buff[a0->part1_len++] = '-';
140
} else if (a0->flags & FLAGS_PLUS) {
141
buff[a0->part1_len++] = '+';
142
} else if (a0->flags & FLAGS_SPACE) {
143
buff[a0->part1_len++] = ' ';
144
}
145
146
a0->buff = (char *) &buff[a0->part1_len];
147
148
_Litob(a0, type);
149
break;
150
151
case 'x':
152
case 'X':
153
case 'u':
154
case 'o':
155
if (a0->length == 'l') {
156
a0->value.s64 = va_arg(*args, s32);
157
} else if (a0->length == 'L') {
158
a0->value.s64 = va_arg(*args, s64);
159
} else {
160
a0->value.s64 = va_arg(*args, s32);
161
}
162
163
if (a0->length == 'h') {
164
a0->value.s64 = (u16) a0->value.s64;
165
} else if (a0->length == 0) {
166
a0->value.s64 = (u32) a0->value.s64;
167
}
168
169
if (a0->flags & FLAGS_HASH) {
170
buff[a0->part1_len++] = '0';
171
if (type == 'x' || type == 'X') {
172
173
buff[a0->part1_len++] = type;
174
}
175
}
176
a0->buff = (char *) &buff[a0->part1_len];
177
_Litob(a0, type);
178
break;
179
180
case 'e':
181
case 'f':
182
case 'g':
183
case 'E':
184
case 'G':
185
//... okay?
186
a0->value.f64 = a0->length == 'L' ? va_arg(*args, f64) : va_arg(*args, f64);
187
188
if (a0->value.u16 & 0x8000) {
189
buff[a0->part1_len++] = '-';
190
} else {
191
if (a0->flags & FLAGS_PLUS) {
192
buff[a0->part1_len++] = '+';
193
} else if (a0->flags & FLAGS_SPACE) {
194
buff[a0->part1_len++] = ' ';
195
}
196
}
197
198
a0->buff = (char *) &buff[a0->part1_len];
199
_Ldtob(a0, type);
200
break;
201
202
case 'n':
203
if (a0->length == 'h') {
204
*(va_arg(*args, u16 *)) = a0->size;
205
} else if (a0->length == 'l') {
206
*va_arg(*args, u32 *) = a0->size;
207
} else if (a0->length == 'L') {
208
*va_arg(*args, u64 *) = a0->size;
209
} else {
210
*va_arg(*args, u32 *) = a0->size;
211
}
212
break;
213
214
case 'p':
215
a0->value.s64 = (intptr_t) va_arg(*args, void *);
216
a0->buff = (char *) &buff[a0->part1_len];
217
_Litob(a0, 'x');
218
break;
219
220
case 's':
221
a0->buff = va_arg(*args, char *);
222
a0->part2_len = strlen(a0->buff);
223
if (a0->precision >= 0 && a0->part2_len > a0->precision) {
224
a0->part2_len = a0->precision;
225
}
226
break;
227
228
case '%':
229
buff[a0->part1_len++] = '%';
230
break;
231
232
default:
233
buff[a0->part1_len++] = type;
234
break;
235
}
236
}
237
238