Added real-time clock support written by Jeff Ward (for the kernel).
git-svn-id: file:///srv/svn/repos/haiku/trunk/current@5137 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
bdd02a033d
commit
2218f79ede
15
headers/private/kernel/arch/real_time_clock.h
Normal file
15
headers/private/kernel/arch/real_time_clock.h
Normal file
@ -0,0 +1,15 @@
|
||||
/*
|
||||
** Copyright 2003, Jeff Ward, jeff@r2d2.stcloudstate.edu. All rights reserved.
|
||||
** Distributed under the terms of the OpenBeOS License.
|
||||
*/
|
||||
#ifndef KERNEL_ARCH_REAL_TIME_CLOCK_H
|
||||
#define KERNEL_ARCH_REAL_TIME_CLOCK_H
|
||||
|
||||
#include <kernel.h>
|
||||
|
||||
void arch_rtc_set_hw_time(uint32 seconds);
|
||||
// Set HW clock to 'seconds' since 1/1/1970
|
||||
uint32 arch_rtc_get_hw_time(void);
|
||||
// Returns number of seconds since 1/1/1970 as stored in HW
|
||||
|
||||
#endif /* KERNEL_ARCH_REAL_TIME_CLOCK_H */
|
22
headers/private/kernel/real_time_clock.h
Normal file
22
headers/private/kernel/real_time_clock.h
Normal file
@ -0,0 +1,22 @@
|
||||
/*
|
||||
** Copyright 2003, Jeff Ward, jeff@r2d2.stcloudstate.edu. All rights reserved.
|
||||
** Distributed under the terms of the OpenBeOS License.
|
||||
*/
|
||||
#ifndef _KERNEL_REAL_TIME_CLOCK_H
|
||||
#define _KERNEL_REAL_TIME_CLOCK_H
|
||||
|
||||
|
||||
#include <KernelExport.h>
|
||||
|
||||
|
||||
int rtc_init(kernel_args *ka);
|
||||
// Initialize the real-time clock.
|
||||
|
||||
void rtc_set_system_time(uint32 current_time);
|
||||
// Set the system time. 'current_time' is the number of seconds elapsed since
|
||||
// Jan. 1, 1970.
|
||||
|
||||
bigtime_t rtc_boot_time(void);
|
||||
// Returns the time at which the system was booted in microseconds since Jan 1, 1970.
|
||||
|
||||
#endif /* _KERNEL_REAL_TIME_CLOCK_H */
|
@ -21,6 +21,7 @@ KernelMergeObject kernel_core.o :
|
||||
# <$(SOURCE_GRIST)>pools.c
|
||||
<$(SOURCE_GRIST)>port.c
|
||||
<$(SOURCE_GRIST)>queue.c
|
||||
<$(SOURCE_GRIST)>real_time_clock.c
|
||||
<$(SOURCE_GRIST)>scheduler.c
|
||||
<$(SOURCE_GRIST)>sem.c
|
||||
<$(SOURCE_GRIST)>signal.c
|
||||
|
@ -8,6 +8,7 @@ KernelStaticLibrary libx86 :
|
||||
<$(SOURCE_GRIST)>arch_faults.c
|
||||
<$(SOURCE_GRIST)>arch_int.c
|
||||
# <$(SOURCE_GRIST)>arch_selector.c
|
||||
<$(SOURCE_GRIST)>arch_real_time_clock.c
|
||||
<$(SOURCE_GRIST)>arch_smp.c
|
||||
<$(SOURCE_GRIST)>arch_thread.c
|
||||
<$(SOURCE_GRIST)>arch_timer.c
|
||||
|
264
src/kernel/core/arch/x86/arch_real_time_clock.c
Normal file
264
src/kernel/core/arch/x86/arch_real_time_clock.c
Normal file
@ -0,0 +1,264 @@
|
||||
/*
|
||||
** Copyright 2003, Jeff Ward, jeff@r2d2.stcloudstate.edu. All rights reserved.
|
||||
** Distributed under the terms of the OpenBeOS License.
|
||||
*/
|
||||
|
||||
|
||||
#include <arch/real_time_clock.h>
|
||||
#include <arch/cpu.h>
|
||||
|
||||
#define CMOS_ADDR_PORT 0x70
|
||||
#define CMOS_DATA_PORT 0x71
|
||||
#define BASE_YEAR 1970
|
||||
#define SECONDS_31 2678400
|
||||
#define SECONDS_30 2592000
|
||||
#define SECONDS_28 2419200
|
||||
#define SECONDS_DAY 86400
|
||||
|
||||
typedef struct {
|
||||
uint8 second;
|
||||
uint8 minute;
|
||||
uint8 hour;
|
||||
uint8 day;
|
||||
uint8 month;
|
||||
uint8 year;
|
||||
uint8 century;
|
||||
} cmos_time;
|
||||
|
||||
uint32 secs_per_month[12] = {SECONDS_31, SECONDS_28, SECONDS_31, SECONDS_30,
|
||||
SECONDS_31, SECONDS_30, SECONDS_31, SECONDS_31, SECONDS_30, SECONDS_31, SECONDS_30,
|
||||
SECONDS_31};
|
||||
|
||||
static uint32
|
||||
bcd_to_int(uint8 bcd) {
|
||||
uint32 numl;
|
||||
uint32 numh;
|
||||
|
||||
numl = bcd & 0x0f;
|
||||
numh = (bcd & 0xf0) >> 4;
|
||||
|
||||
return numh * 10 + numl;
|
||||
}
|
||||
|
||||
static uint8
|
||||
int_to_bcd(uint32 number) {
|
||||
uint8 low;
|
||||
uint8 high;
|
||||
|
||||
if (number > 99)
|
||||
return 0;
|
||||
|
||||
high = number / 10;
|
||||
low = number % 10;
|
||||
|
||||
return (high << 4) | low;
|
||||
}
|
||||
|
||||
static int
|
||||
leap_year(uint32 year) {
|
||||
if (year % 400 == 0)
|
||||
return 1;
|
||||
|
||||
if (year % 100 == 0)
|
||||
return 0;
|
||||
|
||||
if (year % 4 == 0)
|
||||
return 1;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
same_time(const cmos_time* time1, const cmos_time* time2) {
|
||||
return (time1->second == time2->second) &&
|
||||
(time1->minute == time2->minute) &&
|
||||
(time1->hour == time2->hour) &&
|
||||
(time1->day == time2->day) &&
|
||||
(time1->month == time2->month) &&
|
||||
(time1->year == time2->year) &&
|
||||
(time1->century == time2->century);
|
||||
}
|
||||
|
||||
static uint8
|
||||
cmos_read(uint8 addr) {
|
||||
int wait_time;
|
||||
|
||||
wait_time = 10000;
|
||||
|
||||
// Wait until bit 7 of Status Register A (indicating whether or not an update is in
|
||||
// progress) is clear if we are reading one of the clock data registers...
|
||||
if (addr < 0x0a) {
|
||||
out8(0x0a, CMOS_ADDR_PORT);
|
||||
while ( (in8(CMOS_DATA_PORT) & 0x80) && --wait_time );
|
||||
}
|
||||
|
||||
// then read the value.
|
||||
out8(addr, CMOS_ADDR_PORT);
|
||||
return in8(CMOS_DATA_PORT);
|
||||
}
|
||||
|
||||
static void
|
||||
cmos_write(uint8 addr, uint8 data) {
|
||||
out8(addr, CMOS_ADDR_PORT);
|
||||
out8(data, CMOS_DATA_PORT);
|
||||
}
|
||||
|
||||
static void
|
||||
set_24_hour_mode(void) {
|
||||
uint8 status_b;
|
||||
|
||||
status_b = cmos_read(0x0b);
|
||||
status_b |= 0x02;
|
||||
cmos_write(0x0b, status_b);
|
||||
}
|
||||
|
||||
static void
|
||||
read_cmos_clock(cmos_time* cmos) {
|
||||
set_24_hour_mode();
|
||||
|
||||
cmos->century = cmos_read(0x32);
|
||||
cmos->year = cmos_read(0x09);
|
||||
cmos->month = cmos_read(0x08);
|
||||
cmos->day = cmos_read(0x07);
|
||||
cmos->hour = cmos_read(0x04);
|
||||
cmos->minute = cmos_read(0x02);
|
||||
cmos->second = cmos_read(0x00);
|
||||
}
|
||||
|
||||
static void
|
||||
write_cmos_clock(cmos_time* cmos) {
|
||||
set_24_hour_mode();
|
||||
|
||||
cmos_write(0x32, cmos->century);
|
||||
cmos_write(0x09, cmos->year);
|
||||
cmos_write(0x08, cmos->month);
|
||||
cmos_write(0x07, cmos->day);
|
||||
cmos_write(0x04, cmos->hour);
|
||||
cmos_write(0x02, cmos->minute);
|
||||
cmos_write(0x00, cmos->second);
|
||||
}
|
||||
|
||||
static uint32
|
||||
cmos_to_secs(const cmos_time* cmos) {
|
||||
uint32 whole_year;
|
||||
uint32 time = 0;
|
||||
int i;
|
||||
|
||||
whole_year = bcd_to_int(cmos->century) * 100 + bcd_to_int(cmos->year);
|
||||
|
||||
// Add up the seconds from all years since 1970 that have elapsed.
|
||||
for (i = BASE_YEAR; i < whole_year; ++i) {
|
||||
if ( leap_year(i) )
|
||||
time += 31622400;
|
||||
|
||||
else
|
||||
time += 31536000;
|
||||
}
|
||||
|
||||
// Add up the seconds from all months passed this year.
|
||||
for (i = 0; i < bcd_to_int(cmos->month) - 1 && i < 12; ++i)
|
||||
time += secs_per_month[i];
|
||||
|
||||
// Add up the seconds from all days passed this month.
|
||||
if ( leap_year(whole_year) && bcd_to_int(cmos->month) > 2 )
|
||||
time += SECONDS_DAY;
|
||||
|
||||
time += ( bcd_to_int(cmos->day) - 1 ) * SECONDS_DAY;
|
||||
time += bcd_to_int(cmos->hour) * 3600;
|
||||
time += bcd_to_int(cmos->minute) * 60;
|
||||
time += bcd_to_int(cmos->second);
|
||||
|
||||
return time;
|
||||
}
|
||||
|
||||
static void
|
||||
secs_to_cmos(uint32 seconds, cmos_time* cmos) {
|
||||
uint32 whole_year = BASE_YEAR;
|
||||
uint32 secs_this_year;
|
||||
int i;
|
||||
int temp;
|
||||
int is_leapyear;
|
||||
int month;
|
||||
int keep_looping;
|
||||
|
||||
keep_looping = 1;
|
||||
|
||||
// Determine the current year by starting at 1970 and incrementing whole_year as long as
|
||||
// we can keep subtracting secs_this_year from seconds.
|
||||
while (keep_looping) {
|
||||
if ( leap_year(whole_year) )
|
||||
secs_this_year = 31622400;
|
||||
|
||||
else
|
||||
secs_this_year = 31536000;
|
||||
|
||||
if (seconds >= secs_this_year) {
|
||||
seconds -= secs_this_year;
|
||||
++whole_year;
|
||||
}
|
||||
|
||||
else
|
||||
keep_looping = 0;
|
||||
}
|
||||
|
||||
cmos->century = int_to_bcd(whole_year / 100);
|
||||
cmos->year = int_to_bcd(whole_year % 100);
|
||||
|
||||
// Determine the current month
|
||||
month = 1;
|
||||
is_leapyear = leap_year(whole_year);
|
||||
do {
|
||||
temp = seconds - secs_per_month[month - 1];
|
||||
|
||||
if (is_leapyear && month == 2)
|
||||
temp -= SECONDS_DAY;
|
||||
|
||||
if (temp >= 0) {
|
||||
seconds = temp;
|
||||
++month;
|
||||
}
|
||||
} while (temp >= 0 && month < 13);
|
||||
|
||||
cmos->month = int_to_bcd(month);
|
||||
|
||||
cmos->day = int_to_bcd(seconds / SECONDS_DAY + 1);
|
||||
seconds = seconds % SECONDS_DAY;
|
||||
|
||||
cmos->hour = int_to_bcd(seconds / 3600);
|
||||
seconds = seconds % 3600;
|
||||
|
||||
cmos->minute = int_to_bcd(seconds / 60);
|
||||
seconds = seconds % 60;
|
||||
|
||||
cmos->second = int_to_bcd(seconds);
|
||||
}
|
||||
|
||||
uint32
|
||||
arch_rtc_get_hw_time(void) {
|
||||
int wait_time;
|
||||
cmos_time cmos1;
|
||||
cmos_time cmos2;
|
||||
|
||||
wait_time = 1000;
|
||||
|
||||
// We will read the clock twice and make sure both reads are equal. This will prevent
|
||||
// problems that would occur if the clock is read during an update (e.g. if we read the hour
|
||||
// at 8:59:59, the clock gets changed, and then we read the minute and second, we would
|
||||
// be off by a whole hour)
|
||||
do {
|
||||
read_cmos_clock(&cmos1);
|
||||
read_cmos_clock(&cmos2);
|
||||
} while ( !same_time(&cmos1, &cmos2) && --wait_time );
|
||||
|
||||
// Convert the CMOS data to seconds since 1970.
|
||||
return cmos_to_secs(&cmos1);
|
||||
}
|
||||
|
||||
void
|
||||
arch_rtc_set_hw_time(uint32 seconds) {
|
||||
cmos_time cmos;
|
||||
uint32 read_back;
|
||||
|
||||
secs_to_cmos(seconds, &cmos);
|
||||
write_cmos_clock(&cmos);
|
||||
}
|
@ -28,6 +28,7 @@
|
||||
#include <bus.h>
|
||||
#include <kmodule.h>
|
||||
#include <int.h>
|
||||
#include <real_time_clock.h>
|
||||
#include <kernel_daemon.h>
|
||||
|
||||
#include <string.h>
|
||||
@ -83,6 +84,7 @@ _start(kernel_args *oldka, int cpu_num)
|
||||
|
||||
faults_init(&ka);
|
||||
smp_init(&ka);
|
||||
rtc_init(&ka);
|
||||
timer_init(&ka);
|
||||
|
||||
arch_cpu_init2(&ka);
|
||||
|
84
src/kernel/core/real_time_clock.c
Normal file
84
src/kernel/core/real_time_clock.c
Normal file
@ -0,0 +1,84 @@
|
||||
/*
|
||||
** Copyright 2003, Jeff Ward, jeff@r2d2.stcloudstate.edu. All rights reserved.
|
||||
** Distributed under the terms of the OpenBeOS License.
|
||||
*/
|
||||
|
||||
|
||||
#include <OS.h>
|
||||
#include <debug.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#include <arch/real_time_clock.h>
|
||||
#include <real_time_clock.h>
|
||||
|
||||
static bigtime_t boot_time;
|
||||
|
||||
static void
|
||||
rtc_print(void) {
|
||||
uint32 current_time;
|
||||
|
||||
current_time = (boot_time + system_time()) / 1000000;
|
||||
dprintf("system_time: %u\n", (unsigned)system_time());
|
||||
dprintf("boot_time: %u\n", (unsigned)boot_time);
|
||||
dprintf("current_time: %u\n", (unsigned)current_time);
|
||||
}
|
||||
|
||||
static int
|
||||
rtc_debug(int argc, char** argv) {
|
||||
// If no arguments were given, output all usefull data.
|
||||
if (argc < 2)
|
||||
rtc_print();
|
||||
|
||||
// If there was an argument, reset the system and hw time.
|
||||
else {
|
||||
rtc_set_system_time( strtoul(argv[1], NULL, 10) );
|
||||
rtc_system_to_hw();
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int
|
||||
rtc_init(kernel_args *ka) {
|
||||
dprintf("rtc_init: entry\n");
|
||||
add_debugger_command("rtc", &rtc_debug, "Set and test the real-time clock");
|
||||
|
||||
rtc_hw_to_system();
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void
|
||||
rtc_set_system_time(uint32 current_time) {
|
||||
uint64 useconds;
|
||||
|
||||
useconds = (uint64)current_time * 1000000;
|
||||
boot_time = useconds - system_time();
|
||||
}
|
||||
|
||||
|
||||
/** Write the system time to CMOS. */
|
||||
|
||||
void
|
||||
rtc_system_to_hw(void) {
|
||||
uint32 seconds;
|
||||
|
||||
seconds = (boot_time + system_time()) / 1000000;
|
||||
arch_rtc_set_hw_time(seconds);
|
||||
}
|
||||
|
||||
|
||||
/** Read the CMOS clock and update the system time accordingly. */
|
||||
|
||||
void
|
||||
rtc_hw_to_system(void) {
|
||||
uint32 current_time;
|
||||
|
||||
current_time = arch_rtc_get_hw_time();
|
||||
rtc_set_system_time(current_time);
|
||||
}
|
||||
|
||||
bigtime_t
|
||||
rtc_boot_time(void) {
|
||||
return boot_time;
|
||||
}
|
Loading…
Reference in New Issue
Block a user