2014-05-13 17:11:28 +08:00
/*
* Hisilicon SoC reset code
*
* Copyright ( c ) 2014 Hisilicon Ltd .
* Copyright ( c ) 2014 Linaro Ltd .
*
* Author : Haojian Zhuang < haojian . zhuang @ linaro . org >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
# include <linux/delay.h>
# include <linux/io.h>
# include <linux/module.h>
2014-09-30 10:48:35 -07:00
# include <linux/notifier.h>
2014-05-13 17:11:28 +08:00
# include <linux/of_address.h>
# include <linux/platform_device.h>
# include <linux/reboot.h>
# include <asm/proc-fns.h>
static void __iomem * base ;
static u32 reboot_offset ;
2014-09-30 10:48:35 -07:00
static int hisi_restart_handler ( struct notifier_block * this ,
unsigned long mode , void * cmd )
2014-05-13 17:11:28 +08:00
{
writel_relaxed ( 0xdeadbeef , base + reboot_offset ) ;
while ( 1 )
cpu_do_idle ( ) ;
2014-09-30 10:48:35 -07:00
return NOTIFY_DONE ;
2014-05-13 17:11:28 +08:00
}
2014-09-30 10:48:35 -07:00
static struct notifier_block hisi_restart_nb = {
. notifier_call = hisi_restart_handler ,
. priority = 128 ,
} ;
2014-05-13 17:11:28 +08:00
static int hisi_reboot_probe ( struct platform_device * pdev )
{
struct device_node * np = pdev - > dev . of_node ;
2014-09-30 10:48:35 -07:00
int err ;
2014-05-13 17:11:28 +08:00
base = of_iomap ( np , 0 ) ;
if ( ! base ) {
WARN ( 1 , " failed to map base address " ) ;
return - ENODEV ;
}
if ( of_property_read_u32 ( np , " reboot-offset " , & reboot_offset ) < 0 ) {
pr_err ( " failed to find reboot-offset property \n " ) ;
return - EINVAL ;
}
2014-09-30 10:48:35 -07:00
err = register_restart_handler ( & hisi_restart_nb ) ;
if ( err )
dev_err ( & pdev - > dev , " cannot register restart handler (err=%d) \n " ,
err ) ;
2014-05-13 17:11:28 +08:00
2014-09-30 10:48:35 -07:00
return err ;
2014-05-13 17:11:28 +08:00
}
2015-03-16 20:17:12 +01:00
static const struct of_device_id hisi_reboot_of_match [ ] = {
2014-05-13 17:11:28 +08:00
{ . compatible = " hisilicon,sysctrl " } ,
{ }
} ;
static struct platform_driver hisi_reboot_driver = {
. probe = hisi_reboot_probe ,
. driver = {
. name = " hisi-reboot " ,
. of_match_table = hisi_reboot_of_match ,
} ,
} ;
module_platform_driver ( hisi_reboot_driver ) ;