2015-08-19 07:34:34 +03:00
/*
* Copyright ( c ) 2015 , Christoph Hellwig .
* Copyright ( c ) 2015 , Intel Corporation .
*/
# include <linux/platform_device.h>
2015-11-12 03:46:33 +03:00
# include <linux/memory_hotplug.h>
2015-08-19 07:34:34 +03:00
# include <linux/libnvdimm.h>
# include <linux/module.h>
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 int e820_pmem_remove ( struct platform_device * pdev )
{
struct nvdimm_bus * nvdimm_bus = platform_get_drvdata ( pdev ) ;
nvdimm_bus_unregister ( nvdimm_bus ) ;
return 0 ;
}
2015-11-12 03:46:33 +03:00
# ifdef CONFIG_MEMORY_HOTPLUG
static int e820_range_to_nid ( resource_size_t addr )
{
return memory_add_physaddr_to_nid ( addr ) ;
}
# else
static int e820_range_to_nid ( resource_size_t addr )
{
return NUMA_NO_NODE ;
}
# endif
2015-08-19 07:34:34 +03:00
static int e820_pmem_probe ( struct platform_device * pdev )
{
static struct nvdimm_bus_descriptor nd_desc ;
struct device * dev = & pdev - > dev ;
struct nvdimm_bus * nvdimm_bus ;
struct resource * p ;
nd_desc . attr_groups = e820_pmem_attribute_groups ;
nd_desc . provider_name = " e820 " ;
2016-07-22 06:03:19 +03:00
nd_desc . module = THIS_MODULE ;
2015-08-19 07:34:34 +03:00
nvdimm_bus = nvdimm_bus_register ( dev , & nd_desc ) ;
if ( ! nvdimm_bus )
goto err ;
platform_set_drvdata ( pdev , nvdimm_bus ) ;
for ( p = iomem_resource . child ; p ; p = p - > sibling ) {
struct nd_region_desc ndr_desc ;
2016-01-26 23:57:30 +03:00
if ( p - > desc ! = IORES_DESC_PERSISTENT_MEMORY_LEGACY )
2015-08-19 07:34:34 +03:00
continue ;
memset ( & ndr_desc , 0 , sizeof ( ndr_desc ) ) ;
ndr_desc . res = p ;
ndr_desc . attr_groups = e820_pmem_region_attribute_groups ;
2015-11-12 03:46:33 +03:00
ndr_desc . numa_node = e820_range_to_nid ( p - > start ) ;
2015-08-25 02:20:23 +03:00
set_bit ( ND_REGION_PAGEMAP , & ndr_desc . flags ) ;
2015-08-19 07:34:34 +03:00
if ( ! nvdimm_pmem_region_create ( nvdimm_bus , & ndr_desc ) )
goto err ;
}
return 0 ;
err :
nvdimm_bus_unregister ( nvdimm_bus ) ;
dev_err ( dev , " failed to register legacy persistent memory ranges \n " ) ;
return - ENXIO ;
}
static struct platform_driver e820_pmem_driver = {
. probe = e820_pmem_probe ,
. remove = e820_pmem_remove ,
. driver = {
. name = " e820_pmem " ,
} ,
} ;
2016-12-05 11:23:20 +03:00
module_platform_driver ( e820_pmem_driver ) ;
2015-08-19 07:34:34 +03:00
MODULE_ALIAS ( " platform:e820_pmem* " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_AUTHOR ( " Intel Corporation " ) ;