/* kern/cmos_datetime.c - CMOS datetime function. * * VAS_EBOOT -- GRand Unified Bootloader * Copyright (C) 2008,2009 Free Software Foundation, Inc. * * VAS_EBOOT 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. * * VAS_EBOOT 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 VAS_EBOOT. If not, see . */ #include #include #include #include #if defined (__powerpc__) || defined (__sparc__) #include #endif VAS_EBOOT_MOD_LICENSE ("GPLv3+"); static char *rtc = 0; static int no_ieee1275_rtc = 0; /* Helper for find_rtc. */ static int find_rtc_iter (struct VasEBoot_ieee1275_devalias *alias) { if (VasEBoot_strcmp (alias->type, "rtc") == 0) { VasEBoot_dprintf ("datetime", "Found RTC %s\n", alias->path); rtc = VasEBoot_strdup (alias->path); return 1; } return 0; } static void find_rtc (void) { VasEBoot_ieee1275_devices_iterate (find_rtc_iter); if (!rtc) no_ieee1275_rtc = 1; } VasEBoot_err_t VasEBoot_get_datetime (struct VasEBoot_datetime *datetime) { struct get_time_args { struct VasEBoot_ieee1275_common_hdr common; VasEBoot_ieee1275_cell_t method; VasEBoot_ieee1275_cell_t device; VasEBoot_ieee1275_cell_t catch_result; VasEBoot_ieee1275_cell_t year; VasEBoot_ieee1275_cell_t month; VasEBoot_ieee1275_cell_t day; VasEBoot_ieee1275_cell_t hour; VasEBoot_ieee1275_cell_t minute; VasEBoot_ieee1275_cell_t second; } args; int status; VasEBoot_ieee1275_ihandle_t ihandle; if (no_ieee1275_rtc) return VasEBoot_get_datetime_cmos (datetime); if (!rtc) find_rtc (); if (!rtc) return VasEBoot_get_datetime_cmos (datetime); status = VasEBoot_ieee1275_open (rtc, &ihandle); if (status == -1) return VasEBoot_error (VAS_EBOOT_ERR_IO, "couldn't open RTC"); INIT_IEEE1275_COMMON (&args.common, "call-method", 2, 7); args.device = (VasEBoot_ieee1275_cell_t) ihandle; args.method = (VasEBoot_ieee1275_cell_t) "get-time"; status = IEEE1275_CALL_ENTRY_FN (&args); VasEBoot_ieee1275_close (ihandle); if (status == -1 || args.catch_result) return VasEBoot_error (VAS_EBOOT_ERR_IO, "get-time failed"); datetime->year = args.year; datetime->month = args.month; datetime->day = args.day; datetime->hour = args.hour; datetime->minute = args.minute; datetime->second = args.second; return VAS_EBOOT_ERR_NONE; } VasEBoot_err_t VasEBoot_set_datetime (struct VasEBoot_datetime *datetime) { struct set_time_args { struct VasEBoot_ieee1275_common_hdr common; VasEBoot_ieee1275_cell_t method; VasEBoot_ieee1275_cell_t device; VasEBoot_ieee1275_cell_t year; VasEBoot_ieee1275_cell_t month; VasEBoot_ieee1275_cell_t day; VasEBoot_ieee1275_cell_t hour; VasEBoot_ieee1275_cell_t minute; VasEBoot_ieee1275_cell_t second; VasEBoot_ieee1275_cell_t catch_result; } args; int status; VasEBoot_ieee1275_ihandle_t ihandle; if (no_ieee1275_rtc) return VasEBoot_set_datetime_cmos (datetime); if (!rtc) find_rtc (); if (!rtc) return VasEBoot_set_datetime_cmos (datetime); status = VasEBoot_ieee1275_open (rtc, &ihandle); if (status == -1) return VasEBoot_error (VAS_EBOOT_ERR_IO, "couldn't open RTC"); INIT_IEEE1275_COMMON (&args.common, "call-method", 8, 1); args.device = (VasEBoot_ieee1275_cell_t) ihandle; args.method = (VasEBoot_ieee1275_cell_t) "set-time"; args.year = datetime->year; args.month = datetime->month; args.day = datetime->day; args.hour = datetime->hour; args.minute = datetime->minute; args.second = datetime->second; status = IEEE1275_CALL_ENTRY_FN (&args); VasEBoot_ieee1275_close (ihandle); if (status == -1 || args.catch_result) return VasEBoot_error (VAS_EBOOT_ERR_IO, "set-time failed"); return VAS_EBOOT_ERR_NONE; }