aboutsummaryrefslogtreecommitdiff
path: root/src/kernel/lib/stdio/vsprintf.c
blob: 61f4323e3e2de030ca2c26584f10c7f2355152c6 (plain) (blame)
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
#include <lib/stdlib.h>
#include <stdarg.h>
#include <stdint.h>

static void append(char *dest, char *src, int index)
{
	for (int i = index; i < strlen(src) + index; i++)
		dest[i] = src[i - index];
	dest[index + strlen(src)] = 0;
}

int vsprintf(char *str, const char *fmt, va_list args)
{
	u8 ready_to_format = 0;

	int i = 0;
	char buf = 0;
	char format_buffer[20] = "\0";

	for (; *fmt; fmt++) {
		if (ready_to_format) {
			ready_to_format = 0;

			if (*fmt == '%') {
				str[i] = '%';
				continue;
			}

			buf = *fmt;

			// TODO: Improve this repetitive code
			if (buf == 's') {
				char *string = va_arg(args, char *);
				append(str, string, i);
				i = strlen(str);
			} else if (buf == 'x') {
				itoa_base(va_arg(args, u32), format_buffer, 16, 0);
				append(str, format_buffer, i);
				i = strlen(str);
			} else if (buf == 'd' || buf == 'i') {
				itoa_base(va_arg(args, s32), format_buffer, 10, 1);
				append(str, format_buffer, i);
				i = strlen(str);
			} else if (buf == 'u') {
				itoa_base(va_arg(args, u32), format_buffer, 10, 0);
				append(str, format_buffer, i);
				i = strlen(str);
			} else if (buf == 'o') {
				itoa_base(va_arg(args, u32), format_buffer, 8, 0);
				append(str, format_buffer, i);
				i = strlen(str);
			} else if (buf == 'b') {
				itoa_base(va_arg(args, u32), format_buffer, 2, 0);
				append(str, format_buffer, i);
				i = strlen(str);
			} else if (buf == 'c') {
				str[i] = (char)va_arg(args, int);
				i++;
			}
		} else {
			if (*fmt == '%')
				ready_to_format = 1;
			else {
				str[i] = *fmt;
				i++;
			}
		}

		format_buffer[0] = '\0';
	}

	return strlen(str);
}