2015-04-10 23:15:59 +03:00
/* Copyright (c) 2010,2015, The Linux Foundation. All rights reserved.
2015-03-03 02:30:29 +03:00
* Copyright ( C ) 2015 Linaro Ltd .
2010-08-27 21:01:23 +04:00
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 and
* only version 2 as published by the Free Software Foundation .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
*/
2016-06-04 02:25:22 +03:00
# include <linux/platform_device.h>
# include <linux/module.h>
2015-03-12 00:28:10 +03:00
# include <linux/cpumask.h>
# include <linux/export.h>
# include <linux/types.h>
2015-02-27 00:49:09 +03:00
# include <linux/qcom_scm.h>
2016-06-04 02:25:22 +03:00
# include <linux/of.h>
# include <linux/of_platform.h>
# include <linux/clk.h>
2010-08-27 21:01:23 +04:00
2015-03-12 00:28:10 +03:00
# include "qcom_scm.h"
2015-03-03 02:30:28 +03:00
2016-06-04 02:25:22 +03:00
struct qcom_scm {
struct device * dev ;
struct clk * core_clk ;
struct clk * iface_clk ;
struct clk * bus_clk ;
} ;
static struct qcom_scm * __scm ;
static int qcom_scm_clk_enable ( void )
{
int ret ;
ret = clk_prepare_enable ( __scm - > core_clk ) ;
if ( ret )
goto bail ;
ret = clk_prepare_enable ( __scm - > iface_clk ) ;
if ( ret )
goto disable_core ;
ret = clk_prepare_enable ( __scm - > bus_clk ) ;
if ( ret )
goto disable_iface ;
return 0 ;
disable_iface :
clk_disable_unprepare ( __scm - > iface_clk ) ;
disable_core :
clk_disable_unprepare ( __scm - > core_clk ) ;
bail :
return ret ;
}
static void qcom_scm_clk_disable ( void )
{
clk_disable_unprepare ( __scm - > core_clk ) ;
clk_disable_unprepare ( __scm - > iface_clk ) ;
clk_disable_unprepare ( __scm - > bus_clk ) ;
}
2015-03-03 02:30:28 +03:00
/**
* qcom_scm_set_cold_boot_addr ( ) - Set the cold boot address for cpus
* @ entry : Entry point function for the cpus
* @ cpus : The cpumask of cpus that will use the entry point
*
* Set the cold boot address of the cpus . Any cpu outside the supported
* range would be removed from the cpu present mask .
*/
int qcom_scm_set_cold_boot_addr ( void * entry , const cpumask_t * cpus )
{
2015-03-12 00:28:10 +03:00
return __qcom_scm_set_cold_boot_addr ( entry , cpus ) ;
2015-03-03 02:30:28 +03:00
}
EXPORT_SYMBOL ( qcom_scm_set_cold_boot_addr ) ;
2015-03-03 02:30:29 +03:00
/**
* qcom_scm_set_warm_boot_addr ( ) - Set the warm boot address for cpus
* @ entry : Entry point function for the cpus
* @ cpus : The cpumask of cpus that will use the entry point
*
* Set the Linux entry point for the SCM to transfer control to when coming
* out of a power down . CPU power down may be executed on cpuidle or hotplug .
*/
int qcom_scm_set_warm_boot_addr ( void * entry , const cpumask_t * cpus )
{
2016-06-04 02:25:25 +03:00
return __qcom_scm_set_warm_boot_addr ( __scm - > dev , entry , cpus ) ;
2015-03-03 02:30:29 +03:00
}
EXPORT_SYMBOL ( qcom_scm_set_warm_boot_addr ) ;
2015-03-03 02:30:30 +03:00
/**
* qcom_scm_cpu_power_down ( ) - Power down the cpu
* @ flags - Flags to flush cache
*
* This is an end point to power down cpu . If there was a pending interrupt ,
* the control would return from this function , otherwise , the cpu jumps to the
* warm boot entry point set for this cpu upon reset .
*/
void qcom_scm_cpu_power_down ( u32 flags )
{
2015-03-12 00:28:10 +03:00
__qcom_scm_cpu_power_down ( flags ) ;
2015-03-03 02:30:30 +03:00
}
EXPORT_SYMBOL ( qcom_scm_cpu_power_down ) ;
2015-04-10 23:15:59 +03:00
/**
* qcom_scm_hdcp_available ( ) - Check if secure environment supports HDCP .
*
* Return true if HDCP is supported , false if not .
*/
bool qcom_scm_hdcp_available ( void )
{
2016-06-04 02:25:22 +03:00
int ret = qcom_scm_clk_enable ( ) ;
if ( ret )
return ret ;
2015-04-10 23:15:59 +03:00
2016-06-04 02:25:25 +03:00
ret = __qcom_scm_is_call_available ( __scm - > dev , QCOM_SCM_SVC_HDCP ,
2016-06-04 02:25:22 +03:00
QCOM_SCM_CMD_HDCP ) ;
qcom_scm_clk_disable ( ) ;
2015-04-10 23:15:59 +03:00
2016-06-04 02:25:22 +03:00
return ret > 0 ? true : false ;
2015-04-10 23:15:59 +03:00
}
EXPORT_SYMBOL ( qcom_scm_hdcp_available ) ;
/**
* qcom_scm_hdcp_req ( ) - Send HDCP request .
* @ req : HDCP request array
* @ req_cnt : HDCP request array count
* @ resp : response buffer passed to SCM
*
* Write HDCP register ( s ) through SCM .
*/
int qcom_scm_hdcp_req ( struct qcom_scm_hdcp_req * req , u32 req_cnt , u32 * resp )
{
2016-06-04 02:25:22 +03:00
int ret = qcom_scm_clk_enable ( ) ;
if ( ret )
return ret ;
2016-06-04 02:25:25 +03:00
ret = __qcom_scm_hdcp_req ( __scm - > dev , req , req_cnt , resp ) ;
2016-06-04 02:25:22 +03:00
qcom_scm_clk_disable ( ) ;
return ret ;
2015-04-10 23:15:59 +03:00
}
EXPORT_SYMBOL ( qcom_scm_hdcp_req ) ;
2016-06-04 02:25:22 +03:00
static int qcom_scm_probe ( struct platform_device * pdev )
{
struct qcom_scm * scm ;
int ret ;
scm = devm_kzalloc ( & pdev - > dev , sizeof ( * scm ) , GFP_KERNEL ) ;
if ( ! scm )
return - ENOMEM ;
scm - > core_clk = devm_clk_get ( & pdev - > dev , " core " ) ;
if ( IS_ERR ( scm - > core_clk ) ) {
if ( PTR_ERR ( scm - > core_clk ) = = - EPROBE_DEFER )
return PTR_ERR ( scm - > core_clk ) ;
scm - > core_clk = NULL ;
}
if ( of_device_is_compatible ( pdev - > dev . of_node , " qcom,scm " ) ) {
scm - > iface_clk = devm_clk_get ( & pdev - > dev , " iface " ) ;
if ( IS_ERR ( scm - > iface_clk ) ) {
if ( PTR_ERR ( scm - > iface_clk ) ! = - EPROBE_DEFER )
dev_err ( & pdev - > dev , " failed to acquire iface clk \n " ) ;
return PTR_ERR ( scm - > iface_clk ) ;
}
scm - > bus_clk = devm_clk_get ( & pdev - > dev , " bus " ) ;
if ( IS_ERR ( scm - > bus_clk ) ) {
if ( PTR_ERR ( scm - > bus_clk ) ! = - EPROBE_DEFER )
dev_err ( & pdev - > dev , " failed to acquire bus clk \n " ) ;
return PTR_ERR ( scm - > bus_clk ) ;
}
}
/* vote for max clk rate for highest performance */
ret = clk_set_rate ( scm - > core_clk , INT_MAX ) ;
if ( ret )
return ret ;
__scm = scm ;
__scm - > dev = & pdev - > dev ;
return 0 ;
}
static const struct of_device_id qcom_scm_dt_match [ ] = {
{ . compatible = " qcom,scm-apq8064 " , } ,
{ . compatible = " qcom,scm-msm8660 " , } ,
{ . compatible = " qcom,scm-msm8960 " , } ,
{ . compatible = " qcom,scm " , } ,
{ }
} ;
MODULE_DEVICE_TABLE ( of , qcom_scm_dt_match ) ;
static struct platform_driver qcom_scm_driver = {
. driver = {
. name = " qcom_scm " ,
. of_match_table = qcom_scm_dt_match ,
} ,
. probe = qcom_scm_probe ,
} ;
static int __init qcom_scm_init ( void )
{
struct device_node * np , * fw_np ;
int ret ;
fw_np = of_find_node_by_name ( NULL , " firmware " ) ;
if ( ! fw_np )
return - ENODEV ;
np = of_find_matching_node ( fw_np , qcom_scm_dt_match ) ;
if ( ! np ) {
of_node_put ( fw_np ) ;
return - ENODEV ;
}
of_node_put ( np ) ;
ret = of_platform_populate ( fw_np , qcom_scm_dt_match , NULL , NULL ) ;
of_node_put ( fw_np ) ;
if ( ret )
return ret ;
return platform_driver_register ( & qcom_scm_driver ) ;
}
arch_initcall ( qcom_scm_init ) ;
static void __exit qcom_scm_exit ( void )
{
platform_driver_unregister ( & qcom_scm_driver ) ;
}
module_exit ( qcom_scm_exit ) ;
MODULE_DESCRIPTION ( " Qualcomm SCM driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;