2016-08-22 22:57:44 -07:00
/*
2017-01-30 20:33:08 +05:30
* Qualcomm ADSP / SLPI Peripheral Image Loader for MSM8974 and MSM8996
2016-08-22 22:57:44 -07:00
*
* Copyright ( C ) 2016 Linaro Ltd
* Copyright ( C ) 2014 Sony Mobile Communications AB
* Copyright ( c ) 2012 - 2013 , The Linux Foundation . All rights reserved .
*
* 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 .
*
* 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-10-25 13:57:26 -07:00
# include <linux/clk.h>
2016-08-22 22:57:44 -07:00
# include <linux/firmware.h>
# include <linux/interrupt.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/of_address.h>
# include <linux/of_device.h>
# include <linux/platform_device.h>
# include <linux/qcom_scm.h>
# include <linux/regulator/consumer.h>
# include <linux/remoteproc.h>
2017-01-27 03:12:57 -08:00
# include <linux/soc/qcom/mdt_loader.h>
2016-08-22 22:57:44 -07:00
# include <linux/soc/qcom/smem.h>
# include <linux/soc/qcom/smem_state.h>
2017-01-27 02:28:32 -08:00
# include "qcom_common.h"
2018-06-04 13:30:37 -07:00
# include "qcom_q6v5.h"
2016-08-22 22:57:44 -07:00
# include "remoteproc_internal.h"
2017-01-30 20:33:06 +05:30
struct adsp_data {
int crash_reason_smem ;
const char * firmware_name ;
int pas_id ;
2017-01-30 20:33:07 +05:30
bool has_aggre2_clk ;
2017-08-27 21:51:38 -07:00
2017-07-24 22:56:43 -07:00
const char * ssr_name ;
2017-08-27 21:51:38 -07:00
const char * sysmon_name ;
int ssctl_id ;
2017-01-30 20:33:06 +05:30
} ;
2016-08-22 22:57:44 -07:00
struct qcom_adsp {
struct device * dev ;
struct rproc * rproc ;
2018-06-04 13:30:37 -07:00
struct qcom_q6v5 q6v5 ;
2016-08-22 22:57:44 -07:00
2016-10-25 13:57:26 -07:00
struct clk * xo ;
2017-01-30 20:33:07 +05:30
struct clk * aggre2_clk ;
2016-10-25 13:57:26 -07:00
2016-08-22 22:57:44 -07:00
struct regulator * cx_supply ;
2017-01-30 20:33:07 +05:30
struct regulator * px_supply ;
2016-08-22 22:57:44 -07:00
2017-01-30 20:33:06 +05:30
int pas_id ;
int crash_reason_smem ;
2017-01-30 20:33:07 +05:30
bool has_aggre2_clk ;
2017-01-30 20:33:06 +05:30
2016-08-22 22:57:44 -07:00
struct completion start_done ;
struct completion stop_done ;
phys_addr_t mem_phys ;
phys_addr_t mem_reloc ;
void * mem_region ;
size_t mem_size ;
2017-01-29 14:05:50 -08:00
2017-08-29 16:13:35 -07:00
struct qcom_rproc_glink glink_subdev ;
2017-01-29 14:05:50 -08:00
struct qcom_rproc_subdev smd_subdev ;
2017-07-24 22:56:43 -07:00
struct qcom_rproc_ssr ssr_subdev ;
2017-08-27 21:51:38 -07:00
struct qcom_sysmon * sysmon ;
2016-08-22 22:57:44 -07:00
} ;
static int adsp_load ( struct rproc * rproc , const struct firmware * fw )
{
struct qcom_adsp * adsp = ( struct qcom_adsp * ) rproc - > priv ;
2017-01-27 02:17:23 -08:00
return qcom_mdt_load ( adsp - > dev , fw , rproc - > firmware , adsp - > pas_id ,
2018-01-05 16:04:19 -08:00
adsp - > mem_region , adsp - > mem_phys , adsp - > mem_size ,
& adsp - > mem_reloc ) ;
2016-08-22 22:57:44 -07:00
}
static int adsp_start ( struct rproc * rproc )
{
struct qcom_adsp * adsp = ( struct qcom_adsp * ) rproc - > priv ;
int ret ;
2018-06-04 13:30:37 -07:00
qcom_q6v5_prepare ( & adsp - > q6v5 ) ;
2016-10-25 13:57:26 -07:00
ret = clk_prepare_enable ( adsp - > xo ) ;
2016-08-22 22:57:44 -07:00
if ( ret )
return ret ;
2017-01-30 20:33:07 +05:30
ret = clk_prepare_enable ( adsp - > aggre2_clk ) ;
if ( ret )
goto disable_xo_clk ;
2016-10-25 13:57:26 -07:00
ret = regulator_enable ( adsp - > cx_supply ) ;
if ( ret )
2017-01-30 20:33:07 +05:30
goto disable_aggre2_clk ;
ret = regulator_enable ( adsp - > px_supply ) ;
if ( ret )
goto disable_cx_supply ;
2016-10-25 13:57:26 -07:00
2017-01-30 20:33:06 +05:30
ret = qcom_scm_pas_auth_and_reset ( adsp - > pas_id ) ;
2016-08-22 22:57:44 -07:00
if ( ret ) {
dev_err ( adsp - > dev ,
" failed to authenticate image and release reset \n " ) ;
2017-01-30 20:33:07 +05:30
goto disable_px_supply ;
2016-08-22 22:57:44 -07:00
}
2018-06-04 13:30:37 -07:00
ret = qcom_q6v5_wait_for_start ( & adsp - > q6v5 , msecs_to_jiffies ( 5000 ) ) ;
if ( ret = = - ETIMEDOUT ) {
2016-08-22 22:57:44 -07:00
dev_err ( adsp - > dev , " start timed out \n " ) ;
2017-01-30 20:33:06 +05:30
qcom_scm_pas_shutdown ( adsp - > pas_id ) ;
2017-01-30 20:33:07 +05:30
goto disable_px_supply ;
2016-08-22 22:57:44 -07:00
}
2018-06-04 13:30:37 -07:00
return 0 ;
2016-08-22 22:57:44 -07:00
2017-01-30 20:33:07 +05:30
disable_px_supply :
regulator_disable ( adsp - > px_supply ) ;
disable_cx_supply :
2016-08-22 22:57:44 -07:00
regulator_disable ( adsp - > cx_supply ) ;
2017-01-30 20:33:07 +05:30
disable_aggre2_clk :
clk_disable_unprepare ( adsp - > aggre2_clk ) ;
disable_xo_clk :
2016-10-25 13:57:26 -07:00
clk_disable_unprepare ( adsp - > xo ) ;
2016-08-22 22:57:44 -07:00
return ret ;
}
2018-06-04 13:30:37 -07:00
static void qcom_pas_handover ( struct qcom_q6v5 * q6v5 )
{
struct qcom_adsp * adsp = container_of ( q6v5 , struct qcom_adsp , q6v5 ) ;
regulator_disable ( adsp - > px_supply ) ;
regulator_disable ( adsp - > cx_supply ) ;
clk_disable_unprepare ( adsp - > aggre2_clk ) ;
clk_disable_unprepare ( adsp - > xo ) ;
}
2016-08-22 22:57:44 -07:00
static int adsp_stop ( struct rproc * rproc )
{
struct qcom_adsp * adsp = ( struct qcom_adsp * ) rproc - > priv ;
2018-06-04 13:30:37 -07:00
int handover ;
2016-08-22 22:57:44 -07:00
int ret ;
2018-06-04 13:30:37 -07:00
ret = qcom_q6v5_request_stop ( & adsp - > q6v5 ) ;
if ( ret = = - ETIMEDOUT )
2016-08-22 22:57:44 -07:00
dev_err ( adsp - > dev , " timed out on wait \n " ) ;
2017-01-30 20:33:06 +05:30
ret = qcom_scm_pas_shutdown ( adsp - > pas_id ) ;
2016-08-22 22:57:44 -07:00
if ( ret )
dev_err ( adsp - > dev , " failed to shutdown: %d \n " , ret ) ;
2018-06-04 13:30:37 -07:00
handover = qcom_q6v5_unprepare ( & adsp - > q6v5 ) ;
if ( handover )
qcom_pas_handover ( & adsp - > q6v5 ) ;
2016-08-22 22:57:44 -07:00
return ret ;
}
static void * adsp_da_to_va ( struct rproc * rproc , u64 da , int len )
{
struct qcom_adsp * adsp = ( struct qcom_adsp * ) rproc - > priv ;
int offset ;
offset = da - adsp - > mem_reloc ;
if ( offset < 0 | | offset + len > adsp - > mem_size )
return NULL ;
return adsp - > mem_region + offset ;
}
static const struct rproc_ops adsp_ops = {
. start = adsp_start ,
. stop = adsp_stop ,
. da_to_va = adsp_da_to_va ,
2018-01-05 16:04:20 -08:00
. parse_fw = qcom_register_dump_segments ,
2018-01-05 15:58:01 -08:00
. load = adsp_load ,
2016-08-22 22:57:44 -07:00
} ;
2016-10-25 13:57:26 -07:00
static int adsp_init_clock ( struct qcom_adsp * adsp )
{
int ret ;
adsp - > xo = devm_clk_get ( adsp - > dev , " xo " ) ;
if ( IS_ERR ( adsp - > xo ) ) {
ret = PTR_ERR ( adsp - > xo ) ;
if ( ret ! = - EPROBE_DEFER )
dev_err ( adsp - > dev , " failed to get xo clock " ) ;
return ret ;
}
2017-01-30 20:33:07 +05:30
if ( adsp - > has_aggre2_clk ) {
adsp - > aggre2_clk = devm_clk_get ( adsp - > dev , " aggre2 " ) ;
if ( IS_ERR ( adsp - > aggre2_clk ) ) {
ret = PTR_ERR ( adsp - > aggre2_clk ) ;
if ( ret ! = - EPROBE_DEFER )
dev_err ( adsp - > dev ,
" failed to get aggre2 clock " ) ;
return ret ;
}
}
2016-10-25 13:57:26 -07:00
return 0 ;
}
2016-08-22 22:57:44 -07:00
static int adsp_init_regulator ( struct qcom_adsp * adsp )
{
adsp - > cx_supply = devm_regulator_get ( adsp - > dev , " cx " ) ;
if ( IS_ERR ( adsp - > cx_supply ) )
return PTR_ERR ( adsp - > cx_supply ) ;
regulator_set_load ( adsp - > cx_supply , 100000 ) ;
2017-01-30 20:33:07 +05:30
adsp - > px_supply = devm_regulator_get ( adsp - > dev , " px " ) ;
2017-08-29 19:13:18 +05:30
return PTR_ERR_OR_ZERO ( adsp - > px_supply ) ;
2016-08-22 22:57:44 -07:00
}
static int adsp_alloc_memory_region ( struct qcom_adsp * adsp )
{
struct device_node * node ;
struct resource r ;
int ret ;
node = of_parse_phandle ( adsp - > dev - > of_node , " memory-region " , 0 ) ;
if ( ! node ) {
dev_err ( adsp - > dev , " no memory-region specified \n " ) ;
return - EINVAL ;
}
ret = of_address_to_resource ( node , 0 , & r ) ;
if ( ret )
return ret ;
adsp - > mem_phys = adsp - > mem_reloc = r . start ;
adsp - > mem_size = resource_size ( & r ) ;
adsp - > mem_region = devm_ioremap_wc ( adsp - > dev , adsp - > mem_phys , adsp - > mem_size ) ;
if ( ! adsp - > mem_region ) {
dev_err ( adsp - > dev , " unable to map memory region: %pa+%zx \n " ,
& r . start , adsp - > mem_size ) ;
return - EBUSY ;
}
return 0 ;
}
static int adsp_probe ( struct platform_device * pdev )
{
2017-01-30 20:33:06 +05:30
const struct adsp_data * desc ;
2016-08-22 22:57:44 -07:00
struct qcom_adsp * adsp ;
struct rproc * rproc ;
2019-01-15 01:20:01 +05:30
const char * fw_name ;
2016-08-22 22:57:44 -07:00
int ret ;
2017-01-30 20:33:06 +05:30
desc = of_device_get_match_data ( & pdev - > dev ) ;
if ( ! desc )
return - EINVAL ;
2016-08-22 22:57:44 -07:00
if ( ! qcom_scm_is_available ( ) )
return - EPROBE_DEFER ;
2019-01-15 01:20:01 +05:30
fw_name = desc - > firmware_name ;
ret = of_property_read_string ( pdev - > dev . of_node , " firmware-name " ,
& fw_name ) ;
if ( ret < 0 & & ret ! = - EINVAL )
return ret ;
2016-08-22 22:57:44 -07:00
rproc = rproc_alloc ( & pdev - > dev , pdev - > name , & adsp_ops ,
2019-01-15 01:20:01 +05:30
fw_name , sizeof ( * adsp ) ) ;
2016-08-22 22:57:44 -07:00
if ( ! rproc ) {
dev_err ( & pdev - > dev , " unable to allocate remoteproc \n " ) ;
return - ENOMEM ;
}
adsp = ( struct qcom_adsp * ) rproc - > priv ;
adsp - > dev = & pdev - > dev ;
adsp - > rproc = rproc ;
2017-01-30 20:33:06 +05:30
adsp - > pas_id = desc - > pas_id ;
2017-01-30 20:33:07 +05:30
adsp - > has_aggre2_clk = desc - > has_aggre2_clk ;
2016-08-22 22:57:44 -07:00
platform_set_drvdata ( pdev , adsp ) ;
ret = adsp_alloc_memory_region ( adsp ) ;
if ( ret )
goto free_rproc ;
2016-10-25 13:57:26 -07:00
ret = adsp_init_clock ( adsp ) ;
if ( ret )
goto free_rproc ;
2016-08-22 22:57:44 -07:00
ret = adsp_init_regulator ( adsp ) ;
if ( ret )
goto free_rproc ;
2018-06-04 13:30:37 -07:00
ret = qcom_q6v5_init ( & adsp - > q6v5 , pdev , rproc , desc - > crash_reason_smem ,
qcom_pas_handover ) ;
if ( ret )
2016-08-22 22:57:44 -07:00
goto free_rproc ;
2017-08-29 16:13:35 -07:00
qcom_add_glink_subdev ( rproc , & adsp - > glink_subdev ) ;
2017-01-29 14:05:50 -08:00
qcom_add_smd_subdev ( rproc , & adsp - > smd_subdev ) ;
2017-07-24 22:56:43 -07:00
qcom_add_ssr_subdev ( rproc , & adsp - > ssr_subdev , desc - > ssr_name ) ;
2017-08-27 21:51:38 -07:00
adsp - > sysmon = qcom_add_sysmon_subdev ( rproc ,
desc - > sysmon_name ,
desc - > ssctl_id ) ;
2019-01-08 15:53:43 +05:30
if ( IS_ERR ( adsp - > sysmon ) ) {
ret = PTR_ERR ( adsp - > sysmon ) ;
goto free_rproc ;
}
2017-01-29 14:05:50 -08:00
2016-08-22 22:57:44 -07:00
ret = rproc_add ( rproc ) ;
if ( ret )
goto free_rproc ;
return 0 ;
free_rproc :
2016-11-19 22:42:55 -08:00
rproc_free ( rproc ) ;
2016-08-22 22:57:44 -07:00
return ret ;
}
static int adsp_remove ( struct platform_device * pdev )
{
struct qcom_adsp * adsp = platform_get_drvdata ( pdev ) ;
rproc_del ( adsp - > rproc ) ;
2017-01-29 14:05:50 -08:00
2017-08-29 16:13:35 -07:00
qcom_remove_glink_subdev ( adsp - > rproc , & adsp - > glink_subdev ) ;
2017-08-27 21:51:38 -07:00
qcom_remove_sysmon_subdev ( adsp - > sysmon ) ;
2017-01-29 14:05:50 -08:00
qcom_remove_smd_subdev ( adsp - > rproc , & adsp - > smd_subdev ) ;
2017-07-24 22:56:43 -07:00
qcom_remove_ssr_subdev ( adsp - > rproc , & adsp - > ssr_subdev ) ;
2016-11-19 22:42:55 -08:00
rproc_free ( adsp - > rproc ) ;
2016-08-22 22:57:44 -07:00
return 0 ;
}
2017-01-30 20:33:06 +05:30
static const struct adsp_data adsp_resource_init = {
. crash_reason_smem = 423 ,
. firmware_name = " adsp.mdt " ,
. pas_id = 1 ,
2017-01-30 20:33:07 +05:30
. has_aggre2_clk = false ,
2017-07-24 22:56:43 -07:00
. ssr_name = " lpass " ,
2017-08-27 21:51:38 -07:00
. sysmon_name = " adsp " ,
. ssctl_id = 0x14 ,
2017-01-30 20:33:06 +05:30
} ;
2018-08-28 00:14:58 -07:00
static const struct adsp_data cdsp_resource_init = {
. crash_reason_smem = 601 ,
. firmware_name = " cdsp.mdt " ,
. pas_id = 18 ,
. has_aggre2_clk = false ,
. ssr_name = " cdsp " ,
. sysmon_name = " cdsp " ,
. ssctl_id = 0x17 ,
} ;
2017-01-30 20:33:08 +05:30
static const struct adsp_data slpi_resource_init = {
. crash_reason_smem = 424 ,
. firmware_name = " slpi.mdt " ,
. pas_id = 12 ,
. has_aggre2_clk = true ,
2017-07-24 22:56:43 -07:00
. ssr_name = " dsps " ,
2017-08-27 21:51:38 -07:00
. sysmon_name = " slpi " ,
. ssctl_id = 0x16 ,
2017-01-30 20:33:08 +05:30
} ;
2018-09-27 12:03:46 -07:00
static const struct adsp_data wcss_resource_init = {
. crash_reason_smem = 421 ,
. firmware_name = " wcnss.mdt " ,
. pas_id = 6 ,
. ssr_name = " mpss " ,
. sysmon_name = " wcnss " ,
. ssctl_id = 0x12 ,
} ;
2016-08-22 22:57:44 -07:00
static const struct of_device_id adsp_of_match [ ] = {
2017-01-30 20:33:06 +05:30
{ . compatible = " qcom,msm8974-adsp-pil " , . data = & adsp_resource_init } ,
{ . compatible = " qcom,msm8996-adsp-pil " , . data = & adsp_resource_init } ,
2017-01-30 20:33:08 +05:30
{ . compatible = " qcom,msm8996-slpi-pil " , . data = & slpi_resource_init } ,
2018-09-27 12:03:46 -07:00
{ . compatible = " qcom,qcs404-adsp-pas " , . data = & adsp_resource_init } ,
{ . compatible = " qcom,qcs404-cdsp-pas " , . data = & cdsp_resource_init } ,
{ . compatible = " qcom,qcs404-wcss-pas " , . data = & wcss_resource_init } ,
2018-08-28 00:14:58 -07:00
{ . compatible = " qcom,sdm845-adsp-pas " , . data = & adsp_resource_init } ,
{ . compatible = " qcom,sdm845-cdsp-pas " , . data = & cdsp_resource_init } ,
2016-08-22 22:57:44 -07:00
{ } ,
} ;
2016-11-19 22:41:56 -08:00
MODULE_DEVICE_TABLE ( of , adsp_of_match ) ;
2016-08-22 22:57:44 -07:00
static struct platform_driver adsp_driver = {
. probe = adsp_probe ,
. remove = adsp_remove ,
. driver = {
2018-09-24 16:45:25 -07:00
. name = " qcom_q6v5_pas " ,
2016-08-22 22:57:44 -07:00
. of_match_table = adsp_of_match ,
} ,
} ;
module_platform_driver ( adsp_driver ) ;
2018-09-24 16:45:25 -07:00
MODULE_DESCRIPTION ( " Qualcomm Hexagon v5 Peripheral Authentication Service driver " ) ;
2016-08-22 22:57:44 -07:00
MODULE_LICENSE ( " GPL v2 " ) ;