cregit-Linux how code gets into the kernel

Release 4.17 drivers/nvmem/snvs_lpgpr.c

Directory: drivers/nvmem
/*
 * Copyright (c) 2015 Pengutronix, Steffen Trumtrar <kernel@pengutronix.de>
 * Copyright (c) 2017 Pengutronix, Oleksij Rempel <kernel@pengutronix.de>
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2
 * as published by the Free Software Foundation.
 */

#include <linux/mfd/syscon.h>
#include <linux/module.h>
#include <linux/nvmem-provider.h>
#include <linux/of_device.h>
#include <linux/regmap.h>


#define IMX6Q_SNVS_HPLR		0x00

#define IMX6Q_SNVS_LPLR		0x34

#define IMX6Q_SNVS_LPGPR	0x68


#define IMX7D_SNVS_HPLR		0x00

#define IMX7D_SNVS_LPLR		0x34

#define IMX7D_SNVS_LPGPR	0x90


#define IMX_GPR_SL		BIT(5)

#define IMX_GPR_HL		BIT(5)


struct snvs_lpgpr_cfg {
	
int offset;
	
int offset_hplr;
	
int offset_lplr;
	
int size;
};


struct snvs_lpgpr_priv {
	
struct device_d			*dev;
	
struct regmap			*regmap;
	
struct nvmem_config		cfg;
	
const struct snvs_lpgpr_cfg	*dcfg;
};


static const struct snvs_lpgpr_cfg snvs_lpgpr_cfg_imx6q = {
	.offset		= IMX6Q_SNVS_LPGPR,
	.offset_hplr	= IMX6Q_SNVS_HPLR,
	.offset_lplr	= IMX6Q_SNVS_LPLR,
	.size		= 4,
};


static const struct snvs_lpgpr_cfg snvs_lpgpr_cfg_imx7d = {
	.offset		= IMX7D_SNVS_LPGPR,
	.offset_hplr	= IMX7D_SNVS_HPLR,
	.offset_lplr	= IMX7D_SNVS_LPLR,
	.size		= 16,
};


static int snvs_lpgpr_write(void *context, unsigned int offset, void *val, size_t bytes) { struct snvs_lpgpr_priv *priv = context; const struct snvs_lpgpr_cfg *dcfg = priv->dcfg; unsigned int lock_reg; int ret; ret = regmap_read(priv->regmap, dcfg->offset_hplr, &lock_reg); if (ret < 0) return ret; if (lock_reg & IMX_GPR_SL) return -EPERM; ret = regmap_read(priv->regmap, dcfg->offset_lplr, &lock_reg); if (ret < 0) return ret; if (lock_reg & IMX_GPR_HL) return -EPERM; return regmap_bulk_write(priv->regmap, dcfg->offset + offset, val, bytes / 4); }

Contributors

PersonTokensPropCommitsCommitProp
Oleksij Rempel13398.52%150.00%
Andrey Yurovsky21.48%150.00%
Total135100.00%2100.00%


static int snvs_lpgpr_read(void *context, unsigned int offset, void *val, size_t bytes) { struct snvs_lpgpr_priv *priv = context; const struct snvs_lpgpr_cfg *dcfg = priv->dcfg; return regmap_bulk_read(priv->regmap, dcfg->offset + offset, val, bytes / 4); }

Contributors

PersonTokensPropCommitsCommitProp
Oleksij Rempel58100.00%1100.00%
Total58100.00%1100.00%


static int snvs_lpgpr_probe(struct platform_device *pdev) { struct device *dev = &pdev->dev; struct device_node *node = dev->of_node; struct device_node *syscon_node; struct snvs_lpgpr_priv *priv; struct nvmem_config *cfg; struct nvmem_device *nvmem; const struct snvs_lpgpr_cfg *dcfg; if (!node) return -ENOENT; priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); if (!priv) return -ENOMEM; dcfg = of_device_get_match_data(dev); if (!dcfg) return -EINVAL; syscon_node = of_get_parent(node); if (!syscon_node) return -ENODEV; priv->regmap = syscon_node_to_regmap(syscon_node); of_node_put(syscon_node); if (IS_ERR(priv->regmap)) return PTR_ERR(priv->regmap); priv->dcfg = dcfg; cfg = &priv->cfg; cfg->priv = priv; cfg->name = dev_name(dev); cfg->dev = dev; cfg->stride = 4; cfg->word_size = 4; cfg->size = dcfg->size, cfg->owner = THIS_MODULE; cfg->reg_read = snvs_lpgpr_read; cfg->reg_write = snvs_lpgpr_write; nvmem = devm_nvmem_register(dev, cfg); return PTR_ERR_OR_ZERO(nvmem); }

Contributors

PersonTokensPropCommitsCommitProp
Oleksij Rempel22694.56%125.00%
Andrey Smirnov93.77%250.00%
Andrey Yurovsky41.67%125.00%
Total239100.00%4100.00%

static const struct of_device_id snvs_lpgpr_dt_ids[] = { { .compatible = "fsl,imx6q-snvs-lpgpr", .data = &snvs_lpgpr_cfg_imx6q }, { .compatible = "fsl,imx6ul-snvs-lpgpr", .data = &snvs_lpgpr_cfg_imx6q }, { .compatible = "fsl,imx7d-snvs-lpgpr", .data = &snvs_lpgpr_cfg_imx7d }, { }, }; MODULE_DEVICE_TABLE(of, snvs_lpgpr_dt_ids); static struct platform_driver snvs_lpgpr_driver = { .probe = snvs_lpgpr_probe, .driver = { .name = "snvs_lpgpr", .of_match_table = snvs_lpgpr_dt_ids, }, }; module_platform_driver(snvs_lpgpr_driver); MODULE_AUTHOR("Oleksij Rempel <o.rempel@pengutronix.de>"); MODULE_DESCRIPTION("Low Power General Purpose Register in i.MX6 and i.MX7 Secure Non-Volatile Storage"); MODULE_LICENSE("GPL v2");

Overall Contributors

PersonTokensPropCommitsCommitProp
Oleksij Rempel59987.83%125.00%
Andrey Yurovsky7410.85%125.00%
Andrey Smirnov91.32%250.00%
Total682100.00%4100.00%
Directory: drivers/nvmem
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.