From 4f90a0f5c19c1f1902fade40bf5ffa786bc598da Mon Sep 17 00:00:00 2001
From: wartman4404 <wartman4404@users.noreply.github.com>
Date: Sun, 7 Jan 2018 16:38:39 -0600
Subject: [PATCH] Write hpet timer 0 twice to set accumulator and period

This allows booting with implementations that require them to be set separately.
Also, check for the availability of legacy-replacement mode and periodic interrupts before using hpet
---
 src/arch/x86_64/device/hpet.rs | 29 ++++++++++++++++++++++++-----
 src/arch/x86_64/device/mod.rs  |  7 +++++--
 2 files changed, 29 insertions(+), 7 deletions(-)

diff --git a/src/arch/x86_64/device/hpet.rs b/src/arch/x86_64/device/hpet.rs
index 662c1757..6f2c69bc 100644
--- a/src/arch/x86_64/device/hpet.rs
+++ b/src/arch/x86_64/device/hpet.rs
@@ -11,20 +11,39 @@ static CAPABILITY_OFFSET: usize = 0x00;
 static GENERAL_CONFIG_OFFSET: usize = 0x10;
 // static GENERAL_INTERRUPT_OFFSET: usize = 0x20;
 // static MAIN_COUNTER_OFFSET: usize = 0xF0;
+// static NUM_TIMER_CAP_MASK: u64 = 0x0f00;
+static LEG_RT_CAP: u64 = 0x8000;
 static T0_CONFIG_CAPABILITY_OFFSET: usize = 0x100;
 static T0_COMPARATOR_OFFSET: usize = 0x108;
 
-pub unsafe fn init(hpet: &mut Hpet) {
-    let counter_clk_period_fs = hpet.base_address.read_u64(CAPABILITY_OFFSET) >> 32;
+static PER_INT_CAP: u64 = 0x10;
+
+pub unsafe fn init(hpet: &mut Hpet) -> bool {
+    let capability = hpet.base_address.read_u64(CAPABILITY_OFFSET);
+    if capability & LEG_RT_CAP == 0 {
+        return false;
+    }
+
+    let counter_clk_period_fs = capability >> 32;
     let desired_fs_period: u64 = 2_250_286 * 1_000_000;
 
     let clk_periods_per_kernel_tick: u64 = desired_fs_period / counter_clk_period_fs;
 
-    let enable_word: u64 = hpet.base_address.read_u64(GENERAL_CONFIG_OFFSET) | LEG_RT_CNF | ENABLE_CNF;
-    hpet.base_address.write_u64(GENERAL_CONFIG_OFFSET, enable_word);
-    // Enable interrupts from the HPET
+    let t0_capabilities = hpet.base_address.read_u64(T0_CONFIG_CAPABILITY_OFFSET);
+    if t0_capabilities & PER_INT_CAP == 0 {
+        return false;
+    }
 
     let t0_config_word: u64 = TN_VAL_SET_CNF | TN_TYPE_CNF | TN_INT_ENB_CNF;
     hpet.base_address.write_u64(T0_CONFIG_CAPABILITY_OFFSET, t0_config_word);
     hpet.base_address.write_u64(T0_COMPARATOR_OFFSET, clk_periods_per_kernel_tick);
+    // set accumulator value
+    hpet.base_address.write_u64(T0_COMPARATOR_OFFSET, clk_periods_per_kernel_tick);
+    // set interval
+
+    let enable_word: u64 = hpet.base_address.read_u64(GENERAL_CONFIG_OFFSET) | LEG_RT_CNF | ENABLE_CNF;
+    hpet.base_address.write_u64(GENERAL_CONFIG_OFFSET, enable_word);
+    // Enable interrupts from the HPET
+
+    true
 }
diff --git a/src/arch/x86_64/device/mod.rs b/src/arch/x86_64/device/mod.rs
index b7b9ceac..1678e739 100644
--- a/src/arch/x86_64/device/mod.rs
+++ b/src/arch/x86_64/device/mod.rs
@@ -16,9 +16,12 @@ pub unsafe fn init(active_table: &mut ActivePageTable){
 
 pub unsafe fn init_noncore() {
     {
-        if let Some(ref mut hpet) = *ACPI_TABLE.hpet.write() {
-            hpet::init(hpet);
+        let using_hpet = if let Some(ref mut hpet) = *ACPI_TABLE.hpet.write() {
+            hpet::init(hpet)
         } else {
+            false
+        };
+        if !using_hpet {
             pit::init();
         }
     }
-- 
GitLab