264 lines
5.7 KiB
C
264 lines
5.7 KiB
C
#include <efi.h>
|
|
#include <efilib.h>
|
|
#include <stddef.h>
|
|
#include <stdint.h>
|
|
|
|
#include "console.h"
|
|
#include "utility.h"
|
|
|
|
size_t ROWS = 0;
|
|
size_t COLS = 0;
|
|
|
|
static EFI_SIMPLE_TEXT_OUT_PROTOCOL *con_out = 0;
|
|
|
|
const CHAR16 digits[] = {u'0', u'1', u'2', u'3', u'4', u'5', u'6', u'7', u'8', u'9', u'a', u'b', u'c', u'd', u'e', u'f'};
|
|
|
|
EFI_STATUS
|
|
con_initialize(EFI_SYSTEM_TABLE *system_table, const CHAR16 *version)
|
|
{
|
|
EFI_STATUS status;
|
|
|
|
EFI_BOOT_SERVICES *bootsvc = system_table->BootServices;
|
|
con_out = system_table->ConOut;
|
|
|
|
EFI_GRAPHICS_OUTPUT_PROTOCOL *gfx_out_proto;
|
|
EFI_GUID gfx_out_guid = EFI_GRAPHICS_OUTPUT_PROTOCOL_GUID;
|
|
|
|
status = bootsvc->LocateProtocol(&gfx_out_guid, NULL, (void **)&gfx_out_proto);
|
|
CHECK_EFI_STATUS_OR_RETURN(status, "LocateProtocol gfx");
|
|
|
|
const uint32_t modes = gfx_out_proto->Mode->MaxMode;
|
|
uint32_t res = gfx_out_proto->Mode->Info->HorizontalResolution *
|
|
gfx_out_proto->Mode->Info->VerticalResolution;
|
|
uint32_t best = gfx_out_proto->Mode->Mode;
|
|
|
|
for (uint32_t i = 0; i < modes; ++i) {
|
|
UINTN size = 0;
|
|
EFI_GRAPHICS_OUTPUT_MODE_INFORMATION *info = NULL;
|
|
status = gfx_out_proto->QueryMode(gfx_out_proto, i, &size, &info);
|
|
CHECK_EFI_STATUS_OR_RETURN(status, "QueryMode");
|
|
|
|
#ifdef MAX_HRES
|
|
if (info->HorizontalResolution > MAX_HRES) continue;
|
|
#endif
|
|
|
|
const uint32_t new_res = info->HorizontalResolution * info->VerticalResolution;
|
|
|
|
if (new_res > res) {
|
|
best = i;
|
|
res = new_res;
|
|
}
|
|
}
|
|
|
|
status = gfx_out_proto->SetMode(gfx_out_proto, best);
|
|
CHECK_EFI_STATUS_OR_RETURN(status, "SetMode %d/%d", best, modes);
|
|
|
|
status = con_out->QueryMode(con_out, con_out->Mode->Mode, &COLS, &ROWS);
|
|
CHECK_EFI_STATUS_OR_RETURN(status, "QueryMode");
|
|
|
|
status = con_out->ClearScreen(con_out);
|
|
CHECK_EFI_STATUS_OR_RETURN(status, "ClearScreen");
|
|
|
|
con_out->SetAttribute(con_out, EFI_LIGHTCYAN);
|
|
con_out->OutputString(con_out, (CHAR16 *)L"Popcorn loader ");
|
|
|
|
con_out->SetAttribute(con_out, EFI_LIGHTMAGENTA);
|
|
con_out->OutputString(con_out, (CHAR16 *)version);
|
|
|
|
con_out->SetAttribute(con_out, EFI_LIGHTGRAY);
|
|
con_out->OutputString(con_out, (CHAR16 *)L" booting...\r\n\n");
|
|
|
|
con_status_begin(L"Setting console display mode: ");
|
|
con_printf(L"\n %ux%u (%ux%u chars)",
|
|
gfx_out_proto->Mode->Info->HorizontalResolution,
|
|
gfx_out_proto->Mode->Info->VerticalResolution,
|
|
ROWS, COLS);
|
|
|
|
con_status_ok();
|
|
|
|
return status;
|
|
}
|
|
|
|
size_t
|
|
con_wstrlen(const CHAR16 *s)
|
|
{
|
|
size_t count = 0;
|
|
while (s && *s++) count++;
|
|
return count;
|
|
}
|
|
|
|
size_t
|
|
con_print_hex(uint32_t n)
|
|
{
|
|
CHAR16 buffer[9];
|
|
CHAR16 *p = buffer;
|
|
for (int i = 7; i >= 0; --i) {
|
|
uint8_t nibble = (n & (0xf << (i*4))) >> (i*4);
|
|
*p++ = digits[nibble];
|
|
}
|
|
*p = 0;
|
|
con_out->OutputString(con_out, buffer);
|
|
return 8;
|
|
}
|
|
|
|
size_t
|
|
con_print_long_hex(uint64_t n)
|
|
{
|
|
CHAR16 buffer[17];
|
|
CHAR16 *p = buffer;
|
|
for (int i = 15; i >= 0; --i) {
|
|
uint8_t nibble = (n & (0xf << (i*4))) >> (i*4);
|
|
*p++ = digits[nibble];
|
|
}
|
|
*p = 0;
|
|
con_out->OutputString(con_out, buffer);
|
|
return 16;
|
|
}
|
|
|
|
size_t
|
|
con_print_dec(uint32_t n)
|
|
{
|
|
CHAR16 buffer[11];
|
|
CHAR16 *p = buffer + 10;
|
|
*p-- = 0;
|
|
do {
|
|
*p-- = digits[n % 10];
|
|
n /= 10;
|
|
} while (n != 0);
|
|
|
|
con_out->OutputString(con_out, ++p);
|
|
return 10 - (p - buffer);
|
|
}
|
|
|
|
size_t
|
|
con_print_long_dec(uint64_t n)
|
|
{
|
|
CHAR16 buffer[21];
|
|
CHAR16 *p = buffer + 20;
|
|
*p-- = 0;
|
|
do {
|
|
*p-- = digits[n % 10];
|
|
n /= 10;
|
|
} while (n != 0);
|
|
|
|
con_out->OutputString(con_out, ++p);
|
|
return 20 - (p - buffer);
|
|
}
|
|
|
|
size_t
|
|
con_printf(const CHAR16 *fmt, ...)
|
|
{
|
|
CHAR16 buffer[256];
|
|
const CHAR16 *r = fmt;
|
|
CHAR16 *w = buffer;
|
|
va_list args;
|
|
size_t count = 0;
|
|
|
|
va_start(args, fmt);
|
|
|
|
while (r && *r) {
|
|
if (*r != L'%') {
|
|
count++;
|
|
*w++ = *r++;
|
|
continue;
|
|
}
|
|
|
|
*w = 0;
|
|
con_out->OutputString(con_out, buffer);
|
|
w = buffer;
|
|
|
|
r++; // chomp the %
|
|
|
|
switch (*r++) {
|
|
case L'%':
|
|
con_out->OutputString(con_out, L"%");
|
|
count++;
|
|
break;
|
|
|
|
case L'x':
|
|
count += con_print_hex(va_arg(args, uint32_t));
|
|
break;
|
|
|
|
case L'd':
|
|
case L'u':
|
|
count += con_print_dec(va_arg(args, uint32_t));
|
|
break;
|
|
|
|
case L's':
|
|
{
|
|
CHAR16 *s = va_arg(args, CHAR16*);
|
|
count += con_wstrlen(s);
|
|
con_out->OutputString(con_out, s);
|
|
}
|
|
break;
|
|
|
|
case L'l':
|
|
switch (*r++) {
|
|
case L'x':
|
|
count += con_print_long_hex(va_arg(args, uint64_t));
|
|
break;
|
|
|
|
case L'd':
|
|
case L'u':
|
|
count += con_print_long_dec(va_arg(args, uint64_t));
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
*w = 0;
|
|
con_out->OutputString(con_out, buffer);
|
|
|
|
va_end(args);
|
|
return count;
|
|
}
|
|
|
|
void
|
|
con_status_begin(const CHAR16 *message)
|
|
{
|
|
con_out->SetAttribute(con_out, EFI_LIGHTGRAY);
|
|
con_out->OutputString(con_out, (CHAR16 *)message);
|
|
}
|
|
|
|
void
|
|
con_status_ok()
|
|
{
|
|
UINTN row = con_out->Mode->CursorRow;
|
|
con_out->SetCursorPosition(con_out, 4, ++row);
|
|
con_out->SetAttribute(con_out, EFI_LIGHTGRAY);
|
|
con_out->OutputString(con_out, (CHAR16 *)L"[");
|
|
con_out->SetAttribute(con_out, EFI_GREEN);
|
|
con_out->OutputString(con_out, (CHAR16 *)L" ok ");
|
|
con_out->SetAttribute(con_out, EFI_LIGHTGRAY);
|
|
con_out->OutputString(con_out, (CHAR16 *)L"]\r");
|
|
|
|
con_out->SetCursorPosition(con_out, 0, ++row + 1);
|
|
}
|
|
|
|
void
|
|
con_status_fail(const CHAR16 *error)
|
|
{
|
|
UINTN row = con_out->Mode->CursorRow;
|
|
con_out->SetCursorPosition(con_out, COLS - 8, row);
|
|
con_out->SetAttribute(con_out, EFI_LIGHTGRAY);
|
|
con_out->OutputString(con_out, (CHAR16 *)L"[");
|
|
con_out->SetAttribute(con_out, EFI_LIGHTRED);
|
|
con_out->OutputString(con_out, (CHAR16 *)L"failed");
|
|
con_out->SetAttribute(con_out, EFI_LIGHTGRAY);
|
|
con_out->OutputString(con_out, (CHAR16 *)L"]\r");
|
|
|
|
con_out->SetCursorPosition(con_out, 2, row + 1);
|
|
con_out->SetAttribute(con_out, EFI_RED);
|
|
con_out->OutputString(con_out, (CHAR16 *)error);
|
|
|
|
con_out->SetCursorPosition(con_out, 0, row + 2);
|
|
con_out->SetAttribute(con_out, EFI_LIGHTGRAY);
|
|
}
|