2019-10-16 14:37:06 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
|
|
|
/*
|
|
|
|
* pmic-cpcap.c - CPCAP-specific functions for the OPP code
|
|
|
|
*
|
|
|
|
* Adapted from Motorola Mapphone Android Linux kernel
|
|
|
|
* Copyright (C) 2011 Motorola, Inc.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/err.h>
|
|
|
|
#include <linux/io.h>
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
|
|
|
|
#include "soc.h"
|
|
|
|
#include "pm.h"
|
|
|
|
#include "voltage.h"
|
|
|
|
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include "pm.h"
|
|
|
|
#include "vc.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* omap_cpcap_vsel_to_vdc - convert CPCAP VSEL value to microvolts DC
|
|
|
|
* @vsel: CPCAP VSEL value to convert
|
|
|
|
*
|
|
|
|
* Returns the microvolts DC that the CPCAP PMIC should generate when
|
|
|
|
* programmed with @vsel.
|
|
|
|
*/
|
2019-10-18 12:07:01 +00:00
|
|
|
static unsigned long omap_cpcap_vsel_to_uv(unsigned char vsel)
|
2019-10-16 14:37:06 +00:00
|
|
|
{
|
|
|
|
if (vsel > 0x44)
|
|
|
|
vsel = 0x44;
|
|
|
|
return (((vsel * 125) + 6000)) * 100;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* omap_cpcap_uv_to_vsel - convert microvolts DC to CPCAP VSEL value
|
|
|
|
* @uv: microvolts DC to convert
|
|
|
|
*
|
|
|
|
* Returns the VSEL value necessary for the CPCAP PMIC to
|
|
|
|
* generate an output voltage equal to or greater than @uv microvolts DC.
|
|
|
|
*/
|
2019-10-18 12:07:01 +00:00
|
|
|
static unsigned char omap_cpcap_uv_to_vsel(unsigned long uv)
|
2019-10-16 14:37:06 +00:00
|
|
|
{
|
|
|
|
if (uv < 600000)
|
|
|
|
uv = 600000;
|
|
|
|
else if (uv > 1450000)
|
|
|
|
uv = 1450000;
|
|
|
|
return DIV_ROUND_UP(uv - 600000, 12500);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct omap_voltdm_pmic omap_cpcap_core = {
|
|
|
|
.slew_rate = 4000,
|
|
|
|
.step_size = 12500,
|
|
|
|
.vp_erroroffset = OMAP4_VP_CONFIG_ERROROFFSET,
|
|
|
|
.vp_vstepmin = OMAP4_VP_VSTEPMIN_VSTEPMIN,
|
|
|
|
.vp_vstepmax = OMAP4_VP_VSTEPMAX_VSTEPMAX,
|
|
|
|
.vddmin = 900000,
|
|
|
|
.vddmax = 1350000,
|
|
|
|
.vp_timeout_us = OMAP4_VP_VLIMITTO_TIMEOUT_US,
|
|
|
|
.i2c_slave_addr = 0x02,
|
|
|
|
.volt_reg_addr = 0x00,
|
|
|
|
.cmd_reg_addr = 0x01,
|
|
|
|
.i2c_high_speed = true,
|
|
|
|
.vsel_to_uv = omap_cpcap_vsel_to_uv,
|
|
|
|
.uv_to_vsel = omap_cpcap_uv_to_vsel,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct omap_voltdm_pmic omap_cpcap_iva = {
|
|
|
|
.slew_rate = 4000,
|
|
|
|
.step_size = 12500,
|
|
|
|
.vp_erroroffset = OMAP4_VP_CONFIG_ERROROFFSET,
|
|
|
|
.vp_vstepmin = OMAP4_VP_VSTEPMIN_VSTEPMIN,
|
|
|
|
.vp_vstepmax = OMAP4_VP_VSTEPMAX_VSTEPMAX,
|
|
|
|
.vddmin = 900000,
|
|
|
|
.vddmax = 1350000,
|
|
|
|
.vp_timeout_us = OMAP4_VP_VLIMITTO_TIMEOUT_US,
|
|
|
|
.i2c_slave_addr = 0x44,
|
|
|
|
.volt_reg_addr = 0x0,
|
|
|
|
.cmd_reg_addr = 0x01,
|
|
|
|
.i2c_high_speed = true,
|
|
|
|
.vsel_to_uv = omap_cpcap_vsel_to_uv,
|
|
|
|
.uv_to_vsel = omap_cpcap_uv_to_vsel,
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* omap_max8952_vsel_to_vdc - convert MAX8952 VSEL value to microvolts DC
|
|
|
|
* @vsel: MAX8952 VSEL value to convert
|
|
|
|
*
|
|
|
|
* Returns the microvolts DC that the MAX8952 Regulator should generate when
|
|
|
|
* programmed with @vsel.
|
|
|
|
*/
|
2019-10-18 12:07:01 +00:00
|
|
|
static unsigned long omap_max8952_vsel_to_uv(unsigned char vsel)
|
2019-10-16 14:37:06 +00:00
|
|
|
{
|
|
|
|
if (vsel > 0x3F)
|
|
|
|
vsel = 0x3F;
|
|
|
|
return (((vsel * 100) + 7700)) * 100;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* omap_max8952_uv_to_vsel - convert microvolts DC to MAX8952 VSEL value
|
|
|
|
* @uv: microvolts DC to convert
|
|
|
|
*
|
|
|
|
* Returns the VSEL value necessary for the MAX8952 Regulator to
|
|
|
|
* generate an output voltage equal to or greater than @uv microvolts DC.
|
|
|
|
*/
|
2019-10-18 12:07:01 +00:00
|
|
|
static unsigned char omap_max8952_uv_to_vsel(unsigned long uv)
|
2019-10-16 14:37:06 +00:00
|
|
|
{
|
|
|
|
if (uv < 770000)
|
|
|
|
uv = 770000;
|
|
|
|
else if (uv > 1400000)
|
|
|
|
uv = 1400000;
|
|
|
|
return DIV_ROUND_UP(uv - 770000, 10000);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct omap_voltdm_pmic omap443x_max8952_mpu = {
|
|
|
|
.slew_rate = 16000,
|
|
|
|
.step_size = 10000,
|
|
|
|
.vp_erroroffset = OMAP4_VP_CONFIG_ERROROFFSET,
|
|
|
|
.vp_vstepmin = OMAP4_VP_VSTEPMIN_VSTEPMIN,
|
|
|
|
.vp_vstepmax = OMAP4_VP_VSTEPMAX_VSTEPMAX,
|
|
|
|
.vddmin = 900000,
|
|
|
|
.vddmax = 1400000,
|
|
|
|
.vp_timeout_us = OMAP4_VP_VLIMITTO_TIMEOUT_US,
|
|
|
|
.i2c_slave_addr = 0x60,
|
|
|
|
.volt_reg_addr = 0x03,
|
|
|
|
.cmd_reg_addr = 0x03,
|
|
|
|
.i2c_high_speed = true,
|
|
|
|
.vsel_to_uv = omap_max8952_vsel_to_uv,
|
|
|
|
.uv_to_vsel = omap_max8952_uv_to_vsel,
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* omap_fan5355_vsel_to_vdc - convert FAN535503 VSEL value to microvolts DC
|
|
|
|
* @vsel: FAN535503 VSEL value to convert
|
|
|
|
*
|
|
|
|
* Returns the microvolts DC that the FAN535503 Regulator should generate when
|
|
|
|
* programmed with @vsel.
|
|
|
|
*/
|
2019-10-18 12:07:01 +00:00
|
|
|
static unsigned long omap_fan535503_vsel_to_uv(unsigned char vsel)
|
2019-10-16 14:37:06 +00:00
|
|
|
{
|
|
|
|
/* Extract bits[5:0] */
|
|
|
|
vsel &= 0x3F;
|
|
|
|
|
|
|
|
return (((vsel * 125) + 7500)) * 100;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* omap_fan535508_vsel_to_vdc - convert FAN535508 VSEL value to microvolts DC
|
|
|
|
* @vsel: FAN535508 VSEL value to convert
|
|
|
|
*
|
|
|
|
* Returns the microvolts DC that the FAN535508 Regulator should generate when
|
|
|
|
* programmed with @vsel.
|
|
|
|
*/
|
2019-10-18 12:07:01 +00:00
|
|
|
static unsigned long omap_fan535508_vsel_to_uv(unsigned char vsel)
|
2019-10-16 14:37:06 +00:00
|
|
|
{
|
|
|
|
/* Extract bits[5:0] */
|
|
|
|
vsel &= 0x3F;
|
|
|
|
|
|
|
|
if (vsel > 0x37)
|
|
|
|
vsel = 0x37;
|
|
|
|
return (((vsel * 125) + 7500)) * 100;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* omap_fan535503_uv_to_vsel - convert microvolts DC to FAN535503 VSEL value
|
|
|
|
* @uv: microvolts DC to convert
|
|
|
|
*
|
|
|
|
* Returns the VSEL value necessary for the MAX8952 Regulator to
|
|
|
|
* generate an output voltage equal to or greater than @uv microvolts DC.
|
|
|
|
*/
|
2019-10-18 12:07:01 +00:00
|
|
|
static unsigned char omap_fan535503_uv_to_vsel(unsigned long uv)
|
2019-10-16 14:37:06 +00:00
|
|
|
{
|
|
|
|
unsigned char vsel;
|
|
|
|
if (uv < 750000)
|
|
|
|
uv = 750000;
|
|
|
|
else if (uv > 1537500)
|
|
|
|
uv = 1537500;
|
|
|
|
|
|
|
|
vsel = DIV_ROUND_UP(uv - 750000, 12500);
|
|
|
|
return vsel | 0xC0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* omap_fan535508_uv_to_vsel - convert microvolts DC to FAN535508 VSEL value
|
|
|
|
* @uv: microvolts DC to convert
|
|
|
|
*
|
|
|
|
* Returns the VSEL value necessary for the MAX8952 Regulator to
|
|
|
|
* generate an output voltage equal to or greater than @uv microvolts DC.
|
|
|
|
*/
|
2019-10-18 12:07:01 +00:00
|
|
|
static unsigned char omap_fan535508_uv_to_vsel(unsigned long uv)
|
2019-10-16 14:37:06 +00:00
|
|
|
{
|
|
|
|
unsigned char vsel;
|
|
|
|
if (uv < 750000)
|
|
|
|
uv = 750000;
|
|
|
|
else if (uv > 1437500)
|
|
|
|
uv = 1437500;
|
|
|
|
|
|
|
|
vsel = DIV_ROUND_UP(uv - 750000, 12500);
|
|
|
|
return vsel | 0xC0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fan5335-core */
|
|
|
|
static struct omap_voltdm_pmic omap4_fan_core = {
|
|
|
|
.slew_rate = 4000,
|
|
|
|
.step_size = 12500,
|
|
|
|
.vp_erroroffset = OMAP4_VP_CONFIG_ERROROFFSET,
|
|
|
|
.vp_vstepmin = OMAP4_VP_VSTEPMIN_VSTEPMIN,
|
|
|
|
.vp_vstepmax = OMAP4_VP_VSTEPMAX_VSTEPMAX,
|
|
|
|
.vddmin = 850000,
|
|
|
|
.vddmax = 1375000,
|
|
|
|
.vp_timeout_us = OMAP4_VP_VLIMITTO_TIMEOUT_US,
|
|
|
|
.i2c_slave_addr = 0x4A,
|
|
|
|
.i2c_high_speed = true,
|
|
|
|
.volt_reg_addr = 0x01,
|
|
|
|
.cmd_reg_addr = 0x01,
|
|
|
|
.vsel_to_uv = omap_fan535508_vsel_to_uv,
|
|
|
|
.uv_to_vsel = omap_fan535508_uv_to_vsel,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* fan5335 iva */
|
|
|
|
static struct omap_voltdm_pmic omap4_fan_iva = {
|
|
|
|
.slew_rate = 4000,
|
|
|
|
.step_size = 12500,
|
|
|
|
.vp_erroroffset = OMAP4_VP_CONFIG_ERROROFFSET,
|
|
|
|
.vp_vstepmin = OMAP4_VP_VSTEPMIN_VSTEPMIN,
|
|
|
|
.vp_vstepmax = OMAP4_VP_VSTEPMAX_VSTEPMAX,
|
|
|
|
.vddmin = 850000,
|
|
|
|
.vddmax = 1375000,
|
|
|
|
.vp_timeout_us = OMAP4_VP_VLIMITTO_TIMEOUT_US,
|
|
|
|
.i2c_slave_addr = 0x48,
|
|
|
|
.volt_reg_addr = 0x01,
|
|
|
|
.cmd_reg_addr = 0x01,
|
|
|
|
.i2c_high_speed = true,
|
|
|
|
.vsel_to_uv = omap_fan535503_vsel_to_uv,
|
|
|
|
.uv_to_vsel = omap_fan535503_uv_to_vsel,
|
|
|
|
};
|
|
|
|
|
|
|
|
int __init omap4_cpcap_init(void)
|
|
|
|
{
|
|
|
|
struct voltagedomain *voltdm;
|
|
|
|
|
|
|
|
if (!of_find_compatible_node(NULL, NULL, "motorola,cpcap"))
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
voltdm = voltdm_lookup("mpu");
|
|
|
|
omap_voltage_register_pmic(voltdm, &omap443x_max8952_mpu);
|
|
|
|
|
|
|
|
if (of_machine_is_compatible("motorola,droid-bionic")) {
|
|
|
|
voltdm = voltdm_lookup("mpu");
|
|
|
|
omap_voltage_register_pmic(voltdm, &omap_cpcap_core);
|
|
|
|
|
|
|
|
voltdm = voltdm_lookup("mpu");
|
|
|
|
omap_voltage_register_pmic(voltdm, &omap_cpcap_iva);
|
|
|
|
} else {
|
|
|
|
voltdm = voltdm_lookup("core");
|
|
|
|
omap_voltage_register_pmic(voltdm, &omap4_fan_core);
|
|
|
|
|
|
|
|
voltdm = voltdm_lookup("iva");
|
|
|
|
omap_voltage_register_pmic(voltdm, &omap4_fan_iva);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|