summaryrefslogtreecommitdiffstats
path: root/arch/x86/kernel/pmem.c
blob: 0f4ef472ab9e5abc9723d7856125e23111137a34 (plain)
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
/*
 * Copyright (c) 2015, Christoph Hellwig.
 * Copyright (c) 2015, Intel Corporation.
 */
#include <linux/platform_device.h>
#include <linux/libnvdimm.h>
#include <linux/module.h>
#include <asm/e820.h>

static void e820_pmem_release(struct device *dev)
{
	struct nvdimm_bus *nvdimm_bus = dev->platform_data;

	if (nvdimm_bus)
		nvdimm_bus_unregister(nvdimm_bus);
}

static struct platform_device e820_pmem = {
	.name = "e820_pmem",
	.id = -1,
	.dev = {
		.release = e820_pmem_release,
	},
};

static const struct attribute_group *e820_pmem_attribute_groups[] = {
	&nvdimm_bus_attribute_group,
	NULL,
};

static const struct attribute_group *e820_pmem_region_attribute_groups[] = {
	&nd_region_attribute_group,
	&nd_device_attribute_group,
	NULL,
};

static __init int register_e820_pmem(void)
{
	static struct nvdimm_bus_descriptor nd_desc;
	struct device *dev = &e820_pmem.dev;
	struct nvdimm_bus *nvdimm_bus;
	int rc, i;

	rc = platform_device_register(&e820_pmem);
	if (rc)
		return rc;

	nd_desc.attr_groups = e820_pmem_attribute_groups;
	nd_desc.provider_name = "e820";
	nvdimm_bus = nvdimm_bus_register(dev, &nd_desc);
	if (!nvdimm_bus)
		goto err;
	dev->platform_data = nvdimm_bus;

	for (i = 0; i < e820.nr_map; i++) {
		struct e820entry *ei = &e820.map[i];
		struct resource res = {
			.flags	= IORESOURCE_MEM,
			.start	= ei->addr,
			.end	= ei->addr + ei->size - 1,
		};
		struct nd_region_desc ndr_desc;

		if (ei->type != E820_PRAM)
			continue;

		memset(&ndr_desc, 0, sizeof(ndr_desc));
		ndr_desc.res = &res;
		ndr_desc.attr_groups = e820_pmem_region_attribute_groups;
		if (!nvdimm_pmem_region_create(nvdimm_bus, &ndr_desc))
			goto err;
	}

	return 0;

 err:
	dev_err(dev, "failed to register legacy persistent memory ranges\n");
	platform_device_unregister(&e820_pmem);
	return -ENXIO;
}
device_initcall(register_e820_pmem);