mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-10-25 18:18:19 +01:00 
			
		
		
		
	As part of bringing the master branch back in to next, we need to allow for all of these changes to exist here. Reported-by: Jonas Karlman <jonas@kwiboo.se> Signed-off-by: Tom Rini <trini@konsulko.com>
		
			
				
	
	
		
			158 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			158 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /*
 | |
|  * Copyright (c) 2021, Xilinx, Inc.
 | |
|  */
 | |
| 
 | |
| #define LOG_CATEGORY UCLASS_RTC
 | |
| 
 | |
| #include <dm.h>
 | |
| #include <rtc.h>
 | |
| #include <asm/io.h>
 | |
| 
 | |
| /* RTC Registers */
 | |
| #define RTC_SET_TM_WR		0x00
 | |
| #define RTC_SET_TM_RD		0x04
 | |
| #define RTC_CALIB_WR		0x08
 | |
| #define RTC_CUR_TM		0x10
 | |
| #define RTC_INT_STS		0x20
 | |
| #define RTC_CTRL		0x40
 | |
| 
 | |
| #define RTC_INT_SEC		BIT(0)
 | |
| #define RTC_BATT_EN		BIT(31)
 | |
| #define RTC_CALIB_DEF		0x198233
 | |
| #define RTC_CALIB_MASK		0x1FFFFF
 | |
| 
 | |
| struct zynqmp_rtc_priv {
 | |
| 	fdt_addr_t	base;
 | |
| 	unsigned int	calibval;
 | |
| };
 | |
| 
 | |
| static int zynqmp_rtc_get(struct udevice *dev, struct rtc_time *tm)
 | |
| {
 | |
| 	struct zynqmp_rtc_priv *priv = dev_get_priv(dev);
 | |
| 	u32 status;
 | |
| 	unsigned long read_time;
 | |
| 
 | |
| 	status = readl(priv->base + RTC_INT_STS);
 | |
| 
 | |
| 	if (status & RTC_INT_SEC) {
 | |
| 		/*
 | |
| 		 * RTC has updated the CURRENT_TIME with the time written into
 | |
| 		 * SET_TIME_WRITE register.
 | |
| 		 */
 | |
| 		read_time = readl(priv->base + RTC_CUR_TM);
 | |
| 	} else {
 | |
| 		/*
 | |
| 		 * Time written in SET_TIME_WRITE has not yet updated into
 | |
| 		 * the seconds read register, so read the time from the
 | |
| 		 * SET_TIME_WRITE instead of CURRENT_TIME register.
 | |
| 		 * Since we add +1 sec while writing, we need to -1 sec while
 | |
| 		 * reading.
 | |
| 		 */
 | |
| 		read_time = readl(priv->base + RTC_SET_TM_RD) - 1;
 | |
| 	}
 | |
| 
 | |
| 	rtc_to_tm(read_time, tm);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int zynqmp_rtc_set(struct udevice *dev, const struct rtc_time *tm)
 | |
| {
 | |
| 	struct zynqmp_rtc_priv *priv = dev_get_priv(dev);
 | |
| 	unsigned long new_time = 0;
 | |
| 
 | |
| 	if (tm)
 | |
| 		/*
 | |
| 		 * The value written will be updated after 1 sec into the
 | |
| 		 * seconds read register, so we need to program time +1 sec
 | |
| 		 * to get the correct time on read.
 | |
| 		 */
 | |
| 		new_time = rtc_mktime(tm) + 1;
 | |
| 
 | |
| 	/*
 | |
| 	 * Writing into calibration register will clear the Tick Counter and
 | |
| 	 * force the next second to be signaled exactly in 1 second period
 | |
| 	 */
 | |
| 	priv->calibval &= RTC_CALIB_MASK;
 | |
| 	writel(priv->calibval, (priv->base + RTC_CALIB_WR));
 | |
| 
 | |
| 	writel(new_time, priv->base + RTC_SET_TM_WR);
 | |
| 
 | |
| 	/*
 | |
| 	 * Clear the rtc interrupt status register after setting the
 | |
| 	 * time. During a read_time function, the code should read the
 | |
| 	 * RTC_INT_STATUS register and if bit 0 is still 0, it means
 | |
| 	 * that one second has not elapsed yet since RTC was set and
 | |
| 	 * the current time should be read from SET_TIME_READ register;
 | |
| 	 * otherwise, CURRENT_TIME register is read to report the time
 | |
| 	 */
 | |
| 	writel(RTC_INT_SEC, priv->base + RTC_INT_STS);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int zynqmp_rtc_reset(struct udevice *dev)
 | |
| {
 | |
| 	return zynqmp_rtc_set(dev, NULL);
 | |
| }
 | |
| 
 | |
| static int zynqmp_rtc_init(struct udevice *dev)
 | |
| {
 | |
| 	struct zynqmp_rtc_priv *priv = dev_get_priv(dev);
 | |
| 	u32 rtc_ctrl;
 | |
| 
 | |
| 	/* Enable RTC switch to battery when VCC_PSAUX is not available */
 | |
| 	rtc_ctrl = readl(priv->base + RTC_CTRL);
 | |
| 	rtc_ctrl |= RTC_BATT_EN;
 | |
| 	writel(rtc_ctrl, priv->base + RTC_CTRL);
 | |
| 
 | |
| 	/*
 | |
| 	 * Based on crystal freq of 33.330 KHz
 | |
| 	 * set the seconds counter and enable, set fractions counter
 | |
| 	 * to default value suggested as per design spec
 | |
| 	 * to correct RTC delay in frequency over period of time.
 | |
| 	 */
 | |
| 	priv->calibval &= RTC_CALIB_MASK;
 | |
| 	writel(priv->calibval, (priv->base + RTC_CALIB_WR));
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int zynqmp_rtc_probe(struct udevice *dev)
 | |
| {
 | |
| 	struct zynqmp_rtc_priv *priv = dev_get_priv(dev);
 | |
| 	int ret;
 | |
| 
 | |
| 	priv->base = dev_read_addr(dev);
 | |
| 	if (priv->base == FDT_ADDR_T_NONE)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	priv->calibval = dev_read_u32_default(dev, "calibration",
 | |
| 					      RTC_CALIB_DEF);
 | |
| 
 | |
| 	ret = zynqmp_rtc_init(dev);
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static const struct rtc_ops zynqmp_rtc_ops = {
 | |
| 	.get = zynqmp_rtc_get,
 | |
| 	.set = zynqmp_rtc_set,
 | |
| 	.reset = zynqmp_rtc_reset,
 | |
| };
 | |
| 
 | |
| static const struct udevice_id zynqmp_rtc_ids[] = {
 | |
| 	{ .compatible = "xlnx,zynqmp-rtc" },
 | |
| 	{ }
 | |
| };
 | |
| 
 | |
| U_BOOT_DRIVER(rtc_zynqmp) = {
 | |
| 	.name = "rtc-zynqmp",
 | |
| 	.id = UCLASS_RTC,
 | |
| 	.probe = zynqmp_rtc_probe,
 | |
| 	.of_match = zynqmp_rtc_ids,
 | |
| 	.ops = &zynqmp_rtc_ops,
 | |
| 	.priv_auto = sizeof(struct zynqmp_rtc_priv),
 | |
| };
 |