181 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			181 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
// SPDX-License-Identifier: GPL-2.0+
 | 
						|
/*
 | 
						|
 * (C) Copyright 2015 Google, Inc
 | 
						|
 * Written by Simon Glass <sjg@chromium.org>
 | 
						|
 */
 | 
						|
 | 
						|
#include <common.h>
 | 
						|
#include <dm.h>
 | 
						|
#include <errno.h>
 | 
						|
#include <log.h>
 | 
						|
#include <rtc.h>
 | 
						|
 | 
						|
int dm_rtc_get(struct udevice *dev, struct rtc_time *time)
 | 
						|
{
 | 
						|
	struct rtc_ops *ops = rtc_get_ops(dev);
 | 
						|
 | 
						|
	assert(ops);
 | 
						|
	if (!ops->get)
 | 
						|
		return -ENOSYS;
 | 
						|
	return ops->get(dev, time);
 | 
						|
}
 | 
						|
 | 
						|
int dm_rtc_set(struct udevice *dev, struct rtc_time *time)
 | 
						|
{
 | 
						|
	struct rtc_ops *ops = rtc_get_ops(dev);
 | 
						|
 | 
						|
	assert(ops);
 | 
						|
	if (!ops->set)
 | 
						|
		return -ENOSYS;
 | 
						|
	return ops->set(dev, time);
 | 
						|
}
 | 
						|
 | 
						|
int dm_rtc_reset(struct udevice *dev)
 | 
						|
{
 | 
						|
	struct rtc_ops *ops = rtc_get_ops(dev);
 | 
						|
 | 
						|
	assert(ops);
 | 
						|
	if (!ops->reset)
 | 
						|
		return -ENOSYS;
 | 
						|
	return ops->reset(dev);
 | 
						|
}
 | 
						|
 | 
						|
int dm_rtc_read(struct udevice *dev, unsigned int reg, u8 *buf, unsigned int len)
 | 
						|
{
 | 
						|
	struct rtc_ops *ops = rtc_get_ops(dev);
 | 
						|
 | 
						|
	assert(ops);
 | 
						|
	if (ops->read)
 | 
						|
		return ops->read(dev, reg, buf, len);
 | 
						|
	if (!ops->read8)
 | 
						|
		return -ENOSYS;
 | 
						|
	while (len--) {
 | 
						|
		int ret = ops->read8(dev, reg++);
 | 
						|
 | 
						|
		if (ret < 0)
 | 
						|
			return ret;
 | 
						|
		*buf++ = ret;
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int dm_rtc_write(struct udevice *dev, unsigned int reg,
 | 
						|
		 const u8 *buf, unsigned int len)
 | 
						|
{
 | 
						|
	struct rtc_ops *ops = rtc_get_ops(dev);
 | 
						|
 | 
						|
	assert(ops);
 | 
						|
	if (ops->write)
 | 
						|
		return ops->write(dev, reg, buf, len);
 | 
						|
	if (!ops->write8)
 | 
						|
		return -ENOSYS;
 | 
						|
	while (len--) {
 | 
						|
		int ret = ops->write8(dev, reg++, *buf++);
 | 
						|
 | 
						|
		if (ret < 0)
 | 
						|
			return ret;
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int rtc_read8(struct udevice *dev, unsigned int reg)
 | 
						|
{
 | 
						|
	struct rtc_ops *ops = rtc_get_ops(dev);
 | 
						|
 | 
						|
	assert(ops);
 | 
						|
	if (ops->read8)
 | 
						|
		return ops->read8(dev, reg);
 | 
						|
	if (ops->read) {
 | 
						|
		u8 buf[1];
 | 
						|
		int ret = ops->read(dev, reg, buf, 1);
 | 
						|
 | 
						|
		if (ret < 0)
 | 
						|
			return ret;
 | 
						|
		return buf[0];
 | 
						|
	}
 | 
						|
	return -ENOSYS;
 | 
						|
}
 | 
						|
 | 
						|
int rtc_write8(struct udevice *dev, unsigned int reg, int val)
 | 
						|
{
 | 
						|
	struct rtc_ops *ops = rtc_get_ops(dev);
 | 
						|
 | 
						|
	assert(ops);
 | 
						|
	if (ops->write8)
 | 
						|
		return ops->write8(dev, reg, val);
 | 
						|
	if (ops->write) {
 | 
						|
		u8 buf[1] = { val };
 | 
						|
 | 
						|
		return ops->write(dev, reg, buf, 1);
 | 
						|
	}
 | 
						|
	return -ENOSYS;
 | 
						|
}
 | 
						|
 | 
						|
int rtc_read16(struct udevice *dev, unsigned int reg, u16 *valuep)
 | 
						|
{
 | 
						|
	u16 value = 0;
 | 
						|
	int ret;
 | 
						|
	int i;
 | 
						|
 | 
						|
	for (i = 0; i < sizeof(value); i++) {
 | 
						|
		ret = rtc_read8(dev, reg + i);
 | 
						|
		if (ret < 0)
 | 
						|
			return ret;
 | 
						|
		value |= ret << (i << 3);
 | 
						|
	}
 | 
						|
 | 
						|
	*valuep = value;
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int rtc_write16(struct udevice *dev, unsigned int reg, u16 value)
 | 
						|
{
 | 
						|
	int i, ret;
 | 
						|
 | 
						|
	for (i = 0; i < sizeof(value); i++) {
 | 
						|
		ret = rtc_write8(dev, reg + i, (value >> (i << 3)) & 0xff);
 | 
						|
		if (ret)
 | 
						|
			return ret;
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int rtc_read32(struct udevice *dev, unsigned int reg, u32 *valuep)
 | 
						|
{
 | 
						|
	u32 value = 0;
 | 
						|
	int ret;
 | 
						|
	int i;
 | 
						|
 | 
						|
	for (i = 0; i < sizeof(value); i++) {
 | 
						|
		ret = rtc_read8(dev, reg + i);
 | 
						|
		if (ret < 0)
 | 
						|
			return ret;
 | 
						|
		value |= ret << (i << 3);
 | 
						|
	}
 | 
						|
 | 
						|
	*valuep = value;
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int rtc_write32(struct udevice *dev, unsigned int reg, u32 value)
 | 
						|
{
 | 
						|
	int i, ret;
 | 
						|
 | 
						|
	for (i = 0; i < sizeof(value); i++) {
 | 
						|
		ret = rtc_write8(dev, reg + i, (value >> (i << 3)) & 0xff);
 | 
						|
		if (ret)
 | 
						|
			return ret;
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
UCLASS_DRIVER(rtc) = {
 | 
						|
	.name		= "rtc",
 | 
						|
	.id		= UCLASS_RTC,
 | 
						|
#if !CONFIG_IS_ENABLED(OF_PLATDATA)
 | 
						|
	.post_bind	= dm_scan_fdt_dev,
 | 
						|
#endif
 | 
						|
};
 |