// SPDX-License-Identifier: GPL-2.0 /* * Copyright (c) 2019 MediaTek Inc. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef CONFIG_OF #include #include #include #endif #include #include #include #include #include #include #include #include /* for SMC ID table */ #include #include "scp_ipi.h" #include "scp_helper.h" #include "scp_excep.h" #include "scp_dvfs.h" #include "dvfsrc-exp.h" #include "mtk_spm_resource_req.h" #if (defined(CONFIG_MACH_MT6781) \ ||defined(CONFIG_MACH_MT6768) || defined(CONFIG_MACH_MT6771)) #include //#include #if !defined(CONFIG_FPGA_EARLY_PORTING) #include "mtk_pmic_info.h" #include "mtk_pmic_api_buck.h" #include #include #endif #endif #ifdef pr_fmt #undef pr_fmt #endif #define pr_fmt(fmt) "[scp_dvfs]: " fmt #define DRV_Reg32(addr) readl(addr) #define DRV_WriteReg32(addr, val) writel(val, addr) #define DRV_SetReg32(addr, val) DRV_WriteReg32(addr, DRV_Reg32(addr) | (val)) #define DRV_ClrReg32(addr, val) DRV_WriteReg32(addr, DRV_Reg32(addr) & ~(val)) /* -1:SCP DVFS OFF, 1:SCP DVFS ON */ static int scp_dvfs_flag = 1; #define SCP_DVFS_DTSNO_GPIO_CONFIG -1 #define SCP_DVFS_DTSNO_PMIC_CONFIG -2 int scp_pll_ctrl_set(unsigned int pll_ctrl_flag, unsigned int pll_sel); /* * 0: SCP Sleep: OFF, * 1: SCP Sleep: ON, * 2: SCP Sleep: sleep without wakeup, * 3: SCP Sleep: force to sleep */ static int scp_sleep_flag = -1; static int pre_pll_sel = -1; static struct mt_scp_pll_t *mt_scp_pll; static struct wakeup_source *scp_suspend_lock; static int g_scp_dvfs_init_flag = -1; static struct dvfs_data *dvfs; static struct regulator *reg_vcore, *reg_vsram; static struct mtk_pm_qos_request dvfsrc_scp_vcore_req; static struct subsys_data *sd; const char sub_feature_name[SUB_FEATURE_NUM][15] = { "gpio-mode", "pmic-vow-lp", "pmic-pmrc", }; const char subsys_name[SYS_NUM][15] = { "gpio", "pmic", }; /*bool __attribute__((weak)) spm_resource_req(unsigned int user, unsigned int req_mask) { pr_err("ERROR: %s is not linked\n", __func__); return true; }*/ #if defined(CONFIG_MACH_MT6781) #define ADR_GPIO_DIR6 (gpio_base + 0x060) #define BIT_GPIO_DIR6_GPIO200 8 #define MAK_GPIO_DIR6_GPIO200 0x1 /* GPIO200=Lo: SCP uses VASRAM_CORE */ /* GPIO200=High: SCP uses VASRAM_OTHERS */ #define ADR_GPIO_DIN6 (gpio_base + 0x260) #define BIT_GPIO_DIN6_GPIO200 8 #define MAK_GPIO_DIN6_GPIO200 0x1 #define USE_VSRAM_CORE 1 #define USE_VSRAM_OTHERS 2 int Scp_Vsram_Ldo_usage = USE_VSRAM_OTHERS; int scp_resource_req(unsigned int req_type) { unsigned long ret = 0; struct arm_smccc_res res = {0}; /* ret = mt_secure_call(MTK_SIP_KERNEL_SCP_DVFS_CTRL, req_type, 0, 0, 0); */ arm_smccc_smc(MTK_SIP_KERNEL_SCP_DVFS_CTRL, req_type, 0, 0, 0, 0, 0, 0, &res); ret = res.a0; return ret; } EXPORT_SYMBOL(scp_resource_req); #endif static int scp_get_sub_feature_idx(enum subsys_enum sys_e, enum sub_feature_enum comp_e) { int i; if ((sys_e < 0 || sys_e >= SYS_NUM) || (comp_e < 0 || comp_e >= SUB_FEATURE_NUM)) return -EINVAL; if(!sd[sys_e].regmap || !sd[sys_e].fd) { pr_err("cannot find feature index\n"); return -EINVAL; } for (i = 0; i < sd[sys_e].num; i++) { if (!strcmp(sd[sys_e].fd[i].name, sub_feature_name[comp_e])) break; } if (i == SUB_FEATURE_NUM) { pr_err("cannot find feature index\n"); return -EINVAL; } return i; } static struct sub_feature_data *scp_get_sub_feature(enum subsys_enum sys_e, enum sub_feature_enum comp_e) { int idx; if ((sys_e < 0 || sys_e >= SYS_NUM) || (comp_e < 0 || comp_e >= SUB_FEATURE_NUM)) return NULL; idx = scp_get_sub_feature_idx(sys_e, comp_e); if (idx < 0) return NULL; return &sd[sys_e].fd[idx]; } static int scp_get_sub_feature_onoff(enum subsys_enum sys_e, enum sub_feature_enum comp_e) { int idx; if ((sys_e < 0 || sys_e >= SYS_NUM) || (comp_e < 0 || comp_e >= SUB_FEATURE_NUM)) return 0; idx = scp_get_sub_feature_idx(sys_e, comp_e); if (idx < 0) return 0; return sd[sys_e].fd[idx].onoff; } static unsigned int *scp_get_sub_register_cfg(enum subsys_enum sys_e, enum sub_feature_enum comp_e) { struct regmap *regmap = sd[sys_e].regmap; struct sub_feature_data *fd; unsigned int *ret; unsigned int val = 0; int i; if ((sys_e < 0 || sys_e >= SYS_NUM) || (comp_e < 0 || comp_e >= SUB_FEATURE_NUM)) return NULL; fd = scp_get_sub_feature(sys_e, comp_e); if (!fd) { pr_err("fd is NULL\n"); return NULL; } /* alloc memory for return cfg value */ ret = kcalloc(fd->num, sizeof(unsigned int), GFP_KERNEL); if (!ret) return NULL; for (i = 0; i < fd->num; i++) { regmap_read(regmap, fd->reg[i].ofs, &val); ret[i] = (val >> fd->reg[i].bit) & fd->reg[i].msk; } return ret; } static unsigned int scp_set_sub_register_cfg_internal( struct regmap *regmap, struct reg_info *reg, struct reg_cfg *cfg, bool on) { unsigned int val, mask; int ret = 0; /* get mask */ mask = reg->msk << reg->bit; /* get new cfg setting according to feature on/off */ val = on ? cfg->on : cfg->off; val = (val & reg->msk) << reg->bit; if (reg->setclr) { /* clear cfg setting first */ ret = regmap_write(regmap, reg->ofs + 4, mask); /* set cfg with new setting */ ret = regmap_write(regmap, reg->ofs + 2, val); } else /* directly write cfg by new setting */ ret = regmap_update_bits(regmap, reg->ofs, mask, val); return ret; } static int scp_set_sub_register_cfg(enum subsys_enum sys_e, enum sub_feature_enum comp_e, bool on) { struct regmap *regmap = sd[sys_e].regmap; struct sub_feature_data *fd = NULL; int ret = 0; int i; if ((sys_e < 0 || sys_e >= SYS_NUM) || (comp_e < 0 || comp_e >= SUB_FEATURE_NUM)) return -EINVAL; if (!regmap) { pr_err("scp_dvfs: %d regmap is NULL\n", sys_e); return -EINVAL; } fd = scp_get_sub_feature(sys_e, comp_e); if (!fd) { pr_err("fd is NULL\n"); return -EINVAL; } for (i = 0; i < fd->num; i++) { ret = scp_set_sub_register_cfg_internal(regmap, &fd->reg[i], &fd->cfg[i], on); if (ret) { pr_err("fail to set sub feature cfg[%d] : %d\n", i, ret); break; } } return ret; } static int scp_get_freq_idx(unsigned int clk_opp) { int i; for (i = 0; i < dvfs->scp_opp_num; i++) if (clk_opp == dvfs->opp[i].freq) break; if (i == dvfs->scp_opp_num) { pr_err("no available opp\n"); return -EINVAL; } return i; } int scp_set_pmic_vcore(unsigned int cur_freq) { int ret = 0; #if !defined(CONFIG_FPGA_EARLY_PORTING) int idx = 0; unsigned int ret_vc = 0, ret_vs = 0; /* if vcore/vsram define as 0xff, means no pmic op during dvfs */ if (dvfs->opp[dvfs->scp_opp_num - 1].vcore == 0xff && dvfs->opp[dvfs->scp_opp_num - 1].vsram == 0xff) return ret; idx = scp_get_freq_idx(cur_freq); if (idx >= 0 && idx < dvfs->scp_opp_num) { unsigned int vcore; unsigned int uv = dvfs->opp[idx].uv_idx; #if !defined(CONFIG_MACH_MT6768) \ && !defined(CONFIG_MACH_MT6781) && !defined(CONFIG_MACH_MT6771) int max_vcore = dvfs->opp[dvfs->scp_opp_num - 1].vcore + 100000; int max_vsram = dvfs->opp[dvfs->scp_opp_num - 1].vsram + 100000; #endif if (uv != 0xff) vcore = mtk_dvfsrc_vcore_uv_table(uv); else vcore = dvfs->opp[idx].vcore; /* vcore MAX_uV set to highest opp + 100mV */ #if (defined(CONFIG_MACH_MT6768) \ || defined(CONFIG_MACH_MT6781) || defined(CONFIG_MACH_MT6771)) ret_vc = pmic_scp_set_vcore(vcore); ret_vs = pmic_scp_set_vsram_vcore(dvfs->opp[idx].vsram); #else ret_vc = regulator_set_voltage(reg_vcore, vcore, max_vcore); ret_vs = regulator_set_voltage(reg_vsram, dvfs->opp[idx].vsram, max_vsram); #endif } else { ret = -2; pr_err("cur_freq=%d is not supported\n", cur_freq); WARN_ON(1); } if (ret_vc != 0 || ret_vs != 0) { ret = -1; pr_err("ERROR: %s: scp vcore/vsram setting error, (%d, %d)\n", __func__, ret_vc, ret_vs); WARN_ON(1); } if (scp_get_sub_feature_onoff(SYS_PMIC, PMIC_VOW_LP)) { /* vcore > 0.6v cannot hold pmic/vcore in lp mode */ if (idx < 2) /* enable VOW low power mode */ ret = scp_set_sub_register_cfg(SYS_PMIC, PMIC_VOW_LP, true); else /* disable VOW low power mode */ ret = scp_set_sub_register_cfg(SYS_PMIC, PMIC_VOW_LP, false); } #endif /* CONFIG_FPGA_EARLY_PORTING */ return ret; } uint32_t scp_get_freq(void) { uint32_t i; uint32_t sum = 0; uint32_t return_freq = 0; /* * calculate scp frequence */ for (i = 0; i < NUM_FEATURE_ID; i++) { if (feature_table[i].enable == 1) sum += feature_table[i].freq; } /* * calculate scp sensor frequence */ for (i = 0; i < NUM_SENSOR_TYPE; i++) { if (sensor_type_table[i].enable == 1) sum += sensor_type_table[i].freq; } /*pr_debug("[SCP] needed freq sum:%d\n",sum);*/ for (i = 0; i < dvfs->scp_opp_num; i++) { if (sum <= dvfs->opp[i].freq) { return_freq = dvfs->opp[i].freq; break; } } if (i == dvfs->scp_opp_num) { return_freq = dvfs->opp[dvfs->scp_opp_num - 1].freq; pr_notice("warning: request freq %d > max opp %d\n", sum, return_freq); } return return_freq; } void scp_vcore_request(unsigned int clk_opp) { int idx; pr_debug("%s(%d)\n", __func__, clk_opp); /* Set PMIC */ scp_set_pmic_vcore(clk_opp); /* DVFSRC_VCORE_REQUEST [31:30] * 2'b00: scp request 0.575v/0.6v/0.625v * 2'b01: scp request 0.7v * 2'b10: scp request 0.8v */ idx = scp_get_freq_idx(clk_opp); if (idx < 0) { pr_err("%s: invalid clk_opp %d\n", __func__, clk_opp); /* WARN_ON(1); */ return; } /* * dvfs->opp[idx].dvfsrc_opp == 0xff, * means scp opp[idx] not supported * in dvfsrc opp table */ if (dvfs->opp[idx].dvfsrc_opp != 0xff) mtk_pm_qos_update_request(&dvfsrc_scp_vcore_req, dvfs->opp[idx].dvfsrc_opp); /* SCP to SPM voltage level 0x100066C4 (scp reg 0xC0094) * 0x0: scp request 0.575v/0.6v * 0x1: scp request 0.625v * 0x12: scp request 0.7v * 0x28: scp request 0.8v */ DRV_WriteReg32(SCP_SCP2SPM_VOL_LV, dvfs->opp[idx].spm_opp); } /* scp_request_freq * return :-1 means the scp request freq. error * return :0 means the request freq. finished */ int scp_request_freq(void) { int value = 0; int timeout = 250; int ret = 0; unsigned long spin_flags; int is_increasing_freq = 0; pr_debug("%s()\n", __func__); if (scp_dvfs_flag != 1) { pr_debug("warning: SCP DVFS is OFF\n"); return 0; } /* because we are waiting for scp to update register:scp_current_freq * use wake lock to prevent AP from entering suspend state */ __pm_stay_awake(scp_suspend_lock); if (scp_current_freq != scp_expected_freq) { /* do DVS before DFS if increasing frequency */ if (scp_current_freq < scp_expected_freq) { scp_vcore_request(scp_expected_freq); is_increasing_freq = 1; } #ifndef CONFIG_MACH_MT6771 /* Request SPM not to turn off mainpll/26M/infra */ /* because SCP may park in it during DFS process */ #if defined(CONFIG_MACH_MT6781) scp_resource_req(SCP_REQ_26M | SCP_REQ_IFR | SCP_REQ_SYSPLL1); #else spm_resource_req(SPM_RESOURCE_USER_SCP, SPM_RESOURCE_MAINPLL | SPM_RESOURCE_CK_26M | SPM_RESOURCE_AXI_BUS); #endif #endif /* turn on PLL if necessary */ scp_pll_ctrl_set(PLL_ENABLE, scp_expected_freq); do { ret = scp_ipi_send(IPI_DVFS_SET_FREQ, (void *)&value, sizeof(value), 0, SCP_A_ID); if (ret != SCP_IPI_DONE) pr_debug("SCP send IPI fail - %d\n", ret); mdelay(2); timeout -= 1; /*try 50 times, total about 100ms*/ if (timeout <= 0) { pr_err("set freq fail, current(%d) != expect(%d)\n", scp_current_freq, scp_expected_freq); __pm_relax(scp_suspend_lock); WARN_ON(1); return -1; } /* read scp_current_freq again */ spin_lock_irqsave(&scp_awake_spinlock, spin_flags); scp_current_freq = readl(CURRENT_FREQ_REG); spin_unlock_irqrestore(&scp_awake_spinlock, spin_flags); } while (scp_current_freq != scp_expected_freq); /* turn off PLL if necessary */ scp_pll_ctrl_set(PLL_DISABLE, scp_expected_freq); /* do DVS after DFS if decreasing frequency */ if (is_increasing_freq == 0) scp_vcore_request(scp_expected_freq); #ifndef CONFIG_MACH_MT6771 #ifndef CONFIG_MACH_MT6768 if (scp_expected_freq == MAINPLL_273M) #if defined(CONFIG_MACH_MT6781) scp_resource_req(SCP_REQ_26M | SCP_REQ_IFR | SCP_REQ_SYSPLL1); #else spm_resource_req(SPM_RESOURCE_USER_SCP, SPM_RESOURCE_MAINPLL); #endif else if (scp_expected_freq == UNIVPLL_416M) #else if (scp_expected_freq == UNIVPLL_416M) #endif #if defined(CONFIG_MACH_MT6781) scp_resource_req(SCP_REQ_26M | SCP_REQ_IFR); #else spm_resource_req(SPM_RESOURCE_USER_SCP, SPM_RESOURCE_CK_26M | SPM_RESOURCE_AXI_BUS); #endif else #if defined(CONFIG_MACH_MT6781) scp_resource_req(SCP_REQ_RELEASE); #else spm_resource_req(SPM_RESOURCE_USER_SCP, SPM_RESOURCE_RELEASE); #endif #endif } __pm_relax(scp_suspend_lock); pr_debug("[SCP] succeed to set freq, expect=%d, cur=%d\n", scp_expected_freq, scp_current_freq); return 0; } void wait_scp_dvfs_init_done(void) { int count = 0; while (g_scp_dvfs_init_flag != 1) { mdelay(1); count++; if (count > 3000) { pr_err("SCP dvfs driver init fail\n"); WARN_ON(1); } } } int scp_pll_ctrl_set(unsigned int pll_ctrl_flag, unsigned int pll_sel) { int idx; int mux_idx = 0; int ret = 0; pr_debug("%s(%d, %d)\n", __func__, pll_ctrl_flag, pll_sel); if (pll_sel != CLK_26M) { idx = scp_get_freq_idx(pll_sel); if (idx < 0) { pr_notice("invalid idx %d\n", idx); WARN_ON(1); return -EINVAL; } mux_idx = dvfs->opp[idx].clk_mux; if (mux_idx < 0) { pr_notice("invalid mux_idx %d\n", mux_idx); WARN_ON(1); return -EINVAL; } } if (pll_ctrl_flag == PLL_ENABLE) { if ( #ifndef CONFIG_MACH_MT6768 pre_pll_sel != MAINPLL_273M && #endif pre_pll_sel != UNIVPLL_416M) { ret = clk_prepare_enable(mt_scp_pll->clk_mux); if (ret) { pr_err("clk_prepare_enable() failed\n"); WARN_ON(1); } } else { pr_debug("no need to do clk_prepare_enable()\n"); } if (pll_sel == CLK_26M) /* default boot-up clk : 26 MHz */ ret = clk_set_parent(mt_scp_pll->clk_mux, mt_scp_pll->clk_pll[0]); else if (idx >= 0 && idx < dvfs->scp_opp_num && idx < mt_scp_pll->pll_num) ret = clk_set_parent(mt_scp_pll->clk_mux, mt_scp_pll->clk_pll[mux_idx]); else { pr_err("not support opp freq %d\n", pll_sel); WARN_ON(1); } if (ret) { pr_err("clk_set_parent() failed, opp=%d\n", pll_sel); WARN_ON(1); } if (pre_pll_sel != pll_sel) pre_pll_sel = pll_sel; } else if ((pll_ctrl_flag == PLL_DISABLE) &&( #ifndef CONFIG_MACH_MT6768 pll_sel != MAINPLL_273M && #endif pll_sel != UNIVPLL_416M)) { clk_disable_unprepare(mt_scp_pll->clk_mux); /*pr_debug("clk_disable_unprepare()\n");*/ } else { /*pr_debug("no need to do clk_disable_unprepare\n");*/ } return ret; } void scp_pll_ctrl_handler(int id, void *data, unsigned int len) { unsigned int *pll_ctrl_flag = (unsigned int *)data; unsigned int *pll_sel = (unsigned int *) (data + 1); scp_pll_ctrl_set (*pll_ctrl_flag, *pll_sel); } void mt_scp_dvfs_state_dump(void) { unsigned int scp_state; scp_state = readl(SCP_A_SLEEP_DEBUG_REG); printk_deferred("scp status: %s\n", ((scp_state & IN_DEBUG_IDLE) == IN_DEBUG_IDLE) ? "idle mode" : ((scp_state & ENTERING_SLEEP) == ENTERING_SLEEP) ? "enter sleep" : ((scp_state & IN_SLEEP) == IN_SLEEP) ? "sleep mode" : ((scp_state & ENTERING_ACTIVE) == ENTERING_ACTIVE) ? "enter active" : ((scp_state & IN_ACTIVE) == IN_ACTIVE) ? "active mode" : "none of state"); } int mt_scp_dvfs_syscore_suspend(void) { mt_scp_dvfs_state_dump(); return 0; } void mt_scp_dvfs_syscore_resume(void) { mt_scp_dvfs_state_dump(); } static struct syscore_ops mt_scp_dvfs_syscore_ops = { .suspend = mt_scp_dvfs_syscore_suspend, .resume = mt_scp_dvfs_syscore_resume, }; #ifdef CONFIG_PROC_FS /* * PROC */ /**************************** * show SCP state *****************************/ static int mt_scp_dvfs_state_proc_show(struct seq_file *m, void *v) { unsigned int scp_state; scp_state = readl(SCP_A_SLEEP_DEBUG_REG); seq_printf(m, "scp status: %s\n", ((scp_state & IN_DEBUG_IDLE) == IN_DEBUG_IDLE) ? "idle mode" : ((scp_state & ENTERING_SLEEP) == ENTERING_SLEEP) ? "enter sleep" : ((scp_state & IN_SLEEP) == IN_SLEEP) ? "sleep mode" : ((scp_state & ENTERING_ACTIVE) == ENTERING_ACTIVE) ? "enter active" : ((scp_state & IN_ACTIVE) == IN_ACTIVE) ? "active mode" : "none of state"); return 0; } /**************************** * show scp dvfs sleep *****************************/ static int mt_scp_dvfs_sleep_proc_show(struct seq_file *m, void *v) { if (scp_sleep_flag == -1) seq_puts(m, "SCP sleep is not configured by command yet.\n"); else if (scp_sleep_flag == 0) seq_puts(m, "SCP Sleep: OFF\n"); else if (scp_sleep_flag == 1) seq_puts(m, "SCP Sleep: ON\n"); else if (scp_sleep_flag == 2) seq_puts(m, "SCP Sleep: sleep without wakeup\n"); else if (scp_sleep_flag == 3) seq_puts(m, "SCP Sleep: force to sleep\n"); else seq_puts(m, "Warning: invalid SCP Sleep configure\n"); return 0; } /********************************** * write scp dvfs sleep ***********************************/ static ssize_t mt_scp_dvfs_sleep_proc_write( struct file *file, const char __user *buffer, size_t count, loff_t *data) { char desc[64]; unsigned int val = 0; unsigned int len = 0; int ret = 0; if (count <= 0) return 0; len = (count < (sizeof(desc) - 1)) ? count : (sizeof(desc) - 1); if (copy_from_user(desc, buffer, len)) return 0; desc[len] = '\0'; if (kstrtouint(desc, 10, &val) == 0) { if (val >= 0 && val <= 3) { if (val != scp_sleep_flag) { scp_sleep_flag = val; pr_info("scp_sleep_flag = %d\n", scp_sleep_flag); ret = scp_ipi_send(IPI_DVFS_SLEEP, (void *)&scp_sleep_flag, sizeof(scp_sleep_flag), 0, SCP_A_ID); if (ret != SCP_IPI_DONE) pr_info("%s: SCP send IPI fail - %d\n", __func__, ret); } else pr_info("SCP sleep flag is not changed\n"); } else { pr_info("Warning: invalid input value %d\n", val); } } else { pr_info("Warning: invalid input command, val=%d\n", val); } return count; } /**************************** * show scp dvfs ctrl *****************************/ static int mt_scp_dvfs_ctrl_proc_show(struct seq_file *m, void *v) { unsigned long spin_flags; int i; spin_lock_irqsave(&scp_awake_spinlock, spin_flags); scp_current_freq = readl(CURRENT_FREQ_REG); scp_expected_freq = readl(EXPECTED_FREQ_REG); spin_unlock_irqrestore(&scp_awake_spinlock, spin_flags); seq_printf(m, "SCP DVFS: %s\n", (scp_dvfs_flag == 1)?"ON":"OFF"); seq_printf(m, "SCP frequency: cur=%dMHz, expect=%dMHz\n", scp_current_freq, scp_expected_freq); for (i = 0; i < NUM_FEATURE_ID; i++) seq_printf(m, "feature=%d, freq=%d, enable=%d\n", feature_table[i].feature, feature_table[i].freq, feature_table[i].enable); for (i = 0; i < NUM_SENSOR_TYPE; i++) seq_printf(m, "sensor id=%d, freq=%d, enable=%d\n", sensor_type_table[i].feature, sensor_type_table[i].freq, sensor_type_table[i].enable); return 0; } /********************************** * write scp dvfs ctrl ***********************************/ static ssize_t mt_scp_dvfs_ctrl_proc_write( struct file *file, const char __user *buffer, size_t count, loff_t *data) { char desc[64], cmd[32]; unsigned int len = 0; int dvfs_opp; int freq; int n; if (count <= 0) return 0; len = (count < (sizeof(desc) - 1)) ? count : (sizeof(desc) - 1); if (copy_from_user(desc, buffer, len)) return 0; desc[len] = '\0'; n = sscanf(desc, "%31s %d", cmd, &dvfs_opp); if (n == 1 || n == 2) { if (!strcmp(cmd, "on")) { scp_dvfs_flag = 1; pr_info("SCP DVFS: ON\n"); } else if (!strcmp(cmd, "off")) { scp_dvfs_flag = -1; pr_info("SCP DVFS: OFF\n"); } else if (!strcmp(cmd, "opp")) { if (dvfs_opp == -1) { pr_info("remove the opp setting of command\n"); feature_table[VCORE_TEST_FEATURE_ID].freq = 0; scp_deregister_feature( VCORE_TEST_FEATURE_ID); } else if (dvfs_opp >= 0 && dvfs_opp < dvfs->scp_opp_num) { uint32_t i; uint32_t sum = 0, added_freq = 0; pr_info("manually set opp = %d\n", dvfs_opp); /* * calculate scp frequence */ for (i = 0; i < NUM_FEATURE_ID; i++) { if (i != VCORE_TEST_FEATURE_ID && feature_table[i].enable == 1) sum += feature_table[i].freq; } /* * calculate scp sensor frequence */ for (i = 0; i < NUM_SENSOR_TYPE; i++) { if (sensor_type_table[i].enable == 1) sum += sensor_type_table[i].freq; } for (i = 0; i < dvfs->scp_opp_num; i++) { freq = dvfs->opp[i].freq; if (dvfs_opp == i && sum < freq) { added_freq = freq - sum; break; } } for (i = 0; i < NUM_FEATURE_ID; i++) if (VCORE_TEST_FEATURE_ID == feature_table[i].feature) { feature_table[i].freq = added_freq; break; } pr_debug("request freq: %d + %d = %d (MHz)\n", sum, added_freq, sum + added_freq); scp_register_feature( VCORE_TEST_FEATURE_ID); } else { pr_info("invalid opp value %d\n", dvfs_opp); } } else { pr_info("invalid command %s\n", cmd); } } else { pr_info("invalid length %d\n", n); } return count; } #define PROC_FOPS_RW(name) \ static int mt_ ## name ## _proc_open(\ struct inode *inode, \ struct file *file) \ { \ return single_open(file, \ mt_ ## name ## _proc_show, \ PDE_DATA(inode)); \ } \ static const struct file_operations \ mt_ ## name ## _proc_fops = {\ .owner = THIS_MODULE, \ .open = mt_ ## name ## _proc_open, \ .read = seq_read, \ .llseek = seq_lseek, \ .release = single_release, \ .write = mt_ ## name ## _proc_write, \ } #define PROC_FOPS_RO(name) \ static int mt_ ## name ## _proc_open(\ struct inode *inode,\ struct file *file)\ {\ return single_open(file, \ mt_ ## name ## _proc_show, \ PDE_DATA(inode)); \ } \ static const struct file_operations mt_ ## name ## _proc_fops = {\ .owner = THIS_MODULE,\ .open = mt_ ## name ## _proc_open,\ .read = seq_read,\ .llseek = seq_lseek,\ .release = single_release,\ } #define PROC_ENTRY(name) {__stringify(name), &mt_ ## name ## _proc_fops} PROC_FOPS_RO(scp_dvfs_state); PROC_FOPS_RW(scp_dvfs_sleep); PROC_FOPS_RW(scp_dvfs_ctrl); static int mt_scp_dvfs_create_procfs(void) { struct proc_dir_entry *dir = NULL; int i, ret = 0; struct pentry { const char *name; const struct file_operations *fops; }; const struct pentry entries[] = { PROC_ENTRY(scp_dvfs_state), PROC_ENTRY(scp_dvfs_sleep), PROC_ENTRY(scp_dvfs_ctrl) }; dir = proc_mkdir("scp_dvfs", NULL); if (!dir) { pr_err("fail to create /proc/scp_dvfs @ %s()\n", __func__); return -ENOMEM; } for (i = 0; i < ARRAY_SIZE(entries); i++) { if (!proc_create(entries[i].name, 0664, dir, entries[i].fops)) { pr_err("ERROR: %s: create /proc/scp_dvfs/%s failed\n", __func__, entries[i].name); ret = -ENOMEM; } } return ret; } #endif /* CONFIG_PROC_FS */ static const struct of_device_id scpdvfs_of_ids[] = { {.compatible = "mediatek,scp_dvfs",}, {} }; static int mt_scp_dvfs_suspend(struct device *dev) { return 0; } static int mt_scp_dvfs_resume(struct device *dev) { return 0; } static int mt_scp_dvfs_pm_restore_early(struct device *dev) { return 0; } static int __init mt_scp_regmap_init(struct platform_device *pdev, struct device_node *node) { struct platform_device *pmic_pdev; struct device_node *pmic_node; struct mt6397_chip *chip; struct regmap *regmap; /* get GPIO regmap */ regmap = syscon_regmap_lookup_by_phandle(node, subsys_name[SYS_GPIO]); if (IS_ERR(regmap)) { dev_err(&pdev->dev, "Get gpio regmap fail: %ld\n", PTR_ERR(regmap)); goto fail_gpio; } sd[SYS_GPIO].regmap = regmap; /* get PMIC regmap */ pmic_node = of_parse_phandle(node, subsys_name[SYS_PMIC], 0); if (!pmic_node) { dev_notice(&pdev->dev, "fail to find pmic node\n"); goto fail_pmic; } pmic_pdev = of_find_device_by_node(pmic_node); if (!pmic_pdev) { dev_err(&pdev->dev, "fail to find pmic device or some project no pmic config \n"); goto fail_pmic; } chip = dev_get_drvdata(&(pmic_pdev->dev)); if (!chip) { dev_err(&pdev->dev, "fail to find pmic drv data\n"); goto fail_pmic; } regmap = chip->regmap; if (IS_ERR_VALUE(regmap)) { sd[SYS_PMIC].regmap = NULL; dev_err(&pdev->dev, "get pmic regmap fail\n"); goto fail_pmic; } sd[SYS_PMIC].regmap = regmap; return 0; fail_gpio: WARN_ON(1); return SCP_DVFS_DTSNO_GPIO_CONFIG; fail_pmic: return SCP_DVFS_DTSNO_PMIC_CONFIG; } static int __init mt_scp_sub_feature_init_internal(struct device_node *node, struct sub_feature_data *fd) { char *buf = kzalloc(sizeof(char) * 25, GFP_KERNEL); struct reg_info *reg; struct reg_cfg *cfg; unsigned int cfg_num; int ret = 0; int i; if (!buf) return -ENOMEM; ret = snprintf(buf, 25, "%s-reg", fd->name); if (ret < 0 || ret >= 25) goto fail_1; fd->num = of_property_count_u32_elems(node, buf) / 4; if (fd->num <= 0) goto pass; reg = kcalloc(fd->num, sizeof(struct reg_info), GFP_KERNEL); if (!reg) { ret = -ENOMEM; goto fail_1; } for (i = 0; i < fd->num; i++) { ret = of_property_read_u32_index(node, buf, i * 4, ®[i].ofs); if (ret) { pr_err("Cannot get property offset(%d)\n", ret); goto fail_2; } ret = of_property_read_u32_index(node, buf, (i * 4) + 1, ®[i].msk); if (ret) { pr_err("Cannot get property mask(%d)\n", ret); goto fail_2; } ret = of_property_read_u32_index(node, buf, (i * 4) + 2, ®[i].bit); if (ret) { pr_err("Cannot get property bit shift(%d)\n", ret); goto fail_2; } ret = of_property_read_u32_index(node, buf, (i * 4) + 3, ®[i].setclr); if (ret) { pr_err("Cannot get property set/clr type(%d)\n", ret); goto fail_2; } } fd->reg = reg; ret = snprintf(buf, 25, "%s-cfg", fd->name); if (ret < 0 || ret >= 25) goto fail_2; cfg_num = of_property_count_u32_elems(node, buf) / 2; if (cfg_num != fd->num) { pr_debug("cfg number is not matched(%d)\n", cfg_num); goto pass; } cfg = kcalloc(cfg_num, sizeof(struct reg_cfg), GFP_KERNEL); if (!cfg) { ret = -ENOMEM; goto fail_2; } for (i = 0; i < fd->num; i++) { ret = of_property_read_u32_index(node, buf, i * 2, &cfg[i].on); if (ret) { pr_err("Cannot get property cfg on(%d)\n", ret); goto fail_3; } ret = of_property_read_u32_index(node, buf, (i * 2) + 1, &cfg[i].off); if (ret) { pr_err("Cannot get property cfg off(%d)\n", ret); goto fail_3; } } fd->cfg = cfg; pass: kfree(buf); return 0; fail_3: kfree(cfg); fail_2: kfree(reg); fail_1: kfree(buf); WARN_ON(1); return ret; } static int __init mt_scp_sub_feature_init(struct device_node *node, struct subsys_data *sys, const char *str) { struct sub_feature_data *fd; char *buf; int ret; int i, j; /* init feature data struct */ sys->num = of_property_count_strings(node, str); if (sys->num <= 0) goto pass; /* init feature data structure */ fd = kcalloc(sys->num, sizeof(*fd), GFP_KERNEL); buf = kzalloc(sizeof(char) * 25, GFP_KERNEL); if (!fd || !buf) { ret = -ENOMEM; goto fail; } for (i = 0; i < sys->num; i++) { const char *name; ret = of_property_read_string_index(node, str, i, &name); if (ret) { pr_err("Cannot get property string(%d)\n", ret); ret = -EINVAL; goto fail_2; } fd[i].name = name; ret = mt_scp_sub_feature_init_internal(node, &fd[i]); if (ret) { kfree(name); goto fail_2; } /* init feature cfg */ ret = snprintf(buf, 25, "%s-cfg", str); if (ret < 0 || ret >= 25) goto fail_2; ret = of_property_read_u32_index(node, buf, i, &fd[i].onoff); if (ret) { kfree(name); goto fail_2; } } sys->fd = fd; kfree(buf); pass: return 0; fail_2: for (j = i - 1; j >= 0; j--) kfree(fd[j].name); fail: kfree(buf); kfree(fd); WARN_ON(1); return ret; } #if defined(CONFIG_MACH_MT6781) static void mt_pmic_sshub_init_for_mt6781(void) { pmic_scp_set_vcore(600000); pmic_scp_set_vcore_sleep(600000); pmic_set_register_value( PMIC_RG_BUCK_VCORE_SSHUB_EN, 1); pmic_set_register_value( PMIC_RG_BUCK_VCORE_SSHUB_SLEEP_VOSEL_EN, 0); if (Scp_Vsram_Ldo_usage == USE_VSRAM_OTHERS) pr_notice("SCP VSRAM: VSRAM_OTHERS\n"); else if (Scp_Vsram_Ldo_usage == USE_VSRAM_CORE) pr_notice("SCP VSRAM: VSRAM_CORE\n"); else { Scp_Vsram_Ldo_usage = USE_VSRAM_OTHERS; pr_notice("ERROR: unknown VSRAM LDO usage before PMIC setting\n"); WARN_ON(1); } pmic_scp_set_vsram_vcore(850000); pmic_scp_set_vsram_vcore_sleep(850000); if (Scp_Vsram_Ldo_usage == USE_VSRAM_CORE) { pmic_set_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_EN, 0); } else { pmic_set_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_EN, 1); } pmic_set_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_SLEEP_VOSEL_EN, 0); /* Workaround once force BUCK in NML mode */ pmic_set_register_value(PMIC_RG_SRCVOLTEN_LP_EN, 1); return ; } #endif #if defined(CONFIG_MACH_MT6768) static void mt_pmic_sshub_init_for_mt6768(void) { #if !defined(CONFIG_FPGA_EARLY_PORTING) unsigned int val[8]; val[0] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_EN); val[1] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_VOSEL); val[2] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_SLEEP_VOSEL_EN); val[3] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_VOSEL_SLEEP); val[4] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_EN); val[5] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_VOSEL); val[6] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_SLEEP_VOSEL_EN); val[7] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_VOSEL_SLEEP); pr_debug( "Before: vcore=(0x%x,0x%x,0x%x,0x%x), vsram=(0x%x,0x%x,0x%x,0x%x)\n", val[0], val[1], val[2], val[3], val[4], val[5], val[6], val[7]); pmic_scp_set_vcore(650000); pmic_scp_set_vcore_sleep(650000); pmic_set_register_value( PMIC_RG_BUCK_VCORE_SSHUB_EN, 1); pmic_set_register_value( PMIC_RG_BUCK_VCORE_SSHUB_SLEEP_VOSEL_EN, 0); pmic_scp_set_vsram_vcore(900000); pmic_scp_set_vsram_vcore_sleep(900000); pmic_set_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_EN, 1); pmic_set_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_SLEEP_VOSEL_EN, 0); val[0] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_EN); val[1] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_VOSEL); val[2] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_SLEEP_VOSEL_EN); val[3] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_VOSEL_SLEEP); val[4] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_EN); val[5] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_VOSEL); val[6] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_SLEEP_VOSEL_EN); val[7] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_VOSEL_SLEEP); pr_debug( "After: vcore=(0x%x,0x%x,0x%x,0x%x), vsram=(0x%x,0x%x,0x%x,0x%x)\n", val[0], val[1], val[2], val[3], val[4], val[5], val[6], val[7]); /* Workaround once force BUCK in NML mode */ pmic_set_register_value(PMIC_RG_SRCVOLTEN_LP_EN, 1); #endif } #endif #if defined(CONFIG_MACH_MT6771) void mt_pmic_sshub_init_for_mt6771(void) { #if !defined(CONFIG_FPGA_EARLY_PORTING) unsigned int val[8]; val[0] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_EN); val[1] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_VOSEL); val[2] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_SLEEP_VOSEL_EN); val[3] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_VOSEL_SLEEP); val[4] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_EN); val[5] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_VOSEL); val[6] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_SLEEP_VOSEL_EN); val[7] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_VOSEL_SLEEP); pr_debug( "Before: vcore=(0x%x,0x%x,0x%x,0x%x), vsram=(0x%x,0x%x,0x%x,0x%x)\n", val[0], val[1], val[2], val[3], val[4], val[5], val[6], val[7]); pmic_scp_set_vcore(600000); pmic_scp_set_vcore_sleep(600000); pmic_set_register_value( PMIC_RG_BUCK_VCORE_SSHUB_EN, 1); pmic_set_register_value( PMIC_RG_BUCK_VCORE_SSHUB_SLEEP_VOSEL_EN, 0); pmic_scp_set_vsram_vcore(850000); pmic_scp_set_vsram_vcore_sleep(850000); pmic_set_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_EN, 1); pmic_set_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_SLEEP_VOSEL_EN, 0); val[0] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_EN); val[1] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_VOSEL); val[2] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_SLEEP_VOSEL_EN); val[3] = pmic_get_register_value( PMIC_RG_BUCK_VCORE_SSHUB_VOSEL_SLEEP); val[4] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_EN); val[5] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_VOSEL); val[6] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_SLEEP_VOSEL_EN); val[7] = pmic_get_register_value( PMIC_RG_LDO_VSRAM_OTHERS_SSHUB_VOSEL_SLEEP); pr_debug( "After: vcore=(0x%x,0x%x,0x%x,0x%x), vsram=(0x%x,0x%x,0x%x,0x%x)\n", val[0], val[1], val[2], val[3], val[4], val[5], val[6], val[7]); /* Workaround once force BUCK in NML mode */ pmic_set_register_value(PMIC_RG_SRCVOLTEN_LP_EN, 1); #endif } #endif static void __init mt_pmic_sshub_init(void) { #if !defined(CONFIG_FPGA_EARLY_PORTING) #if defined(CONFIG_MACH_MT6768) mt_pmic_sshub_init_for_mt6768(); #elif defined(CONFIG_MACH_MT6781) mt_pmic_sshub_init_for_mt6781(); #elif defined(CONFIG_MACH_MT6771) mt_pmic_sshub_init_for_mt6771(); #else int max_vcore = dvfs->opp[dvfs->scp_opp_num - 1].vcore + 100000; int max_vsram = dvfs->opp[dvfs->scp_opp_num - 1].vsram + 100000; /* if vcore/vsram define as 0xff, means no pmic op during dvfs */ if (max_vcore == 0xff && max_vsram == 0xff) return; /* set SCP VCORE voltage */ if (regulator_set_voltage(reg_vcore, dvfs->opp[0].vcore, max_vcore) != 0) pr_notice("Set wrong vcore voltage\n"); /* set SCP VSRAM voltage */ if (regulator_set_voltage(reg_vsram, dvfs->opp[0].vsram, max_vsram) != 0) pr_notice("Set wrong vsram voltage\n"); if (scp_get_sub_feature_onoff(SYS_PMIC, PMIC_VOW_LP)) /* enable VOW low power mode */ scp_set_sub_register_cfg(SYS_PMIC, PMIC_VOW_LP, true); else /* disable VOW low power mode */ scp_set_sub_register_cfg(SYS_PMIC, PMIC_VOW_LP, false); /* pmrc_mode: OFF */ if (scp_get_sub_feature_onoff(SYS_PMIC, PMIC_PMRC)) scp_set_sub_register_cfg(SYS_PMIC, PMIC_PMRC, true); else scp_set_sub_register_cfg(SYS_PMIC, PMIC_PMRC, false); /* BUCK_VCORE_SSHUB_EN: ON */ /* LDO_VSRAM_OTHERS_SSHUB_EN: ON */ if (regulator_enable(reg_vcore) != 0) pr_notice("Enable vcore failed!!!\n"); if (regulator_enable(reg_vsram) != 0) pr_notice("Enable vsram failed!!!\n"); #endif #endif } static int __init mt_scp_dvfs_pdrv_probe(struct platform_device *pdev) { struct device_node *node; struct dvfs_opp *opp; unsigned int *gpio_mode; char *buf; int i; int ret = 0; #if defined(CONFIG_MACH_MT6781) int gpio_idx, gpio_val; #endif /* find device tree node of scp_dvfs */ node = of_find_matching_node(NULL, scpdvfs_of_ids); if (!node) { dev_notice(&pdev->dev, "fail to find SCPDVFS node\n"); return -ENODEV; } /* init subsys data struct */ sd = kcalloc(SYS_NUM, sizeof(*sd), GFP_KERNEL); if (!sd) return -ENOMEM; memset(sd, 0 ,sizeof(*sd) * SYS_NUM); /* init temp buf */ buf = kzalloc(sizeof(char) * 15, GFP_KERNEL); if (!buf) { kfree(sd); return -ENOMEM; } /* init dvfs data structure */ dvfs = kzalloc(sizeof(*dvfs), GFP_KERNEL); if (!dvfs) { kfree(buf); kfree(sd); return -ENOMEM; } /* get scp dvfs opp count */ ret = of_property_count_u32_elems(node, "dvfs-opp") / 7; if (ret <= 0) { kfree(dvfs); kfree(buf); kfree(sd); return ret; } dvfs->scp_opp_num = ret; /* init opp data structure */ opp = kcalloc(dvfs->scp_opp_num, sizeof(*opp), GFP_KERNEL); if (!dvfs) { kfree(dvfs); kfree(buf); kfree(sd); return -ENOMEM; } /* init regmap */ ret = mt_scp_regmap_init(pdev, node); if (ret == SCP_DVFS_DTSNO_GPIO_CONFIG) goto fail; /* init gpio/pmic feature data */ for (i = 0; i < SYS_NUM; i++) { ret = snprintf(buf, 15, "%s-feature", subsys_name[i]); if (ret < 0 || ret >= 15) goto fail; ret = mt_scp_sub_feature_init(node, &sd[i], buf); if (ret) goto fail; } /* get scp_sel for clk-mux setting */ mt_scp_pll = kzalloc(sizeof(struct mt_scp_pll_t), GFP_KERNEL); if (mt_scp_pll == NULL) { ret = -ENOMEM; goto fail; } mt_scp_pll->clk_mux = devm_clk_get(&pdev->dev, "clk_mux"); if (IS_ERR(mt_scp_pll->clk_mux)) { dev_notice(&pdev->dev, "cannot get clock mux\n"); WARN_ON(1); ret = PTR_ERR(mt_scp_pll->clk_mux); goto fail; } /* scp_sel has most 8 member of clk source */ for (i = 0; i < 8; i++) { ret = snprintf(buf, 15, "clk_pll_%d", i); if (ret < 0 || ret >= 15) goto fail; mt_scp_pll->clk_pll[i] = devm_clk_get(&pdev->dev, buf); if (IS_ERR(mt_scp_pll->clk_pll[i])) { dev_notice(&pdev->dev, "cannot get %dst clock parent\n", i); break; } } mt_scp_pll->pll_num = i; /* check if GPIO is configured correctly for SCP VREQ */ if (scp_get_sub_feature_onoff(SYS_GPIO, GPIO_MODE)) { gpio_mode = scp_get_sub_register_cfg(SYS_GPIO, GPIO_MODE); if (*gpio_mode == 1) pr_notice("v_req muxpin setting is correct\n"); else { pr_notice("wrong V_REQ muxpin setting - %d\n", *gpio_mode); WARN_ON(1); } kfree(gpio_mode); } #if defined(CONFIG_MACH_MT6781) /* get GPIO value by GPIO API */ /* get high/low level of gpio pin */ gpio_idx = of_get_named_gpio(pdev->dev.of_node, "vsram_chk_gpio", 0); gpio_val = gpio_get_value(gpio_idx); pr_notice("vsram_chk_gpio value: %d\n", gpio_val); if (gpio_val == 0) { Scp_Vsram_Ldo_usage = USE_VSRAM_CORE; pr_notice("VSRAM LDO: VSRAM_CORE\n"); } else { Scp_Vsram_Ldo_usage = USE_VSRAM_OTHERS; pr_notice("VSRAM LDO: VSRAM_OTHERS\n"); } #endif /* get each dvfs opp data from dts node */ for (i = 0; i < dvfs->scp_opp_num; i++) { ret = of_property_read_u32_index(node, "dvfs-opp", i * 7, &opp[i].vcore); if (ret) { pr_err("Cannot get property vcore(%d)\n", ret); goto fail; } ret = of_property_read_u32_index(node, "dvfs-opp", (i * 7) + 1, &opp[i].vsram); if (ret) { pr_err("Cannot get property vsram(%d)\n", ret); goto fail; } ret = of_property_read_u32_index(node, "dvfs-opp", (i * 7) + 2, &opp[i].uv_idx); if (ret) { pr_err("Cannot get property uv idx(%d)\n", ret); goto fail; } ret = of_property_read_u32_index(node, "dvfs-opp", (i * 7) + 3, &opp[i].dvfsrc_opp); if (ret) { pr_err("Cannot get property dvfsrc opp(%d)\n", ret); goto fail; } ret = of_property_read_u32_index(node, "dvfs-opp", (i * 7) + 4, &opp[i].spm_opp); if (ret) { pr_err("Cannot get property spm opp(%d)\n", ret); goto fail; } ret = of_property_read_u32_index(node, "dvfs-opp", (i * 7) + 5, &opp[i].freq); if (ret) { pr_err("Cannot get property freq(%d)\n", ret); goto fail; } ret = of_property_read_u32_index(node, "dvfs-opp", (i * 7) + 6, &opp[i].clk_mux); if (ret) { pr_err("Cannot get property clk mux(%d)\n", ret); goto fail; } } dvfs->opp = opp; /* get dvfsrc table opp count */ ret = of_property_read_u32(node, "dvfsrc-opp-num", &dvfs->dvfsrc_opp_num); if (ret) { pr_err("Cannot get property dvfsrc opp num(%d)\n", ret); goto fail; } if (dvfs->dvfsrc_opp_num == 0) { pr_notice("dvfsrc table has zero opp count\n"); } #if (defined (CONFIG_MACH_MT6768) \ ||defined(CONFIG_MACH_MT6781) || defined(CONFIG_MACH_MT6771)) pr_notice("mt6768 6781 6771 no pmic config in dts\n"); mt_pmic_sshub_init(); goto pass; #endif /* get Vcore/Vsram Regulator */ reg_vcore = devm_regulator_get_optional(&pdev->dev, "sshub-vcore"); if (IS_ERR(reg_vcore) || !reg_vcore) { pr_notice("regulator vcore sshub supply is not available\n"); ret = PTR_ERR(reg_vcore); goto pass; } reg_vsram = devm_regulator_get_optional(&pdev->dev, "sshub-vsram"); if (IS_ERR(reg_vsram) || !reg_vsram) { pr_notice("regulator vsram sshub supply is not available\n"); ret = PTR_ERR(reg_vsram); goto pass; } mt_pmic_sshub_init(); pass: kfree(buf); g_scp_dvfs_init_flag = 1; return 0; fail: kfree(mt_scp_pll); kfree(opp); kfree(dvfs); kfree(buf); kfree(sd); WARN_ON(1); return -1; } /*************************************** * this function should never be called ****************************************/ static int mt_scp_dvfs_pdrv_remove(struct platform_device *pdev) { return 0; } static const struct dev_pm_ops mt_scp_dvfs_pm_ops = { .suspend = mt_scp_dvfs_suspend, .resume = mt_scp_dvfs_resume, .restore_early = mt_scp_dvfs_pm_restore_early, }; struct platform_device mt_scp_dvfs_pdev = { .name = "mt-scpdvfs", .id = -1, }; static struct platform_driver mt_scp_dvfs_pdrv __refdata = { .probe = mt_scp_dvfs_pdrv_probe, .remove = mt_scp_dvfs_pdrv_remove, .driver = { .name = "scpdvfs", .pm = &mt_scp_dvfs_pm_ops, .owner = THIS_MODULE, .of_match_table = scpdvfs_of_ids, }, }; /********************************** * mediatek scp dvfs initialization ***********************************/ void __init mt_scp_dvfs_ipi_init(void) { scp_ipi_registration(IPI_SCP_PLL_CTRL, scp_pll_ctrl_handler, "IPI_SCP_PLL_CTRL"); } int __init scp_dvfs_init(void) { int ret = 0; pr_debug("%s\n", __func__); #ifdef CONFIG_PROC_FS /* init proc */ if (mt_scp_dvfs_create_procfs()) { pr_err("mt_scp_dvfs_create_procfs fail..\n"); goto fail; } #endif /* CONFIG_PROC_FS */ /* register platform device/driver */ ret = platform_device_register(&mt_scp_dvfs_pdev); if (ret) { pr_err("fail to register scp dvfs device @ %s()\n", __func__); goto fail; } ret = platform_driver_register(&mt_scp_dvfs_pdrv); if (ret) { pr_err("fail to register scp dvfs driver @ %s()\n", __func__); platform_device_unregister(&mt_scp_dvfs_pdev); goto fail; } scp_suspend_lock = wakeup_source_register(NULL, "scp wakelock"); mt_scp_dvfs_ipi_init(); mtk_pm_qos_add_request(&dvfsrc_scp_vcore_req, MTK_PM_QOS_SCP_VCORE_REQUEST, MTK_PM_QOS_SCP_VCORE_REQUEST_DEFAULT_VALUE); register_syscore_ops(&mt_scp_dvfs_syscore_ops); return 0; fail: WARN_ON(1); return -1; } void __exit scp_dvfs_exit(void) { unregister_syscore_ops(&mt_scp_dvfs_syscore_ops); platform_driver_unregister(&mt_scp_dvfs_pdrv); platform_device_unregister(&mt_scp_dvfs_pdev); }