blob: 3f083050cf891bb9a5480c833e7c35208e9392b9 [file] [log] [blame]
// SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later
/* Copyright 2013-2017 IBM Corp. */
#include <skiboot.h>
#include <opal.h>
#include <mem_region.h>
#include <device.h>
#include <timebase.h>
#include <time-utils.h>
#include <lock.h>
/* timebase when tm_offset was assigned */
static unsigned long tb_synctime;
/*
* Absolute time that was last assigned.
* Current rtc value is calculated from this.
*/
static struct tm tm_offset;
/* protects tm_offset & tb_synctime */
static struct lock emulation_lock;
static int64_t fake_rtc_write(uint32_t ymd, uint64_t hmsm)
{
lock(&emulation_lock);
datetime_to_tm(ymd, hmsm, &tm_offset);
tb_synctime = mftb();
unlock(&emulation_lock);
return OPAL_SUCCESS;
}
static int64_t fake_rtc_read(__be32 *__ymd, __be64 *__hmsm)
{
time_t sec;
struct tm tm_calculated;
uint32_t ymd;
uint64_t hmsm;
if (!__ymd || !__hmsm)
return OPAL_PARAMETER;
/* Compute the emulated clock value */
lock(&emulation_lock);
sec = tb_to_secs(mftb() - tb_synctime) + mktime(&tm_offset);
gmtime_r(&sec, &tm_calculated);
tm_to_datetime(&tm_calculated, &ymd, &hmsm);
unlock(&emulation_lock);
*__ymd = cpu_to_be32(ymd);
*__hmsm = cpu_to_be64(hmsm);
return OPAL_SUCCESS;
}
void fake_rtc_init(void)
{
struct mem_region *rtc_region = NULL;
uint32_t *rtc = NULL, *fake_ymd;
uint64_t *fake_hmsm;
struct dt_node *np;
/* Read initial values from reserved memory */
rtc_region = find_mem_region("ibm,fake-rtc");
/* Should we register anyway? */
if (!rtc_region) {
prlog(PR_TRACE, "No initial RTC value found\n");
return;
}
init_lock(&emulation_lock);
/* Fetch the initial rtc values */
rtc = (uint32_t *) rtc_region->start;
fake_ymd = rtc;
fake_hmsm = ((uint64_t *) &rtc[1]);
fake_rtc_write(*fake_ymd, *fake_hmsm);
/* Register opal calls */
opal_register(OPAL_RTC_READ, fake_rtc_read, 2);
opal_register(OPAL_RTC_WRITE, fake_rtc_write, 2);
/* add the fake rtc dt node */
np = dt_new(opal_node, "rtc");
dt_add_property_strings(np, "compatible", "ibm,opal-rtc");
prlog(PR_TRACE, "Init fake RTC to Date:%d-%d-%d Time:%d-%d-%d\n",
tm_offset.tm_mon, tm_offset.tm_mday, tm_offset.tm_year,
tm_offset.tm_hour, tm_offset.tm_min, tm_offset.tm_sec);
}