summaryrefslogtreecommitdiff
path: root/kernel/io.c
diff options
context:
space:
mode:
authorDanny Holman <dholman@gymli.xyz>2021-01-17 20:18:37 -0600
committerDanny Holman <dholman@gymli.xyz>2021-01-17 20:18:37 -0600
commit611f098bc38256adf2efadbc0b98138a5b4489de (patch)
tree86cecd785c9cee14783d5883f76ff410fd01450d /kernel/io.c
parent0237288ecc558e3da15fcb7ee1b28ad803e28a4d (diff)
kernel: implement a basic printf function
Implement a basic printf function using primitives from the TTY library. Signed-off-by: Danny Holman <dholman@gymli.xyz>
Diffstat (limited to 'kernel/io.c')
-rw-r--r--kernel/io.c72
1 files changed, 72 insertions, 0 deletions
diff --git a/kernel/io.c b/kernel/io.c
new file mode 100644
index 0000000..ca787b3
--- /dev/null
+++ b/kernel/io.c
@@ -0,0 +1,72 @@
+#include <kernel/io.h>
+#include <kernel/string.h>
+
+char* convert(unsigned int num, int base) {
+ static char rep[] = "0123456789ABCDEF";
+ static char buffer[50];
+ char *ptr;
+
+ ptr = &buffer[49];
+ *ptr = '\0';
+
+ do {
+ *--ptr = rep[num%base];
+ num /= base;
+ } while (num != 0);
+
+ return ptr;
+}
+
+int vprintf(const char *fmt, va_list args) {
+ char *traverse;
+ char *s;
+ int i;
+
+ for (traverse = fmt; *traverse != '\0'; traverse++) {
+ if (*traverse != '%') {
+ tty_putchar(*traverse);
+ continue;
+ } else {
+ traverse++;
+ }
+
+ switch (*traverse) {
+ case 'c':
+ i = va_arg(args, int);
+ tty_putchar(i);
+ break;
+ case 'd':
+ i = va_arg(args, int);
+ if (i < 0) {
+ i = -i;
+ tty_putchar('-');
+ }
+ tty_writestring(convert(i, 10));
+ break;
+ case 'o':
+ i = va_arg(args, unsigned int);
+ tty_writestring(convert(i, 8));
+ break;
+ case 's':
+ s = va_arg(args, char*);
+ tty_writestring(s);
+ break;
+ case 'x':
+ i = va_arg(args, unsigned int);
+ tty_writestring(convert(i, 16));
+ break;
+ }
+ }
+ return 0;
+}
+
+int printf(const char *fmt, ...) {
+ va_list args;
+ int done;
+
+ va_start(args, fmt);
+ done = vprintf(fmt, args);
+ va_end(args);
+
+ return done;
+}