2019-05-29 07:18:01 -07:00
// SPDX-License-Identifier: GPL-2.0-only
2017-03-28 18:11:27 +02:00
/*
* coreboot_table . c
*
* Module providing coreboot table access .
*
* Copyright 2017 Google Inc .
2018-01-24 19:41:16 -06:00
* Copyright 2017 Samuel Holland < samuel @ sholland . org >
2017-03-28 18:11:27 +02:00
*/
2018-08-15 13:37:06 -07:00
# include <linux/acpi.h>
2018-01-24 19:41:16 -06:00
# include <linux/device.h>
2017-03-28 18:11:27 +02:00
# include <linux/err.h>
# include <linux/init.h>
# include <linux/io.h>
# include <linux/kernel.h>
# include <linux/module.h>
2018-08-15 13:37:06 -07:00
# include <linux/of.h>
# include <linux/platform_device.h>
2018-01-24 19:41:16 -06:00
# include <linux/slab.h>
2017-03-28 18:11:27 +02:00
# include "coreboot_table.h"
2018-01-24 19:41:16 -06:00
# define CB_DEV(d) container_of(d, struct coreboot_device, dev)
# define CB_DRV(d) container_of(d, struct coreboot_driver, drv)
2017-03-28 18:11:27 +02:00
2018-01-24 19:41:16 -06:00
static int coreboot_bus_match ( struct device * dev , struct device_driver * drv )
{
struct coreboot_device * device = CB_DEV ( dev ) ;
struct coreboot_driver * driver = CB_DRV ( drv ) ;
return device - > entry . tag = = driver - > tag ;
}
static int coreboot_bus_probe ( struct device * dev )
{
int ret = - ENODEV ;
struct coreboot_device * device = CB_DEV ( dev ) ;
struct coreboot_driver * driver = CB_DRV ( dev - > driver ) ;
if ( driver - > probe )
ret = driver - > probe ( device ) ;
return ret ;
}
static int coreboot_bus_remove ( struct device * dev )
{
struct coreboot_device * device = CB_DEV ( dev ) ;
struct coreboot_driver * driver = CB_DRV ( dev - > driver ) ;
if ( driver - > remove )
2021-01-26 22:53:39 +01:00
driver - > remove ( device ) ;
2018-01-24 19:41:16 -06:00
2021-01-26 22:53:39 +01:00
return 0 ;
2018-01-24 19:41:16 -06:00
}
static struct bus_type coreboot_bus_type = {
. name = " coreboot " ,
. match = coreboot_bus_match ,
. probe = coreboot_bus_probe ,
. remove = coreboot_bus_remove ,
} ;
static void coreboot_device_release ( struct device * dev )
{
struct coreboot_device * device = CB_DEV ( dev ) ;
kfree ( device ) ;
}
int coreboot_driver_register ( struct coreboot_driver * driver )
{
driver - > drv . bus = & coreboot_bus_type ;
return driver_register ( & driver - > drv ) ;
}
EXPORT_SYMBOL ( coreboot_driver_register ) ;
void coreboot_driver_unregister ( struct coreboot_driver * driver )
{
driver_unregister ( & driver - > drv ) ;
}
EXPORT_SYMBOL ( coreboot_driver_unregister ) ;
2018-08-15 13:37:08 -07:00
static int coreboot_table_populate ( struct device * dev , void * ptr )
2017-03-28 18:11:27 +02:00
{
2018-01-24 19:41:16 -06:00
int i , ret ;
void * ptr_entry ;
struct coreboot_device * device ;
2018-08-15 13:37:07 -07:00
struct coreboot_table_entry * entry ;
2018-08-15 13:37:08 -07:00
struct coreboot_table_header * header = ptr ;
2017-03-28 18:11:27 +02:00
2018-08-15 13:37:08 -07:00
ptr_entry = ptr + header - > header_bytes ;
2018-08-15 13:37:07 -07:00
for ( i = 0 ; i < header - > table_entries ; i + + ) {
entry = ptr_entry ;
2018-01-24 19:41:16 -06:00
2018-08-15 13:37:07 -07:00
device = kzalloc ( sizeof ( struct device ) + entry - > size , GFP_KERNEL ) ;
2018-08-15 13:37:08 -07:00
if ( ! device )
return - ENOMEM ;
2018-01-24 19:41:16 -06:00
dev_set_name ( & device - > dev , " coreboot%d " , i ) ;
device - > dev . parent = dev ;
device - > dev . bus = & coreboot_bus_type ;
device - > dev . release = coreboot_device_release ;
2018-08-15 13:37:07 -07:00
memcpy ( & device - > entry , ptr_entry , entry - > size ) ;
2018-01-24 19:41:16 -06:00
ret = device_register ( & device - > dev ) ;
if ( ret ) {
put_device ( & device - > dev ) ;
2018-08-15 13:37:08 -07:00
return ret ;
2018-01-24 19:41:16 -06:00
}
2018-08-15 13:37:07 -07:00
ptr_entry + = entry - > size ;
2018-01-24 19:41:16 -06:00
}
2018-08-15 13:37:05 -07:00
2018-08-15 13:37:08 -07:00
return 0 ;
2017-03-28 18:11:27 +02:00
}
2018-08-15 13:37:06 -07:00
static int coreboot_table_probe ( struct platform_device * pdev )
{
resource_size_t len ;
2018-08-15 13:37:07 -07:00
struct coreboot_table_header * header ;
2018-08-15 13:37:06 -07:00
struct resource * res ;
2018-08-15 13:37:08 -07:00
struct device * dev = & pdev - > dev ;
2018-08-15 13:37:07 -07:00
void * ptr ;
2018-08-15 13:37:08 -07:00
int ret ;
2018-08-15 13:37:06 -07:00
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( ! res )
return - EINVAL ;
len = resource_size ( res ) ;
if ( ! res - > start | | ! len )
return - EINVAL ;
2018-08-15 13:37:08 -07:00
/* Check just the header first to make sure things are sane */
2018-08-15 13:37:07 -07:00
header = memremap ( res - > start , sizeof ( * header ) , MEMREMAP_WB ) ;
2018-08-15 13:37:08 -07:00
if ( ! header )
2018-08-15 13:37:06 -07:00
return - ENOMEM ;
2018-08-15 13:37:08 -07:00
len = header - > header_bytes + header - > table_bytes ;
ret = strncmp ( header - > signature , " LBIO " , sizeof ( header - > signature ) ) ;
2018-08-15 13:37:07 -07:00
memunmap ( header ) ;
2018-08-15 13:37:08 -07:00
if ( ret ) {
dev_warn ( dev , " coreboot table missing or corrupt! \n " ) ;
return - ENODEV ;
}
ptr = memremap ( res - > start , len , MEMREMAP_WB ) ;
2018-08-15 13:37:06 -07:00
if ( ! ptr )
return - ENOMEM ;
2018-08-15 13:37:08 -07:00
ret = bus_register ( & coreboot_bus_type ) ;
if ( ! ret ) {
ret = coreboot_table_populate ( dev , ptr ) ;
if ( ret )
bus_unregister ( & coreboot_bus_type ) ;
}
memunmap ( ptr ) ;
return ret ;
2018-08-15 13:37:06 -07:00
}
2019-11-18 11:19:29 +01:00
static int __cb_dev_unregister ( struct device * dev , void * dummy )
{
device_unregister ( dev ) ;
return 0 ;
}
2018-08-15 13:37:06 -07:00
static int coreboot_table_remove ( struct platform_device * pdev )
2017-03-28 18:11:27 +02:00
{
2019-11-18 11:19:29 +01:00
bus_for_each_dev ( & coreboot_bus_type , NULL , NULL , __cb_dev_unregister ) ;
2018-08-15 13:37:08 -07:00
bus_unregister ( & coreboot_bus_type ) ;
2017-03-28 18:11:27 +02:00
return 0 ;
}
2018-08-15 13:37:06 -07:00
# ifdef CONFIG_ACPI
static const struct acpi_device_id cros_coreboot_acpi_match [ ] = {
{ " GOOGCB00 " , 0 } ,
{ " BOOT0000 " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( acpi , cros_coreboot_acpi_match ) ;
# endif
# ifdef CONFIG_OF
static const struct of_device_id coreboot_of_match [ ] = {
{ . compatible = " coreboot " } ,
{ }
} ;
MODULE_DEVICE_TABLE ( of , coreboot_of_match ) ;
# endif
static struct platform_driver coreboot_table_driver = {
. probe = coreboot_table_probe ,
. remove = coreboot_table_remove ,
. driver = {
. name = " coreboot_table " ,
. acpi_match_table = ACPI_PTR ( cros_coreboot_acpi_match ) ,
. of_match_table = of_match_ptr ( coreboot_of_match ) ,
} ,
} ;
module_platform_driver ( coreboot_table_driver ) ;
2017-03-28 18:11:27 +02:00
MODULE_AUTHOR ( " Google, Inc. " ) ;
MODULE_LICENSE ( " GPL " ) ;