lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite for Android: free password hash cracker in your pocket
[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <56D6F1CA.6020302@linaro.org>
Date:	Wed, 2 Mar 2016 13:59:38 +0000
From:	Srinivas Kandagatla <srinivas.kandagatla@...aro.org>
To:	Andrey Smirnov <andrew.smirnov@...il.com>,
	linux-kernel@...r.kernel.org
Cc:	maxime.ripard@...e-electrons.com
Subject: Re: [RESEND RFC 3/3] nvmem: Add 'nvmem-composite' driver


On 01/03/16 16:59, Andrey Smirnov wrote:
> Add 'nvmem-composite' driver which allows to combine multiple chunks of
> various NVMEM cells into a single continuous NVMEM device.

My plan on this feature was add support inside the nvmem_cell_get 
itself, this makes the nvmem bindings more inline with bindings like 
pinctrl.
Also I still want to keep nvmem simple as it can.

DT would look something like this.

nvmem-provider-a {
	cell_a {
		reg = <0 2>;
	};
};

nvmem-provider-b {
	cell_b: cell_c {
		reg = <0 1>;
	};
};

nvmem-provider-c {
	cell_c: cell_c {
		reg = <3 2>;
	}
};

a-node {
	nvmem-cells = <&cell_a &cell_b &cell_c>
	nvmem-cell-names = "some-data";
};


thanks,
srini

>
> Signed-off-by: Andrey Smirnov <andrew.smirnov@...il.com>
> ---
>   .../devicetree/bindings/nvmem/composite.txt        |  44 ++++
>   drivers/nvmem/Makefile                             |   1 +
>   drivers/nvmem/composite.c                          | 225 +++++++++++++++++++++
>   3 files changed, 270 insertions(+)
>   create mode 100644 Documentation/devicetree/bindings/nvmem/composite.txt
>   create mode 100644 drivers/nvmem/composite.c
>
> diff --git a/Documentation/devicetree/bindings/nvmem/composite.txt b/Documentation/devicetree/bindings/nvmem/composite.txt
> new file mode 100644
> index 0000000..e24cf4b
> --- /dev/null
> +++ b/Documentation/devicetree/bindings/nvmem/composite.txt
> @@ -0,0 +1,44 @@
> += NVMEM cell compositor =
> +
> +This binding is designed to provide a way for a developer to combine
> +portions of other NVMEM cell and acces that data as a signle NVMEM
> +cell using NVMEM consumer API.
> +
> +Required properties:
> +- compatible: should be "nvmem-composite"
> +- layout: specifies which sources comprise data in nvmem device
> +  	  format is "<nvmem-cell-phandle offset size>"
> +
> += Data cells =
> +Are child nodes of nvmem-composite, bindings of which as described in
> +bindings/nvmem/nvmem.txt
> +
> +Example:
> +
> +	composite-nvmem {
> +		compatible = "nvmem-composite";
> +		layout = <&another_cell_a 0 2
> + 			  &another_cell_b 0 1
> +			  &another_cell_c 3 2>;
> +
> +		cell1: cell@0 {
> +			reg = <0 5>;
> +		};
> +	}
> +
> +the result of reading variable cell1 would be:
> +
> +[a[0] a[1] b[0] c[3] c[4]],
> +
> +where a[i], b[i], c[i], represnet i-th bytes of NVMEM cells
> +another_cell_a, another_cell_b and another_cell_c respectively
> +
> += Data consumers =
> +Are device nodes which consume nvmem data cells.
> +
> +Example:
> +
> +	a-node {
> +		nvmem-cells = <&cell1>;
> +		nvmem-cell-names = "some-data";
> +	};
> diff --git a/drivers/nvmem/Makefile b/drivers/nvmem/Makefile
> index 1a1adba..49c0eca 100644
> --- a/drivers/nvmem/Makefile
> +++ b/drivers/nvmem/Makefile
> @@ -19,3 +19,4 @@ nvmem_sunxi_sid-y		:= sunxi_sid.o
>   obj-$(CONFIG_NVMEM_VF610_OCOTP)	+= nvmem-vf610-ocotp.o
>   nvmem-vf610-ocotp-y		:= vf610-ocotp.o
>   obj-y += blob.o
> +obj-y += composite.o
> diff --git a/drivers/nvmem/composite.c b/drivers/nvmem/composite.c
> new file mode 100644
> index 0000000..cf01590
> --- /dev/null
> +++ b/drivers/nvmem/composite.c
> @@ -0,0 +1,225 @@
> +#define DEBUG 1
> +
> +#include <linux/device.h>
> +#include <linux/io.h>
> +#include <linux/module.h>
> +#include <linux/nvmem-provider.h>
> +#include <linux/nvmem-consumer.h>
> +#include <linux/of.h>
> +#include <linux/of_device.h>
> +#include <linux/platform_device.h>
> +#include <linux/regmap.h>
> +#include <linux/slab.h>
> +
> +struct nvmem_composite {
> +	struct device *dev;
> +	struct list_head layout;
> +	size_t layout_size;
> +};
> +
> +struct nvmem_composite_item {
> +	struct nvmem_cell *cell;
> +	unsigned int idx, start, end, size;
> +	struct list_head node;
> +};
> +
> +static struct nvmem_composite_item *
> +nvmem_composite_find_first(struct nvmem_composite *ncomp,
> +			   unsigned int offset)
> +{
> +	struct nvmem_composite_item *first;
> +	list_for_each_entry(first, &ncomp->layout, node) {
> +		/*
> +		 * Skip all of the irrelevant items that end before our offset
> +		 */
> +		if (first->end > offset)
> +			return first;
> +	}
> +
> +	return NULL;
> +}
> +
> +static int nvmem_composite_read(void *context,
> +				const void *reg, size_t reg_size,
> +				void *val, size_t val_size)
> +{
> +	struct nvmem_composite *ncomp = context;
> +	const unsigned int offset = *(u32 *)reg;
> +	void *dst = val;
> +	size_t residue = val_size;
> +	struct nvmem_composite_item *item, *first;
> +	uint8_t *data;
> +	unsigned int size, chunk, ii;
> +
> +
> +	first = item = nvmem_composite_find_first(ncomp, offset);
> +	if (!first) {
> +		dev_dbg(ncomp->dev, "Invalid offset\n");
> +		return -EINVAL;
> +	}
> +
> +	list_for_each_entry_from(item, &ncomp->layout, node) {
> +		/*
> +		 * If our first read is not located on item boundary
> +		 * we have to introduce artificial offset
> +		 */
> +		ii = (item == first) ? offset - first->start : 0;
> +
> +		data = nvmem_cell_read(item->cell, &size);
> +		if (IS_ERR(data)) {
> +			dev_dbg(ncomp->dev, "Failed to read nvmem cell\n");
> +			return PTR_ERR(data);
> +		}
> +
> +		chunk = min(residue, item->size - ii);
> +		memcpy(dst, &data[item->idx + ii], chunk);
> +		kfree(data);
> +
> +		dst += chunk;
> +		residue -= chunk;
> +	}
> +
> +	return (residue) ? -EINVAL : 0;
> +}
> +
> +static int nvmem_composite_write(void *context, const void *data,
> +				 size_t count)
> +{
> +	return -ENOTSUPP;
> +}
> +
> +static const struct regmap_bus nvmem_composite_regmap_bus = {
> +	.write = nvmem_composite_write,
> +	.read  = nvmem_composite_read,
> +};
> +
> +static int nvmem_composite_validate_dt(struct device_node *np)
> +{
> +	/* FIXME */
> +	return 0;
> +}
> +
> +static int nvmem_composite_probe(struct platform_device *pdev)
> +{
> +	int i, ret;
> +	unsigned int start = 0;
> +	unsigned int len;
> +	struct device *dev = &pdev->dev;
> +	struct device_node *np = dev->of_node;
> +	struct nvmem_device *nvmem;
> +	struct nvmem_composite *ncomp;
> +	struct regmap *map;
> +	struct nvmem_composite_item *item;
> +	struct nvmem_config  nv_cnf = {0};
> +	struct regmap_config rm_cnf = {0};
> +	const __be32 *addr;
> +	unsigned int item_count;
> +
> +	ret = nvmem_composite_validate_dt(np);
> +	if (ret < 0) {
> +		dev_dbg(dev, "Device validation failed\n");
> +		return ret;
> +	}
> +
> +	ncomp = devm_kzalloc(dev, sizeof(*ncomp), GFP_KERNEL);
> +	INIT_LIST_HEAD(&ncomp->layout);
> +	ncomp->dev = dev;
> +	
> +	addr = of_get_property(np, "layout", &len);
> +	item_count = len / (3 * sizeof(__be32));
> +
> +	for (i = 0; i < item_count; i++) {
> +		struct device_node *cell_np;
> +		uint32_t phandle;
> +
> +		item = devm_kzalloc(dev, sizeof(*item), GFP_KERNEL);
> +
> +		phandle = be32_to_cpup(addr++);
> +		cell_np = of_find_node_by_phandle(phandle);
> +		if (!cell_np) {
> +			dev_dbg(dev,
> +				"Couldn't find nvmem cell by its phandle\n");
> +			return -ENOENT;
> +		}
> +
> +		item->cell = of_nvmem_cell_from_device_node(cell_np);
> +		if (IS_ERR(item->cell)) {
> +			dev_dbg(dev,
> +				"Failed to instantiate nvmem cell from "
> +				"a device tree node\n");
> +			ret = PTR_ERR(item->cell);
> +			goto unwind;
> +		}
> +
> +		item->start = start;
> +		item->idx = be32_to_cpup(addr++);
> +		item->size = be32_to_cpup(addr++);
> +		item->end  = item->size - 1;
> +		ncomp->layout_size += item->size;
> +		start += item->size;
> +
> +		list_add_tail(&item->node, &ncomp->layout);
> +	}
> +
> +	rm_cnf.reg_bits = 32;
> +	rm_cnf.val_bits = 8;
> +	rm_cnf.reg_stride = 1;
> +	rm_cnf.name = "nvmem-composite";
> +	rm_cnf.max_register = ncomp->layout_size - 1;
> +
> +	map = devm_regmap_init(dev,
> +			       &nvmem_composite_regmap_bus,
> +			       ncomp,
> +			       &rm_cnf);
> +	if (IS_ERR(map)) {
> +		dev_dbg(dev, "Failed to initilize regmap\n");
> +		return PTR_ERR(map);
> +	}
> +
> +	nv_cnf.name = "nvmem-composite";
> +	nv_cnf.read_only = true;
> +	nv_cnf.dev = dev;
> +
> +	nvmem = nvmem_register(&nv_cnf);
> +	if (IS_ERR(nvmem)) {
> +		dev_dbg(dev, "Failed to register 'nvmem' device\n");
> +		return PTR_ERR(nvmem);
> +	}
> +
> +	platform_set_drvdata(pdev, nvmem);
> +	return 0;
> +unwind:
> +	/* FIXME  */
> +	return ret;
> +}
> +
> +static int nvmem_composite_remove(struct platform_device *pdev)
> +{
> +	struct nvmem_device *nvmem = platform_get_drvdata(pdev);
> +
> +	/*
> +	   FIXME free allocated nvmem cells
> +	 */
> +	return nvmem_unregister(nvmem);
> +}
> +
> +
> +static const struct of_device_id nvmem_composite_dt_ids[] = {
> +	{ .compatible = "nvmem-composite", },
> +	{ },
> +};
> +MODULE_DEVICE_TABLE(of, nvmem_composite_dt_ids);
> +
> +static struct platform_driver nvmem_composite_driver = {
> +	.probe	= nvmem_composite_probe,
> +	.remove	= nvmem_composite_remove,
> +	.driver = {
> +		.name	= "nvmem-composite",
> +		.of_match_table = nvmem_composite_dt_ids,
> +	},
> +};
> +module_platform_driver(nvmem_composite_driver);
> +
> +MODULE_AUTHOR("Andrey Smirnov <andrew.smirnov@...il.com>");
> +MODULE_DESCRIPTION("FIXME");
> +MODULE_LICENSE("GPL v2");
>

Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ