[resend with thinned out response, since original reply was determined to be spam by vger.kernel.org. wondering that tells me about my review style... ] Thara Gopinath <thara@xxxxxx> writes: > This patch adds voltage driver support for OMAP3. The driver > allows configuring the voltage controller and voltage > processors during init and exports APIs to enable/disable > voltage processors, scale voltage and reset voltage. > The driver also maintains the global voltage table on a per > VDD basis which contains the various voltages supported by the > VDD along with per voltage dependent data like smartreflex > n-target value, errminlimit and voltage processor errorgain. > The driver allows scaling of VDD voltages either through > "vc bypass method" or through "vp forceupdate method" the > choice being configurable through the board file. > > This patch contains code originally in linux omap pm branch > smartreflex driver. Major contributors to this driver are > Lesly A M, Rajendra Nayak, Kalle Jokiniemi, Paul Walmsley, > Nishant Menon, Kevin Hilman. > > Signed-off-by: Thara Gopinath <thara@xxxxxx> All comments below are cut-and-paste from v3 review, and were not addressed in this update. [...] > --- /dev/null > +++ b/arch/arm/mach-omap2/voltage.c > @@ -0,0 +1,1158 @@ > +/* > + * OMAP3/OMAP4 Voltage Management Routines > + * > + * Author: Thara Gopinath <thara@xxxxxx> > + * > + * Copyright (C) 2007 Texas Instruments, Inc. > + * Rajendra Nayak <rnayak@xxxxxx> > + * Lesly A M <x0080970@xxxxxx> > + * > + * Copyright (C) 2008 Nokia Corporation > + * Kalle Jokiniemi > + * > + * Copyright (C) 2010 Texas Instruments, Inc. > + * Thara Gopinath <thara@xxxxxx> > + * > + * This program is free software; you can redistribute it and/or modify > + * it under the terms of the GNU General Public License version 2 as > + * published by the Free Software Foundation. > + */ > + > +#include <linux/delay.h> > +#include <linux/io.h> > +#include <linux/clk.h> > +#include <linux/err.h> > +#include <linux/debugfs.h> > +#include <linux/slab.h> > + > +#include <plat/common.h> > +#include <plat/voltage.h> > + > +#include "prm-regbits-34xx.h" > + > +#define VP_IDLE_TIMEOUT 200 > +#define VP_TRANXDONE_TIMEOUT 300 > +#define VOLTAGE_DIR_SIZE 16 > + > +static struct dentry *voltage_dir; > +/* PRM voltage module */ > +static u32 volt_mod; There's no need for this to be a global, this should be a member of vdd_info (or the voltage domain.) That means the read/write functions will have to take an additional argument (vdd or voltdm) but that's cleaner to me. [...] > +/* OMAP3 specific voltage init functions */ > +/* > + * Intializes the voltage controller registers with the PMIC and board > + * specific parameters and voltage setup times for OMAP3. If the board > + * file does not populate the voltage controller parameters through > + * omap3_pm_init_vc, default values specified in vc_config is used. > + */ > +static void __init omap3_init_voltagecontroller(void) I'd like to see consistent naming. Elsewhere in the code, things are named with either a vdd, vc or vp prefix. Let's keep that here and call this omap3_vc_init(), and for VP below, s/init_voltagecontroller/vp_init/ [...] > + /* > + * Sys clk rate is require to calculate vp timeout value and > + * smpswaittimemin and smpswaittimemax. > + */ > + sys_ck = clk_get(NULL, "sys_ck"); > + if (IS_ERR(sys_ck)) { > + pr_warning("%s: Could not get the sys clk to calculate" > + "various vdd_%s params\n", __func__, vdd->voltdm.name); > + return; > + } > + sys_clk_speed = clk_get_rate(sys_ck); > + clk_put(sys_ck); > + /* Divide to avoid overflow */ > + sys_clk_speed /= 1000; > + > + /* Nominal/Reset voltage of the VDD */ > + vdd->nominal_volt = vdd->curr_volt = 1200000; This should be a #define someplace, something like OMAP3_VDDx_RESET_VOL or similar. Ideally with a TRM/doc reference to where it comes from. I'm a little confused. Is this a hardware reset value? or a software reset value which is part of this layer. It's a little confusing, since if this is a hardware reset voltage why does init_voltageprocessor() have to write it to the VP during init? Even better, rather than hard code this, it would be even better if it just picked one of the nominal voltages from the volt_data table. [...] > +/* vc_bypass_scale_voltage - VC bypass method of voltage scaling */ > +static int vc_bypass_scale_voltage(struct omap_vdd_info *vdd, > + unsigned long target_volt) > +{ > + struct omap_volt_data *volt_data; > + u32 vc_bypass_value, vc_cmdval, vc_valid, vc_bypass_val_reg_offs; > + u32 vp_errgain_val, vc_cmd_on_mask; > + u32 loop_cnt = 0, retries_cnt = 0; > + u32 smps_steps = 0, smps_delay = 0; > + u8 vc_data_shift, vc_slaveaddr_shift, vc_regaddr_shift; > + u8 vc_cmd_on_shift; > + u8 target_vsel, current_vsel, sr_i2c_slave_addr; > + > + if (cpu_is_omap34xx()) { > + vc_cmd_on_shift = OMAP3430_VC_CMD_ON_SHIFT; > + vc_cmd_on_mask = OMAP3430_VC_CMD_ON_MASK; > + vc_data_shift = OMAP3430_DATA_SHIFT; > + vc_slaveaddr_shift = OMAP3430_SLAVEADDR_SHIFT; > + vc_regaddr_shift = OMAP3430_REGADDR_SHIFT; > + vc_valid = OMAP3430_VALID_MASK; > + vc_bypass_val_reg_offs = OMAP3_PRM_VC_BYPASS_VAL_OFFSET; > + sr_i2c_slave_addr = OMAP3_SRI2C_SLAVE_ADDR; > + } else { > + pr_warning("%s: Voltage scaling not yet enabled for" > + "this chip\n", __func__); > + return -EINVAL; > + } cpu_is_* checks are only acceptable at init time. This one happens during every scale. These VC settings should be set at init time only. Same issue with forceupdate_scale. [...] > +/* VP force update method of voltage scaling */ > +static int vp_forceupdate_scale_voltage(struct omap_vdd_info *vdd, > + unsigned long target_volt) > +{ > + struct omap_volt_data *volt_data; > + u32 vc_cmd_on_mask, vc_cmdval, vpconfig; > + u32 smps_steps = 0, smps_delay = 0; > + int timeout = 0; > + u8 target_vsel, current_vsel; > + u8 vc_cmd_on_shift; > + u8 prm_irqst_reg_offs, ocp_mod; > + > + if (cpu_is_omap34xx()) { > + vc_cmd_on_shift = OMAP3430_VC_CMD_ON_SHIFT; > + vc_cmd_on_mask = OMAP3430_VC_CMD_ON_MASK; > + prm_irqst_reg_offs = OMAP3_PRM_IRQSTATUS_MPU_OFFSET; > + ocp_mod = OCP_MOD; > + } else { > + pr_warning("%s: Voltage scaling not yet enabled for" > + "this chip\n", __func__); > + return -EINVAL; > + } > + > + /* Get volt_data corresponding to the target_volt */ > + volt_data = omap_voltage_get_voltdata(&vdd->voltdm, target_volt); > + if (IS_ERR(volt_data)) { > + /* > + * If a match is not found but the target voltage is > + * is the nominal vdd voltage allow scaling > + */ > + if (target_volt != vdd->nominal_volt) { > + pr_warning("%s: Unable to get voltage table for vdd_%s" > + "during voltage scaling. Some really Wrong!", > + __func__, vdd->voltdm.name); > + return -ENODATA; > + } > + volt_data = NULL; > + } > + > + if (!volt_pmic_info.uv_to_vsel) { > + pr_warning("%s: PMIC function to convert voltage in uV to" > + "vsel not registered. Hence unable to scale voltage" > + "for vdd_%s\n", __func__, vdd->voltdm.name); > + return -ENODATA; > + } > + > + target_vsel = volt_pmic_info.uv_to_vsel(target_volt); > + current_vsel = voltage_read_reg(vdd->vp_offs.voltage); > + smps_steps = abs(target_vsel - current_vsel); > + > + /* Setting the ON voltage to the new target voltage */ > + vc_cmdval = voltage_read_reg(vdd->cmdval_reg); > + vc_cmdval &= ~vc_cmd_on_mask; > + vc_cmdval |= (target_vsel << vc_cmd_on_shift); > + voltage_write_reg(vdd->cmdval_reg, vc_cmdval); > + Up 'til here, this looks almost exactly like the vc_bypass version. Maybe they can be combined into a common pre-scale function? > + /* Getting vp errorgain based on the voltage */ > + if (volt_data) > + vdd->vp_reg.vpconfig_errorgain = > + volt_data->vp_errgain; This also looks similar to the vc_bypass version. After the full-series is applied, the comments are identical, but the code is different. I suspect they should be the same though. Possibly a good reason to have a common pre-scale function. > + > + /* > + * Clear all pending TransactionDone interrupt/status. Typical latency > + * is <3us > + */ > + while (timeout++ < VP_TRANXDONE_TIMEOUT) { > + prm_write_mod_reg(vdd->vp_reg.tranxdone_status, > + ocp_mod, prm_irqst_reg_offs); > + if (!(prm_read_mod_reg(ocp_mod, prm_irqst_reg_offs) & > + vdd->vp_reg.tranxdone_status)) > + break; > + udelay(1); > + } > + if (timeout >= VP_TRANXDONE_TIMEOUT) { > + pr_warning("%s: vdd_%s TRANXDONE timeout exceeded." > + "Voltage change aborted", __func__, vdd->voltdm.name); > + return -ETIMEDOUT; > + } > + > + /* Configure for VP-Force Update */ > + vpconfig = voltage_read_reg(vdd->vp_offs.vpconfig); > + vpconfig &= ~(vdd->vp_reg.vpconfig_initvdd | > + vdd->vp_reg.vpconfig_forceupdate | > + vdd->vp_reg.vpconfig_initvoltage_mask | > + vdd->vp_reg.vpconfig_errorgain_mask); > + vpconfig |= ((target_vsel << > + vdd->vp_reg.vpconfig_initvoltage_shift) | > + (vdd->vp_reg.vpconfig_errorgain << > + vdd->vp_reg.vpconfig_errorgain_shift)); > + voltage_write_reg(vdd->vp_offs.vpconfig, vpconfig); > + > + /* Trigger initVDD value copy to voltage processor */ > + vpconfig |= vdd->vp_reg.vpconfig_initvdd; > + voltage_write_reg(vdd->vp_offs.vpconfig, vpconfig); > + > + /* Force update of voltage */ > + vpconfig |= vdd->vp_reg.vpconfig_forceupdate; > + voltage_write_reg(vdd->vp_offs.vpconfig, vpconfig); > + > + /* > + * Wait for TransactionDone. Typical latency is <200us. > + * Depends on SMPSWAITTIMEMIN/MAX and voltage change > + */ > + timeout = 0; > + omap_test_timeout((prm_read_mod_reg(ocp_mod, prm_irqst_reg_offs) & > + vdd->vp_reg.tranxdone_status), > + VP_TRANXDONE_TIMEOUT, timeout); > + if (timeout >= VP_TRANXDONE_TIMEOUT) > + pr_err("%s: vdd_%s TRANXDONE timeout exceeded." > + "TRANXDONE never got set after the voltage update\n", > + __func__, vdd->voltdm.name); > + > + /* > + * Wait for voltage to settle with SW wait-loop. > + * SMPS slew rate / step size. 2us added as buffer. > + */ > + smps_delay = ((smps_steps * volt_pmic_info.step_size) / > + volt_pmic_info.slew_rate) + 2; > + udelay(smps_delay); > + > + /* > + * Disable TransactionDone interrupt , clear all status, clear > + * control registers > + */ > + timeout = 0; > + while (timeout++ < VP_TRANXDONE_TIMEOUT) { > + prm_write_mod_reg(vdd->vp_reg.tranxdone_status, > + ocp_mod, prm_irqst_reg_offs); > + if (!(prm_read_mod_reg(ocp_mod, prm_irqst_reg_offs) & > + vdd->vp_reg.tranxdone_status)) > + break; > + udelay(1); > + } > + if (timeout >= VP_TRANXDONE_TIMEOUT) > + pr_warning("%s: vdd_%s TRANXDONE timeout exceeded while trying" > + "to clear the TRANXDONE status\n", > + __func__, vdd->voltdm.name); > + > + vpconfig = voltage_read_reg(vdd->vp_offs.vpconfig); > + /* Clear initVDD copy trigger bit */ > + vpconfig &= ~vdd->vp_reg.vpconfig_initvdd;; > + voltage_write_reg(vdd->vp_offs.vpconfig, vpconfig); > + /* Clear force bit */ > + vpconfig &= ~vdd->vp_reg.vpconfig_forceupdate; > + voltage_write_reg(vdd->vp_offs.vpconfig, vpconfig); > + > + vdd->curr_volt = target_volt; The smps_delay part and updating ->curr_volt is common to the two scale methods too. Maybe can be combined into a post-scale common function. > + > + return 0; > +} [...] > +void omap_vp_enable(struct voltagedomain *voltdm) > +{ > + struct omap_vdd_info *vdd; > + u32 vpconfig; > + > + if (!voltdm || IS_ERR(voltdm)) { > + pr_warning("%s: VDD specified does not exist!\n", __func__); > + return; > + } > + > + vdd = container_of(voltdm, struct omap_vdd_info, voltdm); > + > + /* If VP is already enabled, do nothing. Return */ > + if (voltage_read_reg(vdd->vp_offs.vpconfig) & > + vdd->vp_reg.vpconfig_vpenable) > + return; Minor: is a register access here required? Why not just keep an 'vp_enabled' flag as part of the vdd struct? > + /* > + * This latching is required only if VC bypass method is used for > + * voltage scaling during dvfs. > + */ > + if (!voltscale_vpforceupdate) > + vp_latch_vsel(vdd); > + > + /* Enable VP */ > + vpconfig = voltage_read_reg(vdd->vp_offs.vpconfig); > + voltage_write_reg(vdd->vp_offs.vpconfig, > + vpconfig | vdd->vp_reg.vpconfig_vpenable); > +} > + > +/** > + * omap_vp_disable() - API to disable a particular VP > + * @voltdm: pointer to the VDD whose VP is to be disabled. > + * > + * This API disables a particular voltage processor. Needed by the smartreflex > + * class drivers. > + */ > +void omap_vp_disable(struct voltagedomain *voltdm) > +{ > + struct omap_vdd_info *vdd; > + u32 vpconfig; > + int timeout; > + > + if (!voltdm || IS_ERR(voltdm)) { > + pr_warning("%s: VDD specified does not exist!\n", __func__); > + return; > + } > + > + vdd = container_of(voltdm, struct omap_vdd_info, voltdm); > + > + /* If VP is already disabled, do nothing. Return */ > + if (!(voltage_read_reg(vdd->vp_offs.vpconfig) & > + vdd->vp_reg.vpconfig_vpenable)) { > + pr_warning("%s: Trying to disable VP for vdd_%s when" > + "it is already disabled\n", __func__, voltdm->name); > + return; > + } Same here... is a register read really needed if we can just check vdd->vp_enabled? > + > + /* Disable VP */ > + vpconfig = voltage_read_reg(vdd->vp_offs.vpconfig); > + vpconfig &= ~vdd->vp_reg.vpconfig_vpenable; > + voltage_write_reg(vdd->vp_offs.vpconfig, vpconfig); > + > + /* > + * Wait for VP idle Typical latency is <2us. Maximum latency is ~100us > + */ > + omap_test_timeout((voltage_read_reg(vdd->vp_offs.vstatus)), > + VP_IDLE_TIMEOUT, timeout); > + > + if (timeout >= VP_IDLE_TIMEOUT) > + pr_warning("%s: vdd_%s idle timedout\n", > + __func__, voltdm->name); > + return; > +} > + > +/** > + * omap_voltage_scale_vdd() - API to scale voltage of a particular > + * voltage domain. > + * @voltdm: pointer to the VDD which is to be scaled. > + * @target_volt: The target voltage of the voltage domain > + * > + * This API should be called by the kernel to do the voltage scaling > + * for a particular voltage domain during dvfs or any other situation. > + */ > +int omap_voltage_scale_vdd(struct voltagedomain *voltdm, > + unsigned long target_volt) > +{ > + struct omap_vdd_info *vdd; > + > + if (!voltdm || IS_ERR(voltdm)) { > + pr_warning("%s: VDD specified does not exist!\n", __func__); > + return -EINVAL; > + } > + > + vdd = container_of(voltdm, struct omap_vdd_info, voltdm); > + > + if (voltscale_vpforceupdate) > + return vp_forceupdate_scale_voltage(vdd, target_volt); > + else > + return vc_bypass_scale_voltage(vdd, target_volt); Rather than the 'if' here, let's have a 'scale' function pointer associated with the VDD which gets updated if the method changes. Not sure if would ever be useful, but that would also allow the scale method to be different across VDDs instead of having it global. [...] > +void omap_change_voltscale_method(int voltscale_method) > +{ > + switch (voltscale_method) { > + case VOLTSCALE_VPFORCEUPDATE: > + voltscale_vpforceupdate = true; > + return; > + case VOLTSCALE_VCBYPASS: > + voltscale_vpforceupdate = false; > + return; > + default: > + pr_warning("%s: Trying to change the method of voltage scaling" > + "to an unsupported one!\n", __func__); > + } > +} And this API should be changed to take a 'vdd' and just change the 'scale' function pointer. [...] Kevin -- To unsubscribe from this list: send the line "unsubscribe linux-omap" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html