mirror of
https://github.com/memtest86plus/memtest86plus.git
synced 2024-11-27 10:00:17 -06:00
e0c0cd55c6
Make this entirely local to hwctrl.c.
95 lines
2.5 KiB
C
95 lines
2.5 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
// Copyright (C) 2020-2022 Martin Whitaker.
|
|
//
|
|
// Derived from an extract of memtest86+ lib.c:
|
|
//
|
|
// lib.c - MemTest-86 Version 3.4
|
|
//
|
|
// Released under version 2 of the Gnu Public License.
|
|
// By Chris Brady
|
|
|
|
#include "stddef.h"
|
|
|
|
#include "boot.h"
|
|
#include "bootparams.h"
|
|
#include "efi.h"
|
|
|
|
#include "io.h"
|
|
|
|
#include "unistd.h"
|
|
|
|
#include "hwctrl.h"
|
|
|
|
//------------------------------------------------------------------------------
|
|
// Private Variables
|
|
//------------------------------------------------------------------------------
|
|
|
|
static efi_runtime_services_t *efi_rs_table = NULL;
|
|
|
|
//------------------------------------------------------------------------------
|
|
// Public Functions
|
|
//------------------------------------------------------------------------------
|
|
|
|
void hwctrl_init(void)
|
|
{
|
|
boot_params_t *boot_params = (boot_params_t *)boot_params_addr;
|
|
#ifdef __x86_64__
|
|
if (boot_params->efi_info.loader_signature == EFI64_LOADER_SIGNATURE) {
|
|
uintptr_t system_table_addr = (uintptr_t)boot_params->efi_info.sys_tab_hi << 32 | boot_params->efi_info.sys_tab;
|
|
if (system_table_addr != 0) {
|
|
efi64_system_table_t *sys_table = (efi64_system_table_t *)system_table_addr;
|
|
efi_rs_table = (efi_runtime_services_t *)sys_table->runtime_services;
|
|
}
|
|
}
|
|
#else
|
|
if (boot_params->efi_info.loader_signature == EFI32_LOADER_SIGNATURE) {
|
|
uintptr_t system_table_addr = boot_params->efi_info.sys_tab;
|
|
if (system_table_addr != 0) {
|
|
efi32_system_table_t *sys_table = (efi32_system_table_t *)system_table_addr;
|
|
efi_rs_table = (efi_runtime_services_t *)(uintptr_t)sys_table->runtime_services;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void reboot(void)
|
|
{
|
|
// Use cf9 method as first try
|
|
uint8_t cf9 = inb(0xcf9) & ~6;
|
|
outb(cf9|2, 0xcf9); // Request hard reset
|
|
usleep(50);
|
|
outb(cf9|6, 0xcf9); // Actually do the reset
|
|
usleep(50);
|
|
|
|
// If we have UEFI, try EFI reset service
|
|
if (efi_rs_table != NULL) {
|
|
efi_rs_table->reset_system(EFI_RESET_COLD, 0, 0);
|
|
usleep(1000000);
|
|
}
|
|
|
|
// Still here? try the PORT 0x64 method
|
|
outb(0xfe, 0x64);
|
|
usleep(150000);
|
|
|
|
if (efi_rs_table == NULL) {
|
|
// In last resort, (very) obsolete reboot method using BIOS
|
|
*((uint16_t *)0x472) = 0x1234;
|
|
}
|
|
}
|
|
|
|
void floppy_off()
|
|
{
|
|
// Stop the floppy motor.
|
|
outb(0x8, 0x3f2);
|
|
}
|
|
|
|
void cursor_off()
|
|
{
|
|
// Set HW cursor off screen.
|
|
outb(0x0f, 0x3d4);
|
|
outb(0xff, 0x3d5);
|
|
|
|
outb(0x0e, 0x3d4);
|
|
outb(0xff, 0x3d5);
|
|
}
|