Hi,
dream21 wrote:I looked at the RTC and CMOS tutorial and made a working implementation. My code is just a copy and goes like this.
This code:
Code: Select all
u8 read_register(int reg)
{
outb(reg, cmos_address);
return inb(cmos_data);
}
needs some kind of protection around it, in case an IRQ or task switch happens causing something else to change "cmos_address" after it's been set but before "cmos_data" has been read.
This code:
Code: Select all
int get_update_in_progress_flag()
{
outb(0x0A, cmos_address);
return (inb(cmos_data) & 0x80);
}
should use the "read_register()" function above instead for reading the register itself.
This code:
Code: Select all
time_t rtc_handler()
{
bool ready = read_register(0x0C) & 0x10;
if (ready)
{
{
current_time.second = read_register(0x00);
current_time.minute = read_register(0x02);
current_time.hour = read_register(0x04);
current_time.month = read_register(0x08);
current_time.year = read_register(0x09);
current_time.day_of_week = read_register(0x06);
current_time.day_of_month = read_register(0x07);
current_time.century = read_register(0x32);
}
}
return current_time;
}
is extremely slow (every "read_register()" costs several thousand cycles of CPU time due to slow IO ports) and shouldn't be used anyway.
Typically an OS gets the current time from the RTC once during boot and convert it into some kind of "time since epoch" format (e.g. "seconds since 1970 in UTC", or "nanoseconds since 2000 in UTC" or ...), and then only does a "time_since_epoch += time_between_ticks" (e.g. when the update IRQ occurs) after that.
This code:
Code: Select all
void rtc_install(void)
{
unsigned char status;
status = read_register(0x0B);
status |= 0x02; // 24 hour clock
status |= 0x10; // update ended interrupts
status &= ~0x20; // no alarm interrupts
status &= ~ 0x40; // no periodic interrupts
bcd = !(status & 0x04); // check if data type is BCD
write_register(0x0B, status);
read_register(0x0C);
//register_interrupt_handler(40, &rtc_handler);
}
is broken. You can't change the way the RTC's time/date registers are configured (e.g. change them from 12-hour to 24-hour) without breaking the BIOS and ruining everything. Unlike most hardware, the RTC is battery backed and remembers how its time/date registers are configured even when the computer is turned off, which means that if you change it then turn the computer off, the next time the computer is turned on the BIOS sees all kinds of unexpected data in the time/date registers because the RTC was left in a state that the BIOS doesn't expect and doesn't support.
Instead; you need to leave the time/date register configuration "as is", and read the pre-existing configuration to figure out how to decode the time/date register values (e.g. convert from BCD to binary if it was using BCD, and convert from 12-hour to 24-hour if it was using 12-hour; so that you end up with "binary 24-hour" regardless of how the time/date registers are configured).
Note: If you do change the way the RTC's time/date registers are configured; then you also have to write corrected values into those registers. For example, if the RTC was configured as "12-hour BCD" and the "hour" register contains the value 0x82 (meaning 2 pm) and you change it to "24-hour BCD", then the "hour" register will still contain the value 0x82 (meaning 82 o'clock, which is pure nonsense).
dream21 wrote:Also I have to give a certain amount of delay between rtc_install and printing the values. What could be the reason for it?
I have no idea - there's no code to print any of the values. Note that it should take up to 1 second before you get your first "update interrupt" (and if you're using an emulator, in some cases "1 virtual second" could take "10 actual seconds").
Cheers,
Brendan