xref: /linux/drivers/dax/hmem/device.c (revision fe098574a93b4e2acb046b583e9857337d807f38)
1 // SPDX-License-Identifier: GPL-2.0
2 #include <linux/platform_device.h>
3 #include <linux/memregion.h>
4 #include <linux/module.h>
5 #include <linux/dax.h>
6 #include <linux/mm.h>
7 
8 static bool nohmem;
9 module_param_named(disable, nohmem, bool, 0444);
10 
11 static struct resource hmem_active = {
12 	.name = "HMEM devices",
13 	.start = 0,
14 	.end = -1,
15 	.flags = IORESOURCE_MEM,
16 };
17 
18 void hmem_register_device(int target_nid, struct resource *res)
19 {
20 	struct platform_device *pdev;
21 	struct memregion_info info;
22 	int rc, id;
23 
24 	if (nohmem)
25 		return;
26 
27 	rc = region_intersects(res->start, resource_size(res), IORESOURCE_MEM,
28 			       IORES_DESC_SOFT_RESERVED);
29 	if (rc != REGION_INTERSECTS)
30 		return;
31 
32 	id = memregion_alloc(GFP_KERNEL);
33 	if (id < 0) {
34 		pr_err("memregion allocation failure for %pr\n", res);
35 		return;
36 	}
37 
38 	pdev = platform_device_alloc("hmem", id);
39 	if (!pdev) {
40 		pr_err("hmem device allocation failure for %pr\n", res);
41 		goto out_pdev;
42 	}
43 
44 	if (!__request_region(&hmem_active, res->start, resource_size(res),
45 			      dev_name(&pdev->dev), 0)) {
46 		dev_dbg(&pdev->dev, "hmem range %pr already active\n", res);
47 		goto out_active;
48 	}
49 
50 	pdev->dev.numa_node = numa_map_to_online_node(target_nid);
51 	info = (struct memregion_info) {
52 		.target_node = target_nid,
53 		.range = {
54 			.start = res->start,
55 			.end = res->end,
56 		},
57 	};
58 	rc = platform_device_add_data(pdev, &info, sizeof(info));
59 	if (rc < 0) {
60 		pr_err("hmem memregion_info allocation failure for %pr\n", res);
61 		goto out_resource;
62 	}
63 
64 	rc = platform_device_add(pdev);
65 	if (rc < 0) {
66 		dev_err(&pdev->dev, "device add failed for %pr\n", res);
67 		goto out_resource;
68 	}
69 
70 	return;
71 
72 out_resource:
73 	__release_region(&hmem_active, res->start, resource_size(res));
74 out_active:
75 	platform_device_put(pdev);
76 out_pdev:
77 	memregion_free(id);
78 }
79 
80 static __init int hmem_register_one(struct resource *res, void *data)
81 {
82 	hmem_register_device(phys_to_target_node(res->start), res);
83 
84 	return 0;
85 }
86 
87 static __init int hmem_init(void)
88 {
89 	walk_iomem_res_desc(IORES_DESC_SOFT_RESERVED,
90 			IORESOURCE_MEM, 0, -1, NULL, hmem_register_one);
91 	return 0;
92 }
93 
94 /*
95  * As this is a fallback for address ranges unclaimed by the ACPI HMAT
96  * parsing it must be at an initcall level greater than hmat_init().
97  */
98 device_initcall(hmem_init);
99