mirror of
https://github.com/Fennix-Project/Kernel.git
synced 2025-05-25 22:14:37 +00:00
209 lines
4.2 KiB
C++
209 lines
4.2 KiB
C++
/*
|
|
This file is part of Fennix Kernel.
|
|
|
|
Fennix Kernel is free software: you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License as
|
|
published by the Free Software Foundation, either version 3 of
|
|
the License, or (at your option) any later version.
|
|
|
|
Fennix Kernel is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with Fennix Kernel. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <time.hpp>
|
|
|
|
#include <memory.hpp>
|
|
#include <acpi.hpp>
|
|
#include <debug.h>
|
|
#include <io.h>
|
|
|
|
|
|
#include "../../kernel.h"
|
|
|
|
namespace Time
|
|
{
|
|
bool time::Sleep(size_t Duration, Units Unit)
|
|
{
|
|
switch (ActiveTimer)
|
|
{
|
|
case NONE:
|
|
error("No timer is active");
|
|
return false;
|
|
case RTC:
|
|
fixme("RTC sleep not implemented");
|
|
return false;
|
|
case PIT:
|
|
fixme("PIT sleep not implemented");
|
|
return false;
|
|
case HPET:
|
|
return this->hpet->Sleep(Duration, Unit);
|
|
case ACPI:
|
|
fixme("ACPI sleep not implemented");
|
|
return false;
|
|
case APIC:
|
|
fixme("APIC sleep not implemented");
|
|
return false;
|
|
case TSC:
|
|
return this->tsc->Sleep(Duration, Unit);
|
|
default:
|
|
error("Unknown timer");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
uint64_t time::GetCounter()
|
|
{
|
|
switch (ActiveTimer)
|
|
{
|
|
case NONE:
|
|
error("No timer is active");
|
|
return 0;
|
|
case RTC:
|
|
fixme("RTC sleep not implemented");
|
|
return 0;
|
|
case PIT:
|
|
fixme("PIT sleep not implemented");
|
|
return 0;
|
|
case HPET:
|
|
return this->hpet->GetCounter();
|
|
case ACPI:
|
|
fixme("ACPI sleep not implemented");
|
|
return 0;
|
|
case APIC:
|
|
fixme("APIC sleep not implemented");
|
|
return 0;
|
|
case TSC:
|
|
return this->tsc->GetCounter();
|
|
default:
|
|
error("Unknown timer");
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
uint64_t time::CalculateTarget(uint64_t Target, Units Unit)
|
|
{
|
|
switch (ActiveTimer)
|
|
{
|
|
case NONE:
|
|
error("No timer is active");
|
|
return 0;
|
|
case RTC:
|
|
fixme("RTC sleep not implemented");
|
|
return 0;
|
|
case PIT:
|
|
fixme("PIT sleep not implemented");
|
|
return 0;
|
|
case HPET:
|
|
return this->hpet->CalculateTarget(Target, Unit);
|
|
case ACPI:
|
|
fixme("ACPI sleep not implemented");
|
|
return 0;
|
|
case APIC:
|
|
fixme("APIC sleep not implemented");
|
|
return 0;
|
|
case TSC:
|
|
return this->tsc->CalculateTarget(Target, Unit);
|
|
default:
|
|
error("Unknown timer");
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
uint64_t time::GetNanosecondsSinceClassCreation()
|
|
{
|
|
switch (ActiveTimer)
|
|
{
|
|
case NONE:
|
|
error("No timer is active");
|
|
return 0;
|
|
case RTC:
|
|
fixme("RTC sleep not implemented");
|
|
return 0;
|
|
case PIT:
|
|
fixme("PIT sleep not implemented");
|
|
return 0;
|
|
case HPET:
|
|
return this->hpet->GetNanosecondsSinceClassCreation();
|
|
case ACPI:
|
|
fixme("ACPI sleep not implemented");
|
|
return 0;
|
|
case APIC:
|
|
fixme("APIC sleep not implemented");
|
|
return 0;
|
|
case TSC:
|
|
return this->tsc->GetNanosecondsSinceClassCreation();
|
|
default:
|
|
error("Unknown timer");
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
void time::FindTimers(void *acpi)
|
|
{
|
|
#if defined(a86)
|
|
/* TODO: RTC check */
|
|
/* TODO: PIT check */
|
|
|
|
if (acpi)
|
|
{
|
|
if (((ACPI::ACPI *)acpi)->HPET)
|
|
{
|
|
hpet = new HighPrecisionEventTimer(((ACPI::ACPI *)acpi)->HPET);
|
|
ActiveTimer = HPET;
|
|
SupportedTimers |= HPET;
|
|
KPrint("\e11FF11HPET found");
|
|
}
|
|
else
|
|
{
|
|
KPrint("\eFF2200HPET not found");
|
|
}
|
|
|
|
/* TODO: ACPI check */
|
|
/* TODO: APIC check */
|
|
}
|
|
else
|
|
{
|
|
KPrint("\eFF2200ACPI not found");
|
|
}
|
|
|
|
bool TSCInvariant = false;
|
|
if (strcmp(CPU::Vendor(), x86_CPUID_VENDOR_AMD) == 0)
|
|
{
|
|
CPU::x86::AMD::CPUID0x80000007 cpuid80000007;
|
|
if (cpuid80000007.EDX.TscInvariant)
|
|
TSCInvariant = true;
|
|
}
|
|
else if (strcmp(CPU::Vendor(), x86_CPUID_VENDOR_INTEL) == 0)
|
|
{
|
|
// TODO: Intel 0x80000007
|
|
CPU::x86::AMD::CPUID0x80000007 cpuid80000007;
|
|
if (cpuid80000007.EDX.TscInvariant)
|
|
TSCInvariant = true;
|
|
}
|
|
|
|
if (TSCInvariant)
|
|
{
|
|
tsc = new TimeStampCounter;
|
|
// FIXME: ActiveTimer = TSC;
|
|
SupportedTimers |= TSC;
|
|
KPrint("\e11FF11Invariant TSC found");
|
|
}
|
|
else
|
|
KPrint("\eFF2200TSC is not invariant");
|
|
#endif
|
|
}
|
|
|
|
time::time()
|
|
{
|
|
}
|
|
|
|
time::~time()
|
|
{
|
|
}
|
|
}
|