245 lines
6.4 KiB
C
245 lines
6.4 KiB
C
/*
|
|
* Copyright (c) 2020 AIIT XUOS Lab
|
|
* XiUOS is licensed under Mulan PSL v2.
|
|
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
|
* You may obtain a copy of Mulan PSL v2 at:
|
|
* http://license.coscl.org.cn/MulanPSL2
|
|
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
|
|
* EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
|
|
* MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
|
|
* See the Mulan PSL v2 for more details.
|
|
*/
|
|
|
|
/**
|
|
* @file drv_rtc.c
|
|
* @brief register rtc drv function using bus driver framework
|
|
* @version 1.0
|
|
* @author AIIT XUOS Lab
|
|
* @date 2021-04-24
|
|
*/
|
|
|
|
#include <bus_rtc.h>
|
|
#include <dev_rtc.h>
|
|
|
|
static DoubleLinklistType rtcdrv_linklist;
|
|
|
|
/*Create the driver linklist*/
|
|
static void RtcDrvLinkInit()
|
|
{
|
|
InitDoubleLinkList(&rtcdrv_linklist);
|
|
}
|
|
|
|
DriverType RtcDriverFind(const char *drv_name, enum DriverType drv_type)
|
|
{
|
|
NULL_PARAM_CHECK(drv_name);
|
|
|
|
struct Driver *driver = NONE;
|
|
|
|
DoubleLinklistType *node = NONE;
|
|
DoubleLinklistType *head = &rtcdrv_linklist;
|
|
|
|
for (node = head->node_next; node != head; node = node->node_next) {
|
|
driver = SYS_DOUBLE_LINKLIST_ENTRY(node, struct Driver, driver_link);
|
|
if ((!strcmp(driver->drv_name, drv_name)) && (drv_type == driver->driver_type)) {
|
|
return driver;
|
|
}
|
|
}
|
|
|
|
KPrintf("RtcDriverFind cannot find the %s driver.return NULL\n", drv_name);
|
|
return NONE;
|
|
}
|
|
|
|
int RtcDriverRegister(struct Driver *driver)
|
|
{
|
|
NULL_PARAM_CHECK(driver);
|
|
|
|
x_err_t ret = EOK;
|
|
static x_bool driver_link_flag = RET_FALSE;
|
|
|
|
if (!driver_link_flag) {
|
|
RtcDrvLinkInit();
|
|
driver_link_flag = RET_TRUE;
|
|
}
|
|
|
|
DoubleLinkListInsertNodeAfter(&rtcdrv_linklist, &(driver->driver_link));
|
|
|
|
return ret;
|
|
}
|
|
|
|
int RtcDrvSetFunction(char *driver_name, struct RtcSetParam *rtc_set_param)
|
|
{
|
|
NULL_PARAM_CHECK(driver_name);
|
|
|
|
x_err_t ret = EOK;
|
|
time_t now;
|
|
struct tm *tm_now;
|
|
struct tm tm_tmp;
|
|
x_base lock;
|
|
|
|
struct Driver *driver;
|
|
struct BusConfigureInfo configure_info;
|
|
struct RtcDrvConfigureParam drv_param;
|
|
configure_info.private_data = &drv_param;
|
|
|
|
driver = RtcDriverFind(driver_name, TYPE_RTC_DRV);
|
|
if (NONE == driver) {
|
|
KPrintf("RtcDrvSetFunction find rtc driver %s error\n", driver_name);
|
|
return ERROR;
|
|
}
|
|
|
|
if (OPER_RTC_SET_TIME == rtc_set_param->rtc_set_cmd) {
|
|
now = time(NONE);
|
|
lock = CriticalAreaLock();
|
|
tm_now = localtime(&now);
|
|
memcpy(&tm_tmp, tm_now, sizeof(struct tm));
|
|
CriticalAreaUnLock(lock);
|
|
|
|
tm_tmp.tm_year = rtc_set_param->date_param.year - 1900;
|
|
tm_tmp.tm_mon = rtc_set_param->date_param.month - 1;
|
|
tm_tmp.tm_mday = rtc_set_param->date_param.day;
|
|
tm_tmp.tm_hour = rtc_set_param->time_param.hour;
|
|
tm_tmp.tm_min = rtc_set_param->time_param.minute;
|
|
tm_tmp.tm_sec = rtc_set_param->time_param.second;
|
|
|
|
now = mktime(&tm_tmp);
|
|
|
|
drv_param.rtc_operation_cmd = OPER_RTC_SET_TIME;
|
|
drv_param.time = &now;
|
|
|
|
ret = driver->configure(driver, &configure_info);
|
|
if (EOK != ret) {
|
|
KPrintf("RtcDrvSetFunction set time error %d\n", ret);
|
|
return ret;
|
|
}
|
|
} else if (OPER_RTC_GET_TIME == rtc_set_param->rtc_set_cmd) {
|
|
drv_param.rtc_operation_cmd = OPER_RTC_GET_TIME;
|
|
drv_param.time = rtc_set_param->time;
|
|
ret = driver->configure(driver, &configure_info);
|
|
if (EOK != ret) {
|
|
KPrintf("RtcDrvSetFunction set time error %d\n", ret);
|
|
return ret;
|
|
}
|
|
}
|
|
return EOK;
|
|
}
|
|
|
|
#ifdef USING_SOFT_RTC
|
|
|
|
static x_ticks_t SoftRtc_InitTick;
|
|
static time_t SoftRtc_InitTime;
|
|
|
|
static int SoftRtcInitTime(struct tm *time, struct RtcDateParam *date_param, struct RtcTimeParam *time_param)
|
|
{
|
|
NULL_PARAM_CHECK(time);
|
|
|
|
time->tm_year = date_param->year - 1900;
|
|
time->tm_mon = date_param->month - 1;
|
|
time->tm_mday = date_param->day;
|
|
time->tm_hour = time_param->hour;
|
|
time->tm_min = time_param->minute;
|
|
time->tm_sec = time_param->second;
|
|
|
|
return EOK;
|
|
}
|
|
|
|
static uint32 SoftRtcConfigure(void *drv, struct BusConfigureInfo *configure_info)
|
|
{
|
|
NULL_PARAM_CHECK(drv);
|
|
|
|
struct RtcDriver *rtc_drv = (struct RtcDriver *)drv;
|
|
struct RtcDrvConfigureParam *drv_param = (struct RtcDrvConfigureParam *)configure_info->private_data;
|
|
|
|
int cmd = drv_param->rtc_operation_cmd;
|
|
time_t *time = drv_param->time;
|
|
|
|
switch(cmd)
|
|
{
|
|
case OPER_RTC_GET_TIME:
|
|
{
|
|
*time = SoftRtc_InitTime + (CurrentTicksGain() - SoftRtc_InitTick) / TICK_PER_SECOND;
|
|
break;
|
|
}
|
|
case OPER_RTC_SET_TIME:
|
|
{
|
|
SoftRtc_InitTime = *time - (CurrentTicksGain() - SoftRtc_InitTick) / TICK_PER_SECOND;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return EOK;
|
|
}
|
|
|
|
static struct RtcDateParam date_param =
|
|
{
|
|
.year = 2021,
|
|
.month = 1,
|
|
.day = 1,
|
|
};
|
|
|
|
static struct RtcTimeParam time_param =
|
|
{
|
|
.hour = 0,
|
|
.minute = 0,
|
|
.second = 0,
|
|
};
|
|
|
|
static int SoftRtcBusInit(struct RtcBus *softrtc_bus, struct RtcDriver *softrtc_driver)
|
|
{
|
|
x_err_t ret = EOK;
|
|
|
|
/*Init the soft rtc bus */
|
|
ret = RtcBusInit(softrtc_bus, SOFT_RTC_BUS_NAME);
|
|
if(EOK != ret)
|
|
{
|
|
KPrintf("SoftRtcBusInit RtcBusInit error %d\n", ret);
|
|
return ERROR;
|
|
}
|
|
|
|
/*Init the soft rtc driver*/
|
|
ret = RtcDriverInit(softrtc_driver, SOFT_RTC_DRV_NAME);
|
|
if(EOK != ret)
|
|
{
|
|
KPrintf("SoftRtcBusInit RtcDriverInit error %d\n", ret);
|
|
return ERROR;
|
|
}
|
|
|
|
/*Attach the soft rtc driver to the soft rtc bus*/
|
|
ret = RtcDriverAttachToBus(SOFT_RTC_DRV_NAME, SOFT_RTC_BUS_NAME);
|
|
if(EOK != ret)
|
|
{
|
|
KPrintf("SoftRtcBusInit RtcDriverAttachToBus error %d\n", ret);
|
|
return ERROR;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
int SoftRtcInit(void)
|
|
{
|
|
x_err_t ret = EOK;
|
|
struct tm time;
|
|
|
|
SoftRtcInitTime(&time, &date_param, &time_param);
|
|
|
|
static struct RtcBus softrtc_bus;
|
|
memset(&softrtc_bus, 0, sizeof(struct RtcBus));
|
|
|
|
static struct RtcDriver softrtc_driver;
|
|
memset(&softrtc_driver, 0, sizeof(struct RtcDriver));
|
|
|
|
softrtc_driver.configure = &(SoftRtcConfigure);
|
|
|
|
ret = SoftRtcBusInit(&softrtc_bus, &softrtc_driver);
|
|
if (EOK != ret) {
|
|
KPrintf("SoftRtcInit error ret %u\n", ret);
|
|
return ERROR;
|
|
}
|
|
|
|
SoftRtc_InitTick = CurrentTicksGain();
|
|
SoftRtc_InitTime = mktime(&time);
|
|
|
|
return ret;
|
|
}
|
|
|
|
#endif
|