/****************************************************************************
* Copyright (C) 2015-2017 by the SotS Team *
* *
* This file is part of Sovereign of the Skies. *
* *
* Sovereign of the Skies is free software: you can redistribute it *
* and/or modify it *
* under the terms of the GNU Lesser General Public License as published *
* by the Free Software Foundation, either version 3 of the License, or *
* (at your option) any later version provided you include a copy of the *
* licence and this header. *
* *
* Sovereign of the Skies is distributed in the hope that it will be *
* useful, but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU Lesser General Public License for more details. *
* *
* You should have received a copy of the GNU Lesser General Public *
* License along with Sovereign of the Skies. *
* If not, see . *
****************************************************************************/
/*
* File: agb_debug.c
* Author: Sturmvogel
*
* Created on 11. April 2017, 04:55
*/
#include
#include
#include
void dprint(const char * sz)
{
__asm__ __volatile__(
"mov r2, %0\n"
"mov r0, #0xC0\n"
"lsl r0, #0x8\n"
"mov r1, #0xDE\n"
"orr r0, r1\n"
"lsl r0, #0x8\n"
"mov r1, #0xD0\n"
"orr r0, r1\n"
"lsl r0, #8\n"
"mov r1, #0x0D\n"
"orr r0, r1\n"
"mov r1, #0\n"
"and r0, r0, r0\n":
:
"r" (sz):
"r0", "r1", "r2");
}
u32 mini_strlen(const char * s) {
u32 len = 0;
while (s[len] != '\0') len++;
return len;
}
u32 mini_itoa(int value, u32 radix, u32 uppercase, u32 unsig, char * buffer, u32 zero_pad) {
char * pbuffer = buffer;
s32 negative = 0;
u32 i, len;
/* No support for unusual radixes. */
if (radix > 16)
return 0;
if (value < 0 && !unsig) {
negative = 1;
value = -value;
}
/* This builds the string back to front ... */
if (radix == 16) {
do {
u32 digit = value & 0xF;
* (pbuffer++) = (digit < 10 ? '0' + digit : (uppercase ? 'A' : 'a') + digit - 10);
value >>= 4;
} while (value > 0);
} else {
do {
u32 digit = value % radix;
* (pbuffer++) = (digit < 10 ? '0' + digit : (uppercase ? 'A' : 'a') + digit - 10);
value /= radix;
} while (value > 0);
}
for (i = (pbuffer - buffer); i < zero_pad; i++)
* (pbuffer++) = '0';
if (negative)
{
* (pbuffer++) = '-'; * (pbuffer) = '\0';
}
/* ... now we reverse it (could do it recursively but will
* conserve the stack space) */
len = (pbuffer - buffer);
for (i = 0; i < len / 2; i++) {
char j = buffer[i];
buffer[i] = buffer[len - i - 1];
buffer[len - i - 1] = j;
}
return len;
}
int mini_vsnprintf(char * buffer, u32 buffer_len,
const char * fmt, va_list va)
{
char * pbuffer = buffer;
char bf[24];
char ch;
int _putc(char ch)
{
if ((u32)((pbuffer - buffer) + 1) >= buffer_len)
return 0;
* (pbuffer++) = ch;
* (pbuffer) = '\0';
return 1;
}
int _puts(char * s, u32 len)
{
u32 i;
if (buffer_len - (pbuffer - buffer) - 1 < len)
len = buffer_len - (pbuffer - buffer) - 1;
/* Copy to buffer */
for (i = 0; i < len; i++)
* (pbuffer++) = s[i];
* (pbuffer) = '\0';
return len;
}
while ((ch = * (fmt++))) {
if ((u32)((pbuffer - buffer) + 1) >= buffer_len)
break;
if (ch != '%')
_putc(ch);
else {
char zero_pad = 0;
char * ptr;
u32 len;
ch = * (fmt++);
/* Zero padding requested */
if (ch == '0') {
ch = * (fmt++);
if (ch == '\0')
return pbuffer - buffer;
if (ch >= '0' && ch <= '9')
zero_pad = ch - '0';
ch = * (fmt++);
}
switch (ch) {
case 0:
return pbuffer - buffer;
case 'u':
case 'd':
len = mini_itoa(va_arg(va, u32), 10, 0, (ch == 'u'), bf, zero_pad);
_puts(bf, len);
break;
case 'x':
case 'X':
len = mini_itoa(va_arg(va, u32), 16, (ch == 'X'), 1, bf, zero_pad);
_puts(bf, len);
break;
case 'c':
_putc((char)(va_arg(va, int)));
break;
case 's':
ptr = va_arg(va, char * );
_puts(ptr, mini_strlen(ptr));
break;
default:
_putc(ch);
break;
}
}
}
return pbuffer - buffer;
}
void dprintf(const char * str, ...)
{
char __outstr[256];
va_list args;
va_start(args, str);
mini_vsnprintf(__outstr, 256, str, args);
va_end(args);
dprint(__outstr);
}