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