156 lines
3.3 KiB
C
156 lines
3.3 KiB
C
|
/* kern/i386/datetime.c - x86 CMOS datetime function.
|
||
|
*
|
||
|
* GRUB -- GRand Unified Bootloader
|
||
|
* Copyright (C) 2008 Free Software Foundation, Inc.
|
||
|
*
|
||
|
* GRUB 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.
|
||
|
*
|
||
|
* GRUB 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 GRUB. If not, see <http://www.gnu.org/licenses/>.
|
||
|
*/
|
||
|
|
||
|
#include <grub/lib/datetime.h>
|
||
|
#include <grub/i386/cmos.h>
|
||
|
|
||
|
grub_err_t
|
||
|
grub_get_datetime (struct grub_datetime *datetime)
|
||
|
{
|
||
|
int is_bcd, is_12hour;
|
||
|
grub_uint8_t value, flag;
|
||
|
|
||
|
flag = grub_cmos_read (GRUB_CMOS_INDEX_STATUS_B);
|
||
|
|
||
|
is_bcd = ! (flag & GRUB_CMOS_STATUS_B_BINARY);
|
||
|
|
||
|
value = grub_cmos_read (GRUB_CMOS_INDEX_YEAR);
|
||
|
if (is_bcd)
|
||
|
value = grub_bcd_to_num (value);
|
||
|
|
||
|
datetime->year = value;
|
||
|
datetime->year += (value < 80) ? 2000 : 1900;
|
||
|
|
||
|
value = grub_cmos_read (GRUB_CMOS_INDEX_MONTH);
|
||
|
if (is_bcd)
|
||
|
value = grub_bcd_to_num (value);
|
||
|
|
||
|
datetime->month = value;
|
||
|
|
||
|
value = grub_cmos_read (GRUB_CMOS_INDEX_DAY_OF_MONTH);
|
||
|
if (is_bcd)
|
||
|
value = grub_bcd_to_num (value);
|
||
|
|
||
|
datetime->day = value;
|
||
|
|
||
|
is_12hour = ! (flag & GRUB_CMOS_STATUS_B_24HOUR);
|
||
|
|
||
|
value = grub_cmos_read (GRUB_CMOS_INDEX_HOUR);
|
||
|
if (is_12hour)
|
||
|
{
|
||
|
is_12hour = (value & 0x80);
|
||
|
|
||
|
value &= 0x7F;
|
||
|
value--;
|
||
|
}
|
||
|
|
||
|
if (is_bcd)
|
||
|
value = grub_bcd_to_num (value);
|
||
|
|
||
|
if (is_12hour)
|
||
|
value += 12;
|
||
|
|
||
|
datetime->hour = value;
|
||
|
|
||
|
value = grub_cmos_read (GRUB_CMOS_INDEX_MINUTE);
|
||
|
if (is_bcd)
|
||
|
value = grub_bcd_to_num (value);
|
||
|
|
||
|
datetime->minute = value;
|
||
|
|
||
|
value = grub_cmos_read (GRUB_CMOS_INDEX_SECOND);
|
||
|
if (is_bcd)
|
||
|
value = grub_bcd_to_num (value);
|
||
|
|
||
|
datetime->second = value;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
grub_err_t
|
||
|
grub_set_datetime (struct grub_datetime *datetime)
|
||
|
{
|
||
|
int is_bcd, is_12hour;
|
||
|
grub_uint8_t value, flag;
|
||
|
|
||
|
flag = grub_cmos_read (GRUB_CMOS_INDEX_STATUS_B);
|
||
|
|
||
|
is_bcd = ! (flag & GRUB_CMOS_STATUS_B_BINARY);
|
||
|
|
||
|
value = ((datetime->year >= 2000) ? datetime->year - 2000 :
|
||
|
datetime->year - 1900);
|
||
|
|
||
|
if (is_bcd)
|
||
|
value = grub_num_to_bcd (value);
|
||
|
|
||
|
grub_cmos_write (GRUB_CMOS_INDEX_YEAR, value);
|
||
|
|
||
|
value = datetime->month;
|
||
|
|
||
|
if (is_bcd)
|
||
|
value = grub_num_to_bcd (value);
|
||
|
|
||
|
grub_cmos_write (GRUB_CMOS_INDEX_MONTH, value);
|
||
|
|
||
|
value = datetime->day;
|
||
|
|
||
|
if (is_bcd)
|
||
|
value = grub_num_to_bcd (value);
|
||
|
|
||
|
grub_cmos_write (GRUB_CMOS_INDEX_DAY_OF_MONTH, value);
|
||
|
|
||
|
value = datetime->hour;
|
||
|
|
||
|
is_12hour = (! (flag & GRUB_CMOS_STATUS_B_24HOUR));
|
||
|
|
||
|
if (is_12hour)
|
||
|
{
|
||
|
value++;
|
||
|
|
||
|
if (value > 12)
|
||
|
value -= 12;
|
||
|
else
|
||
|
is_12hour = 0;
|
||
|
}
|
||
|
|
||
|
if (is_bcd)
|
||
|
value = grub_num_to_bcd (value);
|
||
|
|
||
|
if (is_12hour)
|
||
|
value |= 0x80;
|
||
|
|
||
|
grub_cmos_write (GRUB_CMOS_INDEX_HOUR, value);
|
||
|
|
||
|
value = datetime->minute;
|
||
|
|
||
|
if (is_bcd)
|
||
|
value = grub_num_to_bcd (value);
|
||
|
|
||
|
grub_cmos_write (GRUB_CMOS_INDEX_MINUTE, value);
|
||
|
|
||
|
value = datetime->second;
|
||
|
|
||
|
if (is_bcd)
|
||
|
value = grub_num_to_bcd (value);
|
||
|
|
||
|
grub_cmos_write (GRUB_CMOS_INDEX_SECOND, value);
|
||
|
|
||
|
return 0;
|
||
|
}
|