1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
|
/*
* 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 <common.h>
#include <driver.h>
#include <init.h>
#include <io.h>
#include <of.h>
#include <malloc.h>
#include <regmap.h>
#include <mfd/syscon.h>
#include <linux/nvmem-provider.h>
struct snvs_lpgpr_priv {
struct device_d *dev;
struct regmap *regmap;
int offset;
struct nvmem_config cfg;
};
static int snvs_lpgpr_write(struct device_d *dev, const int reg,
const void *_val, int bytes)
{
struct snvs_lpgpr_priv *priv = dev->parent->priv;
const u32 *val = _val;
int i = 0, words = bytes / 4;
while (words--)
regmap_write(priv->regmap, priv->offset + reg + (i++ * 4),
*val++);
return 0;
}
static int snvs_lpgpr_read(struct device_d *dev, const int reg, void *_val,
int bytes)
{
struct snvs_lpgpr_priv *priv = dev->parent->priv;
u32 *val = _val;
int i = 0, words = bytes / 4;
while (words--)
regmap_read(priv->regmap, priv->offset + reg + (i++ * 4),
val++);
return 0;
}
static const struct nvmem_bus snvs_lpgpr_nvmem_bus = {
.write = snvs_lpgpr_write,
.read = snvs_lpgpr_read,
};
static int snvs_lpgpr_probe(struct device_d *dev)
{
struct device_node *node = dev->device_node;
struct snvs_lpgpr_priv *priv;
struct nvmem_config *cfg;
struct nvmem_device *nvmem;
int err;
if (!node)
return -ENOENT;
priv = kzalloc(sizeof(*priv), GFP_KERNEL);
if (!priv)
return -ENOMEM;
priv->regmap = syscon_node_to_regmap(of_get_parent(node));
if (IS_ERR(priv->regmap)) {
free(priv);
return PTR_ERR(priv->regmap);
}
err = of_property_read_u32(node, "offset", &priv->offset);
if (err)
return err;
cfg = &priv->cfg;
cfg->name = dev_name(dev);
cfg->dev = dev;
cfg->stride = 4,
cfg->word_size = 4,
cfg->size = 4,
cfg->bus = &snvs_lpgpr_nvmem_bus,
nvmem = nvmem_register(cfg);
if (IS_ERR(nvmem)) {
free(priv);
return PTR_ERR(nvmem);
}
dev->priv = priv;
return 0;
}
static __maybe_unused struct of_device_id snvs_lpgpr_dt_ids[] = {
{ .compatible = "fsl,imx6sl-snvs-lpgpr", },
{ .compatible = "fsl,imx6q-snvs-lpgpr", },
{ },
};
static struct driver_d snvs_lpgpr_driver = {
.name = "nvmem-snvs-lpgpr",
.probe = snvs_lpgpr_probe,
.of_compatible = DRV_OF_COMPAT(snvs_lpgpr_dt_ids),
};
device_platform_driver(snvs_lpgpr_driver);
|