mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-11-04 14:00:19 +00: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>
		
			
				
	
	
		
			142 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			142 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// SPDX-License-Identifier: GPL-2.0+
 | 
						|
/*
 | 
						|
 * Copyright (C) 2022 Sean Anderson <sean.anderson@seco.com>
 | 
						|
 */
 | 
						|
 | 
						|
#include <i2c_eeprom.h>
 | 
						|
#include <linker_lists.h>
 | 
						|
#include <misc.h>
 | 
						|
#include <nvmem.h>
 | 
						|
#include <rtc.h>
 | 
						|
#include <dm/device_compat.h>
 | 
						|
#include <dm/ofnode.h>
 | 
						|
#include <dm/read.h>
 | 
						|
#include <dm/uclass.h>
 | 
						|
 | 
						|
int nvmem_cell_read(struct nvmem_cell *cell, void *buf, size_t size)
 | 
						|
{
 | 
						|
	dev_dbg(cell->nvmem, "%s: off=%u size=%zu\n", __func__, cell->offset, size);
 | 
						|
	if (size != cell->size)
 | 
						|
		return -EINVAL;
 | 
						|
 | 
						|
	switch (cell->nvmem->driver->id) {
 | 
						|
	case UCLASS_I2C_EEPROM:
 | 
						|
		return i2c_eeprom_read(cell->nvmem, cell->offset, buf, size);
 | 
						|
	case UCLASS_MISC: {
 | 
						|
		int ret = misc_read(cell->nvmem, cell->offset, buf, size);
 | 
						|
 | 
						|
		if (ret < 0)
 | 
						|
			return ret;
 | 
						|
		if (ret != size)
 | 
						|
			return -EIO;
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
	case UCLASS_RTC:
 | 
						|
		return dm_rtc_read(cell->nvmem, cell->offset, buf, size);
 | 
						|
	default:
 | 
						|
		return -ENOSYS;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
int nvmem_cell_write(struct nvmem_cell *cell, const void *buf, size_t size)
 | 
						|
{
 | 
						|
	dev_dbg(cell->nvmem, "%s: off=%u size=%zu\n", __func__, cell->offset, size);
 | 
						|
	if (size != cell->size)
 | 
						|
		return -EINVAL;
 | 
						|
 | 
						|
	switch (cell->nvmem->driver->id) {
 | 
						|
	case UCLASS_I2C_EEPROM:
 | 
						|
		return i2c_eeprom_write(cell->nvmem, cell->offset, buf, size);
 | 
						|
	case UCLASS_MISC: {
 | 
						|
		int ret = misc_write(cell->nvmem, cell->offset, buf, size);
 | 
						|
 | 
						|
		if (ret < 0)
 | 
						|
			return ret;
 | 
						|
		if (ret != size)
 | 
						|
			return -EIO;
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
	case UCLASS_RTC:
 | 
						|
		return dm_rtc_write(cell->nvmem, cell->offset, buf, size);
 | 
						|
	default:
 | 
						|
		return -ENOSYS;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * nvmem_get_device() - Get an nvmem device for a cell
 | 
						|
 * @node: ofnode of the nvmem device
 | 
						|
 * @cell: Cell to look up
 | 
						|
 *
 | 
						|
 * Try to find a nvmem-compatible device by going through the nvmem interfaces.
 | 
						|
 *
 | 
						|
 * Return:
 | 
						|
 * * 0 on success
 | 
						|
 * * -ENODEV if we didn't find anything
 | 
						|
 * * A negative error if there was a problem looking up the device
 | 
						|
 */
 | 
						|
static int nvmem_get_device(ofnode node, struct nvmem_cell *cell)
 | 
						|
{
 | 
						|
	int i, ret;
 | 
						|
	enum uclass_id ids[] = {
 | 
						|
		UCLASS_I2C_EEPROM,
 | 
						|
		UCLASS_MISC,
 | 
						|
		UCLASS_RTC,
 | 
						|
	};
 | 
						|
 | 
						|
	for (i = 0; i < ARRAY_SIZE(ids); i++) {
 | 
						|
		ret = uclass_get_device_by_ofnode(ids[i], node, &cell->nvmem);
 | 
						|
		if (!ret)
 | 
						|
			return 0;
 | 
						|
		if (ret != -ENODEV && ret != -EPFNOSUPPORT)
 | 
						|
			return ret;
 | 
						|
	}
 | 
						|
 | 
						|
	return -ENODEV;
 | 
						|
}
 | 
						|
 | 
						|
int nvmem_cell_get_by_index(struct udevice *dev, int index,
 | 
						|
			    struct nvmem_cell *cell)
 | 
						|
{
 | 
						|
	fdt_addr_t offset;
 | 
						|
	fdt_size_t size = FDT_SIZE_T_NONE;
 | 
						|
	int ret;
 | 
						|
	struct ofnode_phandle_args args;
 | 
						|
 | 
						|
	dev_dbg(dev, "%s: index=%d\n", __func__, index);
 | 
						|
 | 
						|
	ret = dev_read_phandle_with_args(dev, "nvmem-cells", NULL, 0, index,
 | 
						|
					 &args);
 | 
						|
	if (ret)
 | 
						|
		return ret;
 | 
						|
 | 
						|
	ret = nvmem_get_device(ofnode_get_parent(args.node), cell);
 | 
						|
	if (ret)
 | 
						|
		return ret;
 | 
						|
 | 
						|
	offset = ofnode_get_addr_size_index_notrans(args.node, 0, &size);
 | 
						|
	if (offset == FDT_ADDR_T_NONE || size == FDT_SIZE_T_NONE) {
 | 
						|
		dev_dbg(cell->nvmem, "missing address or size for %s\n",
 | 
						|
			ofnode_get_name(args.node));
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
 | 
						|
	cell->offset = offset;
 | 
						|
	cell->size = size;
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int nvmem_cell_get_by_name(struct udevice *dev, const char *name,
 | 
						|
			   struct nvmem_cell *cell)
 | 
						|
{
 | 
						|
	int index;
 | 
						|
 | 
						|
	dev_dbg(dev, "%s, name=%s\n", __func__, name);
 | 
						|
 | 
						|
	index = dev_read_stringlist_search(dev, "nvmem-cell-names", name);
 | 
						|
	if (index < 0)
 | 
						|
		return index;
 | 
						|
 | 
						|
	return nvmem_cell_get_by_index(dev, index, cell);
 | 
						|
}
 |