You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

139 lines
2.4 KiB
C

#define TEXT_IO_ADDR 0xff0000
#define TEXT_IO_BUFLEN 64
void dbgln(char*, int);
char* itoa(int value, char* str, int base);
int thread(void* args);
int main()
{
dbgln("main", 4);
int arg = 144;
// manually invoke syscall to spawn thread
// syscall code (a7): 1
// args: target function, arg ptr
__asm__ (
"mv a0, %0\n"
"mv a1, %1\n"
"li a7, 1\n"
"ecall" :: "r"(thread), "r"(&arg)
);
// wait for child thread to modify value
while (arg == 144) { }
dbgln("child exited!", 13);
return 0;
}
int thread(void* args)
{
// read value
int arg = *((int*) args);
char buff[32] = "the magic number is: ";
char* end = itoa(arg, &buff[21], 10);
// print given number
dbgln(buff, (int) (end - buff));
// set value to free parent thread
*((int*) args) = 0;
// return value as exit code
return arg;
}
/*
* Additional functions
*/
void dbgln(char* text, int len)
{
while (len > TEXT_IO_BUFLEN) {
dbgln(text, TEXT_IO_BUFLEN);
text += TEXT_IO_BUFLEN;
len -= TEXT_IO_BUFLEN;
}
char* ioaddr = (char*) TEXT_IO_ADDR + 4;
for (int i = 0; i < len; i++) {
if (*text == 0)
break;
*ioaddr++ = *text++;
}
if (len < TEXT_IO_BUFLEN)
*ioaddr = '\n';
// write a 1 to the start of the textIO to signal a buffer flush
*((char*) TEXT_IO_ADDR) = 1;
}
char alpha[16] = "0123456789abcdef";
char* itoa(int value, char* str, int base)
{
if (base > 16 || base < 2) {
*str++ = '?';
return str;
}
if (value < 0) {
*str++ = '-';
value *= -1;
}
int digits = 0;
int num = 0;
// reverse number
do {
num = num * base;
num += value % base;
value = value / base;
digits++;
} while (value > 0);
value = num;
do {
num = value % base;
value = value / base;
*str++ = alpha[num];
digits--;
}while (digits > 0);
return str;
}
void wrap_main()
{
dbgln("start", 5);
register int code asm("s1") = main();
dbgln("end", 3);
__asm__ __volatile__(
"mv a0, s1\n"
"li a7, 5\n"
"ecall\n"
"ebreak\n"
);
}
void _start()
{
__asm__ __volatile__(
".option push\n"
".option norelax\n"
" la gp, _gp\n"
".option pop\n"
);
wrap_main();
}