2018-12-12 15:51:47 -08:00
// SPDX-License-Identifier: GPL-2.0+
/*
* PM MFD driver for Broadcom BCM2835
*
* This driver binds to the PM block and creates the MFD device for
2018-12-12 15:51:48 -08:00
* the WDT and power drivers .
2018-12-12 15:51:47 -08:00
*/
# include <linux/delay.h>
# include <linux/io.h>
# include <linux/mfd/bcm2835-pm.h>
# include <linux/mfd/core.h>
# include <linux/module.h>
# include <linux/of_address.h>
# include <linux/of_platform.h>
# include <linux/platform_device.h>
# include <linux/types.h>
# include <linux/watchdog.h>
static const struct mfd_cell bcm2835_pm_devs [ ] = {
{ . name = " bcm2835-wdt " } ,
} ;
2018-12-12 15:51:48 -08:00
static const struct mfd_cell bcm2835_power_devs [ ] = {
{ . name = " bcm2835-power " } ,
} ;
2022-06-25 13:36:14 +02:00
static int bcm2835_pm_get_pdata ( struct platform_device * pdev ,
struct bcm2835_pm * pm )
{
2023-03-10 08:47:12 -06:00
if ( of_property_present ( pm - > dev - > of_node , " reg-names " ) ) {
2022-06-25 13:36:14 +02:00
struct resource * res ;
pm - > base = devm_platform_ioremap_resource_byname ( pdev , " pm " ) ;
if ( IS_ERR ( pm - > base ) )
return PTR_ERR ( pm - > base ) ;
res = platform_get_resource_byname ( pdev , IORESOURCE_MEM , " asb " ) ;
if ( res ) {
pm - > asb = devm_ioremap_resource ( & pdev - > dev , res ) ;
if ( IS_ERR ( pm - > asb ) )
pm - > asb = NULL ;
}
2022-06-25 13:36:15 +02:00
res = platform_get_resource_byname ( pdev , IORESOURCE_MEM ,
" rpivid_asb " ) ;
if ( res ) {
pm - > rpivid_asb = devm_ioremap_resource ( & pdev - > dev , res ) ;
if ( IS_ERR ( pm - > rpivid_asb ) )
pm - > rpivid_asb = NULL ;
}
2022-06-25 13:36:14 +02:00
return 0 ;
}
/* If no 'reg-names' property is found we can assume we're using old DTB. */
pm - > base = devm_platform_ioremap_resource ( pdev , 0 ) ;
if ( IS_ERR ( pm - > base ) )
return PTR_ERR ( pm - > base ) ;
pm - > asb = devm_platform_ioremap_resource ( pdev , 1 ) ;
if ( IS_ERR ( pm - > asb ) )
pm - > asb = NULL ;
2022-06-25 13:36:15 +02:00
pm - > rpivid_asb = devm_platform_ioremap_resource ( pdev , 2 ) ;
if ( IS_ERR ( pm - > rpivid_asb ) )
pm - > rpivid_asb = NULL ;
2022-06-25 13:36:14 +02:00
return 0 ;
}
2018-12-12 15:51:47 -08:00
static int bcm2835_pm_probe ( struct platform_device * pdev )
{
struct device * dev = & pdev - > dev ;
struct bcm2835_pm * pm ;
2018-12-12 15:51:48 -08:00
int ret ;
2018-12-12 15:51:47 -08:00
pm = devm_kzalloc ( dev , sizeof ( * pm ) , GFP_KERNEL ) ;
if ( ! pm )
return - ENOMEM ;
platform_set_drvdata ( pdev , pm ) ;
pm - > dev = dev ;
2022-06-25 13:36:14 +02:00
ret = bcm2835_pm_get_pdata ( pdev , pm ) ;
if ( ret )
return ret ;
2018-12-12 15:51:47 -08:00
2018-12-12 15:51:48 -08:00
ret = devm_mfd_add_devices ( dev , - 1 ,
bcm2835_pm_devs , ARRAY_SIZE ( bcm2835_pm_devs ) ,
NULL , 0 , NULL ) ;
if ( ret )
return ret ;
2022-06-25 13:36:14 +02:00
/*
* We ' ll use the presence of the AXI ASB regs in the
2018-12-12 15:51:48 -08:00
* bcm2835 - pm binding as the key for whether we can reference
* the full PM register range and support power domains .
*/
2022-06-25 13:36:14 +02:00
if ( pm - > asb )
return devm_mfd_add_devices ( dev , - 1 , bcm2835_power_devs ,
ARRAY_SIZE ( bcm2835_power_devs ) ,
NULL , 0 , NULL ) ;
2018-12-12 15:51:48 -08:00
return 0 ;
2018-12-12 15:51:47 -08:00
}
static const struct of_device_id bcm2835_pm_of_match [ ] = {
{ . compatible = " brcm,bcm2835-pm-wdt " , } ,
2018-12-12 15:51:48 -08:00
{ . compatible = " brcm,bcm2835-pm " , } ,
2022-06-25 13:36:15 +02:00
{ . compatible = " brcm,bcm2711-pm " , } ,
2018-12-12 15:51:47 -08:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , bcm2835_pm_of_match ) ;
static struct platform_driver bcm2835_pm_driver = {
. probe = bcm2835_pm_probe ,
. driver = {
. name = " bcm2835-pm " ,
. of_match_table = bcm2835_pm_of_match ,
} ,
} ;
module_platform_driver ( bcm2835_pm_driver ) ;
MODULE_AUTHOR ( " Eric Anholt <eric@anholt.net> " ) ;
MODULE_DESCRIPTION ( " Driver for Broadcom BCM2835 PM MFD " ) ;