2018-05-05 22:55:40 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <errno.h>
|
2018-05-09 21:20:14 +01:00
|
|
|
#include <malloc.h>
|
2018-05-05 22:55:40 +01:00
|
|
|
#include <sys/iosupport.h>
|
|
|
|
#include "console.h"
|
2018-08-18 17:59:33 +01:00
|
|
|
#include "di.h"
|
2018-05-05 22:55:40 +01:00
|
|
|
#include "display/video_fb.h"
|
|
|
|
|
2018-05-09 21:20:14 +01:00
|
|
|
static void *g_framebuffer = NULL;
|
|
|
|
static bool g_display_initialized = false;
|
|
|
|
|
2018-05-05 22:55:40 +01:00
|
|
|
static ssize_t console_write(struct _reent *r, void *fd, const char *ptr, size_t len);
|
2018-05-09 21:20:14 +01:00
|
|
|
static void console_init_display(void);
|
2018-05-05 22:55:40 +01:00
|
|
|
|
|
|
|
static const devoptab_t dotab_stdout = {
|
|
|
|
.name = "con",
|
|
|
|
.write_r = console_write,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* https://github.com/switchbrew/libnx/blob/master/nx/source/runtime/util/utf/decode_utf8.c */
|
2018-05-09 21:20:14 +01:00
|
|
|
static ssize_t decode_utf8(uint32_t *out, const uint8_t *in) {
|
2018-05-05 22:55:40 +01:00
|
|
|
uint8_t code1, code2, code3, code4;
|
|
|
|
|
|
|
|
code1 = *in++;
|
|
|
|
if(code1 < 0x80) {
|
|
|
|
/* 1-byte sequence */
|
|
|
|
*out = code1;
|
|
|
|
return 1;
|
|
|
|
} else if(code1 < 0xC2) {
|
|
|
|
return -1;
|
|
|
|
} else if(code1 < 0xE0) {
|
|
|
|
/* 2-byte sequence */
|
|
|
|
code2 = *in++;
|
|
|
|
if((code2 & 0xC0) != 0x80) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out = (code1 << 6) + code2 - 0x3080;
|
|
|
|
return 2;
|
|
|
|
} else if(code1 < 0xF0) {
|
|
|
|
/* 3-byte sequence */
|
|
|
|
code2 = *in++;
|
|
|
|
if((code2 & 0xC0) != 0x80) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if(code1 == 0xE0 && code2 < 0xA0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
code3 = *in++;
|
|
|
|
if((code3 & 0xC0) != 0x80) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out = (code1 << 12) + (code2 << 6) + code3 - 0xE2080;
|
|
|
|
return 3;
|
|
|
|
} else if(code1 < 0xF5) {
|
|
|
|
/* 4-byte sequence */
|
|
|
|
code2 = *in++;
|
|
|
|
if((code2 & 0xC0) != 0x80) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if(code1 == 0xF0 && code2 < 0x90) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if(code1 == 0xF4 && code2 >= 0x90) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
code3 = *in++;
|
|
|
|
if((code3 & 0xC0) != 0x80) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
code4 = *in++;
|
|
|
|
if((code4 & 0xC0) != 0x80) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out = (code1 << 18) + (code2 << 12) + (code3 << 6) + code4 - 0x3C82080;
|
|
|
|
return 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-05-09 21:20:14 +01:00
|
|
|
static void console_init_display(void) {
|
2018-06-04 18:17:23 +01:00
|
|
|
if (!g_display_initialized) {
|
|
|
|
/* Initialize the display. */
|
|
|
|
display_init();
|
|
|
|
}
|
2018-05-09 21:20:14 +01:00
|
|
|
|
|
|
|
/* Set the framebuffer. */
|
|
|
|
display_init_framebuffer(g_framebuffer);
|
|
|
|
|
|
|
|
/* Turn on the backlight after initializing the lfb */
|
|
|
|
/* to avoid flickering. */
|
2018-06-04 18:17:23 +01:00
|
|
|
if (!g_display_initialized) {
|
2018-08-18 17:59:33 +01:00
|
|
|
display_backlight(true);
|
2018-06-04 18:17:23 +01:00
|
|
|
}
|
2018-05-09 21:20:14 +01:00
|
|
|
|
|
|
|
g_display_initialized = true;
|
|
|
|
}
|
|
|
|
|
2018-05-05 22:55:40 +01:00
|
|
|
static ssize_t console_write(struct _reent *r, void *fd, const char *ptr, size_t len) {
|
|
|
|
size_t i = 0;
|
2018-05-09 21:20:14 +01:00
|
|
|
if (!g_display_initialized) {
|
|
|
|
console_init_display();
|
|
|
|
}
|
2018-05-05 22:55:40 +01:00
|
|
|
while (i < len) {
|
|
|
|
uint32_t chr;
|
|
|
|
ssize_t n = decode_utf8(&chr, (uint8_t *)(ptr + i));
|
|
|
|
if (n == -1) {
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
i += (size_t)n;
|
|
|
|
video_putc(chr > 0xFF ? '?' : (char)chr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int console_create(void) {
|
2018-05-09 21:20:14 +01:00
|
|
|
if (g_framebuffer != NULL) {
|
2018-05-05 22:55:40 +01:00
|
|
|
errno = EEXIST;
|
|
|
|
return -1;
|
|
|
|
}
|
2018-05-09 21:20:14 +01:00
|
|
|
g_framebuffer = memalign(0x1000, CONFIG_VIDEO_VISIBLE_ROWS * CONFIG_VIDEO_COLS * CONFIG_VIDEO_PIXEL_SIZE);
|
|
|
|
|
|
|
|
if (g_framebuffer == NULL) {
|
|
|
|
errno = ENOMEM;
|
|
|
|
return -1;
|
|
|
|
}
|
2018-05-05 22:55:40 +01:00
|
|
|
|
|
|
|
devoptab_list[STD_OUT] = &dotab_stdout;
|
|
|
|
devoptab_list[STD_ERR] = &dotab_stdout;
|
|
|
|
|
2018-05-09 10:49:10 +01:00
|
|
|
setvbuf(stdout, NULL , _IOLBF, 4096);
|
2018-05-05 22:55:40 +01:00
|
|
|
setvbuf(stderr, NULL , _IONBF, 0);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-06-04 18:17:23 +01:00
|
|
|
int console_init(bool display_initialized) {
|
|
|
|
g_display_initialized = display_initialized;
|
|
|
|
|
2018-05-09 21:20:14 +01:00
|
|
|
if (console_create() == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Zero-fill the framebuffer, etc. */
|
|
|
|
if (video_init(g_framebuffer) == -1) {
|
2018-05-05 22:55:40 +01:00
|
|
|
errno = EIO;
|
2018-05-09 21:20:14 +01:00
|
|
|
console_end();
|
2018-05-05 22:55:40 +01:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-05-09 21:20:14 +01:00
|
|
|
return 0;
|
2018-05-05 22:55:40 +01:00
|
|
|
}
|
|
|
|
|
2018-05-12 20:39:29 +01:00
|
|
|
void *console_get_framebuffer(bool enable_display) {
|
|
|
|
if (g_framebuffer != NULL && enable_display) {
|
|
|
|
console_init_display();
|
|
|
|
}
|
|
|
|
|
|
|
|
return g_framebuffer;
|
|
|
|
}
|
|
|
|
|
2018-05-09 21:20:14 +01:00
|
|
|
int console_display(const void *framebuffer) {
|
|
|
|
if (!g_display_initialized) {
|
|
|
|
console_init_display();
|
2018-05-05 22:55:40 +01:00
|
|
|
}
|
2018-05-09 21:20:14 +01:00
|
|
|
|
|
|
|
/* TODO: does this work? */
|
|
|
|
display_init_framebuffer((void *)framebuffer);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int console_resume(void) {
|
|
|
|
if (!g_display_initialized) {
|
|
|
|
console_init_display();
|
|
|
|
} else {
|
|
|
|
/* TODO: does this work? */
|
|
|
|
display_init_framebuffer(g_framebuffer);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int console_end(void) {
|
|
|
|
/* Deinitialize the framebuffer and display */
|
|
|
|
if (g_display_initialized) {
|
2018-08-18 17:59:33 +01:00
|
|
|
display_backlight(false);
|
2018-05-09 21:20:14 +01:00
|
|
|
display_end();
|
|
|
|
}
|
|
|
|
free(g_framebuffer);
|
|
|
|
g_framebuffer = NULL;
|
|
|
|
return 0;
|
2018-05-05 22:55:40 +01:00
|
|
|
}
|