1*83d290c5STom Rini // SPDX-License-Identifier: GPL-2.0+
258714166SSimon Glass /*
358714166SSimon Glass * (C) Copyright 2015 Google, Inc
458714166SSimon Glass * Written by Simon Glass <sjg@chromium.org>
558714166SSimon Glass */
658714166SSimon Glass
758714166SSimon Glass #include <common.h>
858714166SSimon Glass #include <dm.h>
958714166SSimon Glass #include <i2c.h>
1058714166SSimon Glass #include <rtc.h>
1158714166SSimon Glass #include <asm/rtc.h>
1258714166SSimon Glass
1358714166SSimon Glass #define REG_COUNT 0x80
1458714166SSimon Glass
sandbox_rtc_get(struct udevice * dev,struct rtc_time * time)1558714166SSimon Glass static int sandbox_rtc_get(struct udevice *dev, struct rtc_time *time)
1658714166SSimon Glass {
1758714166SSimon Glass time->tm_sec = dm_i2c_reg_read(dev, REG_SEC);
1858714166SSimon Glass if (time->tm_sec < 0)
1958714166SSimon Glass return time->tm_sec;
2058714166SSimon Glass time->tm_min = dm_i2c_reg_read(dev, REG_MIN);
2158714166SSimon Glass if (time->tm_min < 0)
2258714166SSimon Glass return time->tm_min;
2358714166SSimon Glass time->tm_hour = dm_i2c_reg_read(dev, REG_HOUR);
2458714166SSimon Glass if (time->tm_hour < 0)
2558714166SSimon Glass return time->tm_hour;
2658714166SSimon Glass time->tm_mday = dm_i2c_reg_read(dev, REG_MDAY);
2758714166SSimon Glass if (time->tm_mday < 0)
2858714166SSimon Glass return time->tm_mday;
2958714166SSimon Glass time->tm_mon = dm_i2c_reg_read(dev, REG_MON);
3058714166SSimon Glass if (time->tm_mon < 0)
3158714166SSimon Glass return time->tm_mon;
3258714166SSimon Glass time->tm_year = dm_i2c_reg_read(dev, REG_YEAR);
3358714166SSimon Glass if (time->tm_year < 0)
3458714166SSimon Glass return time->tm_year;
3558714166SSimon Glass time->tm_year += 1900;
3658714166SSimon Glass time->tm_wday = dm_i2c_reg_read(dev, REG_WDAY);
3758714166SSimon Glass if (time->tm_wday < 0)
3858714166SSimon Glass return time->tm_wday;
3958714166SSimon Glass
4058714166SSimon Glass return 0;
4158714166SSimon Glass }
4258714166SSimon Glass
sandbox_rtc_set(struct udevice * dev,const struct rtc_time * time)4358714166SSimon Glass static int sandbox_rtc_set(struct udevice *dev, const struct rtc_time *time)
4458714166SSimon Glass {
4558714166SSimon Glass int ret;
4658714166SSimon Glass
4758714166SSimon Glass ret = dm_i2c_reg_write(dev, REG_SEC, time->tm_sec);
4858714166SSimon Glass if (ret < 0)
4958714166SSimon Glass return ret;
5058714166SSimon Glass ret = dm_i2c_reg_write(dev, REG_MIN, time->tm_min);
5158714166SSimon Glass if (ret < 0)
5258714166SSimon Glass return ret;
5358714166SSimon Glass ret = dm_i2c_reg_write(dev, REG_HOUR, time->tm_hour);
5458714166SSimon Glass if (ret < 0)
5558714166SSimon Glass return ret;
5658714166SSimon Glass ret = dm_i2c_reg_write(dev, REG_MDAY, time->tm_mday);
5758714166SSimon Glass if (ret < 0)
5858714166SSimon Glass return ret;
5958714166SSimon Glass ret = dm_i2c_reg_write(dev, REG_MON, time->tm_mon);
6058714166SSimon Glass if (ret < 0)
6158714166SSimon Glass return ret;
6258714166SSimon Glass ret = dm_i2c_reg_write(dev, REG_YEAR, time->tm_year - 1900);
6358714166SSimon Glass if (ret < 0)
6458714166SSimon Glass return ret;
6558714166SSimon Glass ret = dm_i2c_reg_write(dev, REG_WDAY, time->tm_wday);
6658714166SSimon Glass if (ret < 0)
6758714166SSimon Glass return ret;
6858714166SSimon Glass
6958714166SSimon Glass return 0;
7058714166SSimon Glass }
7158714166SSimon Glass
sandbox_rtc_reset(struct udevice * dev)7258714166SSimon Glass static int sandbox_rtc_reset(struct udevice *dev)
7358714166SSimon Glass {
7458714166SSimon Glass return dm_i2c_reg_write(dev, REG_RESET, 0);
7558714166SSimon Glass }
7658714166SSimon Glass
sandbox_rtc_read8(struct udevice * dev,unsigned int reg)7758714166SSimon Glass static int sandbox_rtc_read8(struct udevice *dev, unsigned int reg)
7858714166SSimon Glass {
7958714166SSimon Glass return dm_i2c_reg_read(dev, reg);
8058714166SSimon Glass }
8158714166SSimon Glass
sandbox_rtc_write8(struct udevice * dev,unsigned int reg,int val)8258714166SSimon Glass static int sandbox_rtc_write8(struct udevice *dev, unsigned int reg, int val)
8358714166SSimon Glass {
8458714166SSimon Glass return dm_i2c_reg_write(dev, reg, val);
8558714166SSimon Glass }
8658714166SSimon Glass
8758714166SSimon Glass static const struct rtc_ops sandbox_rtc_ops = {
8858714166SSimon Glass .get = sandbox_rtc_get,
8958714166SSimon Glass .set = sandbox_rtc_set,
9058714166SSimon Glass .reset = sandbox_rtc_reset,
9158714166SSimon Glass .read8 = sandbox_rtc_read8,
9258714166SSimon Glass .write8 = sandbox_rtc_write8,
9358714166SSimon Glass };
9458714166SSimon Glass
9558714166SSimon Glass static const struct udevice_id sandbox_rtc_ids[] = {
9658714166SSimon Glass { .compatible = "sandbox-rtc" },
9758714166SSimon Glass { }
9858714166SSimon Glass };
9958714166SSimon Glass
10058714166SSimon Glass U_BOOT_DRIVER(rtc_sandbox) = {
10158714166SSimon Glass .name = "rtc-sandbox",
10258714166SSimon Glass .id = UCLASS_RTC,
10358714166SSimon Glass .of_match = sandbox_rtc_ids,
10458714166SSimon Glass .ops = &sandbox_rtc_ops,
10558714166SSimon Glass };
106