aboutsummaryrefslogtreecommitdiff
path: root/src/kernel/acpi
diff options
context:
space:
mode:
authorMarvin Borner2020-04-18 19:27:00 +0200
committerMarvin Borner2020-04-18 19:27:00 +0200
commitdce9106bcb47ef37a4cf5a221340f4eb438aedfe (patch)
tree2b1ec8efdba9b31510ad6f49a9610df701aab025 /src/kernel/acpi
parentf943fad362a0e17f6e294341b32e49b9afd6e9d7 (diff)
Added fadt flags and ACPI reboot
Diffstat (limited to 'src/kernel/acpi')
-rw-r--r--src/kernel/acpi/acpi.c49
-rw-r--r--src/kernel/acpi/acpi.h32
2 files changed, 50 insertions, 31 deletions
diff --git a/src/kernel/acpi/acpi.c b/src/kernel/acpi/acpi.c
index f826e19..5485042 100644
--- a/src/kernel/acpi/acpi.c
+++ b/src/kernel/acpi/acpi.c
@@ -37,6 +37,7 @@ void acpi_init(struct rsdp *rsdp)
if (strncmp(rsdp->signature, "RSD PTR ", 8) == 0) {
memcpy(rsdt, rsdp->rsdt_address, sizeof(struct rsdt) + 32);
+ debug("Found RSDT");
if (!check_sum((struct sdt_header *)rsdt)) {
warn("Corrupted RSDT!");
} else {
@@ -48,17 +49,17 @@ void acpi_init(struct rsdp *rsdp)
memcpy(header, (void *)address, sizeof(struct sdt_header));
if (strncmp(header->signature, "FACP", 4) == 0) {
- info("Found FADT");
+ debug("Found FADT");
memcpy(fadt, (void *)address, sizeof(struct fadt));
if (!check_sum((struct sdt_header *)fadt))
warn("Corrupted FADT!");
} else if (strncmp(header->signature, "HPET", 4) == 0) {
- info("Found HPET");
+ debug("Found HPET");
memcpy(hpet, (void *)address, sizeof(struct hpet));
if (!check_sum((struct sdt_header *)hpet))
warn("Corrupted HPET!");
} else if (strncmp(header->signature, "APIC", 4) == 0) {
- info("Found MADT");
+ debug("Found MADT");
memcpy(madt, (void *)address, sizeof(struct madt));
if (!check_sum((struct sdt_header *)madt))
warn("Corrupted MADT!");
@@ -83,36 +84,26 @@ void acpi_new_init(struct multiboot_tag_new_acpi *tag)
void acpi_poweroff()
{
+ // TODO: Add APCI poweroff support
cli();
- /*
- if (SCI_EN == 0) {
- warn("ACPI shutdown is not supported");
- return;
- }
-
- // Send shutdown command
- outw((uint16_t)(unsigned int)PM1a_CNT, (uint16_t)(SLP_TYPa | SLP_EN));
- if (PM1b_CNT != 0)
- outw((uint16_t)(unsigned int)PM1b_CNT, (uint16_t)(SLP_TYPb | SLP_EN));
- else {
- outw(0xB004, 0x2000); // Bochs
- outw(0x604, 0x2000); // QEMU
- outw(0x4004, 0x3400); // VirtualBox
- }
- */
+ outw(0x604, 0x2000); // QEMU
+ outw(0xB004, 0x2000); // Bochs
+ outw(0x4004, 0x3400); // VirtualBox
+ halt_loop();
}
void reboot()
{
cli();
- outb(fadt->reset_reg.address, fadt->reset_value);
- halt_loop();
-
- /* else?
- uint8_t good = 0x02;
- while (good & 0x02)
- good = inb(0x64);
- outb(0x64, 0xFE);
- halt_loop();
- */
+ if (fadt->header.revision >= 2 && fadt->flags.reset_support) {
+ debug("Reset support!");
+ outb(fadt->reset_reg.address, fadt->reset_value);
+ halt_loop();
+ } else {
+ uint8_t good = 0x02;
+ while (good & 0x02)
+ good = inb(0x64);
+ outb(0x64, 0xFE);
+ halt_loop();
+ }
}
diff --git a/src/kernel/acpi/acpi.h b/src/kernel/acpi/acpi.h
index c9df040..055af2c 100644
--- a/src/kernel/acpi/acpi.h
+++ b/src/kernel/acpi/acpi.h
@@ -28,6 +28,33 @@ struct address_structure {
uint64_t address;
};
+// p. 138
+struct fadt_flags {
+ uint8_t WBINVD : 1;
+ uint8_t WBINVD_flush : 1;
+ uint8_t C1_support : 1;
+ uint8_t C2_mp_support : 1;
+ uint8_t power_button : 1; // 1 if not present
+ uint8_t sleep_button : 1; // 1 if not present
+ uint8_t rtc_fix_reg : 1;
+ uint8_t rtc_wakes_S4 : 1;
+ uint8_t timer_32 : 1;
+ uint8_t dock_support : 1;
+ uint8_t reset_support : 1;
+ uint8_t sealed_case : 1;
+ uint8_t headless : 1;
+ uint8_t slp_instruction : 1;
+ uint8_t pci_wake_support : 1;
+ uint8_t use_platform_clock : 1;
+ uint8_t rtc_valid_S4 : 1;
+ uint8_t remote_on_support : 1;
+ uint8_t force_apic_cluster : 1;
+ uint8_t force_apic_physical : 1;
+ uint8_t hw_reduced_acpi : 1;
+ uint8_t low_power_S0_support : 1;
+ uint16_t reserved : 10;
+};
+
struct rsdp {
char signature[8];
char checksum;
@@ -50,11 +77,12 @@ struct sdt_header {
struct rsdt {
struct sdt_header header;
+ uint32_t sdt_pointer[];
};
struct xsdt {
struct sdt_header header;
- uint32_t *sdt_pointer;
+ uint32_t sdt_pointer[];
};
struct fadt {
@@ -96,7 +124,7 @@ struct fadt {
uint8_t century;
uint16_t boot_architecture_flags; // Reserved in 1.0
uint8_t reserved2;
- uint32_t flags;
+ struct fadt_flags flags;
struct address_structure reset_reg;
uint8_t reset_value;
uint8_t reserved3[3];