// SPDX-License-Identifier: GPL-2.0+ /* * Copyright (C) 2022 Sean Anderson */ #include #include #include #include #include #include #include #include #include #include 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); }