2013-02-26 21:48:49 -08:00
/*
* Copyright ( C ) 2013 Broadcom Corporation
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation version 2.
*
* This program is distributed " as is " WITHOUT ANY WARRANTY of any
* kind , whether express or implied ; without even the implied warranty
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*/
# include <stdarg.h>
# include <linux/smp.h>
# include <linux/io.h>
# include <linux/ioport.h>
# include <asm/cacheflush.h>
# include <linux/of_address.h>
# include "bcm_kona_smc.h"
struct secure_bridge_data {
void __iomem * bounce ; /* virtual address */
u32 __iomem buffer_addr ; /* physical address */
int initialized ;
} bridge_data ;
struct bcm_kona_smc_data {
unsigned service_id ;
unsigned arg0 ;
unsigned arg1 ;
unsigned arg2 ;
unsigned arg3 ;
} ;
static const struct of_device_id bcm_kona_smc_ids [ ] __initconst = {
2013-08-20 08:37:19 -07:00
{ . compatible = " brcm,kona-smc " } ,
{ . compatible = " bcm,kona-smc " } , /* deprecated name */
2013-02-26 21:48:49 -08:00
{ } ,
} ;
/* Map in the bounce area */
2013-08-06 17:04:54 -07:00
int __init bcm_kona_smc_init ( void )
2013-02-26 21:48:49 -08:00
{
struct device_node * node ;
2014-04-21 16:53:03 -05:00
const __be32 * prop_val ;
2013-02-26 21:48:49 -08:00
/* Read buffer addr and size from the device tree node */
node = of_find_matching_node ( NULL , bcm_kona_smc_ids ) ;
2013-08-06 17:04:54 -07:00
if ( ! node )
return - ENODEV ;
2013-02-26 21:48:49 -08:00
/* Don't care about size or flags of the DT node */
2014-04-21 16:53:03 -05:00
prop_val = of_get_address ( node , 0 , NULL , NULL ) ;
if ( ! prop_val )
return - EINVAL ;
bridge_data . buffer_addr = be32_to_cpu ( * prop_val ) ;
if ( ! bridge_data . buffer_addr )
return - EINVAL ;
2013-02-26 21:48:49 -08:00
bridge_data . bounce = of_iomap ( node , 0 ) ;
2014-04-21 16:53:03 -05:00
if ( ! bridge_data . bounce )
return - ENOMEM ;
2013-02-26 21:48:49 -08:00
bridge_data . initialized = 1 ;
2013-08-06 17:04:54 -07:00
pr_info ( " Kona Secure API initialized \n " ) ;
return 0 ;
2013-02-26 21:48:49 -08:00
}
/* __bcm_kona_smc() should only run on CPU 0, with pre-emption disabled */
static void __bcm_kona_smc ( void * info )
{
struct bcm_kona_smc_data * data = info ;
u32 * args = bridge_data . bounce ;
int rc = 0 ;
/* Must run on CPU 0 */
BUG_ON ( smp_processor_id ( ) ! = 0 ) ;
/* Check map in the bounce area */
BUG_ON ( ! bridge_data . initialized ) ;
2014-04-21 16:53:02 -05:00
/* Copy the four 32 bit argument values into the bounce area */
writel_relaxed ( data - > arg0 , args + + ) ;
writel_relaxed ( data - > arg1 , args + + ) ;
writel_relaxed ( data - > arg2 , args + + ) ;
writel ( data - > arg3 , args ) ;
2013-02-26 21:48:49 -08:00
/* Flush caches for input data passed to Secure Monitor */
if ( data - > service_id ! = SSAPI_BRCM_START_VC_CORE )
flush_cache_all ( ) ;
/* Trap into Secure Monitor */
rc = bcm_kona_smc_asm ( data - > service_id , bridge_data . buffer_addr ) ;
if ( rc ! = SEC_ROM_RET_OK )
pr_err ( " Secure Monitor call failed (0x%x)! \n " , rc ) ;
}
unsigned bcm_kona_smc ( unsigned service_id , unsigned arg0 , unsigned arg1 ,
unsigned arg2 , unsigned arg3 )
{
struct bcm_kona_smc_data data ;
data . service_id = service_id ;
data . arg0 = arg0 ;
data . arg1 = arg1 ;
data . arg2 = arg2 ;
data . arg3 = arg3 ;
/*
* Due to a limitation of the secure monitor , we must use the SMP
* infrastructure to forward all secure monitor calls to Core 0.
*/
if ( get_cpu ( ) ! = 0 )
smp_call_function_single ( 0 , __bcm_kona_smc , ( void * ) & data , 1 ) ;
else
__bcm_kona_smc ( & data ) ;
put_cpu ( ) ;
return 0 ;
}