2019-05-27 08:55:21 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2016-06-20 14:28:41 -07:00
/*
2018-09-24 16:45:26 -07:00
* Qualcomm self - authenticating modem subsystem remoteproc driver
2016-06-20 14:28:41 -07:00
*
* Copyright ( C ) 2016 Linaro Ltd .
* Copyright ( C ) 2014 Sony Mobile Communications AB
* Copyright ( c ) 2012 - 2013 , The Linux Foundation . All rights reserved .
*/
# include <linux/clk.h>
# include <linux/delay.h>
2020-07-21 16:59:35 +05:30
# include <linux/devcoredump.h>
2022-05-11 11:27:05 +05:30
# include <linux/dma-map-ops.h>
2016-06-20 14:28:41 -07:00
# include <linux/dma-mapping.h>
# include <linux/interrupt.h>
# include <linux/kernel.h>
# include <linux/mfd/syscon.h>
# include <linux/module.h>
# include <linux/of_address.h>
2016-12-30 19:24:00 +05:30
# include <linux/of_device.h>
2016-06-20 14:28:41 -07:00
# include <linux/platform_device.h>
2019-01-30 16:39:30 -08:00
# include <linux/pm_domain.h>
# include <linux/pm_runtime.h>
2016-06-20 14:28:41 -07:00
# include <linux/regmap.h>
# include <linux/regulator/consumer.h>
# include <linux/remoteproc.h>
# include <linux/reset.h>
2017-01-27 03:12:57 -08:00
# include <linux/soc/qcom/mdt_loader.h>
2017-10-24 21:22:27 +05:30
# include <linux/iopoll.h>
2020-06-12 16:57:37 +10:00
# include <linux/slab.h>
2016-06-20 14:28:41 -07:00
# include "remoteproc_internal.h"
2017-01-27 02:28:32 -08:00
# include "qcom_common.h"
2020-06-22 12:19:40 -07:00
# include "qcom_pil_info.h"
2018-06-04 13:30:38 -07:00
# include "qcom_q6v5.h"
2016-06-20 14:28:41 -07:00
# include <linux/qcom_scm.h>
# define MPSS_CRASH_REASON_SMEM 421
2020-07-21 16:59:35 +05:30
# define MBA_LOG_SIZE SZ_4K
2016-06-20 14:28:41 -07:00
/* RMB Status Register Values */
# define RMB_PBL_SUCCESS 0x1
# define RMB_MBA_XPU_UNLOCKED 0x1
# define RMB_MBA_XPU_UNLOCKED_SCRIBBLED 0x2
# define RMB_MBA_META_DATA_AUTH_SUCCESS 0x3
# define RMB_MBA_AUTH_COMPLETE 0x4
/* PBL/MBA interface registers */
# define RMB_MBA_IMAGE_REG 0x00
# define RMB_PBL_STATUS_REG 0x04
# define RMB_MBA_COMMAND_REG 0x08
# define RMB_MBA_STATUS_REG 0x0C
# define RMB_PMI_META_DATA_REG 0x10
# define RMB_PMI_CODE_START_REG 0x14
# define RMB_PMI_CODE_LENGTH_REG 0x18
2018-05-21 22:57:13 +05:30
# define RMB_MBA_MSS_STATUS 0x40
# define RMB_MBA_ALT_RESET 0x44
2016-06-20 14:28:41 -07:00
# define RMB_CMD_META_DATA_READY 0x1
# define RMB_CMD_LOAD_READY 0x2
/* QDSP6SS Register Offsets */
# define QDSP6SS_RESET_REG 0x014
# define QDSP6SS_GFMUX_CTL_REG 0x020
# define QDSP6SS_PWR_CTL_REG 0x030
2017-10-24 21:22:27 +05:30
# define QDSP6SS_MEM_PWR_CTL 0x0B0
2019-10-31 19:45:01 -07:00
# define QDSP6V6SS_MEM_PWR_CTL 0x034
2017-10-24 21:22:27 +05:30
# define QDSP6SS_STRAP_ACC 0x110
2016-06-20 14:28:41 -07:00
/* AXI Halt Register Offsets */
# define AXI_HALTREQ_REG 0x0
# define AXI_HALTACK_REG 0x4
# define AXI_IDLE_REG 0x8
2020-01-23 18:42:36 +05:30
# define AXI_GATING_VALID_OVERRIDE BIT(0)
2016-06-20 14:28:41 -07:00
2020-01-23 18:42:35 +05:30
# define HALT_ACK_TIMEOUT_US 100000
2016-06-20 14:28:41 -07:00
2021-09-17 19:25:30 +05:30
/* QACCEPT Register Offsets */
# define QACCEPT_ACCEPT_REG 0x0
# define QACCEPT_ACTIVE_REG 0x4
# define QACCEPT_DENY_REG 0x8
# define QACCEPT_REQ_REG 0xC
# define QACCEPT_TIMEOUT_US 50
2016-06-20 14:28:41 -07:00
/* QDSP6SS_RESET */
# define Q6SS_STOP_CORE BIT(0)
# define Q6SS_CORE_ARES BIT(1)
# define Q6SS_BUS_ARES_ENABLE BIT(2)
2020-01-17 19:21:28 +05:30
/* QDSP6SS CBCR */
# define Q6SS_CBCR_CLKEN BIT(0)
# define Q6SS_CBCR_CLKOFF BIT(31)
# define Q6SS_CBCR_TIMEOUT_US 200
2016-06-20 14:28:41 -07:00
/* QDSP6SS_GFMUX_CTL */
# define Q6SS_CLK_ENABLE BIT(1)
/* QDSP6SS_PWR_CTL */
# define Q6SS_L2DATA_SLP_NRET_N_0 BIT(0)
# define Q6SS_L2DATA_SLP_NRET_N_1 BIT(1)
# define Q6SS_L2DATA_SLP_NRET_N_2 BIT(2)
# define Q6SS_L2TAG_SLP_NRET_N BIT(16)
# define Q6SS_ETB_SLP_NRET_N BIT(17)
# define Q6SS_L2DATA_STBY_N BIT(18)
# define Q6SS_SLP_RET_N BIT(19)
# define Q6SS_CLAMP_IO BIT(20)
# define QDSS_BHS_ON BIT(21)
# define QDSS_LDO_BYP BIT(22)
2017-10-24 21:22:27 +05:30
/* QDSP6v56 parameters */
# define QDSP6v56_LDO_BYP BIT(25)
# define QDSP6v56_BHS_ON BIT(24)
# define QDSP6v56_CLAMP_WL BIT(21)
# define QDSP6v56_CLAMP_QMC_MEM BIT(22)
# define QDSP6SS_XO_CBCR 0x0038
# define QDSP6SS_ACC_OVERRIDE_VAL 0x20
2018-05-21 22:57:13 +05:30
/* QDSP6v65 parameters */
2019-12-19 11:15:06 +05:30
# define QDSP6SS_CORE_CBCR 0x20
2018-05-21 22:57:13 +05:30
# define QDSP6SS_SLEEP 0x3C
# define QDSP6SS_BOOT_CORE_START 0x400
# define QDSP6SS_BOOT_CMD 0x404
# define BOOT_FSM_TIMEOUT 10000
2016-12-30 19:24:02 +05:30
struct reg_info {
struct regulator * reg ;
int uV ;
int uA ;
} ;
struct qcom_mss_reg_res {
const char * supply ;
int uV ;
int uA ;
} ;
2016-12-30 19:24:00 +05:30
struct rproc_hexagon_res {
const char * hexagon_mba_image ;
2017-02-01 17:56:28 +01:00
struct qcom_mss_reg_res * proxy_supply ;
2020-09-16 12:41:31 +02:00
struct qcom_mss_reg_res * fallback_proxy_supply ;
2017-02-01 17:56:28 +01:00
struct qcom_mss_reg_res * active_supply ;
2016-12-30 19:24:01 +05:30
char * * proxy_clk_names ;
2018-05-21 22:57:13 +05:30
char * * reset_clk_names ;
2016-12-30 19:24:01 +05:30
char * * active_clk_names ;
2019-01-30 16:39:30 -08:00
char * * proxy_pd_names ;
2017-10-24 21:22:27 +05:30
int version ;
2017-10-24 21:22:26 +05:30
bool need_mem_protection ;
2018-05-21 22:57:13 +05:30
bool has_alt_reset ;
2020-07-21 16:59:35 +05:30
bool has_mba_logs ;
2020-04-15 20:21:10 +05:30
bool has_spare_reg ;
2021-09-17 19:25:30 +05:30
bool has_qaccept_regs ;
bool has_ext_cntl_regs ;
bool has_vq6 ;
2016-12-30 19:24:00 +05:30
} ;
2016-06-20 14:28:41 -07:00
struct q6v5 {
struct device * dev ;
struct rproc * rproc ;
void __iomem * reg_base ;
void __iomem * rmb_base ;
struct regmap * halt_map ;
2019-12-19 11:15:06 +05:30
struct regmap * conn_map ;
2016-06-20 14:28:41 -07:00
u32 halt_q6 ;
u32 halt_modem ;
u32 halt_nc ;
2021-09-17 19:25:30 +05:30
u32 halt_vq6 ;
2019-12-19 11:15:06 +05:30
u32 conn_box ;
2016-06-20 14:28:41 -07:00
2021-09-17 19:25:30 +05:30
u32 qaccept_mdm ;
u32 qaccept_cx ;
u32 qaccept_axi ;
u32 axim1_clk_off ;
u32 crypto_clk_off ;
u32 force_clk_on ;
u32 rscc_disable ;
2016-06-20 14:28:41 -07:00
struct reset_control * mss_restart ;
2018-08-30 00:42:15 +05:30
struct reset_control * pdc_reset ;
2016-06-20 14:28:41 -07:00
2018-06-04 13:30:38 -07:00
struct qcom_q6v5 q6v5 ;
2018-05-21 22:57:09 +05:30
2016-12-30 19:24:01 +05:30
struct clk * active_clks [ 8 ] ;
2018-05-21 22:57:13 +05:30
struct clk * reset_clks [ 4 ] ;
2016-12-30 19:24:01 +05:30
struct clk * proxy_clks [ 4 ] ;
2019-01-30 16:39:30 -08:00
struct device * proxy_pds [ 3 ] ;
2016-12-30 19:24:01 +05:30
int active_clk_count ;
2018-05-21 22:57:13 +05:30
int reset_clk_count ;
2016-12-30 19:24:01 +05:30
int proxy_clk_count ;
2019-01-30 16:39:30 -08:00
int proxy_pd_count ;
2016-12-30 19:24:01 +05:30
2016-12-30 19:24:02 +05:30
struct reg_info active_regs [ 1 ] ;
2020-09-16 12:41:31 +02:00
struct reg_info proxy_regs [ 1 ] ;
struct reg_info fallback_proxy_regs [ 2 ] ;
2016-12-30 19:24:02 +05:30
int active_reg_count ;
int proxy_reg_count ;
2020-09-16 12:41:31 +02:00
int fallback_proxy_reg_count ;
2016-06-20 14:28:41 -07:00
2018-10-17 19:25:25 +05:30
bool dump_mba_loaded ;
2020-07-16 15:20:32 -07:00
size_t current_dump_size ;
size_t total_dump_size ;
2018-10-17 19:25:26 +05:30
2016-06-20 14:28:41 -07:00
phys_addr_t mba_phys ;
size_t mba_size ;
2020-07-23 01:40:47 +05:30
size_t dp_size ;
2016-06-20 14:28:41 -07:00
phys_addr_t mpss_phys ;
phys_addr_t mpss_reloc ;
size_t mpss_size ;
2017-01-29 14:05:50 -08:00
2018-05-21 22:57:14 +05:30
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 ;
2022-02-28 23:53:59 +01:00
struct platform_device * bam_dmux ;
2017-10-24 21:22:26 +05:30
bool need_mem_protection ;
2018-05-21 22:57:13 +05:30
bool has_alt_reset ;
2020-07-21 16:59:35 +05:30
bool has_mba_logs ;
2020-04-15 20:21:10 +05:30
bool has_spare_reg ;
2021-09-17 19:25:30 +05:30
bool has_qaccept_regs ;
bool has_ext_cntl_regs ;
bool has_vq6 ;
2017-10-24 21:22:26 +05:30
int mpss_perm ;
int mba_perm ;
2019-01-15 01:20:01 +05:30
const char * hexagon_mdt_image ;
2017-10-24 21:22:27 +05:30
int version ;
} ;
2017-10-24 21:22:26 +05:30
2017-10-24 21:22:27 +05:30
enum {
MSS_MSM8916 ,
MSS_MSM8974 ,
MSS_MSM8996 ,
2019-10-31 19:45:01 -07:00
MSS_MSM8998 ,
2019-12-19 11:15:06 +05:30
MSS_SC7180 ,
2021-09-17 19:25:30 +05:30
MSS_SC7280 ,
2018-05-21 22:57:13 +05:30
MSS_SDM845 ,
2016-06-20 14:28:41 -07:00
} ;
2016-12-30 19:24:02 +05:30
static int q6v5_regulator_init ( struct device * dev , struct reg_info * regs ,
const struct qcom_mss_reg_res * reg_res )
2016-06-20 14:28:41 -07:00
{
2016-12-30 19:24:02 +05:30
int rc ;
int i ;
2016-06-20 14:28:41 -07:00
2017-01-30 03:20:27 -08:00
if ( ! reg_res )
return 0 ;
2016-12-30 19:24:02 +05:30
for ( i = 0 ; reg_res [ i ] . supply ; i + + ) {
regs [ i ] . reg = devm_regulator_get ( dev , reg_res [ i ] . supply ) ;
if ( IS_ERR ( regs [ i ] . reg ) ) {
rc = PTR_ERR ( regs [ i ] . reg ) ;
if ( rc ! = - EPROBE_DEFER )
dev_err ( dev , " Failed to get %s \n regulator " ,
reg_res [ i ] . supply ) ;
return rc ;
}
2016-06-20 14:28:41 -07:00
2016-12-30 19:24:02 +05:30
regs [ i ] . uV = reg_res [ i ] . uV ;
regs [ i ] . uA = reg_res [ i ] . uA ;
2016-06-20 14:28:41 -07:00
}
2016-12-30 19:24:02 +05:30
return i ;
2016-06-20 14:28:41 -07:00
}
2016-12-30 19:24:02 +05:30
static int q6v5_regulator_enable ( struct q6v5 * qproc ,
struct reg_info * regs , int count )
2016-06-20 14:28:41 -07:00
{
int ret ;
2016-12-30 19:24:02 +05:30
int i ;
2016-06-20 14:28:41 -07:00
2016-12-30 19:24:02 +05:30
for ( i = 0 ; i < count ; i + + ) {
if ( regs [ i ] . uV > 0 ) {
ret = regulator_set_voltage ( regs [ i ] . reg ,
regs [ i ] . uV , INT_MAX ) ;
if ( ret ) {
dev_err ( qproc - > dev ,
" Failed to request voltage for %d. \n " ,
i ) ;
goto err ;
}
}
2016-06-20 14:28:41 -07:00
2016-12-30 19:24:02 +05:30
if ( regs [ i ] . uA > 0 ) {
ret = regulator_set_load ( regs [ i ] . reg ,
regs [ i ] . uA ) ;
if ( ret < 0 ) {
dev_err ( qproc - > dev ,
" Failed to set regulator mode \n " ) ;
goto err ;
}
}
ret = regulator_enable ( regs [ i ] . reg ) ;
if ( ret ) {
dev_err ( qproc - > dev , " Regulator enable failed \n " ) ;
goto err ;
}
}
return 0 ;
err :
for ( ; i > = 0 ; i - - ) {
if ( regs [ i ] . uV > 0 )
regulator_set_voltage ( regs [ i ] . reg , 0 , INT_MAX ) ;
if ( regs [ i ] . uA > 0 )
regulator_set_load ( regs [ i ] . reg , 0 ) ;
2016-06-20 14:28:41 -07:00
2016-12-30 19:24:02 +05:30
regulator_disable ( regs [ i ] . reg ) ;
}
2016-06-20 14:28:41 -07:00
2016-12-30 19:24:02 +05:30
return ret ;
2016-06-20 14:28:41 -07:00
}
2016-12-30 19:24:02 +05:30
static void q6v5_regulator_disable ( struct q6v5 * qproc ,
struct reg_info * regs , int count )
2016-06-20 14:28:41 -07:00
{
2016-12-30 19:24:02 +05:30
int i ;
for ( i = 0 ; i < count ; i + + ) {
if ( regs [ i ] . uV > 0 )
regulator_set_voltage ( regs [ i ] . reg , 0 , INT_MAX ) ;
2016-06-20 14:28:41 -07:00
2016-12-30 19:24:02 +05:30
if ( regs [ i ] . uA > 0 )
regulator_set_load ( regs [ i ] . reg , 0 ) ;
2016-06-20 14:28:41 -07:00
2016-12-30 19:24:02 +05:30
regulator_disable ( regs [ i ] . reg ) ;
}
2016-06-20 14:28:41 -07:00
}
2016-12-30 19:24:01 +05:30
static int q6v5_clk_enable ( struct device * dev ,
struct clk * * clks , int count )
{
int rc ;
int i ;
for ( i = 0 ; i < count ; i + + ) {
rc = clk_prepare_enable ( clks [ i ] ) ;
if ( rc ) {
dev_err ( dev , " Clock enable failed \n " ) ;
goto err ;
}
}
return 0 ;
err :
for ( i - - ; i > = 0 ; i - - )
clk_disable_unprepare ( clks [ i ] ) ;
return rc ;
}
static void q6v5_clk_disable ( struct device * dev ,
struct clk * * clks , int count )
{
int i ;
for ( i = 0 ; i < count ; i + + )
clk_disable_unprepare ( clks [ i ] ) ;
}
2019-01-30 16:39:30 -08:00
static int q6v5_pds_enable ( struct q6v5 * qproc , struct device * * pds ,
size_t pd_count )
{
int ret ;
int i ;
for ( i = 0 ; i < pd_count ; i + + ) {
dev_pm_genpd_set_performance_state ( pds [ i ] , INT_MAX ) ;
ret = pm_runtime_get_sync ( pds [ i ] ) ;
2020-11-02 22:34:33 +08:00
if ( ret < 0 ) {
pm_runtime_put_noidle ( pds [ i ] ) ;
dev_pm_genpd_set_performance_state ( pds [ i ] , 0 ) ;
2019-01-30 16:39:30 -08:00
goto unroll_pd_votes ;
2020-11-02 22:34:33 +08:00
}
2019-01-30 16:39:30 -08:00
}
return 0 ;
unroll_pd_votes :
for ( i - - ; i > = 0 ; i - - ) {
dev_pm_genpd_set_performance_state ( pds [ i ] , 0 ) ;
pm_runtime_put ( pds [ i ] ) ;
}
return ret ;
2020-04-03 12:50:05 -05:00
}
2019-01-30 16:39:30 -08:00
static void q6v5_pds_disable ( struct q6v5 * qproc , struct device * * pds ,
size_t pd_count )
{
int i ;
for ( i = 0 ; i < pd_count ; i + + ) {
dev_pm_genpd_set_performance_state ( pds [ i ] , 0 ) ;
pm_runtime_put ( pds [ i ] ) ;
}
}
2017-10-24 21:22:26 +05:30
static int q6v5_xfer_mem_ownership ( struct q6v5 * qproc , int * current_perm ,
2020-03-05 01:17:28 +05:30
bool local , bool remote , phys_addr_t addr ,
2017-10-24 21:22:26 +05:30
size_t size )
{
2020-03-05 01:17:28 +05:30
struct qcom_scm_vmperm next [ 2 ] ;
int perms = 0 ;
2017-10-24 21:22:26 +05:30
if ( ! qproc - > need_mem_protection )
return 0 ;
2020-03-05 01:17:28 +05:30
if ( local = = ! ! ( * current_perm & BIT ( QCOM_SCM_VMID_HLOS ) ) & &
remote = = ! ! ( * current_perm & BIT ( QCOM_SCM_VMID_MSS_MSA ) ) )
2017-10-24 21:22:26 +05:30
return 0 ;
2020-03-05 01:17:28 +05:30
if ( local ) {
next [ perms ] . vmid = QCOM_SCM_VMID_HLOS ;
next [ perms ] . perm = QCOM_SCM_PERM_RWX ;
perms + + ;
}
if ( remote ) {
next [ perms ] . vmid = QCOM_SCM_VMID_MSS_MSA ;
next [ perms ] . perm = QCOM_SCM_PERM_RW ;
perms + + ;
}
2017-10-24 21:22:26 +05:30
2017-11-06 22:26:41 -08:00
return qcom_scm_assign_mem ( addr , ALIGN ( size , SZ_4K ) ,
2020-03-05 01:17:28 +05:30
current_perm , next , perms ) ;
2017-10-24 21:22:26 +05:30
}
2020-11-04 12:33:42 +05:30
static void q6v5_debug_policy_load ( struct q6v5 * qproc , void * mba_region )
2020-07-23 01:40:47 +05:30
{
const struct firmware * dp_fw ;
if ( request_firmware_direct ( & dp_fw , " msadp " , qproc - > dev ) )
return ;
if ( SZ_1M + dp_fw - > size < = qproc - > mba_size ) {
2020-11-04 12:33:42 +05:30
memcpy ( mba_region + SZ_1M , dp_fw - > data , dp_fw - > size ) ;
2020-07-23 01:40:47 +05:30
qproc - > dp_size = dp_fw - > size ;
}
release_firmware ( dp_fw ) ;
}
2016-06-20 14:28:41 -07:00
static int q6v5_load ( struct rproc * rproc , const struct firmware * fw )
{
struct q6v5 * qproc = rproc - > priv ;
2020-11-04 12:33:42 +05:30
void * mba_region ;
2016-06-20 14:28:41 -07:00
2020-07-23 01:40:45 +05:30
/* MBA is restricted to a maximum size of 1M */
if ( fw - > size > qproc - > mba_size | | fw - > size > SZ_1M ) {
dev_err ( qproc - > dev , " MBA firmware load failed \n " ) ;
return - EINVAL ;
}
2020-11-04 12:33:42 +05:30
mba_region = memremap ( qproc - > mba_phys , qproc - > mba_size , MEMREMAP_WC ) ;
if ( ! mba_region ) {
dev_err ( qproc - > dev , " unable to map memory region: %pa+%zx \n " ,
& qproc - > mba_phys , qproc - > mba_size ) ;
return - EBUSY ;
}
memcpy ( mba_region , fw - > data , fw - > size ) ;
q6v5_debug_policy_load ( qproc , mba_region ) ;
memunmap ( mba_region ) ;
2016-06-20 14:28:41 -07:00
return 0 ;
}
2018-05-21 22:57:12 +05:30
static int q6v5_reset_assert ( struct q6v5 * qproc )
{
2018-08-30 00:42:15 +05:30
int ret ;
if ( qproc - > has_alt_reset ) {
reset_control_assert ( qproc - > pdc_reset ) ;
ret = reset_control_reset ( qproc - > mss_restart ) ;
reset_control_deassert ( qproc - > pdc_reset ) ;
2020-04-15 20:21:10 +05:30
} else if ( qproc - > has_spare_reg ) {
2020-01-23 18:42:36 +05:30
/*
* When the AXI pipeline is being reset with the Q6 modem partly
* operational there is possibility of AXI valid signal to
* glitch , leading to spurious transactions and Q6 hangs . A work
* around is employed by asserting the AXI_GATING_VALID_OVERRIDE
2020-04-15 20:21:10 +05:30
* BIT before triggering Q6 MSS reset . AXI_GATING_VALID_OVERRIDE
* is withdrawn post MSS assert followed by a MSS deassert ,
* while holding the PDC reset .
2020-01-23 18:42:36 +05:30
*/
2019-12-19 11:15:06 +05:30
reset_control_assert ( qproc - > pdc_reset ) ;
regmap_update_bits ( qproc - > conn_map , qproc - > conn_box ,
2020-01-23 18:42:36 +05:30
AXI_GATING_VALID_OVERRIDE , 1 ) ;
2019-12-19 11:15:06 +05:30
reset_control_assert ( qproc - > mss_restart ) ;
reset_control_deassert ( qproc - > pdc_reset ) ;
regmap_update_bits ( qproc - > conn_map , qproc - > conn_box ,
2020-01-23 18:42:36 +05:30
AXI_GATING_VALID_OVERRIDE , 0 ) ;
2019-12-19 11:15:06 +05:30
ret = reset_control_deassert ( qproc - > mss_restart ) ;
2021-09-17 19:25:30 +05:30
} else if ( qproc - > has_ext_cntl_regs ) {
regmap_write ( qproc - > conn_map , qproc - > rscc_disable , 0 ) ;
reset_control_assert ( qproc - > pdc_reset ) ;
reset_control_assert ( qproc - > mss_restart ) ;
reset_control_deassert ( qproc - > pdc_reset ) ;
ret = reset_control_deassert ( qproc - > mss_restart ) ;
2018-08-30 00:42:15 +05:30
} else {
ret = reset_control_assert ( qproc - > mss_restart ) ;
}
return ret ;
2018-05-21 22:57:12 +05:30
}
static int q6v5_reset_deassert ( struct q6v5 * qproc )
{
2018-05-21 22:57:13 +05:30
int ret ;
if ( qproc - > has_alt_reset ) {
2018-08-30 00:42:15 +05:30
reset_control_assert ( qproc - > pdc_reset ) ;
2018-05-21 22:57:13 +05:30
writel ( 1 , qproc - > rmb_base + RMB_MBA_ALT_RESET ) ;
ret = reset_control_reset ( qproc - > mss_restart ) ;
writel ( 0 , qproc - > rmb_base + RMB_MBA_ALT_RESET ) ;
2018-08-30 00:42:15 +05:30
reset_control_deassert ( qproc - > pdc_reset ) ;
2021-09-17 19:25:30 +05:30
} else if ( qproc - > has_spare_reg | | qproc - > has_ext_cntl_regs ) {
2019-12-19 11:15:06 +05:30
ret = reset_control_reset ( qproc - > mss_restart ) ;
2018-05-21 22:57:13 +05:30
} else {
ret = reset_control_deassert ( qproc - > mss_restart ) ;
}
return ret ;
2018-05-21 22:57:12 +05:30
}
2016-06-20 14:28:41 -07:00
static int q6v5_rmb_pbl_wait ( struct q6v5 * qproc , int ms )
{
unsigned long timeout ;
s32 val ;
timeout = jiffies + msecs_to_jiffies ( ms ) ;
for ( ; ; ) {
val = readl ( qproc - > rmb_base + RMB_PBL_STATUS_REG ) ;
if ( val )
break ;
if ( time_after ( jiffies , timeout ) )
return - ETIMEDOUT ;
msleep ( 1 ) ;
}
return val ;
}
static int q6v5_rmb_mba_wait ( struct q6v5 * qproc , u32 status , int ms )
{
unsigned long timeout ;
s32 val ;
timeout = jiffies + msecs_to_jiffies ( ms ) ;
for ( ; ; ) {
val = readl ( qproc - > rmb_base + RMB_MBA_STATUS_REG ) ;
if ( val < 0 )
break ;
if ( ! status & & val )
break ;
else if ( status & & val = = status )
break ;
if ( time_after ( jiffies , timeout ) )
return - ETIMEDOUT ;
msleep ( 1 ) ;
}
return val ;
}
2020-07-21 16:59:35 +05:30
static void q6v5_dump_mba_logs ( struct q6v5 * qproc )
{
struct rproc * rproc = qproc - > rproc ;
void * data ;
2020-11-04 12:33:42 +05:30
void * mba_region ;
2020-07-21 16:59:35 +05:30
if ( ! qproc - > has_mba_logs )
return ;
if ( q6v5_xfer_mem_ownership ( qproc , & qproc - > mba_perm , true , false , qproc - > mba_phys ,
qproc - > mba_size ) )
return ;
2020-11-04 12:33:42 +05:30
mba_region = memremap ( qproc - > mba_phys , qproc - > mba_size , MEMREMAP_WC ) ;
if ( ! mba_region )
2020-07-21 16:59:35 +05:30
return ;
2020-11-04 12:33:42 +05:30
data = vmalloc ( MBA_LOG_SIZE ) ;
if ( data ) {
memcpy ( data , mba_region , MBA_LOG_SIZE ) ;
dev_coredumpv ( & rproc - > dev , data , MBA_LOG_SIZE , GFP_KERNEL ) ;
}
memunmap ( mba_region ) ;
2020-07-21 16:59:35 +05:30
}
2016-06-20 14:28:41 -07:00
static int q6v5proc_reset ( struct q6v5 * qproc )
{
u32 val ;
int ret ;
2017-10-24 21:22:27 +05:30
int i ;
2016-06-20 14:28:41 -07:00
2018-05-21 22:57:13 +05:30
if ( qproc - > version = = MSS_SDM845 ) {
val = readl ( qproc - > reg_base + QDSP6SS_SLEEP ) ;
2020-01-17 19:21:28 +05:30
val | = Q6SS_CBCR_CLKEN ;
2018-05-21 22:57:13 +05:30
writel ( val , qproc - > reg_base + QDSP6SS_SLEEP ) ;
2016-06-20 14:28:41 -07:00
2018-05-21 22:57:13 +05:30
ret = readl_poll_timeout ( qproc - > reg_base + QDSP6SS_SLEEP ,
2020-01-17 19:21:28 +05:30
val , ! ( val & Q6SS_CBCR_CLKOFF ) , 1 ,
Q6SS_CBCR_TIMEOUT_US ) ;
2018-05-21 22:57:13 +05:30
if ( ret ) {
dev_err ( qproc - > dev , " QDSP6SS Sleep clock timed out \n " ) ;
return - ETIMEDOUT ;
}
/* De-assert QDSP6 stop core */
writel ( 1 , qproc - > reg_base + QDSP6SS_BOOT_CORE_START ) ;
/* Trigger boot FSM */
writel ( 1 , qproc - > reg_base + QDSP6SS_BOOT_CMD ) ;
ret = readl_poll_timeout ( qproc - > rmb_base + RMB_MBA_MSS_STATUS ,
val , ( val & BIT ( 0 ) ) ! = 0 , 10 , BOOT_FSM_TIMEOUT ) ;
if ( ret ) {
dev_err ( qproc - > dev , " Boot FSM failed to complete. \n " ) ;
/* Reset the modem so that boot FSM is in reset state */
q6v5_reset_deassert ( qproc ) ;
return ret ;
}
2019-12-19 11:15:06 +05:30
goto pbl_wait ;
2021-09-17 19:25:30 +05:30
} else if ( qproc - > version = = MSS_SC7180 | | qproc - > version = = MSS_SC7280 ) {
2019-12-19 11:15:06 +05:30
val = readl ( qproc - > reg_base + QDSP6SS_SLEEP ) ;
2020-01-17 19:21:28 +05:30
val | = Q6SS_CBCR_CLKEN ;
2019-12-19 11:15:06 +05:30
writel ( val , qproc - > reg_base + QDSP6SS_SLEEP ) ;
ret = readl_poll_timeout ( qproc - > reg_base + QDSP6SS_SLEEP ,
2020-01-17 19:21:28 +05:30
val , ! ( val & Q6SS_CBCR_CLKOFF ) , 1 ,
Q6SS_CBCR_TIMEOUT_US ) ;
2019-12-19 11:15:06 +05:30
if ( ret ) {
dev_err ( qproc - > dev , " QDSP6SS Sleep clock timed out \n " ) ;
return - ETIMEDOUT ;
}
/* Turn on the XO clock needed for PLL setup */
val = readl ( qproc - > reg_base + QDSP6SS_XO_CBCR ) ;
2020-01-17 19:21:28 +05:30
val | = Q6SS_CBCR_CLKEN ;
2019-12-19 11:15:06 +05:30
writel ( val , qproc - > reg_base + QDSP6SS_XO_CBCR ) ;
ret = readl_poll_timeout ( qproc - > reg_base + QDSP6SS_XO_CBCR ,
2020-01-17 19:21:28 +05:30
val , ! ( val & Q6SS_CBCR_CLKOFF ) , 1 ,
Q6SS_CBCR_TIMEOUT_US ) ;
2019-12-19 11:15:06 +05:30
if ( ret ) {
dev_err ( qproc - > dev , " QDSP6SS XO clock timed out \n " ) ;
return - ETIMEDOUT ;
}
/* Configure Q6 core CBCR to auto-enable after reset sequence */
val = readl ( qproc - > reg_base + QDSP6SS_CORE_CBCR ) ;
2020-01-17 19:21:28 +05:30
val | = Q6SS_CBCR_CLKEN ;
2019-12-19 11:15:06 +05:30
writel ( val , qproc - > reg_base + QDSP6SS_CORE_CBCR ) ;
/* De-assert the Q6 stop core signal */
writel ( 1 , qproc - > reg_base + QDSP6SS_BOOT_CORE_START ) ;
2020-07-16 17:35:14 +05:30
/* Wait for 10 us for any staggering logic to settle */
usleep_range ( 10 , 20 ) ;
2019-12-19 11:15:06 +05:30
/* Trigger the boot FSM to start the Q6 out-of-reset sequence */
writel ( 1 , qproc - > reg_base + QDSP6SS_BOOT_CMD ) ;
2020-07-16 17:35:14 +05:30
/* Poll the MSS_STATUS for FSM completion */
ret = readl_poll_timeout ( qproc - > rmb_base + RMB_MBA_MSS_STATUS ,
val , ( val & BIT ( 0 ) ) ! = 0 , 10 , BOOT_FSM_TIMEOUT ) ;
2019-12-19 11:15:06 +05:30
if ( ret ) {
dev_err ( qproc - > dev , " Boot FSM failed to complete. \n " ) ;
/* Reset the modem so that boot FSM is in reset state */
q6v5_reset_deassert ( qproc ) ;
return ret ;
}
2018-05-21 22:57:13 +05:30
goto pbl_wait ;
2019-10-31 19:45:01 -07:00
} else if ( qproc - > version = = MSS_MSM8996 | |
qproc - > version = = MSS_MSM8998 ) {
int mem_pwr_ctl ;
2017-10-24 21:22:27 +05:30
/* Override the ACC value if required */
writel ( QDSP6SS_ACC_OVERRIDE_VAL ,
qproc - > reg_base + QDSP6SS_STRAP_ACC ) ;
2016-06-20 14:28:41 -07:00
2017-10-24 21:22:27 +05:30
/* Assert resets, stop core */
val = readl ( qproc - > reg_base + QDSP6SS_RESET_REG ) ;
val | = Q6SS_CORE_ARES | Q6SS_BUS_ARES_ENABLE | Q6SS_STOP_CORE ;
writel ( val , qproc - > reg_base + QDSP6SS_RESET_REG ) ;
/* BHS require xo cbcr to be enabled */
val = readl ( qproc - > reg_base + QDSP6SS_XO_CBCR ) ;
2020-01-17 19:21:28 +05:30
val | = Q6SS_CBCR_CLKEN ;
2017-10-24 21:22:27 +05:30
writel ( val , qproc - > reg_base + QDSP6SS_XO_CBCR ) ;
/* Read CLKOFF bit to go low indicating CLK is enabled */
ret = readl_poll_timeout ( qproc - > reg_base + QDSP6SS_XO_CBCR ,
2020-01-17 19:21:28 +05:30
val , ! ( val & Q6SS_CBCR_CLKOFF ) , 1 ,
Q6SS_CBCR_TIMEOUT_US ) ;
2017-10-24 21:22:27 +05:30
if ( ret ) {
dev_err ( qproc - > dev ,
" xo cbcr enabling timed out (rc:%d) \n " , ret ) ;
return ret ;
}
/* Enable power block headswitch and wait for it to stabilize */
val = readl ( qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
val | = QDSP6v56_BHS_ON ;
writel ( val , qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
val | = readl ( qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
udelay ( 1 ) ;
/* Put LDO in bypass mode */
val | = QDSP6v56_LDO_BYP ;
writel ( val , qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
/* Deassert QDSP6 compiler memory clamp */
val = readl ( qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
val & = ~ QDSP6v56_CLAMP_QMC_MEM ;
writel ( val , qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
/* Deassert memory peripheral sleep and L2 memory standby */
val | = Q6SS_L2DATA_STBY_N | Q6SS_SLP_RET_N ;
writel ( val , qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
/* Turn on L1, L2, ETB and JU memories 1 at a time */
2019-10-31 19:45:01 -07:00
if ( qproc - > version = = MSS_MSM8996 ) {
mem_pwr_ctl = QDSP6SS_MEM_PWR_CTL ;
i = 19 ;
} else {
/* MSS_MSM8998 */
mem_pwr_ctl = QDSP6V6SS_MEM_PWR_CTL ;
i = 28 ;
}
val = readl ( qproc - > reg_base + mem_pwr_ctl ) ;
for ( ; i > = 0 ; i - - ) {
2017-10-24 21:22:27 +05:30
val | = BIT ( i ) ;
2019-10-31 19:45:01 -07:00
writel ( val , qproc - > reg_base + mem_pwr_ctl ) ;
2017-10-24 21:22:27 +05:30
/*
* Read back value to ensure the write is done then
* wait for 1u s for both memory peripheral and data
* array to turn on .
*/
2019-10-31 19:45:01 -07:00
val | = readl ( qproc - > reg_base + mem_pwr_ctl ) ;
2017-10-24 21:22:27 +05:30
udelay ( 1 ) ;
}
/* Remove word line clamp */
val = readl ( qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
val & = ~ QDSP6v56_CLAMP_WL ;
writel ( val , qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
} else {
/* Assert resets, stop core */
val = readl ( qproc - > reg_base + QDSP6SS_RESET_REG ) ;
val | = Q6SS_CORE_ARES | Q6SS_BUS_ARES_ENABLE | Q6SS_STOP_CORE ;
writel ( val , qproc - > reg_base + QDSP6SS_RESET_REG ) ;
/* Enable power block headswitch and wait for it to stabilize */
val = readl ( qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
val | = QDSS_BHS_ON | QDSS_LDO_BYP ;
writel ( val , qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
val | = readl ( qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
udelay ( 1 ) ;
/*
* Turn on memories . L2 banks should be done individually
* to minimize inrush current .
*/
val = readl ( qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
val | = Q6SS_SLP_RET_N | Q6SS_L2TAG_SLP_NRET_N |
Q6SS_ETB_SLP_NRET_N | Q6SS_L2DATA_STBY_N ;
writel ( val , qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
val | = Q6SS_L2DATA_SLP_NRET_N_2 ;
writel ( val , qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
val | = Q6SS_L2DATA_SLP_NRET_N_1 ;
writel ( val , qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
val | = Q6SS_L2DATA_SLP_NRET_N_0 ;
writel ( val , qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
}
2016-06-20 14:28:41 -07:00
/* Remove IO clamp */
val & = ~ Q6SS_CLAMP_IO ;
writel ( val , qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
/* Bring core out of reset */
val = readl ( qproc - > reg_base + QDSP6SS_RESET_REG ) ;
val & = ~ Q6SS_CORE_ARES ;
writel ( val , qproc - > reg_base + QDSP6SS_RESET_REG ) ;
/* Turn on core clock */
val = readl ( qproc - > reg_base + QDSP6SS_GFMUX_CTL_REG ) ;
val | = Q6SS_CLK_ENABLE ;
writel ( val , qproc - > reg_base + QDSP6SS_GFMUX_CTL_REG ) ;
/* Start core execution */
val = readl ( qproc - > reg_base + QDSP6SS_RESET_REG ) ;
val & = ~ Q6SS_STOP_CORE ;
writel ( val , qproc - > reg_base + QDSP6SS_RESET_REG ) ;
2018-05-21 22:57:13 +05:30
pbl_wait :
2016-06-20 14:28:41 -07:00
/* Wait for PBL status */
ret = q6v5_rmb_pbl_wait ( qproc , 1000 ) ;
if ( ret = = - ETIMEDOUT ) {
dev_err ( qproc - > dev , " PBL boot timed out \n " ) ;
} else if ( ret ! = RMB_PBL_SUCCESS ) {
dev_err ( qproc - > dev , " PBL returned unexpected status %d \n " , ret ) ;
ret = - EINVAL ;
} else {
ret = 0 ;
}
return ret ;
}
2021-09-17 19:25:30 +05:30
static int q6v5proc_enable_qchannel ( struct q6v5 * qproc , struct regmap * map , u32 offset )
{
unsigned int val ;
int ret ;
if ( ! qproc - > has_qaccept_regs )
return 0 ;
if ( qproc - > has_ext_cntl_regs ) {
regmap_write ( qproc - > conn_map , qproc - > rscc_disable , 0 ) ;
regmap_write ( qproc - > conn_map , qproc - > force_clk_on , 1 ) ;
ret = regmap_read_poll_timeout ( qproc - > halt_map , qproc - > axim1_clk_off , val ,
! val , 1 , Q6SS_CBCR_TIMEOUT_US ) ;
if ( ret ) {
dev_err ( qproc - > dev , " failed to enable axim1 clock \n " ) ;
return - ETIMEDOUT ;
}
}
regmap_write ( map , offset + QACCEPT_REQ_REG , 1 ) ;
/* Wait for accept */
ret = regmap_read_poll_timeout ( map , offset + QACCEPT_ACCEPT_REG , val , val , 5 ,
QACCEPT_TIMEOUT_US ) ;
if ( ret ) {
dev_err ( qproc - > dev , " qchannel enable failed \n " ) ;
return - ETIMEDOUT ;
}
return 0 ;
}
static void q6v5proc_disable_qchannel ( struct q6v5 * qproc , struct regmap * map , u32 offset )
{
int ret ;
unsigned int val , retry ;
unsigned int nretry = 10 ;
bool takedown_complete = false ;
if ( ! qproc - > has_qaccept_regs )
return ;
while ( ! takedown_complete & & nretry ) {
nretry - - ;
/* Wait for active transactions to complete */
regmap_read_poll_timeout ( map , offset + QACCEPT_ACTIVE_REG , val , ! val , 5 ,
QACCEPT_TIMEOUT_US ) ;
/* Request Q-channel transaction takedown */
regmap_write ( map , offset + QACCEPT_REQ_REG , 0 ) ;
/*
* If the request is denied , reset the Q - channel takedown request ,
* wait for active transactions to complete and retry takedown .
*/
retry = 10 ;
while ( retry ) {
usleep_range ( 5 , 10 ) ;
retry - - ;
ret = regmap_read ( map , offset + QACCEPT_DENY_REG , & val ) ;
if ( ! ret & & val ) {
regmap_write ( map , offset + QACCEPT_REQ_REG , 1 ) ;
break ;
}
ret = regmap_read ( map , offset + QACCEPT_ACCEPT_REG , & val ) ;
if ( ! ret & & ! val ) {
takedown_complete = true ;
break ;
}
}
if ( ! retry )
break ;
}
/* Rely on mss_restart to clear out pending transactions on takedown failure */
if ( ! takedown_complete )
dev_err ( qproc - > dev , " qchannel takedown failed \n " ) ;
}
2016-06-20 14:28:41 -07:00
static void q6v5proc_halt_axi_port ( struct q6v5 * qproc ,
struct regmap * halt_map ,
u32 offset )
{
unsigned int val ;
int ret ;
/* Check if we're already idle */
ret = regmap_read ( halt_map , offset + AXI_IDLE_REG , & val ) ;
if ( ! ret & & val )
return ;
/* Assert halt request */
regmap_write ( halt_map , offset + AXI_HALTREQ_REG , 1 ) ;
/* Wait for halt */
2020-01-23 18:42:35 +05:30
regmap_read_poll_timeout ( halt_map , offset + AXI_HALTACK_REG , val ,
val , 1000 , HALT_ACK_TIMEOUT_US ) ;
2016-06-20 14:28:41 -07:00
ret = regmap_read ( halt_map , offset + AXI_IDLE_REG , & val ) ;
if ( ret | | ! val )
dev_err ( qproc - > dev , " port failed halt \n " ) ;
/* Clear halt request (port will remain halted until reset) */
regmap_write ( halt_map , offset + AXI_HALTREQ_REG , 0 ) ;
}
2022-01-27 18:55:03 -08:00
static int q6v5_mpss_init_image ( struct q6v5 * qproc , const struct firmware * fw ,
const char * fw_name )
2016-06-20 14:28:41 -07:00
{
2022-05-11 11:27:05 +05:30
unsigned long dma_attrs = DMA_ATTR_FORCE_CONTIGUOUS | DMA_ATTR_NO_KERNEL_MAPPING ;
unsigned long flags = VM_DMA_COHERENT | VM_FLUSH_RESET_PERMS ;
struct page * * pages ;
struct page * page ;
2016-06-20 14:28:41 -07:00
dma_addr_t phys ;
2019-06-21 18:21:46 -07:00
void * metadata ;
2017-10-24 21:22:26 +05:30
int mdata_perm ;
int xferop_ret ;
2019-06-21 18:21:46 -07:00
size_t size ;
2022-05-11 11:27:05 +05:30
void * vaddr ;
int count ;
2016-06-20 14:28:41 -07:00
int ret ;
2022-05-11 11:27:05 +05:30
int i ;
2016-06-20 14:28:41 -07:00
2022-01-27 18:55:03 -08:00
metadata = qcom_mdt_read_metadata ( fw , & size , fw_name , qproc - > dev ) ;
2019-06-21 18:21:46 -07:00
if ( IS_ERR ( metadata ) )
return PTR_ERR ( metadata ) ;
2022-05-11 11:27:05 +05:30
page = dma_alloc_attrs ( qproc - > dev , size , & phys , GFP_KERNEL , dma_attrs ) ;
if ( ! page ) {
2019-06-21 18:21:46 -07:00
kfree ( metadata ) ;
2016-06-20 14:28:41 -07:00
dev_err ( qproc - > dev , " failed to allocate mdt buffer \n " ) ;
return - ENOMEM ;
}
2022-05-11 11:27:05 +05:30
count = PAGE_ALIGN ( size ) > > PAGE_SHIFT ;
pages = kmalloc_array ( count , sizeof ( struct page * ) , GFP_KERNEL ) ;
if ( ! pages ) {
ret = - ENOMEM ;
goto free_dma_attrs ;
}
for ( i = 0 ; i < count ; i + + )
pages [ i ] = nth_page ( page , i ) ;
vaddr = vmap ( pages , count , flags , pgprot_dmacoherent ( PAGE_KERNEL ) ) ;
kfree ( pages ) ;
if ( ! vaddr ) {
dev_err ( qproc - > dev , " unable to map memory region: %pa+%zx \n " , & phys , size ) ;
ret = - EBUSY ;
goto free_dma_attrs ;
}
memcpy ( vaddr , metadata , size ) ;
vunmap ( vaddr ) ;
2016-06-20 14:28:41 -07:00
2017-10-24 21:22:26 +05:30
/* Hypervisor mapping to access metadata by modem */
mdata_perm = BIT ( QCOM_SCM_VMID_HLOS ) ;
2020-03-05 01:17:28 +05:30
ret = q6v5_xfer_mem_ownership ( qproc , & mdata_perm , false , true ,
phys , size ) ;
2017-11-06 22:26:41 -08:00
if ( ret ) {
dev_err ( qproc - > dev ,
" assigning Q6 access to metadata failed: %d \n " , ret ) ;
2017-11-15 07:58:35 +01:00
ret = - EAGAIN ;
goto free_dma_attrs ;
2017-11-06 22:26:41 -08:00
}
2017-10-24 21:22:26 +05:30
2016-06-20 14:28:41 -07:00
writel ( phys , qproc - > rmb_base + RMB_PMI_META_DATA_REG ) ;
writel ( RMB_CMD_META_DATA_READY , qproc - > rmb_base + RMB_MBA_COMMAND_REG ) ;
ret = q6v5_rmb_mba_wait ( qproc , RMB_MBA_META_DATA_AUTH_SUCCESS , 1000 ) ;
if ( ret = = - ETIMEDOUT )
dev_err ( qproc - > dev , " MPSS header authentication timed out \n " ) ;
else if ( ret < 0 )
dev_err ( qproc - > dev , " MPSS header authentication failed: %d \n " , ret ) ;
2017-10-24 21:22:26 +05:30
/* Metadata authentication done, remove modem access */
2020-03-05 01:17:28 +05:30
xferop_ret = q6v5_xfer_mem_ownership ( qproc , & mdata_perm , true , false ,
phys , size ) ;
2017-10-24 21:22:26 +05:30
if ( xferop_ret )
dev_warn ( qproc - > dev ,
" mdt buffer not reclaimed system may become unstable \n " ) ;
2017-11-15 07:58:35 +01:00
free_dma_attrs :
2022-05-11 11:27:05 +05:30
dma_free_attrs ( qproc - > dev , size , page , phys , dma_attrs ) ;
2019-06-21 18:21:46 -07:00
kfree ( metadata ) ;
2016-06-20 14:28:41 -07:00
return ret < 0 ? ret : 0 ;
}
2017-01-26 13:58:35 -08:00
static bool q6v5_phdr_valid ( const struct elf32_phdr * phdr )
{
if ( phdr - > p_type ! = PT_LOAD )
return false ;
if ( ( phdr - > p_flags & QCOM_MDT_TYPE_MASK ) = = QCOM_MDT_TYPE_HASH )
return false ;
if ( ! phdr - > p_memsz )
return false ;
return true ;
}
2018-10-17 19:25:25 +05:30
static int q6v5_mba_load ( struct q6v5 * qproc )
{
int ret ;
int xfermemop_ret ;
2020-07-21 16:59:35 +05:30
bool mba_load_err = false ;
2018-10-17 19:25:25 +05:30
2021-09-16 19:29:21 +05:30
ret = qcom_q6v5_prepare ( & qproc - > q6v5 ) ;
if ( ret )
return ret ;
2019-01-30 16:39:31 -08:00
2019-01-30 16:39:30 -08:00
ret = q6v5_pds_enable ( qproc , qproc - > proxy_pds , qproc - > proxy_pd_count ) ;
if ( ret < 0 ) {
dev_err ( qproc - > dev , " failed to enable proxy power domains \n " ) ;
2021-09-16 19:29:21 +05:30
goto disable_irqs ;
2019-01-30 16:39:30 -08:00
}
2020-09-16 12:41:31 +02:00
ret = q6v5_regulator_enable ( qproc , qproc - > fallback_proxy_regs ,
qproc - > fallback_proxy_reg_count ) ;
if ( ret ) {
dev_err ( qproc - > dev , " failed to enable fallback proxy supplies \n " ) ;
goto disable_proxy_pds ;
}
2018-10-17 19:25:25 +05:30
ret = q6v5_regulator_enable ( qproc , qproc - > proxy_regs ,
qproc - > proxy_reg_count ) ;
if ( ret ) {
dev_err ( qproc - > dev , " failed to enable proxy supplies \n " ) ;
2020-09-16 12:41:31 +02:00
goto disable_fallback_proxy_reg ;
2018-10-17 19:25:25 +05:30
}
ret = q6v5_clk_enable ( qproc - > dev , qproc - > proxy_clks ,
qproc - > proxy_clk_count ) ;
if ( ret ) {
dev_err ( qproc - > dev , " failed to enable proxy clocks \n " ) ;
goto disable_proxy_reg ;
}
ret = q6v5_regulator_enable ( qproc , qproc - > active_regs ,
qproc - > active_reg_count ) ;
if ( ret ) {
dev_err ( qproc - > dev , " failed to enable supplies \n " ) ;
goto disable_proxy_clk ;
}
ret = q6v5_clk_enable ( qproc - > dev , qproc - > reset_clks ,
qproc - > reset_clk_count ) ;
if ( ret ) {
dev_err ( qproc - > dev , " failed to enable reset clocks \n " ) ;
goto disable_vdd ;
}
ret = q6v5_reset_deassert ( qproc ) ;
if ( ret ) {
dev_err ( qproc - > dev , " failed to deassert mss restart \n " ) ;
goto disable_reset_clks ;
}
ret = q6v5_clk_enable ( qproc - > dev , qproc - > active_clks ,
qproc - > active_clk_count ) ;
if ( ret ) {
dev_err ( qproc - > dev , " failed to enable clocks \n " ) ;
goto assert_reset ;
}
2021-09-17 19:25:30 +05:30
ret = q6v5proc_enable_qchannel ( qproc , qproc - > halt_map , qproc - > qaccept_axi ) ;
if ( ret ) {
dev_err ( qproc - > dev , " failed to enable axi bridge \n " ) ;
goto disable_active_clks ;
}
2020-09-17 23:28:40 +05:30
/*
* Some versions of the MBA firmware will upon boot wipe the MPSS region as well , so provide
* the Q6 access to this region .
*/
ret = q6v5_xfer_mem_ownership ( qproc , & qproc - > mpss_perm , false , true ,
qproc - > mpss_phys , qproc - > mpss_size ) ;
if ( ret ) {
dev_err ( qproc - > dev , " assigning Q6 access to mpss memory failed: %d \n " , ret ) ;
goto disable_active_clks ;
}
2018-10-17 19:25:25 +05:30
/* Assign MBA image access in DDR to q6 */
2020-03-05 01:17:28 +05:30
ret = q6v5_xfer_mem_ownership ( qproc , & qproc - > mba_perm , false , true ,
2018-10-17 19:25:25 +05:30
qproc - > mba_phys , qproc - > mba_size ) ;
if ( ret ) {
dev_err ( qproc - > dev ,
" assigning Q6 access to mba memory failed: %d \n " , ret ) ;
goto disable_active_clks ;
}
writel ( qproc - > mba_phys , qproc - > rmb_base + RMB_MBA_IMAGE_REG ) ;
2020-07-23 01:40:47 +05:30
if ( qproc - > dp_size ) {
writel ( qproc - > mba_phys + SZ_1M , qproc - > rmb_base + RMB_PMI_CODE_START_REG ) ;
writel ( qproc - > dp_size , qproc - > rmb_base + RMB_PMI_CODE_LENGTH_REG ) ;
}
2018-10-17 19:25:25 +05:30
ret = q6v5proc_reset ( qproc ) ;
if ( ret )
goto reclaim_mba ;
2022-05-24 18:15:34 +05:30
if ( qproc - > has_mba_logs )
qcom_pil_info_store ( " mba " , qproc - > mba_phys , MBA_LOG_SIZE ) ;
2018-10-17 19:25:25 +05:30
ret = q6v5_rmb_mba_wait ( qproc , 0 , 5000 ) ;
if ( ret = = - ETIMEDOUT ) {
dev_err ( qproc - > dev , " MBA boot timed out \n " ) ;
goto halt_axi_ports ;
} else if ( ret ! = RMB_MBA_XPU_UNLOCKED & &
ret ! = RMB_MBA_XPU_UNLOCKED_SCRIBBLED ) {
dev_err ( qproc - > dev , " MBA returned unexpected status %d \n " , ret ) ;
ret = - EINVAL ;
goto halt_axi_ports ;
}
qproc - > dump_mba_loaded = true ;
return 0 ;
halt_axi_ports :
q6v5proc_halt_axi_port ( qproc , qproc - > halt_map , qproc - > halt_q6 ) ;
2021-09-17 19:25:30 +05:30
if ( qproc - > has_vq6 )
q6v5proc_halt_axi_port ( qproc , qproc - > halt_map , qproc - > halt_vq6 ) ;
2018-10-17 19:25:25 +05:30
q6v5proc_halt_axi_port ( qproc , qproc - > halt_map , qproc - > halt_modem ) ;
q6v5proc_halt_axi_port ( qproc , qproc - > halt_map , qproc - > halt_nc ) ;
2021-09-17 19:25:30 +05:30
q6v5proc_disable_qchannel ( qproc , qproc - > halt_map , qproc - > qaccept_mdm ) ;
q6v5proc_disable_qchannel ( qproc , qproc - > halt_map , qproc - > qaccept_cx ) ;
q6v5proc_disable_qchannel ( qproc , qproc - > halt_map , qproc - > qaccept_axi ) ;
2020-07-21 16:59:35 +05:30
mba_load_err = true ;
2018-10-17 19:25:25 +05:30
reclaim_mba :
2020-03-05 01:17:28 +05:30
xfermemop_ret = q6v5_xfer_mem_ownership ( qproc , & qproc - > mba_perm , true ,
false , qproc - > mba_phys ,
2018-10-17 19:25:25 +05:30
qproc - > mba_size ) ;
if ( xfermemop_ret ) {
dev_err ( qproc - > dev ,
" Failed to reclaim mba buffer, system may become unstable \n " ) ;
2020-07-21 16:59:35 +05:30
} else if ( mba_load_err ) {
q6v5_dump_mba_logs ( qproc ) ;
2018-10-17 19:25:25 +05:30
}
disable_active_clks :
q6v5_clk_disable ( qproc - > dev , qproc - > active_clks ,
qproc - > active_clk_count ) ;
assert_reset :
q6v5_reset_assert ( qproc ) ;
disable_reset_clks :
q6v5_clk_disable ( qproc - > dev , qproc - > reset_clks ,
qproc - > reset_clk_count ) ;
disable_vdd :
q6v5_regulator_disable ( qproc , qproc - > active_regs ,
qproc - > active_reg_count ) ;
disable_proxy_clk :
q6v5_clk_disable ( qproc - > dev , qproc - > proxy_clks ,
qproc - > proxy_clk_count ) ;
disable_proxy_reg :
q6v5_regulator_disable ( qproc , qproc - > proxy_regs ,
qproc - > proxy_reg_count ) ;
2020-09-16 12:41:31 +02:00
disable_fallback_proxy_reg :
q6v5_regulator_disable ( qproc , qproc - > fallback_proxy_regs ,
qproc - > fallback_proxy_reg_count ) ;
2019-01-30 16:39:30 -08:00
disable_proxy_pds :
q6v5_pds_disable ( qproc , qproc - > proxy_pds , qproc - > proxy_pd_count ) ;
2018-10-17 19:25:25 +05:30
disable_irqs :
qcom_q6v5_unprepare ( & qproc - > q6v5 ) ;
return ret ;
}
static void q6v5_mba_reclaim ( struct q6v5 * qproc )
{
int ret ;
u32 val ;
qproc - > dump_mba_loaded = false ;
2020-07-23 01:40:47 +05:30
qproc - > dp_size = 0 ;
2018-10-17 19:25:25 +05:30
q6v5proc_halt_axi_port ( qproc , qproc - > halt_map , qproc - > halt_q6 ) ;
2021-09-17 19:25:30 +05:30
if ( qproc - > has_vq6 )
q6v5proc_halt_axi_port ( qproc , qproc - > halt_map , qproc - > halt_vq6 ) ;
2018-10-17 19:25:25 +05:30
q6v5proc_halt_axi_port ( qproc , qproc - > halt_map , qproc - > halt_modem ) ;
q6v5proc_halt_axi_port ( qproc , qproc - > halt_map , qproc - > halt_nc ) ;
if ( qproc - > version = = MSS_MSM8996 ) {
/*
* To avoid high MX current during LPASS / MSS restart .
*/
val = readl ( qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
val | = Q6SS_CLAMP_IO | QDSP6v56_CLAMP_WL |
QDSP6v56_CLAMP_QMC_MEM ;
writel ( val , qproc - > reg_base + QDSP6SS_PWR_CTL_REG ) ;
}
2021-09-17 19:25:30 +05:30
if ( qproc - > has_ext_cntl_regs ) {
regmap_write ( qproc - > conn_map , qproc - > rscc_disable , 1 ) ;
ret = regmap_read_poll_timeout ( qproc - > halt_map , qproc - > axim1_clk_off , val ,
! val , 1 , Q6SS_CBCR_TIMEOUT_US ) ;
if ( ret )
dev_err ( qproc - > dev , " failed to enable axim1 clock \n " ) ;
ret = regmap_read_poll_timeout ( qproc - > halt_map , qproc - > crypto_clk_off , val ,
! val , 1 , Q6SS_CBCR_TIMEOUT_US ) ;
if ( ret )
dev_err ( qproc - > dev , " failed to enable crypto clock \n " ) ;
}
q6v5proc_disable_qchannel ( qproc , qproc - > halt_map , qproc - > qaccept_mdm ) ;
q6v5proc_disable_qchannel ( qproc , qproc - > halt_map , qproc - > qaccept_cx ) ;
q6v5proc_disable_qchannel ( qproc , qproc - > halt_map , qproc - > qaccept_axi ) ;
2018-10-17 19:25:25 +05:30
q6v5_reset_assert ( qproc ) ;
q6v5_clk_disable ( qproc - > dev , qproc - > reset_clks ,
qproc - > reset_clk_count ) ;
q6v5_clk_disable ( qproc - > dev , qproc - > active_clks ,
qproc - > active_clk_count ) ;
q6v5_regulator_disable ( qproc , qproc - > active_regs ,
qproc - > active_reg_count ) ;
/* In case of failure or coredump scenario where reclaiming MBA memory
* could not happen reclaim it here .
*/
2020-03-05 01:17:28 +05:30
ret = q6v5_xfer_mem_ownership ( qproc , & qproc - > mba_perm , true , false ,
2018-10-17 19:25:25 +05:30
qproc - > mba_phys ,
qproc - > mba_size ) ;
WARN_ON ( ret ) ;
ret = qcom_q6v5_unprepare ( & qproc - > q6v5 ) ;
if ( ret ) {
2019-01-30 16:39:30 -08:00
q6v5_pds_disable ( qproc , qproc - > proxy_pds ,
qproc - > proxy_pd_count ) ;
2018-10-17 19:25:25 +05:30
q6v5_clk_disable ( qproc - > dev , qproc - > proxy_clks ,
qproc - > proxy_clk_count ) ;
2020-09-16 12:41:31 +02:00
q6v5_regulator_disable ( qproc , qproc - > fallback_proxy_regs ,
qproc - > fallback_proxy_reg_count ) ;
2018-10-17 19:25:25 +05:30
q6v5_regulator_disable ( qproc , qproc - > proxy_regs ,
qproc - > proxy_reg_count ) ;
}
}
2020-03-05 01:17:29 +05:30
static int q6v5_reload_mba ( struct rproc * rproc )
{
struct q6v5 * qproc = rproc - > priv ;
const struct firmware * fw ;
int ret ;
ret = request_firmware ( & fw , rproc - > firmware , qproc - > dev ) ;
if ( ret < 0 )
return ret ;
q6v5_load ( rproc , fw ) ;
ret = q6v5_mba_load ( qproc ) ;
release_firmware ( fw ) ;
return ret ;
}
2017-01-26 13:58:35 -08:00
static int q6v5_mpss_load ( struct q6v5 * qproc )
2016-06-20 14:28:41 -07:00
{
const struct elf32_phdr * phdrs ;
const struct elf32_phdr * phdr ;
2017-01-26 13:58:35 -08:00
const struct firmware * seg_fw ;
const struct firmware * fw ;
2016-06-20 14:28:41 -07:00
struct elf32_hdr * ehdr ;
2017-01-26 13:58:35 -08:00
phys_addr_t mpss_reloc ;
2016-06-20 14:28:41 -07:00
phys_addr_t boot_addr ;
2018-06-14 15:28:02 -07:00
phys_addr_t min_addr = PHYS_ADDR_MAX ;
2017-01-26 13:58:35 -08:00
phys_addr_t max_addr = 0 ;
2020-03-05 01:17:28 +05:30
u32 code_length ;
2017-01-26 13:58:35 -08:00
bool relocate = false ;
2019-01-15 01:20:01 +05:30
char * fw_name ;
size_t fw_name_len ;
2017-02-15 14:00:41 -08:00
ssize_t offset ;
2017-10-24 21:22:25 +05:30
size_t size = 0 ;
2017-01-26 13:58:35 -08:00
void * ptr ;
2016-06-20 14:28:41 -07:00
int ret ;
int i ;
2019-01-15 01:20:01 +05:30
fw_name_len = strlen ( qproc - > hexagon_mdt_image ) ;
if ( fw_name_len < = 4 )
return - EINVAL ;
fw_name = kstrdup ( qproc - > hexagon_mdt_image , GFP_KERNEL ) ;
if ( ! fw_name )
return - ENOMEM ;
ret = request_firmware ( & fw , fw_name , qproc - > dev ) ;
2017-01-26 13:58:35 -08:00
if ( ret < 0 ) {
2019-01-15 01:20:01 +05:30
dev_err ( qproc - > dev , " unable to load %s \n " , fw_name ) ;
goto out ;
2016-06-20 14:28:41 -07:00
}
2017-01-26 13:58:35 -08:00
/* Initialize the RMB validator */
writel ( 0 , qproc - > rmb_base + RMB_PMI_CODE_LENGTH_REG ) ;
2022-01-27 18:55:03 -08:00
ret = q6v5_mpss_init_image ( qproc , fw , qproc - > hexagon_mdt_image ) ;
2017-01-26 13:58:35 -08:00
if ( ret )
goto release_firmware ;
2016-06-20 14:28:41 -07:00
ehdr = ( struct elf32_hdr * ) fw - > data ;
phdrs = ( struct elf32_phdr * ) ( ehdr + 1 ) ;
2017-01-26 13:58:35 -08:00
for ( i = 0 ; i < ehdr - > e_phnum ; i + + ) {
2016-06-20 14:28:41 -07:00
phdr = & phdrs [ i ] ;
2017-01-26 13:58:35 -08:00
if ( ! q6v5_phdr_valid ( phdr ) )
2016-06-20 14:28:41 -07:00
continue ;
2017-01-26 13:58:35 -08:00
if ( phdr - > p_flags & QCOM_MDT_RELOCATABLE )
relocate = true ;
2016-06-20 14:28:41 -07:00
2017-01-26 13:58:35 -08:00
if ( phdr - > p_paddr < min_addr )
min_addr = phdr - > p_paddr ;
if ( phdr - > p_paddr + phdr - > p_memsz > max_addr )
max_addr = ALIGN ( phdr - > p_paddr + phdr - > p_memsz , SZ_4K ) ;
}
2020-09-17 23:28:40 +05:30
/*
2020-03-05 01:17:27 +05:30
* In case of a modem subsystem restart on secure devices , the modem
2020-09-17 23:28:40 +05:30
* memory can be reclaimed only after MBA is loaded .
2020-03-05 01:17:27 +05:30
*/
2020-03-05 01:17:28 +05:30
q6v5_xfer_mem_ownership ( qproc , & qproc - > mpss_perm , true , false ,
2020-03-05 01:17:27 +05:30
qproc - > mpss_phys , qproc - > mpss_size ) ;
2020-03-05 01:17:28 +05:30
/* Share ownership between Linux and MSS, during segment loading */
ret = q6v5_xfer_mem_ownership ( qproc , & qproc - > mpss_perm , true , true ,
qproc - > mpss_phys , qproc - > mpss_size ) ;
if ( ret ) {
dev_err ( qproc - > dev ,
" assigning Q6 access to mpss memory failed: %d \n " , ret ) ;
ret = - EAGAIN ;
goto release_firmware ;
}
2017-01-26 13:58:35 -08:00
mpss_reloc = relocate ? min_addr : qproc - > mpss_phys ;
2018-07-27 20:50:03 +05:30
qproc - > mpss_reloc = mpss_reloc ;
2017-10-24 21:22:25 +05:30
/* Load firmware segments */
2017-01-26 13:58:35 -08:00
for ( i = 0 ; i < ehdr - > e_phnum ; i + + ) {
phdr = & phdrs [ i ] ;
if ( ! q6v5_phdr_valid ( phdr ) )
2016-06-20 14:28:41 -07:00
continue ;
2017-01-26 13:58:35 -08:00
offset = phdr - > p_paddr - mpss_reloc ;
if ( offset < 0 | | offset + phdr - > p_memsz > qproc - > mpss_size ) {
dev_err ( qproc - > dev , " segment outside memory range \n " ) ;
ret = - EINVAL ;
goto release_firmware ;
}
2021-03-12 15:20:02 -08:00
if ( phdr - > p_filesz > phdr - > p_memsz ) {
dev_err ( qproc - > dev ,
" refusing to load segment %d with p_filesz > p_memsz \n " ,
i ) ;
ret = - EINVAL ;
goto release_firmware ;
}
2020-11-04 12:33:41 +05:30
ptr = memremap ( qproc - > mpss_phys + offset , phdr - > p_memsz , MEMREMAP_WC ) ;
2020-04-15 12:46:18 +05:30
if ( ! ptr ) {
dev_err ( qproc - > dev ,
" unable to map memory region: %pa+%zx-%x \n " ,
& qproc - > mpss_phys , offset , phdr - > p_memsz ) ;
goto release_firmware ;
}
2017-01-26 13:58:35 -08:00
2019-06-21 18:21:46 -07:00
if ( phdr - > p_filesz & & phdr - > p_offset < fw - > size ) {
/* Firmware is large enough to be non-split */
if ( phdr - > p_offset + phdr - > p_filesz > fw - > size ) {
dev_err ( qproc - > dev ,
" failed to load segment %d from truncated file %s \n " ,
i , fw_name ) ;
ret = - EINVAL ;
2020-11-04 12:33:41 +05:30
memunmap ( ptr ) ;
2019-06-21 18:21:46 -07:00
goto release_firmware ;
}
memcpy ( ptr , fw - > data + phdr - > p_offset , phdr - > p_filesz ) ;
} else if ( phdr - > p_filesz ) {
2019-01-15 01:20:01 +05:30
/* Replace "xxx.xxx" with "xxx.bxx" */
sprintf ( fw_name + fw_name_len - 3 , " b%02d " , i ) ;
2020-07-23 01:40:46 +05:30
ret = request_firmware_into_buf ( & seg_fw , fw_name , qproc - > dev ,
ptr , phdr - > p_filesz ) ;
2017-01-26 13:58:35 -08:00
if ( ret ) {
2019-01-15 01:20:01 +05:30
dev_err ( qproc - > dev , " failed to load %s \n " , fw_name ) ;
2020-11-04 12:33:41 +05:30
memunmap ( ptr ) ;
2017-01-26 13:58:35 -08:00
goto release_firmware ;
}
2021-03-12 15:20:02 -08:00
if ( seg_fw - > size ! = phdr - > p_filesz ) {
dev_err ( qproc - > dev ,
" failed to load segment %d from truncated file %s \n " ,
i , fw_name ) ;
ret = - EINVAL ;
release_firmware ( seg_fw ) ;
memunmap ( ptr ) ;
goto release_firmware ;
}
2017-01-26 13:58:35 -08:00
release_firmware ( seg_fw ) ;
}
if ( phdr - > p_memsz > phdr - > p_filesz ) {
memset ( ptr + phdr - > p_filesz , 0 ,
phdr - > p_memsz - phdr - > p_filesz ) ;
}
2020-11-04 12:33:41 +05:30
memunmap ( ptr ) ;
2016-06-20 14:28:41 -07:00
size + = phdr - > p_memsz ;
2020-03-05 01:17:28 +05:30
code_length = readl ( qproc - > rmb_base + RMB_PMI_CODE_LENGTH_REG ) ;
if ( ! code_length ) {
boot_addr = relocate ? qproc - > mpss_phys : min_addr ;
writel ( boot_addr , qproc - > rmb_base + RMB_PMI_CODE_START_REG ) ;
writel ( RMB_CMD_LOAD_READY , qproc - > rmb_base + RMB_MBA_COMMAND_REG ) ;
}
writel ( size , qproc - > rmb_base + RMB_PMI_CODE_LENGTH_REG ) ;
ret = readl ( qproc - > rmb_base + RMB_MBA_STATUS_REG ) ;
if ( ret < 0 ) {
dev_err ( qproc - > dev , " MPSS authentication failed: %d \n " ,
ret ) ;
goto release_firmware ;
}
2016-06-20 14:28:41 -07:00
}
2017-10-24 21:22:26 +05:30
/* Transfer ownership of modem ddr region to q6 */
2020-03-05 01:17:28 +05:30
ret = q6v5_xfer_mem_ownership ( qproc , & qproc - > mpss_perm , false , true ,
2017-10-24 21:22:26 +05:30
qproc - > mpss_phys , qproc - > mpss_size ) ;
2017-11-06 22:26:41 -08:00
if ( ret ) {
dev_err ( qproc - > dev ,
" assigning Q6 access to mpss memory failed: %d \n " , ret ) ;
2017-11-15 07:58:35 +01:00
ret = - EAGAIN ;
goto release_firmware ;
2017-11-06 22:26:41 -08:00
}
2017-10-24 21:22:26 +05:30
2016-07-12 17:15:45 -07:00
ret = q6v5_rmb_mba_wait ( qproc , RMB_MBA_AUTH_COMPLETE , 10000 ) ;
if ( ret = = - ETIMEDOUT )
dev_err ( qproc - > dev , " MPSS authentication timed out \n " ) ;
else if ( ret < 0 )
dev_err ( qproc - > dev , " MPSS authentication failed: %d \n " , ret ) ;
2020-06-22 12:19:40 -07:00
qcom_pil_info_store ( " modem " , qproc - > mpss_phys , qproc - > mpss_size ) ;
2016-06-20 14:28:41 -07:00
release_firmware :
release_firmware ( fw ) ;
2019-01-15 01:20:01 +05:30
out :
kfree ( fw_name ) ;
2016-06-20 14:28:41 -07:00
return ret < 0 ? ret : 0 ;
}
2018-10-17 19:25:26 +05:30
static void qcom_q6v5_dump_segment ( struct rproc * rproc ,
struct rproc_dump_segment * segment ,
2020-07-16 15:20:33 -07:00
void * dest , size_t cp_offset , size_t size )
2018-10-17 19:25:26 +05:30
{
int ret = 0 ;
struct q6v5 * qproc = rproc - > priv ;
2020-04-15 12:46:18 +05:30
int offset = segment - > da - qproc - > mpss_reloc ;
void * ptr = NULL ;
2018-10-17 19:25:26 +05:30
/* Unlock mba before copying segments */
2020-03-05 01:17:27 +05:30
if ( ! qproc - > dump_mba_loaded ) {
2020-03-05 01:17:29 +05:30
ret = q6v5_reload_mba ( rproc ) ;
2020-03-05 01:17:27 +05:30
if ( ! ret ) {
/* Reset ownership back to Linux to copy segments */
ret = q6v5_xfer_mem_ownership ( qproc , & qproc - > mpss_perm ,
2020-03-05 01:17:28 +05:30
true , false ,
2020-03-05 01:17:27 +05:30
qproc - > mpss_phys ,
qproc - > mpss_size ) ;
}
}
2018-10-17 19:25:26 +05:30
2020-04-15 12:46:18 +05:30
if ( ! ret )
2020-11-04 12:33:41 +05:30
ptr = memremap ( qproc - > mpss_phys + offset + cp_offset , size , MEMREMAP_WC ) ;
2020-04-15 12:46:18 +05:30
if ( ptr ) {
2020-07-16 15:20:33 -07:00
memcpy ( dest , ptr , size ) ;
2020-11-04 12:33:41 +05:30
memunmap ( ptr ) ;
2020-04-15 12:46:18 +05:30
} else {
2020-07-16 15:20:33 -07:00
memset ( dest , 0xff , size ) ;
2020-04-15 12:46:18 +05:30
}
2018-10-17 19:25:26 +05:30
2020-07-16 15:20:33 -07:00
qproc - > current_dump_size + = size ;
2018-10-17 19:25:26 +05:30
/* Reclaim mba after copying segments */
2020-07-16 15:20:32 -07:00
if ( qproc - > current_dump_size = = qproc - > total_dump_size ) {
2020-03-05 01:17:27 +05:30
if ( qproc - > dump_mba_loaded ) {
/* Try to reset ownership back to Q6 */
q6v5_xfer_mem_ownership ( qproc , & qproc - > mpss_perm ,
2020-03-05 01:17:28 +05:30
false , true ,
2020-03-05 01:17:27 +05:30
qproc - > mpss_phys ,
qproc - > mpss_size ) ;
2018-10-17 19:25:26 +05:30
q6v5_mba_reclaim ( qproc ) ;
2020-03-05 01:17:27 +05:30
}
2018-10-17 19:25:26 +05:30
}
}
2016-06-20 14:28:41 -07:00
static int q6v5_start ( struct rproc * rproc )
{
struct q6v5 * qproc = ( struct q6v5 * ) rproc - > priv ;
2017-10-24 21:22:26 +05:30
int xfermemop_ret ;
2016-06-20 14:28:41 -07:00
int ret ;
2018-10-17 19:25:25 +05:30
ret = q6v5_mba_load ( qproc ) ;
2016-06-20 14:28:41 -07:00
if ( ret )
2018-10-17 19:25:25 +05:30
return ret ;
2016-06-20 14:28:41 -07:00
2020-07-23 01:40:47 +05:30
dev_info ( qproc - > dev , " MBA booted with%s debug policy, loading mpss \n " ,
qproc - > dp_size ? " " : " out " ) ;
2016-06-20 14:28:41 -07:00
ret = q6v5_mpss_load ( qproc ) ;
if ( ret )
2017-10-24 21:22:26 +05:30
goto reclaim_mpss ;
2016-06-20 14:28:41 -07:00
2018-06-04 13:30:38 -07:00
ret = qcom_q6v5_wait_for_start ( & qproc - > q6v5 , msecs_to_jiffies ( 5000 ) ) ;
if ( ret = = - ETIMEDOUT ) {
2016-06-20 14:28:41 -07:00
dev_err ( qproc - > dev , " start timed out \n " ) ;
2017-10-24 21:22:26 +05:30
goto reclaim_mpss ;
2016-06-20 14:28:41 -07:00
}
2020-03-05 01:17:28 +05:30
xfermemop_ret = q6v5_xfer_mem_ownership ( qproc , & qproc - > mba_perm , true ,
false , qproc - > mba_phys ,
2017-10-24 21:22:26 +05:30
qproc - > mba_size ) ;
if ( xfermemop_ret )
dev_err ( qproc - > dev ,
" Failed to reclaim mba buffer system may become unstable \n " ) ;
2018-10-17 19:25:26 +05:30
/* Reset Dump Segment Mask */
2020-07-16 15:20:32 -07:00
qproc - > current_dump_size = 0 ;
2016-06-20 14:28:41 -07:00
return 0 ;
2017-10-24 21:22:26 +05:30
reclaim_mpss :
2018-10-17 19:25:25 +05:30
q6v5_mba_reclaim ( qproc ) ;
2020-07-21 16:59:35 +05:30
q6v5_dump_mba_logs ( qproc ) ;
2018-05-21 22:57:09 +05:30
2016-06-20 14:28:41 -07:00
return ret ;
}
static int q6v5_stop ( struct rproc * rproc )
{
struct q6v5 * qproc = ( struct q6v5 * ) rproc - > priv ;
int ret ;
2020-11-21 21:41:34 -08:00
ret = qcom_q6v5_request_stop ( & qproc - > q6v5 , qproc - > sysmon ) ;
2018-06-04 13:30:38 -07:00
if ( ret = = - ETIMEDOUT )
2016-06-20 14:28:41 -07:00
dev_err ( qproc - > dev , " timed out on wait \n " ) ;
2018-10-17 19:25:25 +05:30
q6v5_mba_reclaim ( qproc ) ;
2016-06-20 14:28:41 -07:00
return 0 ;
}
2018-10-17 19:25:27 +05:30
static int qcom_q6v5_register_dump_segments ( struct rproc * rproc ,
const struct firmware * mba_fw )
{
const struct firmware * fw ;
const struct elf32_phdr * phdrs ;
const struct elf32_phdr * phdr ;
const struct elf32_hdr * ehdr ;
struct q6v5 * qproc = rproc - > priv ;
unsigned long i ;
int ret ;
2019-01-15 01:20:01 +05:30
ret = request_firmware ( & fw , qproc - > hexagon_mdt_image , qproc - > dev ) ;
2018-10-17 19:25:27 +05:30
if ( ret < 0 ) {
2019-01-15 01:20:01 +05:30
dev_err ( qproc - > dev , " unable to load %s \n " ,
qproc - > hexagon_mdt_image ) ;
2018-10-17 19:25:27 +05:30
return ret ;
}
2020-04-10 12:24:33 +02:00
rproc_coredump_set_elf_info ( rproc , ELFCLASS32 , EM_NONE ) ;
2018-10-17 19:25:27 +05:30
ehdr = ( struct elf32_hdr * ) fw - > data ;
phdrs = ( struct elf32_phdr * ) ( ehdr + 1 ) ;
2020-07-16 15:20:32 -07:00
qproc - > total_dump_size = 0 ;
2018-10-17 19:25:27 +05:30
for ( i = 0 ; i < ehdr - > e_phnum ; i + + ) {
phdr = & phdrs [ i ] ;
if ( ! q6v5_phdr_valid ( phdr ) )
continue ;
ret = rproc_coredump_add_custom_segment ( rproc , phdr - > p_paddr ,
phdr - > p_memsz ,
qcom_q6v5_dump_segment ,
2020-07-16 15:20:32 -07:00
NULL ) ;
2018-10-17 19:25:27 +05:30
if ( ret )
break ;
2020-07-16 15:20:32 -07:00
qproc - > total_dump_size + = phdr - > p_memsz ;
2018-10-17 19:25:27 +05:30
}
release_firmware ( fw ) ;
return ret ;
}
2022-05-24 18:15:35 +05:30
static unsigned long q6v5_panic ( struct rproc * rproc )
{
struct q6v5 * qproc = ( struct q6v5 * ) rproc - > priv ;
return qcom_q6v5_panic ( & qproc - > q6v5 ) ;
}
2016-06-20 14:28:41 -07:00
static const struct rproc_ops q6v5_ops = {
. start = q6v5_start ,
. stop = q6v5_stop ,
2018-10-17 19:25:27 +05:30
. parse_fw = qcom_q6v5_register_dump_segments ,
2018-01-05 15:58:01 -08:00
. load = q6v5_load ,
2022-05-24 18:15:35 +05:30
. panic = q6v5_panic ,
2016-06-20 14:28:41 -07:00
} ;
2018-06-04 13:30:38 -07:00
static void qcom_msa_handover ( struct qcom_q6v5 * q6v5 )
2018-05-21 22:57:09 +05:30
{
2018-06-04 13:30:38 -07:00
struct q6v5 * qproc = container_of ( q6v5 , struct q6v5 , q6v5 ) ;
2018-05-21 22:57:09 +05:30
q6v5_clk_disable ( qproc - > dev , qproc - > proxy_clks ,
qproc - > proxy_clk_count ) ;
q6v5_regulator_disable ( qproc , qproc - > proxy_regs ,
qproc - > proxy_reg_count ) ;
2020-09-16 12:41:31 +02:00
q6v5_regulator_disable ( qproc , qproc - > fallback_proxy_regs ,
qproc - > fallback_proxy_reg_count ) ;
2019-01-30 16:39:30 -08:00
q6v5_pds_disable ( qproc , qproc - > proxy_pds , qproc - > proxy_pd_count ) ;
2016-06-20 14:28:41 -07:00
}
static int q6v5_init_mem ( struct q6v5 * qproc , struct platform_device * pdev )
{
struct of_phandle_args args ;
2021-09-17 19:25:30 +05:30
int halt_cell_cnt = 3 ;
2016-06-20 14:28:41 -07:00
int ret ;
2021-09-06 15:11:47 +08:00
qproc - > reg_base = devm_platform_ioremap_resource_byname ( pdev , " qdsp6 " ) ;
2016-07-14 12:57:44 +00:00
if ( IS_ERR ( qproc - > reg_base ) )
2016-06-20 14:28:41 -07:00
return PTR_ERR ( qproc - > reg_base ) ;
2021-09-06 15:11:47 +08:00
qproc - > rmb_base = devm_platform_ioremap_resource_byname ( pdev , " rmb " ) ;
2016-07-14 12:57:44 +00:00
if ( IS_ERR ( qproc - > rmb_base ) )
2016-06-20 14:28:41 -07:00
return PTR_ERR ( qproc - > rmb_base ) ;
2021-09-17 19:25:30 +05:30
if ( qproc - > has_vq6 )
halt_cell_cnt + + ;
2016-06-20 14:28:41 -07:00
ret = of_parse_phandle_with_fixed_args ( pdev - > dev . of_node ,
2021-09-17 19:25:30 +05:30
" qcom,halt-regs " , halt_cell_cnt , 0 , & args ) ;
2016-06-20 14:28:41 -07:00
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " failed to parse qcom,halt-regs \n " ) ;
return - EINVAL ;
}
qproc - > halt_map = syscon_node_to_regmap ( args . np ) ;
of_node_put ( args . np ) ;
if ( IS_ERR ( qproc - > halt_map ) )
return PTR_ERR ( qproc - > halt_map ) ;
qproc - > halt_q6 = args . args [ 0 ] ;
qproc - > halt_modem = args . args [ 1 ] ;
qproc - > halt_nc = args . args [ 2 ] ;
2021-09-17 19:25:30 +05:30
if ( qproc - > has_vq6 )
qproc - > halt_vq6 = args . args [ 3 ] ;
if ( qproc - > has_qaccept_regs ) {
ret = of_parse_phandle_with_fixed_args ( pdev - > dev . of_node ,
" qcom,qaccept-regs " ,
3 , 0 , & args ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " failed to parse qaccept-regs \n " ) ;
return - EINVAL ;
}
qproc - > qaccept_mdm = args . args [ 0 ] ;
qproc - > qaccept_cx = args . args [ 1 ] ;
qproc - > qaccept_axi = args . args [ 2 ] ;
}
if ( qproc - > has_ext_cntl_regs ) {
ret = of_parse_phandle_with_fixed_args ( pdev - > dev . of_node ,
" qcom,ext-regs " ,
2 , 0 , & args ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " failed to parse ext-regs index 0 \n " ) ;
return - EINVAL ;
}
qproc - > conn_map = syscon_node_to_regmap ( args . np ) ;
of_node_put ( args . np ) ;
if ( IS_ERR ( qproc - > conn_map ) )
return PTR_ERR ( qproc - > conn_map ) ;
qproc - > force_clk_on = args . args [ 0 ] ;
qproc - > rscc_disable = args . args [ 1 ] ;
ret = of_parse_phandle_with_fixed_args ( pdev - > dev . of_node ,
" qcom,ext-regs " ,
2 , 1 , & args ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " failed to parse ext-regs index 1 \n " ) ;
return - EINVAL ;
}
qproc - > axim1_clk_off = args . args [ 0 ] ;
qproc - > crypto_clk_off = args . args [ 1 ] ;
}
2020-04-15 20:21:10 +05:30
if ( qproc - > has_spare_reg ) {
2019-12-19 11:15:06 +05:30
ret = of_parse_phandle_with_fixed_args ( pdev - > dev . of_node ,
2020-04-15 20:21:10 +05:30
" qcom,spare-regs " ,
2019-12-19 11:15:06 +05:30
1 , 0 , & args ) ;
if ( ret < 0 ) {
2020-04-15 20:21:10 +05:30
dev_err ( & pdev - > dev , " failed to parse spare-regs \n " ) ;
2019-12-19 11:15:06 +05:30
return - EINVAL ;
}
qproc - > conn_map = syscon_node_to_regmap ( args . np ) ;
of_node_put ( args . np ) ;
if ( IS_ERR ( qproc - > conn_map ) )
return PTR_ERR ( qproc - > conn_map ) ;
qproc - > conn_box = args . args [ 0 ] ;
}
2016-06-20 14:28:41 -07:00
return 0 ;
}
2016-12-30 19:24:01 +05:30
static int q6v5_init_clocks ( struct device * dev , struct clk * * clks ,
char * * clk_names )
2016-06-20 14:28:41 -07:00
{
2016-12-30 19:24:01 +05:30
int i ;
2016-06-20 14:28:41 -07:00
2016-12-30 19:24:01 +05:30
if ( ! clk_names )
return 0 ;
for ( i = 0 ; clk_names [ i ] ; i + + ) {
clks [ i ] = devm_clk_get ( dev , clk_names [ i ] ) ;
if ( IS_ERR ( clks [ i ] ) ) {
int rc = PTR_ERR ( clks [ i ] ) ;
2016-06-20 14:28:41 -07:00
2016-12-30 19:24:01 +05:30
if ( rc ! = - EPROBE_DEFER )
dev_err ( dev , " Failed to get %s clock \n " ,
clk_names [ i ] ) ;
return rc ;
}
2016-06-20 14:28:41 -07:00
}
2016-12-30 19:24:01 +05:30
return i ;
2016-06-20 14:28:41 -07:00
}
2019-01-30 16:39:30 -08:00
static int q6v5_pds_attach ( struct device * dev , struct device * * devs ,
char * * pd_names )
{
size_t num_pds = 0 ;
int ret ;
int i ;
if ( ! pd_names )
return 0 ;
while ( pd_names [ num_pds ] )
num_pds + + ;
for ( i = 0 ; i < num_pds ; i + + ) {
devs [ i ] = dev_pm_domain_attach_by_name ( dev , pd_names [ i ] ) ;
2019-08-21 23:35:48 +05:30
if ( IS_ERR_OR_NULL ( devs [ i ] ) ) {
ret = PTR_ERR ( devs [ i ] ) ? : - ENODATA ;
2019-01-30 16:39:30 -08:00
goto unroll_attach ;
}
}
return num_pds ;
unroll_attach :
for ( i - - ; i > = 0 ; i - - )
dev_pm_domain_detach ( devs [ i ] , false ) ;
return ret ;
2020-04-03 12:50:05 -05:00
}
2019-01-30 16:39:30 -08:00
static void q6v5_pds_detach ( struct q6v5 * qproc , struct device * * pds ,
size_t pd_count )
{
int i ;
for ( i = 0 ; i < pd_count ; i + + )
dev_pm_domain_detach ( pds [ i ] , false ) ;
}
2016-06-20 14:28:41 -07:00
static int q6v5_init_reset ( struct q6v5 * qproc )
{
2017-07-19 17:26:16 +02:00
qproc - > mss_restart = devm_reset_control_get_exclusive ( qproc - > dev ,
2018-08-30 00:42:14 +05:30
" mss_restart " ) ;
2016-06-20 14:28:41 -07:00
if ( IS_ERR ( qproc - > mss_restart ) ) {
dev_err ( qproc - > dev , " failed to acquire mss restart \n " ) ;
return PTR_ERR ( qproc - > mss_restart ) ;
}
2021-09-17 19:25:30 +05:30
if ( qproc - > has_alt_reset | | qproc - > has_spare_reg | | qproc - > has_ext_cntl_regs ) {
2018-08-30 00:42:15 +05:30
qproc - > pdc_reset = devm_reset_control_get_exclusive ( qproc - > dev ,
" pdc_reset " ) ;
if ( IS_ERR ( qproc - > pdc_reset ) ) {
dev_err ( qproc - > dev , " failed to acquire pdc reset \n " ) ;
return PTR_ERR ( qproc - > pdc_reset ) ;
}
}
2016-06-20 14:28:41 -07:00
return 0 ;
}
static int q6v5_alloc_memory_region ( struct q6v5 * qproc )
{
struct device_node * child ;
struct device_node * node ;
struct resource r ;
int ret ;
2020-04-21 20:02:25 +05:30
/*
* In the absence of mba / mpss sub - child , extract the mba and mpss
* reserved memory regions from device ' s memory - region property .
*/
2016-06-20 14:28:41 -07:00
child = of_get_child_by_name ( qproc - > dev - > of_node , " mba " ) ;
2022-03-08 06:45:21 +00:00
if ( ! child ) {
2020-04-21 20:02:25 +05:30
node = of_parse_phandle ( qproc - > dev - > of_node ,
" memory-region " , 0 ) ;
2022-03-08 06:45:21 +00:00
} else {
2020-04-21 20:02:25 +05:30
node = of_parse_phandle ( child , " memory-region " , 0 ) ;
2022-03-08 06:45:21 +00:00
of_node_put ( child ) ;
}
2020-04-21 20:02:25 +05:30
2016-06-20 14:28:41 -07:00
ret = of_address_to_resource ( node , 0 , & r ) ;
2022-03-08 06:45:21 +00:00
of_node_put ( node ) ;
2016-06-20 14:28:41 -07:00
if ( ret ) {
dev_err ( qproc - > dev , " unable to resolve mba region \n " ) ;
return ret ;
}
qproc - > mba_phys = r . start ;
qproc - > mba_size = resource_size ( & r ) ;
2020-04-21 20:02:25 +05:30
if ( ! child ) {
node = of_parse_phandle ( qproc - > dev - > of_node ,
" memory-region " , 1 ) ;
} else {
child = of_get_child_by_name ( qproc - > dev - > of_node , " mpss " ) ;
node = of_parse_phandle ( child , " memory-region " , 0 ) ;
2022-03-08 06:45:21 +00:00
of_node_put ( child ) ;
2020-04-21 20:02:25 +05:30
}
2016-06-20 14:28:41 -07:00
ret = of_address_to_resource ( node , 0 , & r ) ;
2022-03-08 06:45:21 +00:00
of_node_put ( node ) ;
2016-06-20 14:28:41 -07:00
if ( ret ) {
dev_err ( qproc - > dev , " unable to resolve mpss region \n " ) ;
return ret ;
}
qproc - > mpss_phys = qproc - > mpss_reloc = r . start ;
qproc - > mpss_size = resource_size ( & r ) ;
return 0 ;
}
static int q6v5_probe ( struct platform_device * pdev )
{
2016-12-30 19:24:00 +05:30
const struct rproc_hexagon_res * desc ;
2022-02-28 23:53:59 +01:00
struct device_node * node ;
2016-06-20 14:28:41 -07:00
struct q6v5 * qproc ;
struct rproc * rproc ;
2019-01-15 01:20:01 +05:30
const char * mba_image ;
2016-06-20 14:28:41 -07:00
int ret ;
2016-12-30 19:24:00 +05:30
desc = of_device_get_match_data ( & pdev - > dev ) ;
if ( ! desc )
return - EINVAL ;
2018-10-08 19:08:05 -07:00
if ( desc - > need_mem_protection & & ! qcom_scm_is_available ( ) )
return - EPROBE_DEFER ;
2019-01-15 01:20:01 +05:30
mba_image = desc - > hexagon_mba_image ;
ret = of_property_read_string_index ( pdev - > dev . of_node , " firmware-name " ,
0 , & mba_image ) ;
2021-03-11 16:26:05 -08:00
if ( ret < 0 & & ret ! = - EINVAL ) {
dev_err ( & pdev - > dev , " unable to read mba firmware-name \n " ) ;
2019-01-15 01:20:01 +05:30
return ret ;
2021-03-11 16:26:05 -08:00
}
2019-01-15 01:20:01 +05:30
2016-06-20 14:28:41 -07:00
rproc = rproc_alloc ( & pdev - > dev , pdev - > name , & q6v5_ops ,
2019-01-15 01:20:01 +05:30
mba_image , sizeof ( * qproc ) ) ;
2016-06-20 14:28:41 -07:00
if ( ! rproc ) {
dev_err ( & pdev - > dev , " failed to allocate rproc \n " ) ;
return - ENOMEM ;
}
2018-05-24 22:21:41 +03:00
rproc - > auto_boot = false ;
2020-04-10 12:24:33 +02:00
rproc_coredump_set_elf_info ( rproc , ELFCLASS32 , EM_NONE ) ;
2018-05-24 22:21:41 +03:00
2016-06-20 14:28:41 -07:00
qproc = ( struct q6v5 * ) rproc - > priv ;
qproc - > dev = & pdev - > dev ;
qproc - > rproc = rproc ;
2019-01-15 01:20:01 +05:30
qproc - > hexagon_mdt_image = " modem.mdt " ;
ret = of_property_read_string_index ( pdev - > dev . of_node , " firmware-name " ,
1 , & qproc - > hexagon_mdt_image ) ;
2021-03-11 16:26:05 -08:00
if ( ret < 0 & & ret ! = - EINVAL ) {
dev_err ( & pdev - > dev , " unable to read mpss firmware-name \n " ) ;
2020-04-03 12:50:04 -05:00
goto free_rproc ;
2021-03-11 16:26:05 -08:00
}
2019-01-15 01:20:01 +05:30
2016-06-20 14:28:41 -07:00
platform_set_drvdata ( pdev , qproc ) ;
2021-09-17 19:25:30 +05:30
qproc - > has_qaccept_regs = desc - > has_qaccept_regs ;
qproc - > has_ext_cntl_regs = desc - > has_ext_cntl_regs ;
qproc - > has_vq6 = desc - > has_vq6 ;
2020-04-15 20:21:10 +05:30
qproc - > has_spare_reg = desc - > has_spare_reg ;
2016-06-20 14:28:41 -07:00
ret = q6v5_init_mem ( qproc , pdev ) ;
if ( ret )
goto free_rproc ;
ret = q6v5_alloc_memory_region ( qproc ) ;
if ( ret )
goto free_rproc ;
2016-12-30 19:24:01 +05:30
ret = q6v5_init_clocks ( & pdev - > dev , qproc - > proxy_clks ,
desc - > proxy_clk_names ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " Failed to get proxy clocks. \n " ) ;
2016-06-20 14:28:41 -07:00
goto free_rproc ;
2016-12-30 19:24:01 +05:30
}
qproc - > proxy_clk_count = ret ;
2018-05-21 22:57:13 +05:30
ret = q6v5_init_clocks ( & pdev - > dev , qproc - > reset_clks ,
desc - > reset_clk_names ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " Failed to get reset clocks. \n " ) ;
goto free_rproc ;
}
qproc - > reset_clk_count = ret ;
2016-12-30 19:24:01 +05:30
ret = q6v5_init_clocks ( & pdev - > dev , qproc - > active_clks ,
desc - > active_clk_names ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " Failed to get active clocks. \n " ) ;
goto free_rproc ;
}
qproc - > active_clk_count = ret ;
2016-06-20 14:28:41 -07:00
2016-12-30 19:24:02 +05:30
ret = q6v5_regulator_init ( & pdev - > dev , qproc - > proxy_regs ,
desc - > proxy_supply ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " Failed to get proxy regulators. \n " ) ;
2016-06-20 14:28:41 -07:00
goto free_rproc ;
2016-12-30 19:24:02 +05:30
}
qproc - > proxy_reg_count = ret ;
ret = q6v5_regulator_init ( & pdev - > dev , qproc - > active_regs ,
desc - > active_supply ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " Failed to get active regulators. \n " ) ;
goto free_rproc ;
}
qproc - > active_reg_count = ret ;
2016-06-20 14:28:41 -07:00
2019-01-30 16:39:30 -08:00
ret = q6v5_pds_attach ( & pdev - > dev , qproc - > proxy_pds ,
desc - > proxy_pd_names ) ;
2020-09-16 12:41:31 +02:00
/* Fallback to regulators for old device trees */
if ( ret = = - ENODATA & & desc - > fallback_proxy_supply ) {
ret = q6v5_regulator_init ( & pdev - > dev ,
qproc - > fallback_proxy_regs ,
desc - > fallback_proxy_supply ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " Failed to get fallback proxy regulators. \n " ) ;
2021-09-16 19:29:21 +05:30
goto free_rproc ;
2020-09-16 12:41:31 +02:00
}
qproc - > fallback_proxy_reg_count = ret ;
} else if ( ret < 0 ) {
2019-01-30 16:39:30 -08:00
dev_err ( & pdev - > dev , " Failed to init power domains \n " ) ;
2021-09-16 19:29:21 +05:30
goto free_rproc ;
2020-09-16 12:41:31 +02:00
} else {
qproc - > proxy_pd_count = ret ;
2019-01-30 16:39:30 -08:00
}
2018-08-30 00:42:15 +05:30
qproc - > has_alt_reset = desc - > has_alt_reset ;
2016-06-20 14:28:41 -07:00
ret = q6v5_init_reset ( qproc ) ;
if ( ret )
2019-01-30 16:39:30 -08:00
goto detach_proxy_pds ;
2016-06-20 14:28:41 -07:00
2017-10-24 21:22:27 +05:30
qproc - > version = desc - > version ;
2017-10-24 21:22:26 +05:30
qproc - > need_mem_protection = desc - > need_mem_protection ;
2020-07-21 16:59:35 +05:30
qproc - > has_mba_logs = desc - > has_mba_logs ;
2016-06-20 14:28:41 -07:00
2021-09-16 19:29:21 +05:30
ret = qcom_q6v5_init ( & qproc - > q6v5 , pdev , rproc , MPSS_CRASH_REASON_SMEM , " modem " ,
2018-06-04 13:30:38 -07:00
qcom_msa_handover ) ;
if ( ret )
2019-01-30 16:39:30 -08:00
goto detach_proxy_pds ;
2016-06-20 14:28:41 -07:00
2017-10-24 21:22:26 +05:30
qproc - > mpss_perm = BIT ( QCOM_SCM_VMID_HLOS ) ;
qproc - > mba_perm = BIT ( QCOM_SCM_VMID_HLOS ) ;
2020-04-22 17:37:33 -07:00
qcom_add_glink_subdev ( rproc , & qproc - > glink_subdev , " mpss " ) ;
2017-01-29 14:05:50 -08:00
qcom_add_smd_subdev ( rproc , & qproc - > smd_subdev ) ;
2017-07-24 22:56:43 -07:00
qcom_add_ssr_subdev ( rproc , & qproc - > ssr_subdev , " mpss " ) ;
2017-08-27 21:51:38 -07:00
qproc - > sysmon = qcom_add_sysmon_subdev ( rproc , " modem " , 0x12 ) ;
2019-01-08 15:53:43 +05:30
if ( IS_ERR ( qproc - > sysmon ) ) {
ret = PTR_ERR ( qproc - > sysmon ) ;
2020-04-03 12:50:05 -05:00
goto remove_subdevs ;
2019-01-08 15:53:43 +05:30
}
2017-01-29 14:05:50 -08:00
2016-06-20 14:28:41 -07:00
ret = rproc_add ( rproc ) ;
if ( ret )
2020-04-03 12:50:05 -05:00
goto remove_sysmon_subdev ;
2016-06-20 14:28:41 -07:00
2022-02-28 23:53:59 +01:00
node = of_get_compatible_child ( pdev - > dev . of_node , " qcom,bam-dmux " ) ;
qproc - > bam_dmux = of_platform_device_create ( node , NULL , & pdev - > dev ) ;
of_node_put ( node ) ;
2016-06-20 14:28:41 -07:00
return 0 ;
2020-04-03 12:50:05 -05:00
remove_sysmon_subdev :
qcom_remove_sysmon_subdev ( qproc - > sysmon ) ;
remove_subdevs :
qcom_remove_ssr_subdev ( rproc , & qproc - > ssr_subdev ) ;
qcom_remove_smd_subdev ( rproc , & qproc - > smd_subdev ) ;
qcom_remove_glink_subdev ( rproc , & qproc - > glink_subdev ) ;
detach_proxy_pds :
2019-01-30 16:39:30 -08:00
q6v5_pds_detach ( qproc , qproc - > proxy_pds , qproc - > proxy_pd_count ) ;
2016-06-20 14:28:41 -07:00
free_rproc :
2016-10-02 17:46:38 -07:00
rproc_free ( rproc ) ;
2016-06-20 14:28:41 -07:00
return ret ;
}
static int q6v5_remove ( struct platform_device * pdev )
{
struct q6v5 * qproc = platform_get_drvdata ( pdev ) ;
2020-04-03 12:50:05 -05:00
struct rproc * rproc = qproc - > rproc ;
2016-06-20 14:28:41 -07:00
2022-02-28 23:53:59 +01:00
if ( qproc - > bam_dmux )
of_platform_device_destroy ( & qproc - > bam_dmux - > dev , NULL ) ;
2020-04-03 12:50:05 -05:00
rproc_del ( rproc ) ;
2017-01-29 14:05:50 -08:00
2021-09-16 19:29:21 +05:30
qcom_q6v5_deinit ( & qproc - > q6v5 ) ;
2017-08-27 21:51:38 -07:00
qcom_remove_sysmon_subdev ( qproc - > sysmon ) ;
2020-04-03 12:50:05 -05:00
qcom_remove_ssr_subdev ( rproc , & qproc - > ssr_subdev ) ;
qcom_remove_smd_subdev ( rproc , & qproc - > smd_subdev ) ;
qcom_remove_glink_subdev ( rproc , & qproc - > glink_subdev ) ;
2019-01-30 16:39:30 -08:00
q6v5_pds_detach ( qproc , qproc - > proxy_pds , qproc - > proxy_pd_count ) ;
2020-04-03 12:50:05 -05:00
rproc_free ( rproc ) ;
2016-06-20 14:28:41 -07:00
return 0 ;
}
2019-12-19 11:15:06 +05:30
static const struct rproc_hexagon_res sc7180_mss = {
. hexagon_mba_image = " mba.mbn " ,
. proxy_clk_names = ( char * [ ] ) {
" xo " ,
NULL
} ,
. reset_clk_names = ( char * [ ] ) {
" iface " ,
" bus " ,
" snoc_axi " ,
NULL
} ,
. active_clk_names = ( char * [ ] ) {
" mnoc_axi " ,
" nav " ,
NULL
} ,
. proxy_pd_names = ( char * [ ] ) {
" cx " ,
" mx " ,
" mss " ,
NULL
} ,
. need_mem_protection = true ,
. has_alt_reset = false ,
2020-07-21 16:59:35 +05:30
. has_mba_logs = true ,
2020-04-15 20:21:10 +05:30
. has_spare_reg = true ,
2021-09-17 19:25:30 +05:30
. has_qaccept_regs = false ,
. has_ext_cntl_regs = false ,
. has_vq6 = false ,
2019-12-19 11:15:06 +05:30
. version = MSS_SC7180 ,
} ;
2021-09-17 19:25:30 +05:30
static const struct rproc_hexagon_res sc7280_mss = {
. hexagon_mba_image = " mba.mbn " ,
. proxy_clk_names = ( char * [ ] ) {
" xo " ,
" pka " ,
NULL
} ,
. active_clk_names = ( char * [ ] ) {
" iface " ,
" offline " ,
" snoc_axi " ,
NULL
} ,
. proxy_pd_names = ( char * [ ] ) {
" cx " ,
" mss " ,
NULL
} ,
. need_mem_protection = true ,
. has_alt_reset = false ,
. has_mba_logs = true ,
. has_spare_reg = false ,
. has_qaccept_regs = true ,
. has_ext_cntl_regs = true ,
. has_vq6 = true ,
. version = MSS_SC7280 ,
} ;
2018-05-21 22:57:13 +05:30
static const struct rproc_hexagon_res sdm845_mss = {
. hexagon_mba_image = " mba.mbn " ,
. proxy_clk_names = ( char * [ ] ) {
" xo " ,
" prng " ,
NULL
} ,
. reset_clk_names = ( char * [ ] ) {
" iface " ,
" snoc_axi " ,
NULL
} ,
. active_clk_names = ( char * [ ] ) {
" bus " ,
" mem " ,
" gpll0_mss " ,
" mnoc_axi " ,
NULL
} ,
2019-01-30 16:39:30 -08:00
. proxy_pd_names = ( char * [ ] ) {
" cx " ,
" mx " ,
" mss " ,
NULL
} ,
2018-05-21 22:57:13 +05:30
. need_mem_protection = true ,
. has_alt_reset = true ,
2020-07-21 16:59:35 +05:30
. has_mba_logs = false ,
2020-04-15 20:21:10 +05:30
. has_spare_reg = false ,
2021-09-17 19:25:30 +05:30
. has_qaccept_regs = false ,
. has_ext_cntl_regs = false ,
. has_vq6 = false ,
2018-05-21 22:57:13 +05:30
. version = MSS_SDM845 ,
} ;
2019-10-31 19:45:01 -07:00
static const struct rproc_hexagon_res msm8998_mss = {
. hexagon_mba_image = " mba.mbn " ,
. proxy_clk_names = ( char * [ ] ) {
" xo " ,
" qdss " ,
" mem " ,
NULL
} ,
. active_clk_names = ( char * [ ] ) {
" iface " ,
" bus " ,
" gpll0_mss " ,
" mnoc_axi " ,
" snoc_axi " ,
NULL
} ,
. proxy_pd_names = ( char * [ ] ) {
" cx " ,
" mx " ,
NULL
} ,
. need_mem_protection = true ,
. has_alt_reset = false ,
2020-07-21 16:59:35 +05:30
. has_mba_logs = false ,
2020-04-15 20:21:10 +05:30
. has_spare_reg = false ,
2021-09-17 19:25:30 +05:30
. has_qaccept_regs = false ,
. has_ext_cntl_regs = false ,
. has_vq6 = false ,
2019-10-31 19:45:01 -07:00
. version = MSS_MSM8998 ,
} ;
2017-10-24 21:22:27 +05:30
static const struct rproc_hexagon_res msm8996_mss = {
. hexagon_mba_image = " mba.mbn " ,
2018-12-29 00:23:05 +05:30
. proxy_supply = ( struct qcom_mss_reg_res [ ] ) {
{
. supply = " pll " ,
. uA = 100000 ,
} ,
{ }
} ,
2017-10-24 21:22:27 +05:30
. proxy_clk_names = ( char * [ ] ) {
" xo " ,
" pnoc " ,
2018-12-29 00:23:03 +05:30
" qdss " ,
2017-10-24 21:22:27 +05:30
NULL
} ,
. active_clk_names = ( char * [ ] ) {
" iface " ,
" bus " ,
" mem " ,
2018-12-29 00:23:03 +05:30
" gpll0_mss " ,
" snoc_axi " ,
" mnoc_axi " ,
2017-10-24 21:22:27 +05:30
NULL
} ,
2022-07-04 19:22:02 +03:00
. proxy_pd_names = ( char * [ ] ) {
" mx " ,
" cx " ,
NULL
} ,
2017-10-24 21:22:27 +05:30
. need_mem_protection = true ,
2018-05-21 22:57:13 +05:30
. has_alt_reset = false ,
2020-07-21 16:59:35 +05:30
. has_mba_logs = false ,
2020-04-15 20:21:10 +05:30
. has_spare_reg = false ,
2021-09-17 19:25:30 +05:30
. has_qaccept_regs = false ,
. has_ext_cntl_regs = false ,
. has_vq6 = false ,
2017-10-24 21:22:27 +05:30
. version = MSS_MSM8996 ,
} ;
2016-12-30 19:24:00 +05:30
static const struct rproc_hexagon_res msm8916_mss = {
. hexagon_mba_image = " mba.mbn " ,
2016-12-30 19:24:02 +05:30
. proxy_supply = ( struct qcom_mss_reg_res [ ] ) {
2020-09-16 12:41:31 +02:00
{
. supply = " pll " ,
. uA = 100000 ,
} ,
{ }
} ,
. fallback_proxy_supply = ( struct qcom_mss_reg_res [ ] ) {
2016-12-30 19:24:02 +05:30
{
. supply = " mx " ,
. uV = 1050000 ,
} ,
{
. supply = " cx " ,
. uA = 100000 ,
} ,
{ }
} ,
2016-12-30 19:24:01 +05:30
. proxy_clk_names = ( char * [ ] ) {
" xo " ,
NULL
} ,
. active_clk_names = ( char * [ ] ) {
" iface " ,
" bus " ,
" mem " ,
NULL
} ,
2020-09-16 12:41:31 +02:00
. proxy_pd_names = ( char * [ ] ) {
" mx " ,
" cx " ,
NULL
} ,
2017-10-24 21:22:26 +05:30
. need_mem_protection = false ,
2018-05-21 22:57:13 +05:30
. has_alt_reset = false ,
2020-07-21 16:59:35 +05:30
. has_mba_logs = false ,
2020-04-15 20:21:10 +05:30
. has_spare_reg = false ,
2021-09-17 19:25:30 +05:30
. has_qaccept_regs = false ,
. has_ext_cntl_regs = false ,
. has_vq6 = false ,
2017-10-24 21:22:27 +05:30
. version = MSS_MSM8916 ,
2016-12-30 19:24:00 +05:30
} ;
static const struct rproc_hexagon_res msm8974_mss = {
. hexagon_mba_image = " mba.b00 " ,
2016-12-30 19:24:02 +05:30
. proxy_supply = ( struct qcom_mss_reg_res [ ] ) {
2020-09-16 12:41:31 +02:00
{
. supply = " pll " ,
. uA = 100000 ,
} ,
{ }
} ,
. fallback_proxy_supply = ( struct qcom_mss_reg_res [ ] ) {
2016-12-30 19:24:02 +05:30
{
. supply = " mx " ,
. uV = 1050000 ,
} ,
{
. supply = " cx " ,
. uA = 100000 ,
} ,
{ }
} ,
. active_supply = ( struct qcom_mss_reg_res [ ] ) {
{
. supply = " mss " ,
. uV = 1050000 ,
. uA = 100000 ,
} ,
{ }
} ,
2016-12-30 19:24:01 +05:30
. proxy_clk_names = ( char * [ ] ) {
" xo " ,
NULL
} ,
. active_clk_names = ( char * [ ] ) {
" iface " ,
" bus " ,
" mem " ,
NULL
} ,
2020-09-16 12:41:31 +02:00
. proxy_pd_names = ( char * [ ] ) {
" mx " ,
" cx " ,
NULL
} ,
2017-10-24 21:22:26 +05:30
. need_mem_protection = false ,
2018-05-21 22:57:13 +05:30
. has_alt_reset = false ,
2020-07-21 16:59:35 +05:30
. has_mba_logs = false ,
2020-04-15 20:21:10 +05:30
. has_spare_reg = false ,
2021-09-17 19:25:30 +05:30
. has_qaccept_regs = false ,
. has_ext_cntl_regs = false ,
. has_vq6 = false ,
2017-10-24 21:22:27 +05:30
. version = MSS_MSM8974 ,
2016-12-30 19:24:00 +05:30
} ;
2016-06-20 14:28:41 -07:00
static const struct of_device_id q6v5_of_match [ ] = {
2016-12-30 19:24:00 +05:30
{ . compatible = " qcom,q6v5-pil " , . data = & msm8916_mss } ,
{ . compatible = " qcom,msm8916-mss-pil " , . data = & msm8916_mss } ,
{ . compatible = " qcom,msm8974-mss-pil " , . data = & msm8974_mss } ,
2017-10-24 21:22:27 +05:30
{ . compatible = " qcom,msm8996-mss-pil " , . data = & msm8996_mss } ,
2019-10-31 19:45:01 -07:00
{ . compatible = " qcom,msm8998-mss-pil " , . data = & msm8998_mss } ,
2019-12-19 11:15:06 +05:30
{ . compatible = " qcom,sc7180-mss-pil " , . data = & sc7180_mss } ,
2021-09-17 19:25:30 +05:30
{ . compatible = " qcom,sc7280-mss-pil " , . data = & sc7280_mss } ,
2018-05-21 22:57:13 +05:30
{ . compatible = " qcom,sdm845-mss-pil " , . data = & sdm845_mss } ,
2016-06-20 14:28:41 -07:00
{ } ,
} ;
2016-10-18 18:24:19 -03:00
MODULE_DEVICE_TABLE ( of , q6v5_of_match ) ;
2016-06-20 14:28:41 -07:00
static struct platform_driver q6v5_driver = {
. probe = q6v5_probe ,
. remove = q6v5_remove ,
. driver = {
2018-09-24 16:45:26 -07:00
. name = " qcom-q6v5-mss " ,
2016-06-20 14:28:41 -07:00
. of_match_table = q6v5_of_match ,
} ,
} ;
module_platform_driver ( q6v5_driver ) ;
2018-09-24 16:45:26 -07:00
MODULE_DESCRIPTION ( " Qualcomm Self-authenticating modem remoteproc driver " ) ;
2016-06-20 14:28:41 -07:00
MODULE_LICENSE ( " GPL v2 " ) ;