1 // SPDX-License-Identifier: GPL-2.0+
3 #define pr_fmt(fmt) "of_pmem: " fmt
6 #include <linux/libnvdimm.h>
7 #include <linux/module.h>
8 #include <linux/ioport.h>
9 #include <linux/platform_device.h>
10 #include <linux/slab.h>
12 struct of_pmem_private
{
13 struct nvdimm_bus_descriptor bus_desc
;
14 struct nvdimm_bus
*bus
;
17 static int of_pmem_region_probe(struct platform_device
*pdev
)
19 struct of_pmem_private
*priv
;
20 struct device_node
*np
;
21 struct nvdimm_bus
*bus
;
25 np
= dev_of_node(&pdev
->dev
);
29 priv
= kzalloc(sizeof(*priv
), GFP_KERNEL
);
33 priv
->bus_desc
.provider_name
= devm_kstrdup(&pdev
->dev
, pdev
->name
,
35 if (!priv
->bus_desc
.provider_name
) {
40 priv
->bus_desc
.module
= THIS_MODULE
;
41 priv
->bus_desc
.of_node
= np
;
43 priv
->bus
= bus
= nvdimm_bus_register(&pdev
->dev
, &priv
->bus_desc
);
48 platform_set_drvdata(pdev
, priv
);
50 is_volatile
= of_property_read_bool(np
, "volatile");
51 dev_dbg(&pdev
->dev
, "Registering %s regions from %pOF\n",
52 is_volatile
? "volatile" : "non-volatile", np
);
54 for (i
= 0; i
< pdev
->num_resources
; i
++) {
55 struct nd_region_desc ndr_desc
;
56 struct nd_region
*region
;
59 * NB: libnvdimm copies the data from ndr_desc into it's own
60 * structures so passing a stack pointer is fine.
62 memset(&ndr_desc
, 0, sizeof(ndr_desc
));
63 ndr_desc
.numa_node
= dev_to_node(&pdev
->dev
);
64 ndr_desc
.target_node
= ndr_desc
.numa_node
;
65 ndr_desc
.res
= &pdev
->resource
[i
];
66 ndr_desc
.of_node
= np
;
67 set_bit(ND_REGION_PAGEMAP
, &ndr_desc
.flags
);
70 region
= nvdimm_volatile_region_create(bus
, &ndr_desc
);
72 set_bit(ND_REGION_PERSIST_MEMCTRL
, &ndr_desc
.flags
);
73 region
= nvdimm_pmem_region_create(bus
, &ndr_desc
);
77 dev_warn(&pdev
->dev
, "Unable to register region %pR from %pOF\n",
80 dev_dbg(&pdev
->dev
, "Registered region %pR from %pOF\n",
87 static void of_pmem_region_remove(struct platform_device
*pdev
)
89 struct of_pmem_private
*priv
= platform_get_drvdata(pdev
);
91 nvdimm_bus_unregister(priv
->bus
);
95 static const struct of_device_id of_pmem_region_match
[] = {
96 { .compatible
= "pmem-region" },
97 { .compatible
= "pmem-region-v2" },
101 static struct platform_driver of_pmem_region_driver
= {
102 .probe
= of_pmem_region_probe
,
103 .remove_new
= of_pmem_region_remove
,
106 .of_match_table
= of_pmem_region_match
,
110 module_platform_driver(of_pmem_region_driver
);
111 MODULE_DEVICE_TABLE(of
, of_pmem_region_match
);
112 MODULE_DESCRIPTION("NVDIMM Device Tree support");
113 MODULE_LICENSE("GPL");
114 MODULE_AUTHOR("IBM Corporation");