* [PATCH 1/3] MFD: TWL4030: Add support for TWL4030/5030 dynamic power switching [not found] <1247050046-31104-1-git-send-email-./0000-cover-letter.patch> @ 2009-07-08 10:49 ` Amit Kucheria 2009-08-04 17:55 ` Samuel Ortiz 2009-07-08 10:49 ` [PATCH 2/3] MFD: TWL4030: print warning for out-of-order script loading Amit Kucheria 2009-07-08 10:49 ` [PATCH 3/3] MFD: TWL4030: OMAP: Board-specifc DPS scripts for RX51 board Amit Kucheria 2 siblings, 1 reply; 11+ messages in thread From: Amit Kucheria @ 2009-07-08 10:49 UTC (permalink / raw) To: sameo; +Cc: dbrownell, linux-kernel, linux-omap The TWL4030/5030 family of multifunction devices allows board-specific control of the the various regulators, clock and reset lines through 'scripts' that are loaded into its memory. This allows for Dynamic Power Switching (DPS). Implement board-independent core support for DPS that is then used by board-specific code to load custom DPS scripts. Signed-off-by: Amit Kucheria <amit.kucheria@verdurent.com> --- drivers/mfd/Kconfig | 13 ++ drivers/mfd/Makefile | 1 + drivers/mfd/twl4030-core.c | 12 ++ drivers/mfd/twl4030-power.c | 384 +++++++++++++++++++++++++++++++++++++++++++ include/linux/i2c/twl4030.h | 91 +++++++++- 5 files changed, 491 insertions(+), 10 deletions(-) create mode 100644 drivers/mfd/twl4030-power.c diff --git a/drivers/mfd/Kconfig b/drivers/mfd/Kconfig index 491ac0f..94fa9a0 100644 --- a/drivers/mfd/Kconfig +++ b/drivers/mfd/Kconfig @@ -108,6 +108,19 @@ config TWL4030_CORE high speed USB OTG transceiver, an audio codec (on most versions) and many other features. +config TWL4030_POWER + bool "Support power resources on TWL4030 family chips" + depends on TWL4030_CORE + help + Say yes here if you want to use the power resources on the + TWL4030 family chips. Most of these resources are regulators, + which have a separate driver; some are control signals, such + as clock request handshaking. + + This driver uses board-specific data to initialize the resources + and load scripts controling which resources are switched off/on + or reset when a sleep, wakeup or warm reset event occurs. + config MFD_TMIO bool default n diff --git a/drivers/mfd/Makefile b/drivers/mfd/Makefile index 6f8a9a1..84b9eda 100644 --- a/drivers/mfd/Makefile +++ b/drivers/mfd/Makefile @@ -23,6 +23,7 @@ obj-$(CONFIG_TPS65010) += tps65010.o obj-$(CONFIG_MENELAUS) += menelaus.o obj-$(CONFIG_TWL4030_CORE) += twl4030-core.o twl4030-irq.o +obj-$(CONFIG_TWL4030_POWER) += twl4030-power.o obj-$(CONFIG_MFD_CORE) += mfd-core.o diff --git a/drivers/mfd/twl4030-core.c b/drivers/mfd/twl4030-core.c index ca54996..ee8ede8 100644 --- a/drivers/mfd/twl4030-core.c +++ b/drivers/mfd/twl4030-core.c @@ -89,6 +89,12 @@ #define twl_has_madc() false #endif +#ifdef CONFIG_TWL4030_POWER +#define twl_has_power() true +#else +#define twl_has_power() false +#endif + #if defined(CONFIG_RTC_DRV_TWL4030) || defined(CONFIG_RTC_DRV_TWL4030_MODULE) #define twl_has_rtc() true #else @@ -231,6 +237,8 @@ static struct twl4030mapping twl4030_map[TWL4030_MODULE_LAST + 1] = { { 3, TWL4030_BASEADD_SECURED_REG }, }; +extern void twl4030_power_init(struct twl4030_power_data *triton2_scripts); + /*----------------------------------------------------------------------*/ /* Exported Functions */ @@ -788,6 +796,10 @@ twl4030_probe(struct i2c_client *client, const struct i2c_device_id *id) /* setup clock framework */ clocks_init(&client->dev); + /* load power event scripts */ + if (twl_has_power() && pdata->power) + twl4030_power_init(pdata->power); + /* Maybe init the T2 Interrupt subsystem */ if (client->irq && pdata->irq_base diff --git a/drivers/mfd/twl4030-power.c b/drivers/mfd/twl4030-power.c new file mode 100644 index 0000000..bb9e45f --- /dev/null +++ b/drivers/mfd/twl4030-power.c @@ -0,0 +1,384 @@ +/* + * linux/drivers/i2c/chips/twl4030-power.c + * + * Handle TWL4030 Power initialization + * + * Copyright (C) 2008 Nokia Corporation + * Copyright (C) 2006 Texas Instruments, Inc + * + * Written by Kalle Jokiniemi + * Peter De Schrijver <peter.de-schrijver@nokia.com> + * Several fixes by Amit Kucheria <amit.kucheria@verdurent.com> + * + * This file is subject to the terms and conditions of the GNU General + * Public License. See the file "COPYING" in the main directory of this + * archive for more details. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#include <linux/module.h> +#include <linux/pm.h> +#include <linux/i2c/twl4030.h> +#include <linux/platform_device.h> + +#include <asm/mach-types.h> + +static u8 triton_next_free_address = 0x2b; + +#define PWR_P1_SW_EVENTS 0x10 +#define PWR_DEVOFF (1<<0) + +#define PHY_TO_OFF_PM_MASTER(p) (p - 0x36) +#define PHY_TO_OFF_PM_RECEIVER(p) (p - 0x5b) + +#define NUM_OF_RESOURCES 28 + +/* resource - hfclk */ +#define R_HFCLKOUT_DEV_GRP PHY_TO_OFF_PM_RECEIVER(0xe6) + +/* PM events */ +#define R_P1_SW_EVENTS PHY_TO_OFF_PM_MASTER(0x46) +#define R_P2_SW_EVENTS PHY_TO_OFF_PM_MASTER(0x47) +#define R_P3_SW_EVENTS PHY_TO_OFF_PM_MASTER(0x48) +#define R_CFG_P1_TRANSITION PHY_TO_OFF_PM_MASTER(0x36) +#define R_CFG_P2_TRANSITION PHY_TO_OFF_PM_MASTER(0x37) +#define R_CFG_P3_TRANSITION PHY_TO_OFF_PM_MASTER(0x38) + +#define LVL_WAKEUP 0x08 + +#define ENABLE_WARMRESET (1<<4) + +#define END_OF_SCRIPT 0x3f + +#define R_SEQ_ADD_A2S PHY_TO_OFF_PM_MASTER(0x55) +#define R_SEQ_ADD_S2A12 PHY_TO_OFF_PM_MASTER(0x56) +#define R_SEQ_ADD_S2A3 PHY_TO_OFF_PM_MASTER(0x57) +#define R_SEQ_ADD_WARM PHY_TO_OFF_PM_MASTER(0x58) +#define R_MEMORY_ADDRESS PHY_TO_OFF_PM_MASTER(0x59) +#define R_MEMORY_DATA PHY_TO_OFF_PM_MASTER(0x5a) + +#define R_PROTECT_KEY 0x0E +#define KEY_1 0xC0 +#define KEY_2 0x0C + +/* resource configuration registers */ + +#define DEVGROUP_OFFSET 0 +#define TYPE_OFFSET 1 + +static u8 res_config_addrs[] = { + [RES_VAUX1] = 0x17, + [RES_VAUX2] = 0x1b, + [RES_VAUX3] = 0x1f, + [RES_VAUX4] = 0x23, + [RES_VMMC1] = 0x27, + [RES_VMMC2] = 0x2b, + [RES_VPLL1] = 0x2f, + [RES_VPLL2] = 0x33, + [RES_VSIM] = 0x37, + [RES_VDAC] = 0x3b, + [RES_VINTANA1] = 0x3f, + [RES_VINTANA2] = 0x43, + [RES_VINTDIG] = 0x47, + [RES_VIO] = 0x4b, + [RES_VDD1] = 0x55, + [RES_VDD2] = 0x63, + [RES_VUSB_1V5] = 0x71, + [RES_VUSB_1V8] = 0x74, + [RES_VUSB_3V1] = 0x77, + [RES_VUSBCP] = 0x7a, + [RES_REGEN] = 0x7f, + [RES_NRES_PWRON] = 0x82, + [RES_CLKEN] = 0x85, + [RES_SYSEN] = 0x88, + [RES_HFCLKOUT] = 0x8b, + [RES_32KCLKOUT] = 0x8e, + [RES_RESET] = 0x91, + [RES_Main_Ref] = 0x94, +}; + +static int __init twl4030_write_script_byte(u8 address, u8 byte) +{ + int err; + + err = twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, + R_MEMORY_ADDRESS); + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, byte, + R_MEMORY_DATA); + + return err; +} + +static int __init twl4030_write_script_ins(u8 address, u16 pmb_message, + u8 delay, u8 next) +{ + int err = 0; + + address *= 4; + err |= twl4030_write_script_byte(address++, pmb_message >> 8); + err |= twl4030_write_script_byte(address++, pmb_message & 0xff); + err |= twl4030_write_script_byte(address++, delay); + err |= twl4030_write_script_byte(address++, next); + + return err; +} + +static int __init twl4030_write_script(u8 address, struct twl4030_ins *script, + int len) +{ + int err = 0; + + for (; len; len--, address++, script++) { + if (len == 1) + err |= twl4030_write_script_ins(address, + script->pmb_message, + script->delay, + END_OF_SCRIPT); + else + err |= twl4030_write_script_ins(address, + script->pmb_message, + script->delay, + address + 1); + } + + return err; +} + +static int __init config_wakeup3_sequence(u8 address) +{ + int err = 0; + + /* Set SLEEP to ACTIVE SEQ address for P3 */ + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, + R_SEQ_ADD_S2A3); + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, LVL_WAKEUP, + R_P3_SW_EVENTS); + if (err) + printk(KERN_ERR "TWL4030 wakeup sequence for P3" \ + "config error\n"); + + return err; +} + +static int __init config_wakeup12_sequence(u8 address) +{ + int err = 0; + u8 data; + + /* Set SLEEP to ACTIVE SEQ address for P1 and P2 */ + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, + R_SEQ_ADD_S2A12); + + /* P1/P2 LVL_WAKEUP should be on LEVEL */ + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &data, + R_P1_SW_EVENTS); + data |= LVL_WAKEUP; + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, data, + R_P1_SW_EVENTS); + + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &data, + R_P2_SW_EVENTS); + data |= LVL_WAKEUP; + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, data, + R_P2_SW_EVENTS); + + if (machine_is_omap_3430sdp() || machine_is_omap_ldp()) { + /* Disabling AC charger effect on sleep-active transitions */ + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &data, + R_CFG_P1_TRANSITION); + data &= ~(1<<1); + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, data , + R_CFG_P1_TRANSITION); + } + + if (err) + printk(KERN_ERR "TWL4030 wakeup sequence for P1 and P2" \ + "config error\n"); + + return err; +} + +static int __init config_sleep_sequence(u8 address) +{ + int err = 0; + + /* Set ACTIVE to SLEEP SEQ address in T2 memory*/ + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, + R_SEQ_ADD_A2S); + + if (err) + printk(KERN_ERR "TWL4030 sleep sequence config error\n"); + + return err; +} + +static int __init config_warmreset_sequence(u8 address) +{ + int err = 0; + u8 rd_data; + + /* Set WARM RESET SEQ address for P1 */ + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, + R_SEQ_ADD_WARM); + + /* P1/P2/P3 enable WARMRESET */ + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &rd_data, + R_P1_SW_EVENTS); + rd_data |= ENABLE_WARMRESET; + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, rd_data, + R_P1_SW_EVENTS); + + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &rd_data, + R_P2_SW_EVENTS); + rd_data |= ENABLE_WARMRESET; + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, rd_data, + R_P2_SW_EVENTS); + + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &rd_data, + R_P3_SW_EVENTS); + rd_data |= ENABLE_WARMRESET; + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, rd_data, + R_P3_SW_EVENTS); + + if (err) + printk(KERN_ERR + "TWL4030 warmreset seq config error\n"); + return err; +} + +static int __init twl4030_configure_resource(struct twl4030_resconfig *rconfig) +{ + int rconfig_addr; + int err; + u8 type; + + if (rconfig->resource > NUM_OF_RESOURCES) { + printk(KERN_ERR + "TWL4030 Resource %d does not exist\n", + rconfig->resource); + return -EINVAL; + } + + rconfig_addr = res_config_addrs[rconfig->resource]; + + /* Set resource group */ + if (rconfig->devgroup >= 0) + err = twl4030_i2c_write_u8(TWL4030_MODULE_PM_RECEIVER, + rconfig->devgroup << 5, + rconfig_addr + DEVGROUP_OFFSET); + if (err < 0) { + printk(KERN_ERR + "TWL4030 failed to program devgroup"); + return err; + } + + /* Set resource types */ + err = twl4030_i2c_read_u8(TWL4030_MODULE_PM_RECEIVER, &type, + rconfig_addr + TYPE_OFFSET); + if (err < 0) { + printk(KERN_ERR + "TWL4030 Resource %d type could not be read\n", + rconfig->resource); + return err; + } + + if (rconfig->type >= 0) { + type &= ~7; + type |= rconfig->type; + } + + if (rconfig->type2 >= 0) { + type &= ~(3 << 3); + type |= rconfig->type2 << 3; + } + + err = twl4030_i2c_write_u8(TWL4030_MODULE_PM_RECEIVER, + type, rconfig_addr + TYPE_OFFSET); + if (err < 0) { + printk(KERN_ERR + "TWL4030 failed to program resource type"); + return err; + } + + return 0; +} + +static int __init load_triton_script(struct twl4030_script *tscript) +{ + u8 address = triton_next_free_address; + int err; + + /* Make sure the script isn't going beyond last valid address */ + if ((address + tscript->size) > (END_OF_SCRIPT-1)) { + printk(KERN_ERR "TWL4030 script too big error\n"); + return -EINVAL; + } + + err = twl4030_write_script(address, tscript->script, tscript->size); + if (err) + return err; + + triton_next_free_address += tscript->size; + + if (tscript->flags & TRITON_WRST_SCRIPT) + err |= config_warmreset_sequence(address); + + if (tscript->flags & TRITON_WAKEUP12_SCRIPT) + err |= config_wakeup12_sequence(address); + + if (tscript->flags & TRITON_WAKEUP3_SCRIPT) + err |= config_wakeup3_sequence(address); + + if (tscript->flags & TRITON_SLEEP_SCRIPT) + err |= config_sleep_sequence(address); + + return err; +} + +void __init twl4030_power_init(struct twl4030_power_data *triton2_scripts) +{ + int err = 0; + int i; + struct twl4030_resconfig *resconfig; + + err = twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, KEY_1, + R_PROTECT_KEY); + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, KEY_2, + R_PROTECT_KEY); + if (err) + printk(KERN_ERR + "TWL4030 Unable to unlock registers\n"); + + for (i = 0; i < triton2_scripts->size; i++) { + err = load_triton_script(triton2_scripts->scripts[i]); + if (err < 0) { + printk(KERN_ERR "TWL4030 failed to load scripts"); + break; + } + } + + resconfig = triton2_scripts->resource_config; + if (resconfig) { + while (resconfig->resource) { + err = twl4030_configure_resource(resconfig); + resconfig++; + if (err < 0) { + printk(KERN_ERR + "TWL4030 failed to configure resource"); + break; + } + } + } + + if (twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, 0, R_PROTECT_KEY)) + printk(KERN_ERR + "TWL4030 Unable to relock registers\n"); +} diff --git a/include/linux/i2c/twl4030.h b/include/linux/i2c/twl4030.h index 0dc80ef..6f59bad 100644 --- a/include/linux/i2c/twl4030.h +++ b/include/linux/i2c/twl4030.h @@ -220,19 +220,28 @@ int twl4030_i2c_read(u8 mod_no, u8 *value, u8 reg, unsigned num_bytes); /* Power bus message definitions */ -#define DEV_GRP_NULL 0x0 -#define DEV_GRP_P1 0x1 -#define DEV_GRP_P2 0x2 -#define DEV_GRP_P3 0x4 +/* The TWL4030/5030 splits its power-management resources (the various + * regulators, clock and reset lines) into 3 processor groups - P1, P2 and + * P3. These groups can then be configured to transition between sleep, wait-on + * and active states by sending messages to the power bus. See Section 5.4.2 + * Power Resources of TWL4030 TRM + */ -#define RES_GRP_RES 0x0 -#define RES_GRP_PP 0x1 -#define RES_GRP_RC 0x2 +/* Processor groups */ +#define DEV_GRP_NULL 0x0 +#define DEV_GRP_P1 0x1 /* P1: all OMAP devices */ +#define DEV_GRP_P2 0x2 /* P2: all Modem devices */ +#define DEV_GRP_P3 0x4 /* P3: all peripheral devices */ + +/* Resource groups */ +#define RES_GRP_RES 0x0 /* Reserved */ +#define RES_GRP_PP 0x1 /* Power providers */ +#define RES_GRP_RC 0x2 /* Reset and control */ #define RES_GRP_PP_RC 0x3 -#define RES_GRP_PR 0x4 +#define RES_GRP_PR 0x4 /* Power references */ #define RES_GRP_PP_PR 0x5 #define RES_GRP_RC_PR 0x6 -#define RES_GRP_ALL 0x7 +#define RES_GRP_ALL 0x7 /* All resource groups */ #define RES_TYPE2_R0 0x0 @@ -243,6 +252,40 @@ int twl4030_i2c_read(u8 mod_no, u8 *value, u8 reg, unsigned num_bytes); #define RES_STATE_SLEEP 0x8 #define RES_STATE_OFF 0x0 +/* Power resources */ + +/* Power providers */ +#define RES_VAUX1 1 +#define RES_VAUX2 2 +#define RES_VAUX3 3 +#define RES_VAUX4 4 +#define RES_VMMC1 5 +#define RES_VMMC2 6 +#define RES_VPLL1 7 +#define RES_VPLL2 8 +#define RES_VSIM 9 +#define RES_VDAC 10 +#define RES_VINTANA1 11 +#define RES_VINTANA2 12 +#define RES_VINTDIG 13 +#define RES_VIO 14 +#define RES_VDD1 15 +#define RES_VDD2 16 +#define RES_VUSB_1V5 17 +#define RES_VUSB_1V8 18 +#define RES_VUSB_3V1 19 +#define RES_VUSBCP 20 +#define RES_REGEN 21 +/* Reset and control */ +#define RES_NRES_PWRON 22 +#define RES_CLKEN 23 +#define RES_SYSEN 24 +#define RES_HFCLKOUT 25 +#define RES_32KCLKOUT 26 +#define RES_RESET 27 +/* Power Reference */ +#define RES_Main_Ref 28 + /* * Power Bus Message Format ... these can be sent individually by Linux, * but are usually part of downloaded scripts that are run when various @@ -320,6 +363,34 @@ struct twl4030_usb_data { enum twl4030_usb_mode usb_mode; }; +struct twl4030_ins { + u16 pmb_message; + u8 delay; +}; + +struct twl4030_script { + struct twl4030_ins *script; + unsigned size; + u8 flags; +#define TRITON_WRST_SCRIPT (1<<0) +#define TRITON_WAKEUP12_SCRIPT (1<<1) +#define TRITON_WAKEUP3_SCRIPT (1<<2) +#define TRITON_SLEEP_SCRIPT (1<<3) +}; + +struct twl4030_resconfig { + u8 resource; + u8 devgroup; + u8 type; + u8 type2; +}; + +struct twl4030_power_data { + struct twl4030_script **scripts; + unsigned size; + struct twl4030_resconfig *resource_config; +}; + struct twl4030_platform_data { unsigned irq_base, irq_end; struct twl4030_bci_platform_data *bci; @@ -327,6 +398,7 @@ struct twl4030_platform_data { struct twl4030_madc_platform_data *madc; struct twl4030_keypad_data *keypad; struct twl4030_usb_data *usb; + struct twl4030_power_data *power; /* LDO regulators */ struct regulator_init_data *vdac; @@ -357,7 +429,6 @@ int twl4030_sih_setup(int module); #define TWL4030_VAUX3_DEV_GRP 0x1F #define TWL4030_VAUX3_DEDICATED 0x22 - #if defined(CONFIG_TWL4030_BCI_BATTERY) || \ defined(CONFIG_TWL4030_BCI_BATTERY_MODULE) extern int twl4030charger_usb_en(int enable); -- 1.6.3.3 ^ permalink raw reply related [flat|nested] 11+ messages in thread
* Re: [PATCH 1/3] MFD: TWL4030: Add support for TWL4030/5030 dynamic power switching 2009-07-08 10:49 ` [PATCH 1/3] MFD: TWL4030: Add support for TWL4030/5030 dynamic power switching Amit Kucheria @ 2009-08-04 17:55 ` Samuel Ortiz 2009-08-17 11:42 ` Amit Kucheria 0 siblings, 1 reply; 11+ messages in thread From: Samuel Ortiz @ 2009-08-04 17:55 UTC (permalink / raw) To: Amit Kucheria; +Cc: dbrownell, linux-kernel, linux-omap Hi Amit, My apologies for the late answer, but I'm going through my mail backlog after a 3 weeks holidays. On Wed, Jul 08, 2009 at 01:49:27PM +0300, Amit Kucheria wrote: > The TWL4030/5030 family of multifunction devices allows board-specific > control of the the various regulators, clock and reset lines through > 'scripts' that are loaded into its memory. This allows for Dynamic Power > Switching (DPS). > > Implement board-independent core support for DPS that is then used by > board-specific code to load custom DPS scripts. > > Signed-off-by: Amit Kucheria <amit.kucheria@verdurent.com> Some comments here: > --- > drivers/mfd/Kconfig | 13 ++ > drivers/mfd/Makefile | 1 + > drivers/mfd/twl4030-core.c | 12 ++ > drivers/mfd/twl4030-power.c | 384 +++++++++++++++++++++++++++++++++++++++++++ > include/linux/i2c/twl4030.h | 91 +++++++++- > 5 files changed, 491 insertions(+), 10 deletions(-) > create mode 100644 drivers/mfd/twl4030-power.c > > diff --git a/drivers/mfd/Kconfig b/drivers/mfd/Kconfig > index 491ac0f..94fa9a0 100644 > --- a/drivers/mfd/Kconfig > +++ b/drivers/mfd/Kconfig > @@ -108,6 +108,19 @@ config TWL4030_CORE > high speed USB OTG transceiver, an audio codec (on most > versions) and many other features. > > +config TWL4030_POWER > + bool "Support power resources on TWL4030 family chips" > + depends on TWL4030_CORE > + help > + Say yes here if you want to use the power resources on the > + TWL4030 family chips. Most of these resources are regulators, > + which have a separate driver; some are control signals, such > + as clock request handshaking. > + > + This driver uses board-specific data to initialize the resources > + and load scripts controling which resources are switched off/on > + or reset when a sleep, wakeup or warm reset event occurs. > + > config MFD_TMIO > bool > default n > diff --git a/drivers/mfd/Makefile b/drivers/mfd/Makefile > index 6f8a9a1..84b9eda 100644 > --- a/drivers/mfd/Makefile > +++ b/drivers/mfd/Makefile > @@ -23,6 +23,7 @@ obj-$(CONFIG_TPS65010) += tps65010.o > obj-$(CONFIG_MENELAUS) += menelaus.o > > obj-$(CONFIG_TWL4030_CORE) += twl4030-core.o twl4030-irq.o > +obj-$(CONFIG_TWL4030_POWER) += twl4030-power.o > > obj-$(CONFIG_MFD_CORE) += mfd-core.o > > diff --git a/drivers/mfd/twl4030-core.c b/drivers/mfd/twl4030-core.c > index ca54996..ee8ede8 100644 > --- a/drivers/mfd/twl4030-core.c > +++ b/drivers/mfd/twl4030-core.c > @@ -89,6 +89,12 @@ > #define twl_has_madc() false > #endif > > +#ifdef CONFIG_TWL4030_POWER > +#define twl_has_power() true > +#else > +#define twl_has_power() false > +#endif > + > #if defined(CONFIG_RTC_DRV_TWL4030) || defined(CONFIG_RTC_DRV_TWL4030_MODULE) > #define twl_has_rtc() true > #else > @@ -231,6 +237,8 @@ static struct twl4030mapping twl4030_map[TWL4030_MODULE_LAST + 1] = { > { 3, TWL4030_BASEADD_SECURED_REG }, > }; > > +extern void twl4030_power_init(struct twl4030_power_data *triton2_scripts); Maybe this one should be defined in twl4030.h. > /*----------------------------------------------------------------------*/ > > /* Exported Functions */ > @@ -788,6 +796,10 @@ twl4030_probe(struct i2c_client *client, const struct i2c_device_id *id) > /* setup clock framework */ > clocks_init(&client->dev); > > + /* load power event scripts */ > + if (twl_has_power() && pdata->power) > + twl4030_power_init(pdata->power); > + > /* Maybe init the T2 Interrupt subsystem */ > if (client->irq > && pdata->irq_base > diff --git a/drivers/mfd/twl4030-power.c b/drivers/mfd/twl4030-power.c > new file mode 100644 > index 0000000..bb9e45f > --- /dev/null > +++ b/drivers/mfd/twl4030-power.c > @@ -0,0 +1,384 @@ > +/* > + * linux/drivers/i2c/chips/twl4030-power.c > + * > + * Handle TWL4030 Power initialization > + * > + * Copyright (C) 2008 Nokia Corporation > + * Copyright (C) 2006 Texas Instruments, Inc > + * > + * Written by Kalle Jokiniemi > + * Peter De Schrijver <peter.de-schrijver@nokia.com> > + * Several fixes by Amit Kucheria <amit.kucheria@verdurent.com> > + * > + * This file is subject to the terms and conditions of the GNU General > + * Public License. See the file "COPYING" in the main directory of this > + * archive for more details. > + * > + * This program is distributed in the hope that it will be useful, > + * but WITHOUT ANY WARRANTY; without even the implied warranty of > + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the > + * GNU General Public License for more details. > + * > + * You should have received a copy of the GNU General Public License > + * along with this program; if not, write to the Free Software > + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA > + */ > + > +#include <linux/module.h> > +#include <linux/pm.h> > +#include <linux/i2c/twl4030.h> > +#include <linux/platform_device.h> > + > +#include <asm/mach-types.h> > + > +static u8 triton_next_free_address = 0x2b; Are triton and twl4030 the same thing ? If that's so could we use one and only one prefix in this code (which would be twl4030) ? I'm nitpicking here, but this is kind of confusing unless triton is only a subset of the 4030 family. > +#define PWR_P1_SW_EVENTS 0x10 > +#define PWR_DEVOFF (1<<0) > + > +#define PHY_TO_OFF_PM_MASTER(p) (p - 0x36) > +#define PHY_TO_OFF_PM_RECEIVER(p) (p - 0x5b) > + > +#define NUM_OF_RESOURCES 28 > + > +/* resource - hfclk */ > +#define R_HFCLKOUT_DEV_GRP PHY_TO_OFF_PM_RECEIVER(0xe6) > + > +/* PM events */ > +#define R_P1_SW_EVENTS PHY_TO_OFF_PM_MASTER(0x46) > +#define R_P2_SW_EVENTS PHY_TO_OFF_PM_MASTER(0x47) > +#define R_P3_SW_EVENTS PHY_TO_OFF_PM_MASTER(0x48) > +#define R_CFG_P1_TRANSITION PHY_TO_OFF_PM_MASTER(0x36) > +#define R_CFG_P2_TRANSITION PHY_TO_OFF_PM_MASTER(0x37) > +#define R_CFG_P3_TRANSITION PHY_TO_OFF_PM_MASTER(0x38) > + > +#define LVL_WAKEUP 0x08 > + > +#define ENABLE_WARMRESET (1<<4) > + > +#define END_OF_SCRIPT 0x3f > + > +#define R_SEQ_ADD_A2S PHY_TO_OFF_PM_MASTER(0x55) > +#define R_SEQ_ADD_S2A12 PHY_TO_OFF_PM_MASTER(0x56) > +#define R_SEQ_ADD_S2A3 PHY_TO_OFF_PM_MASTER(0x57) > +#define R_SEQ_ADD_WARM PHY_TO_OFF_PM_MASTER(0x58) > +#define R_MEMORY_ADDRESS PHY_TO_OFF_PM_MASTER(0x59) > +#define R_MEMORY_DATA PHY_TO_OFF_PM_MASTER(0x5a) > + > +#define R_PROTECT_KEY 0x0E > +#define KEY_1 0xC0 > +#define KEY_2 0x0C > + > +/* resource configuration registers */ > + > +#define DEVGROUP_OFFSET 0 > +#define TYPE_OFFSET 1 > + > +static u8 res_config_addrs[] = { > + [RES_VAUX1] = 0x17, > + [RES_VAUX2] = 0x1b, > + [RES_VAUX3] = 0x1f, > + [RES_VAUX4] = 0x23, > + [RES_VMMC1] = 0x27, > + [RES_VMMC2] = 0x2b, > + [RES_VPLL1] = 0x2f, > + [RES_VPLL2] = 0x33, > + [RES_VSIM] = 0x37, > + [RES_VDAC] = 0x3b, > + [RES_VINTANA1] = 0x3f, > + [RES_VINTANA2] = 0x43, > + [RES_VINTDIG] = 0x47, > + [RES_VIO] = 0x4b, > + [RES_VDD1] = 0x55, > + [RES_VDD2] = 0x63, > + [RES_VUSB_1V5] = 0x71, > + [RES_VUSB_1V8] = 0x74, > + [RES_VUSB_3V1] = 0x77, > + [RES_VUSBCP] = 0x7a, > + [RES_REGEN] = 0x7f, > + [RES_NRES_PWRON] = 0x82, > + [RES_CLKEN] = 0x85, > + [RES_SYSEN] = 0x88, > + [RES_HFCLKOUT] = 0x8b, > + [RES_32KCLKOUT] = 0x8e, > + [RES_RESET] = 0x91, > + [RES_Main_Ref] = 0x94, > +}; > + > +static int __init twl4030_write_script_byte(u8 address, u8 byte) > +{ > + int err; > + > + err = twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, > + R_MEMORY_ADDRESS); > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, byte, > + R_MEMORY_DATA); In most of this piece of code, you're ORing the return code, and that seems wrong to me. First of all, do you really want to continue reading or writing after the first error ? Then, your ORed err may eventually be meaningless. I'd prefer to always check for err and return in the error case. > + return err; > +} > + > +static int __init twl4030_write_script_ins(u8 address, u16 pmb_message, > + u8 delay, u8 next) > +{ > + int err = 0; > + > + address *= 4; > + err |= twl4030_write_script_byte(address++, pmb_message >> 8); > + err |= twl4030_write_script_byte(address++, pmb_message & 0xff); > + err |= twl4030_write_script_byte(address++, delay); > + err |= twl4030_write_script_byte(address++, next); > + > + return err; > +} > + > +static int __init twl4030_write_script(u8 address, struct twl4030_ins *script, > + int len) > +{ > + int err = 0; > + > + for (; len; len--, address++, script++) { > + if (len == 1) > + err |= twl4030_write_script_ins(address, > + script->pmb_message, > + script->delay, > + END_OF_SCRIPT); > + else > + err |= twl4030_write_script_ins(address, > + script->pmb_message, > + script->delay, > + address + 1); > + } > + > + return err; > +} > + > +static int __init config_wakeup3_sequence(u8 address) Some of your static routines are not twl4030_ prefixed, and some are. I guess we could be more consistent here. > +{ > + int err = 0; > + > + /* Set SLEEP to ACTIVE SEQ address for P3 */ > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, > + R_SEQ_ADD_S2A3); > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, LVL_WAKEUP, > + R_P3_SW_EVENTS); > + if (err) > + printk(KERN_ERR "TWL4030 wakeup sequence for P3" \ > + "config error\n"); The rest of the twl4030 code uses pr_err(), so you probably should use it too. > + return err; > +} > + > +static int __init config_wakeup12_sequence(u8 address) > +{ > + int err = 0; > + u8 data; > + > + /* Set SLEEP to ACTIVE SEQ address for P1 and P2 */ > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, > + R_SEQ_ADD_S2A12); > + > + /* P1/P2 LVL_WAKEUP should be on LEVEL */ > + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &data, > + R_P1_SW_EVENTS); > + data |= LVL_WAKEUP; > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, data, > + R_P1_SW_EVENTS); > + > + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &data, > + R_P2_SW_EVENTS); > + data |= LVL_WAKEUP; > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, data, > + R_P2_SW_EVENTS); > + > + if (machine_is_omap_3430sdp() || machine_is_omap_ldp()) { > + /* Disabling AC charger effect on sleep-active transitions */ > + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &data, > + R_CFG_P1_TRANSITION); > + data &= ~(1<<1); > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, data , > + R_CFG_P1_TRANSITION); > + } > + > + if (err) > + printk(KERN_ERR "TWL4030 wakeup sequence for P1 and P2" \ > + "config error\n"); > + > + return err; > +} > + > +static int __init config_sleep_sequence(u8 address) > +{ > + int err = 0; > + > + /* Set ACTIVE to SLEEP SEQ address in T2 memory*/ > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, > + R_SEQ_ADD_A2S); > + > + if (err) > + printk(KERN_ERR "TWL4030 sleep sequence config error\n"); > + > + return err; > +} > + > +static int __init config_warmreset_sequence(u8 address) > +{ > + int err = 0; > + u8 rd_data; > + > + /* Set WARM RESET SEQ address for P1 */ > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, > + R_SEQ_ADD_WARM); > + > + /* P1/P2/P3 enable WARMRESET */ > + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &rd_data, > + R_P1_SW_EVENTS); > + rd_data |= ENABLE_WARMRESET; > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, rd_data, > + R_P1_SW_EVENTS); > + > + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &rd_data, > + R_P2_SW_EVENTS); > + rd_data |= ENABLE_WARMRESET; > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, rd_data, > + R_P2_SW_EVENTS); > + > + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &rd_data, > + R_P3_SW_EVENTS); > + rd_data |= ENABLE_WARMRESET; > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, rd_data, > + R_P3_SW_EVENTS); > + > + if (err) > + printk(KERN_ERR > + "TWL4030 warmreset seq config error\n"); > + return err; > +} > + > +static int __init twl4030_configure_resource(struct twl4030_resconfig *rconfig) > +{ > + int rconfig_addr; > + int err; > + u8 type; > + > + if (rconfig->resource > NUM_OF_RESOURCES) { > + printk(KERN_ERR > + "TWL4030 Resource %d does not exist\n", > + rconfig->resource); > + return -EINVAL; > + } > + > + rconfig_addr = res_config_addrs[rconfig->resource]; > + > + /* Set resource group */ > + if (rconfig->devgroup >= 0) > + err = twl4030_i2c_write_u8(TWL4030_MODULE_PM_RECEIVER, > + rconfig->devgroup << 5, > + rconfig_addr + DEVGROUP_OFFSET); > + if (err < 0) { > + printk(KERN_ERR > + "TWL4030 failed to program devgroup"); > + return err; > + } > + > + /* Set resource types */ > + err = twl4030_i2c_read_u8(TWL4030_MODULE_PM_RECEIVER, &type, > + rconfig_addr + TYPE_OFFSET); > + if (err < 0) { > + printk(KERN_ERR > + "TWL4030 Resource %d type could not be read\n", > + rconfig->resource); > + return err; > + } > + > + if (rconfig->type >= 0) { > + type &= ~7; Could we have some definition for this 7 ? > + type |= rconfig->type; > + } > + > + if (rconfig->type2 >= 0) { > + type &= ~(3 << 3); Same here. > + type |= rconfig->type2 << 3; > + } > + > + err = twl4030_i2c_write_u8(TWL4030_MODULE_PM_RECEIVER, > + type, rconfig_addr + TYPE_OFFSET); > + if (err < 0) { > + printk(KERN_ERR > + "TWL4030 failed to program resource type"); > + return err; > + } > + > + return 0; > +} > + > +static int __init load_triton_script(struct twl4030_script *tscript) > +{ > + u8 address = triton_next_free_address; > + int err; Here I'd prefer to have address as a function argument, and then set it first to 0x2b from the load_triton_script() loop in power_init. That would be one way to get rid of this static triton_next_free_address variable which I dont like too much. > + /* Make sure the script isn't going beyond last valid address */ > + if ((address + tscript->size) > (END_OF_SCRIPT-1)) { > + printk(KERN_ERR "TWL4030 script too big error\n"); > + return -EINVAL; > + } > + > + err = twl4030_write_script(address, tscript->script, tscript->size); > + if (err) > + return err; > + > + triton_next_free_address += tscript->size; > + > + if (tscript->flags & TRITON_WRST_SCRIPT) > + err |= config_warmreset_sequence(address); > + > + if (tscript->flags & TRITON_WAKEUP12_SCRIPT) > + err |= config_wakeup12_sequence(address); > + > + if (tscript->flags & TRITON_WAKEUP3_SCRIPT) > + err |= config_wakeup3_sequence(address); > + > + if (tscript->flags & TRITON_SLEEP_SCRIPT) > + err |= config_sleep_sequence(address); > + > + return err; > +} > + > +void __init twl4030_power_init(struct twl4030_power_data *triton2_scripts) > +{ > + int err = 0; > + int i; > + struct twl4030_resconfig *resconfig; > + > + err = twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, KEY_1, > + R_PROTECT_KEY); > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, KEY_2, > + R_PROTECT_KEY); > + if (err) > + printk(KERN_ERR > + "TWL4030 Unable to unlock registers\n"); > + > + for (i = 0; i < triton2_scripts->size; i++) { > + err = load_triton_script(triton2_scripts->scripts[i]); > + if (err < 0) { > + printk(KERN_ERR "TWL4030 failed to load scripts"); > + break; > + } > + } > + > + resconfig = triton2_scripts->resource_config; > + if (resconfig) { > + while (resconfig->resource) { > + err = twl4030_configure_resource(resconfig); > + resconfig++; > + if (err < 0) { > + printk(KERN_ERR > + "TWL4030 failed to configure resource"); > + break; > + } > + } > + } > + > + if (twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, 0, R_PROTECT_KEY)) > + printk(KERN_ERR > + "TWL4030 Unable to relock registers\n"); > +} > diff --git a/include/linux/i2c/twl4030.h b/include/linux/i2c/twl4030.h > index 0dc80ef..6f59bad 100644 > --- a/include/linux/i2c/twl4030.h > +++ b/include/linux/i2c/twl4030.h > @@ -220,19 +220,28 @@ int twl4030_i2c_read(u8 mod_no, u8 *value, u8 reg, unsigned num_bytes); > > /* Power bus message definitions */ > > -#define DEV_GRP_NULL 0x0 > -#define DEV_GRP_P1 0x1 > -#define DEV_GRP_P2 0x2 > -#define DEV_GRP_P3 0x4 > +/* The TWL4030/5030 splits its power-management resources (the various > + * regulators, clock and reset lines) into 3 processor groups - P1, P2 and > + * P3. These groups can then be configured to transition between sleep, wait-on > + * and active states by sending messages to the power bus. See Section 5.4.2 > + * Power Resources of TWL4030 TRM > + */ > > -#define RES_GRP_RES 0x0 > -#define RES_GRP_PP 0x1 > -#define RES_GRP_RC 0x2 > +/* Processor groups */ > +#define DEV_GRP_NULL 0x0 > +#define DEV_GRP_P1 0x1 /* P1: all OMAP devices */ > +#define DEV_GRP_P2 0x2 /* P2: all Modem devices */ > +#define DEV_GRP_P3 0x4 /* P3: all peripheral devices */ > + > +/* Resource groups */ > +#define RES_GRP_RES 0x0 /* Reserved */ > +#define RES_GRP_PP 0x1 /* Power providers */ > +#define RES_GRP_RC 0x2 /* Reset and control */ > #define RES_GRP_PP_RC 0x3 > -#define RES_GRP_PR 0x4 > +#define RES_GRP_PR 0x4 /* Power references */ > #define RES_GRP_PP_PR 0x5 > #define RES_GRP_RC_PR 0x6 > -#define RES_GRP_ALL 0x7 > +#define RES_GRP_ALL 0x7 /* All resource groups */ > > #define RES_TYPE2_R0 0x0 > > @@ -243,6 +252,40 @@ int twl4030_i2c_read(u8 mod_no, u8 *value, u8 reg, unsigned num_bytes); > #define RES_STATE_SLEEP 0x8 > #define RES_STATE_OFF 0x0 > > +/* Power resources */ > + > +/* Power providers */ > +#define RES_VAUX1 1 > +#define RES_VAUX2 2 > +#define RES_VAUX3 3 > +#define RES_VAUX4 4 > +#define RES_VMMC1 5 > +#define RES_VMMC2 6 > +#define RES_VPLL1 7 > +#define RES_VPLL2 8 > +#define RES_VSIM 9 > +#define RES_VDAC 10 > +#define RES_VINTANA1 11 > +#define RES_VINTANA2 12 > +#define RES_VINTDIG 13 > +#define RES_VIO 14 > +#define RES_VDD1 15 > +#define RES_VDD2 16 > +#define RES_VUSB_1V5 17 > +#define RES_VUSB_1V8 18 > +#define RES_VUSB_3V1 19 > +#define RES_VUSBCP 20 > +#define RES_REGEN 21 > +/* Reset and control */ > +#define RES_NRES_PWRON 22 > +#define RES_CLKEN 23 > +#define RES_SYSEN 24 > +#define RES_HFCLKOUT 25 > +#define RES_32KCLKOUT 26 > +#define RES_RESET 27 > +/* Power Reference */ > +#define RES_Main_Ref 28 > + > /* > * Power Bus Message Format ... these can be sent individually by Linux, > * but are usually part of downloaded scripts that are run when various > @@ -320,6 +363,34 @@ struct twl4030_usb_data { > enum twl4030_usb_mode usb_mode; > }; > > +struct twl4030_ins { > + u16 pmb_message; > + u8 delay; > +}; > + > +struct twl4030_script { > + struct twl4030_ins *script; > + unsigned size; > + u8 flags; > +#define TRITON_WRST_SCRIPT (1<<0) > +#define TRITON_WAKEUP12_SCRIPT (1<<1) > +#define TRITON_WAKEUP3_SCRIPT (1<<2) > +#define TRITON_SLEEP_SCRIPT (1<<3) > +}; > + > +struct twl4030_resconfig { > + u8 resource; > + u8 devgroup; > + u8 type; > + u8 type2; type and type2 are not really self explanative. We could have better names or at least some comments here. > +}; > + > +struct twl4030_power_data { > + struct twl4030_script **scripts; > + unsigned size; > + struct twl4030_resconfig *resource_config; > +}; > + > struct twl4030_platform_data { > unsigned irq_base, irq_end; > struct twl4030_bci_platform_data *bci; > @@ -327,6 +398,7 @@ struct twl4030_platform_data { > struct twl4030_madc_platform_data *madc; > struct twl4030_keypad_data *keypad; > struct twl4030_usb_data *usb; > + struct twl4030_power_data *power; > > /* LDO regulators */ > struct regulator_init_data *vdac; > @@ -357,7 +429,6 @@ int twl4030_sih_setup(int module); > #define TWL4030_VAUX3_DEV_GRP 0x1F > #define TWL4030_VAUX3_DEDICATED 0x22 > > - > #if defined(CONFIG_TWL4030_BCI_BATTERY) || \ > defined(CONFIG_TWL4030_BCI_BATTERY_MODULE) > extern int twl4030charger_usb_en(int enable); > -- > 1.6.3.3 > -- Intel Open Source Technology Centre http://oss.intel.com/ ^ permalink raw reply [flat|nested] 11+ messages in thread
* Re: [PATCH 1/3] MFD: TWL4030: Add support for TWL4030/5030 dynamic power switching 2009-08-04 17:55 ` Samuel Ortiz @ 2009-08-17 11:42 ` Amit Kucheria 0 siblings, 0 replies; 11+ messages in thread From: Amit Kucheria @ 2009-08-17 11:42 UTC (permalink / raw) To: Samuel Ortiz; +Cc: dbrownell, linux-kernel, linux-omap Hi Samuel, On 09 Aug 04, Samuel Ortiz wrote: > Hi Amit, > > My apologies for the late answer, but I'm going through my mail backlog after > a 3 weeks holidays. And you replied just when I was away :) Replies inline. Refreshed patchset follows in a new thread. > On Wed, Jul 08, 2009 at 01:49:27PM +0300, Amit Kucheria wrote: > > The TWL4030/5030 family of multifunction devices allows board-specific > > control of the the various regulators, clock and reset lines through > > 'scripts' that are loaded into its memory. This allows for Dynamic Power > > Switching (DPS). > > > > Implement board-independent core support for DPS that is then used by > > board-specific code to load custom DPS scripts. > > > > Signed-off-by: Amit Kucheria <amit.kucheria@verdurent.com> > Some comments here: > > > --- > > drivers/mfd/Kconfig | 13 ++ > > drivers/mfd/Makefile | 1 + > > drivers/mfd/twl4030-core.c | 12 ++ > > drivers/mfd/twl4030-power.c | 384 +++++++++++++++++++++++++++++++++++++++++++ > > include/linux/i2c/twl4030.h | 91 +++++++++- > > 5 files changed, 491 insertions(+), 10 deletions(-) > > create mode 100644 drivers/mfd/twl4030-power.c > > > > diff --git a/drivers/mfd/Kconfig b/drivers/mfd/Kconfig > > index 491ac0f..94fa9a0 100644 > > --- a/drivers/mfd/Kconfig > > +++ b/drivers/mfd/Kconfig > > @@ -108,6 +108,19 @@ config TWL4030_CORE > > high speed USB OTG transceiver, an audio codec (on most > > versions) and many other features. > > > > +config TWL4030_POWER > > + bool "Support power resources on TWL4030 family chips" > > + depends on TWL4030_CORE > > + help > > + Say yes here if you want to use the power resources on the > > + TWL4030 family chips. Most of these resources are regulators, > > + which have a separate driver; some are control signals, such > > + as clock request handshaking. > > + > > + This driver uses board-specific data to initialize the resources > > + and load scripts controling which resources are switched off/on > > + or reset when a sleep, wakeup or warm reset event occurs. > > + > > config MFD_TMIO > > bool > > default n > > diff --git a/drivers/mfd/Makefile b/drivers/mfd/Makefile > > index 6f8a9a1..84b9eda 100644 > > --- a/drivers/mfd/Makefile > > +++ b/drivers/mfd/Makefile > > @@ -23,6 +23,7 @@ obj-$(CONFIG_TPS65010) += tps65010.o > > obj-$(CONFIG_MENELAUS) += menelaus.o > > > > obj-$(CONFIG_TWL4030_CORE) += twl4030-core.o twl4030-irq.o > > +obj-$(CONFIG_TWL4030_POWER) += twl4030-power.o > > > > obj-$(CONFIG_MFD_CORE) += mfd-core.o > > > > diff --git a/drivers/mfd/twl4030-core.c b/drivers/mfd/twl4030-core.c > > index ca54996..ee8ede8 100644 > > --- a/drivers/mfd/twl4030-core.c > > +++ b/drivers/mfd/twl4030-core.c > > @@ -89,6 +89,12 @@ > > #define twl_has_madc() false > > #endif > > > > +#ifdef CONFIG_TWL4030_POWER > > +#define twl_has_power() true > > +#else > > +#define twl_has_power() false > > +#endif > > + > > #if defined(CONFIG_RTC_DRV_TWL4030) || defined(CONFIG_RTC_DRV_TWL4030_MODULE) > > #define twl_has_rtc() true > > #else > > @@ -231,6 +237,8 @@ static struct twl4030mapping twl4030_map[TWL4030_MODULE_LAST + 1] = { > > { 3, TWL4030_BASEADD_SECURED_REG }, > > }; > > > > +extern void twl4030_power_init(struct twl4030_power_data *triton2_scripts); > Maybe this one should be defined in twl4030.h. Done. > > > /*----------------------------------------------------------------------*/ > > > > /* Exported Functions */ > > @@ -788,6 +796,10 @@ twl4030_probe(struct i2c_client *client, const struct i2c_device_id *id) > > /* setup clock framework */ > > clocks_init(&client->dev); > > > > + /* load power event scripts */ > > + if (twl_has_power() && pdata->power) > > + twl4030_power_init(pdata->power); > > + > > /* Maybe init the T2 Interrupt subsystem */ > > if (client->irq > > && pdata->irq_base > > diff --git a/drivers/mfd/twl4030-power.c b/drivers/mfd/twl4030-power.c > > new file mode 100644 > > index 0000000..bb9e45f > > --- /dev/null > > +++ b/drivers/mfd/twl4030-power.c > > @@ -0,0 +1,384 @@ > > +/* > > + * linux/drivers/i2c/chips/twl4030-power.c > > + * > > + * Handle TWL4030 Power initialization > > + * > > + * Copyright (C) 2008 Nokia Corporation > > + * Copyright (C) 2006 Texas Instruments, Inc > > + * > > + * Written by Kalle Jokiniemi > > + * Peter De Schrijver <peter.de-schrijver@nokia.com> > > + * Several fixes by Amit Kucheria <amit.kucheria@verdurent.com> > > + * > > + * This file is subject to the terms and conditions of the GNU General > > + * Public License. See the file "COPYING" in the main directory of this > > + * archive for more details. > > + * > > + * This program is distributed in the hope that it will be useful, > > + * but WITHOUT ANY WARRANTY; without even the implied warranty of > > + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the > > + * GNU General Public License for more details. > > + * > > + * You should have received a copy of the GNU General Public License > > + * along with this program; if not, write to the Free Software > > + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA > > + */ > > + > > +#include <linux/module.h> > > +#include <linux/pm.h> > > +#include <linux/i2c/twl4030.h> > > +#include <linux/platform_device.h> > > + > > +#include <asm/mach-types.h> > > + > > +static u8 triton_next_free_address = 0x2b; > Are triton and twl4030 the same thing ? If that's so could we use one and only > one prefix in this code (which would be twl4030) ? I'm nitpicking here, but > this is kind of confusing unless triton is only a subset of the 4030 family. Done. > > > +#define PWR_P1_SW_EVENTS 0x10 > > +#define PWR_DEVOFF (1<<0) > > + > > +#define PHY_TO_OFF_PM_MASTER(p) (p - 0x36) > > +#define PHY_TO_OFF_PM_RECEIVER(p) (p - 0x5b) > > + > > +#define NUM_OF_RESOURCES 28 > > + > > +/* resource - hfclk */ > > +#define R_HFCLKOUT_DEV_GRP PHY_TO_OFF_PM_RECEIVER(0xe6) > > + > > +/* PM events */ > > +#define R_P1_SW_EVENTS PHY_TO_OFF_PM_MASTER(0x46) > > +#define R_P2_SW_EVENTS PHY_TO_OFF_PM_MASTER(0x47) > > +#define R_P3_SW_EVENTS PHY_TO_OFF_PM_MASTER(0x48) > > +#define R_CFG_P1_TRANSITION PHY_TO_OFF_PM_MASTER(0x36) > > +#define R_CFG_P2_TRANSITION PHY_TO_OFF_PM_MASTER(0x37) > > +#define R_CFG_P3_TRANSITION PHY_TO_OFF_PM_MASTER(0x38) > > + > > +#define LVL_WAKEUP 0x08 > > + > > +#define ENABLE_WARMRESET (1<<4) > > + > > +#define END_OF_SCRIPT 0x3f > > + > > +#define R_SEQ_ADD_A2S PHY_TO_OFF_PM_MASTER(0x55) > > +#define R_SEQ_ADD_S2A12 PHY_TO_OFF_PM_MASTER(0x56) > > +#define R_SEQ_ADD_S2A3 PHY_TO_OFF_PM_MASTER(0x57) > > +#define R_SEQ_ADD_WARM PHY_TO_OFF_PM_MASTER(0x58) > > +#define R_MEMORY_ADDRESS PHY_TO_OFF_PM_MASTER(0x59) > > +#define R_MEMORY_DATA PHY_TO_OFF_PM_MASTER(0x5a) > > + > > +#define R_PROTECT_KEY 0x0E > > +#define KEY_1 0xC0 > > +#define KEY_2 0x0C > > + > > +/* resource configuration registers */ > > + > > +#define DEVGROUP_OFFSET 0 > > +#define TYPE_OFFSET 1 > > + > > +static u8 res_config_addrs[] = { > > + [RES_VAUX1] = 0x17, > > + [RES_VAUX2] = 0x1b, > > + [RES_VAUX3] = 0x1f, > > + [RES_VAUX4] = 0x23, > > + [RES_VMMC1] = 0x27, > > + [RES_VMMC2] = 0x2b, > > + [RES_VPLL1] = 0x2f, > > + [RES_VPLL2] = 0x33, > > + [RES_VSIM] = 0x37, > > + [RES_VDAC] = 0x3b, > > + [RES_VINTANA1] = 0x3f, > > + [RES_VINTANA2] = 0x43, > > + [RES_VINTDIG] = 0x47, > > + [RES_VIO] = 0x4b, > > + [RES_VDD1] = 0x55, > > + [RES_VDD2] = 0x63, > > + [RES_VUSB_1V5] = 0x71, > > + [RES_VUSB_1V8] = 0x74, > > + [RES_VUSB_3V1] = 0x77, > > + [RES_VUSBCP] = 0x7a, > > + [RES_REGEN] = 0x7f, > > + [RES_NRES_PWRON] = 0x82, > > + [RES_CLKEN] = 0x85, > > + [RES_SYSEN] = 0x88, > > + [RES_HFCLKOUT] = 0x8b, > > + [RES_32KCLKOUT] = 0x8e, > > + [RES_RESET] = 0x91, > > + [RES_Main_Ref] = 0x94, > > +}; > > + > > +static int __init twl4030_write_script_byte(u8 address, u8 byte) > > +{ > > + int err; > > + > > + err = twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, > > + R_MEMORY_ADDRESS); > > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, byte, > > + R_MEMORY_DATA); > In most of this piece of code, you're ORing the return code, and that seems > wrong to me. First of all, do you really want to continue reading or writing > after the first error ? Then, your ORed err may eventually be meaningless. > I'd prefer to always check for err and return in the error case. > There are 30 calls to i2c_{read/write} in twl4030_power.c alone. We were trying a shortcut. Hopefully something like the following is acceptable: err = twl4030_i2c_write_u8(...); if (err) goto out; err = twl4030_i2c_read_u8(...); if (err) goto out out: return err; > > + return err; > > +} > > + > > +static int __init twl4030_write_script_ins(u8 address, u16 pmb_message, > > + u8 delay, u8 next) > > +{ > > + int err = 0; > > + > > + address *= 4; > > + err |= twl4030_write_script_byte(address++, pmb_message >> 8); > > + err |= twl4030_write_script_byte(address++, pmb_message & 0xff); > > + err |= twl4030_write_script_byte(address++, delay); > > + err |= twl4030_write_script_byte(address++, next); > > + > > + return err; > > +} > > + > > +static int __init twl4030_write_script(u8 address, struct twl4030_ins *script, > > + int len) > > +{ > > + int err = 0; > > + > > + for (; len; len--, address++, script++) { > > + if (len == 1) > > + err |= twl4030_write_script_ins(address, > > + script->pmb_message, > > + script->delay, > > + END_OF_SCRIPT); > > + else > > + err |= twl4030_write_script_ins(address, > > + script->pmb_message, > > + script->delay, > > + address + 1); > > + } > > + > > + return err; > > +} > > + > > +static int __init config_wakeup3_sequence(u8 address) > Some of your static routines are not twl4030_ prefixed, and some are. I guess > we could be more consistent here. Fixed > > +{ > > + int err = 0; > > + > > + /* Set SLEEP to ACTIVE SEQ address for P3 */ > > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, > > + R_SEQ_ADD_S2A3); > > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, LVL_WAKEUP, > > + R_P3_SW_EVENTS); > > + if (err) > > + printk(KERN_ERR "TWL4030 wakeup sequence for P3" \ > > + "config error\n"); > The rest of the twl4030 code uses pr_err(), so you probably should use it too. Fixed > > + return err; > > +} > > + > > +static int __init config_wakeup12_sequence(u8 address) > > +{ > > + int err = 0; > > + u8 data; > > + > > + /* Set SLEEP to ACTIVE SEQ address for P1 and P2 */ > > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, > > + R_SEQ_ADD_S2A12); > > + > > + /* P1/P2 LVL_WAKEUP should be on LEVEL */ > > + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &data, > > + R_P1_SW_EVENTS); > > + data |= LVL_WAKEUP; > > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, data, > > + R_P1_SW_EVENTS); > > + > > + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &data, > > + R_P2_SW_EVENTS); > > + data |= LVL_WAKEUP; > > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, data, > > + R_P2_SW_EVENTS); > > + > > + if (machine_is_omap_3430sdp() || machine_is_omap_ldp()) { > > + /* Disabling AC charger effect on sleep-active transitions */ > > + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &data, > > + R_CFG_P1_TRANSITION); > > + data &= ~(1<<1); > > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, data , > > + R_CFG_P1_TRANSITION); > > + } > > + > > + if (err) > > + printk(KERN_ERR "TWL4030 wakeup sequence for P1 and P2" \ > > + "config error\n"); > > + > > + return err; > > +} > > + > > +static int __init config_sleep_sequence(u8 address) > > +{ > > + int err = 0; > > + > > + /* Set ACTIVE to SLEEP SEQ address in T2 memory*/ > > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, > > + R_SEQ_ADD_A2S); > > + > > + if (err) > > + printk(KERN_ERR "TWL4030 sleep sequence config error\n"); > > + > > + return err; > > +} > > + > > +static int __init config_warmreset_sequence(u8 address) > > +{ > > + int err = 0; > > + u8 rd_data; > > + > > + /* Set WARM RESET SEQ address for P1 */ > > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, address, > > + R_SEQ_ADD_WARM); > > + > > + /* P1/P2/P3 enable WARMRESET */ > > + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &rd_data, > > + R_P1_SW_EVENTS); > > + rd_data |= ENABLE_WARMRESET; > > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, rd_data, > > + R_P1_SW_EVENTS); > > + > > + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &rd_data, > > + R_P2_SW_EVENTS); > > + rd_data |= ENABLE_WARMRESET; > > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, rd_data, > > + R_P2_SW_EVENTS); > > + > > + err |= twl4030_i2c_read_u8(TWL4030_MODULE_PM_MASTER, &rd_data, > > + R_P3_SW_EVENTS); > > + rd_data |= ENABLE_WARMRESET; > > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, rd_data, > > + R_P3_SW_EVENTS); > > + > > + if (err) > > + printk(KERN_ERR > > + "TWL4030 warmreset seq config error\n"); > > + return err; > > +} > > + > > +static int __init twl4030_configure_resource(struct twl4030_resconfig *rconfig) > > +{ > > + int rconfig_addr; > > + int err; > > + u8 type; > > + > > + if (rconfig->resource > NUM_OF_RESOURCES) { > > + printk(KERN_ERR > > + "TWL4030 Resource %d does not exist\n", > > + rconfig->resource); > > + return -EINVAL; > > + } > > + > > + rconfig_addr = res_config_addrs[rconfig->resource]; > > + > > + /* Set resource group */ > > + if (rconfig->devgroup >= 0) > > + err = twl4030_i2c_write_u8(TWL4030_MODULE_PM_RECEIVER, > > + rconfig->devgroup << 5, > > + rconfig_addr + DEVGROUP_OFFSET); > > + if (err < 0) { > > + printk(KERN_ERR > > + "TWL4030 failed to program devgroup"); > > + return err; > > + } > > + > > + /* Set resource types */ > > + err = twl4030_i2c_read_u8(TWL4030_MODULE_PM_RECEIVER, &type, > > + rconfig_addr + TYPE_OFFSET); > > + if (err < 0) { > > + printk(KERN_ERR > > + "TWL4030 Resource %d type could not be read\n", > > + rconfig->resource); > > + return err; > > + } > > + > > + if (rconfig->type >= 0) { > > + type &= ~7; > Could we have some definition for this 7 ? Fixed > > > + type |= rconfig->type; > > + } > > + > > + if (rconfig->type2 >= 0) { > > + type &= ~(3 << 3); > Same here. Fixed > > > + type |= rconfig->type2 << 3; > > + } > > + > > + err = twl4030_i2c_write_u8(TWL4030_MODULE_PM_RECEIVER, > > + type, rconfig_addr + TYPE_OFFSET); > > + if (err < 0) { > > + printk(KERN_ERR > > + "TWL4030 failed to program resource type"); > > + return err; > > + } > > + > > + return 0; > > +} > > + > > +static int __init load_triton_script(struct twl4030_script *tscript) > > +{ > > + u8 address = triton_next_free_address; > > + int err; > Here I'd prefer to have address as a function argument, and then set it first > to 0x2b from the load_triton_script() loop in power_init. That would be one > way to get rid of this static triton_next_free_address variable which I dont > like too much. Fixed > > + /* Make sure the script isn't going beyond last valid address */ > > + if ((address + tscript->size) > (END_OF_SCRIPT-1)) { > > + printk(KERN_ERR "TWL4030 script too big error\n"); > > + return -EINVAL; > > + } > > + > > + err = twl4030_write_script(address, tscript->script, tscript->size); > > + if (err) > > + return err; > > + > > + triton_next_free_address += tscript->size; > > + > > + if (tscript->flags & TRITON_WRST_SCRIPT) > > + err |= config_warmreset_sequence(address); > > + > > + if (tscript->flags & TRITON_WAKEUP12_SCRIPT) > > + err |= config_wakeup12_sequence(address); > > + > > + if (tscript->flags & TRITON_WAKEUP3_SCRIPT) > > + err |= config_wakeup3_sequence(address); > > + > > + if (tscript->flags & TRITON_SLEEP_SCRIPT) > > + err |= config_sleep_sequence(address); > > + > > + return err; > > +} > > + > > +void __init twl4030_power_init(struct twl4030_power_data *triton2_scripts) > > +{ > > + int err = 0; > > + int i; > > + struct twl4030_resconfig *resconfig; > > + > > + err = twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, KEY_1, > > + R_PROTECT_KEY); > > + err |= twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, KEY_2, > > + R_PROTECT_KEY); > > + if (err) > > + printk(KERN_ERR > > + "TWL4030 Unable to unlock registers\n"); > > + > > + for (i = 0; i < triton2_scripts->size; i++) { > > + err = load_triton_script(triton2_scripts->scripts[i]); > > + if (err < 0) { > > + printk(KERN_ERR "TWL4030 failed to load scripts"); > > + break; > > + } > > + } > > + > > + resconfig = triton2_scripts->resource_config; > > + if (resconfig) { > > + while (resconfig->resource) { > > + err = twl4030_configure_resource(resconfig); > > + resconfig++; > > + if (err < 0) { > > + printk(KERN_ERR > > + "TWL4030 failed to configure resource"); > > + break; > > + } > > + } > > + } > > + > > + if (twl4030_i2c_write_u8(TWL4030_MODULE_PM_MASTER, 0, R_PROTECT_KEY)) > > + printk(KERN_ERR > > + "TWL4030 Unable to relock registers\n"); > > +} > > diff --git a/include/linux/i2c/twl4030.h b/include/linux/i2c/twl4030.h > > index 0dc80ef..6f59bad 100644 > > --- a/include/linux/i2c/twl4030.h > > +++ b/include/linux/i2c/twl4030.h > > @@ -220,19 +220,28 @@ int twl4030_i2c_read(u8 mod_no, u8 *value, u8 reg, unsigned num_bytes); > > > > /* Power bus message definitions */ > > > > -#define DEV_GRP_NULL 0x0 > > -#define DEV_GRP_P1 0x1 > > -#define DEV_GRP_P2 0x2 > > -#define DEV_GRP_P3 0x4 > > +/* The TWL4030/5030 splits its power-management resources (the various > > + * regulators, clock and reset lines) into 3 processor groups - P1, P2 and > > + * P3. These groups can then be configured to transition between sleep, wait-on > > + * and active states by sending messages to the power bus. See Section 5.4.2 > > + * Power Resources of TWL4030 TRM > > + */ > > > > -#define RES_GRP_RES 0x0 > > -#define RES_GRP_PP 0x1 > > -#define RES_GRP_RC 0x2 > > +/* Processor groups */ > > +#define DEV_GRP_NULL 0x0 > > +#define DEV_GRP_P1 0x1 /* P1: all OMAP devices */ > > +#define DEV_GRP_P2 0x2 /* P2: all Modem devices */ > > +#define DEV_GRP_P3 0x4 /* P3: all peripheral devices */ > > + > > +/* Resource groups */ > > +#define RES_GRP_RES 0x0 /* Reserved */ > > +#define RES_GRP_PP 0x1 /* Power providers */ > > +#define RES_GRP_RC 0x2 /* Reset and control */ > > #define RES_GRP_PP_RC 0x3 > > -#define RES_GRP_PR 0x4 > > +#define RES_GRP_PR 0x4 /* Power references */ > > #define RES_GRP_PP_PR 0x5 > > #define RES_GRP_RC_PR 0x6 > > -#define RES_GRP_ALL 0x7 > > +#define RES_GRP_ALL 0x7 /* All resource groups */ > > > > #define RES_TYPE2_R0 0x0 > > > > @@ -243,6 +252,40 @@ int twl4030_i2c_read(u8 mod_no, u8 *value, u8 reg, unsigned num_bytes); > > #define RES_STATE_SLEEP 0x8 > > #define RES_STATE_OFF 0x0 > > > > +/* Power resources */ > > + > > +/* Power providers */ > > +#define RES_VAUX1 1 > > +#define RES_VAUX2 2 > > +#define RES_VAUX3 3 > > +#define RES_VAUX4 4 > > +#define RES_VMMC1 5 > > +#define RES_VMMC2 6 > > +#define RES_VPLL1 7 > > +#define RES_VPLL2 8 > > +#define RES_VSIM 9 > > +#define RES_VDAC 10 > > +#define RES_VINTANA1 11 > > +#define RES_VINTANA2 12 > > +#define RES_VINTDIG 13 > > +#define RES_VIO 14 > > +#define RES_VDD1 15 > > +#define RES_VDD2 16 > > +#define RES_VUSB_1V5 17 > > +#define RES_VUSB_1V8 18 > > +#define RES_VUSB_3V1 19 > > +#define RES_VUSBCP 20 > > +#define RES_REGEN 21 > > +/* Reset and control */ > > +#define RES_NRES_PWRON 22 > > +#define RES_CLKEN 23 > > +#define RES_SYSEN 24 > > +#define RES_HFCLKOUT 25 > > +#define RES_32KCLKOUT 26 > > +#define RES_RESET 27 > > +/* Power Reference */ > > +#define RES_Main_Ref 28 > > + > > /* > > * Power Bus Message Format ... these can be sent individually by Linux, > > * but are usually part of downloaded scripts that are run when various > > @@ -320,6 +363,34 @@ struct twl4030_usb_data { > > enum twl4030_usb_mode usb_mode; > > }; > > > > +struct twl4030_ins { > > + u16 pmb_message; > > + u8 delay; > > +}; > > + > > +struct twl4030_script { > > + struct twl4030_ins *script; > > + unsigned size; > > + u8 flags; > > +#define TRITON_WRST_SCRIPT (1<<0) > > +#define TRITON_WAKEUP12_SCRIPT (1<<1) > > +#define TRITON_WAKEUP3_SCRIPT (1<<2) > > +#define TRITON_SLEEP_SCRIPT (1<<3) > > +}; > > + > > +struct twl4030_resconfig { > > + u8 resource; > > + u8 devgroup; > > + u8 type; > > + u8 type2; > type and type2 are not really self explanative. We could have better names or > at least some comments here. These are the names the reference manual uses. :) I've added some comments. > > > +}; > > + > > +struct twl4030_power_data { > > + struct twl4030_script **scripts; > > + unsigned size; > > + struct twl4030_resconfig *resource_config; > > +}; > > + > > struct twl4030_platform_data { > > unsigned irq_base, irq_end; > > struct twl4030_bci_platform_data *bci; > > @@ -327,6 +398,7 @@ struct twl4030_platform_data { > > struct twl4030_madc_platform_data *madc; > > struct twl4030_keypad_data *keypad; > > struct twl4030_usb_data *usb; > > + struct twl4030_power_data *power; > > > > /* LDO regulators */ > > struct regulator_init_data *vdac; > > @@ -357,7 +429,6 @@ int twl4030_sih_setup(int module); > > #define TWL4030_VAUX3_DEV_GRP 0x1F > > #define TWL4030_VAUX3_DEDICATED 0x22 > > > > - > > #if defined(CONFIG_TWL4030_BCI_BATTERY) || \ > > defined(CONFIG_TWL4030_BCI_BATTERY_MODULE) > > extern int twl4030charger_usb_en(int enable); > > -- > > 1.6.3.3 > > > > -- > Intel Open Source Technology Centre > http://oss.intel.com/ -- ------------------------------------------------------------------------- Amit Kucheria, Kernel Developer, Verdurent ------------------------------------------------------------------------- ^ permalink raw reply [flat|nested] 11+ messages in thread
* [PATCH 2/3] MFD: TWL4030: print warning for out-of-order script loading [not found] <1247050046-31104-1-git-send-email-./0000-cover-letter.patch> 2009-07-08 10:49 ` [PATCH 1/3] MFD: TWL4030: Add support for TWL4030/5030 dynamic power switching Amit Kucheria @ 2009-07-08 10:49 ` Amit Kucheria 2009-08-04 18:00 ` Samuel Ortiz 2009-07-08 10:49 ` [PATCH 3/3] MFD: TWL4030: OMAP: Board-specifc DPS scripts for RX51 board Amit Kucheria 2 siblings, 1 reply; 11+ messages in thread From: Amit Kucheria @ 2009-07-08 10:49 UTC (permalink / raw) To: sameo; +Cc: dbrownell, linux-kernel, linux-omap When the sleep script is loaded before the wakeup script, there is a chance that the system might go to sleep before the wakeup script loading is completed. This will lead to a system that does not wakeup and has been observed to cause non-booting boards. Various options were considered to solve this problem, including modification of the core twl4030 power code to be smart enough to reorder the loading of the scripts. But it felt too over-engineered. Hence this patch just warns the DPS script developer so that they may be reordered in the board-code itself. Signed-off-by: Amit Kucheria <amit.kucheria@verdurent.com> --- drivers/mfd/twl4030-power.c | 11 +++++++++-- 1 files changed, 9 insertions(+), 2 deletions(-) diff --git a/drivers/mfd/twl4030-power.c b/drivers/mfd/twl4030-power.c index bb9e45f..ef4cc4e 100644 --- a/drivers/mfd/twl4030-power.c +++ b/drivers/mfd/twl4030-power.c @@ -315,6 +315,7 @@ static int __init load_triton_script(struct twl4030_script *tscript) { u8 address = triton_next_free_address; int err; + static u8 mask = 0; /* Make sure the script isn't going beyond last valid address */ if ((address + tscript->size) > (END_OF_SCRIPT-1)) { @@ -331,14 +332,20 @@ static int __init load_triton_script(struct twl4030_script *tscript) if (tscript->flags & TRITON_WRST_SCRIPT) err |= config_warmreset_sequence(address); - if (tscript->flags & TRITON_WAKEUP12_SCRIPT) + if (tscript->flags & TRITON_WAKEUP12_SCRIPT) { err |= config_wakeup12_sequence(address); + mask |= TRITON_WAKEUP12_SCRIPT; + } if (tscript->flags & TRITON_WAKEUP3_SCRIPT) err |= config_wakeup3_sequence(address); - if (tscript->flags & TRITON_SLEEP_SCRIPT) + if (tscript->flags & TRITON_SLEEP_SCRIPT) { + if (!(mask & TRITON_WAKEUP12_SCRIPT)) + printk(KERN_WARNING + "TWL4030: Wakeup script not yet loaded. Might lead to boot failure on some boards\n"); err |= config_sleep_sequence(address); + } return err; } -- 1.6.3.3 ^ permalink raw reply related [flat|nested] 11+ messages in thread
* Re: [PATCH 2/3] MFD: TWL4030: print warning for out-of-order script loading 2009-07-08 10:49 ` [PATCH 2/3] MFD: TWL4030: print warning for out-of-order script loading Amit Kucheria @ 2009-08-04 18:00 ` Samuel Ortiz 2009-08-17 11:43 ` Amit Kucheria 0 siblings, 1 reply; 11+ messages in thread From: Samuel Ortiz @ 2009-08-04 18:00 UTC (permalink / raw) To: Amit Kucheria; +Cc: dbrownell, linux-kernel, linux-omap Hi Amit, On Wed, Jul 08, 2009 at 01:49:35PM +0300, Amit Kucheria wrote: > When the sleep script is loaded before the wakeup script, there is a > chance that the system might go to sleep before the wakeup script > loading is completed. This will lead to a system that does not wakeup > and has been observed to cause non-booting boards. > > Various options were considered to solve this problem, including > modification of the core twl4030 power code to be smart enough to > reorder the loading of the scripts. But it felt too over-engineered. > > Hence this patch just warns the DPS script developer so that they may be > reordered in the board-code itself. > > Signed-off-by: Amit Kucheria <amit.kucheria@verdurent.com> > --- > drivers/mfd/twl4030-power.c | 11 +++++++++-- > 1 files changed, 9 insertions(+), 2 deletions(-) > > diff --git a/drivers/mfd/twl4030-power.c b/drivers/mfd/twl4030-power.c > index bb9e45f..ef4cc4e 100644 > --- a/drivers/mfd/twl4030-power.c > +++ b/drivers/mfd/twl4030-power.c > @@ -315,6 +315,7 @@ static int __init load_triton_script(struct twl4030_script *tscript) > { > u8 address = triton_next_free_address; > int err; > + static u8 mask = 0; > > /* Make sure the script isn't going beyond last valid address */ > if ((address + tscript->size) > (END_OF_SCRIPT-1)) { > @@ -331,14 +332,20 @@ static int __init load_triton_script(struct twl4030_script *tscript) > if (tscript->flags & TRITON_WRST_SCRIPT) > err |= config_warmreset_sequence(address); > > - if (tscript->flags & TRITON_WAKEUP12_SCRIPT) > + if (tscript->flags & TRITON_WAKEUP12_SCRIPT) { > err |= config_wakeup12_sequence(address); > + mask |= TRITON_WAKEUP12_SCRIPT; > + } > > if (tscript->flags & TRITON_WAKEUP3_SCRIPT) > err |= config_wakeup3_sequence(address); > > - if (tscript->flags & TRITON_SLEEP_SCRIPT) > + if (tscript->flags & TRITON_SLEEP_SCRIPT) { Wouldnt something like: if (tscript->flags & (TRITON_SLEEP_SCRIPT | TRITON_WAKEUP12_SCRIPT)) { save us from using an additional mask variable and simplify this patch ? Cheers, Samuel. > + if (!(mask & TRITON_WAKEUP12_SCRIPT)) > + printk(KERN_WARNING > + "TWL4030: Wakeup script not yet loaded. Might lead to boot failure on some boards\n"); > err |= config_sleep_sequence(address); > + } > > return err; > } > -- > 1.6.3.3 > -- Intel Open Source Technology Centre http://oss.intel.com/ ^ permalink raw reply [flat|nested] 11+ messages in thread
* Re: [PATCH 2/3] MFD: TWL4030: print warning for out-of-order script loading 2009-08-04 18:00 ` Samuel Ortiz @ 2009-08-17 11:43 ` Amit Kucheria 0 siblings, 0 replies; 11+ messages in thread From: Amit Kucheria @ 2009-08-17 11:43 UTC (permalink / raw) To: Samuel Ortiz; +Cc: dbrownell, linux-kernel, linux-omap On 09 Aug 04, Samuel Ortiz wrote: > Hi Amit, > > On Wed, Jul 08, 2009 at 01:49:35PM +0300, Amit Kucheria wrote: > > When the sleep script is loaded before the wakeup script, there is a > > chance that the system might go to sleep before the wakeup script > > loading is completed. This will lead to a system that does not wakeup > > and has been observed to cause non-booting boards. > > > > Various options were considered to solve this problem, including > > modification of the core twl4030 power code to be smart enough to > > reorder the loading of the scripts. But it felt too over-engineered. > > > > Hence this patch just warns the DPS script developer so that they may be > > reordered in the board-code itself. > > > > Signed-off-by: Amit Kucheria <amit.kucheria@verdurent.com> > > --- > > drivers/mfd/twl4030-power.c | 11 +++++++++-- > > 1 files changed, 9 insertions(+), 2 deletions(-) > > > > diff --git a/drivers/mfd/twl4030-power.c b/drivers/mfd/twl4030-power.c > > index bb9e45f..ef4cc4e 100644 > > --- a/drivers/mfd/twl4030-power.c > > +++ b/drivers/mfd/twl4030-power.c > > @@ -315,6 +315,7 @@ static int __init load_triton_script(struct twl4030_script *tscript) > > { > > u8 address = triton_next_free_address; > > int err; > > + static u8 mask = 0; > > > > /* Make sure the script isn't going beyond last valid address */ > > if ((address + tscript->size) > (END_OF_SCRIPT-1)) { > > @@ -331,14 +332,20 @@ static int __init load_triton_script(struct twl4030_script *tscript) > > if (tscript->flags & TRITON_WRST_SCRIPT) > > err |= config_warmreset_sequence(address); > > > > - if (tscript->flags & TRITON_WAKEUP12_SCRIPT) > > + if (tscript->flags & TRITON_WAKEUP12_SCRIPT) { > > err |= config_wakeup12_sequence(address); > > + mask |= TRITON_WAKEUP12_SCRIPT; > > + } > > > > if (tscript->flags & TRITON_WAKEUP3_SCRIPT) > > err |= config_wakeup3_sequence(address); > > > > - if (tscript->flags & TRITON_SLEEP_SCRIPT) > > + if (tscript->flags & TRITON_SLEEP_SCRIPT) { > Wouldnt something like: > > if (tscript->flags & (TRITON_SLEEP_SCRIPT | TRITON_WAKEUP12_SCRIPT)) { > > save us from using an additional mask variable and simplify this patch ? > I don't see how this will save the mask variable. What I need to detect that the SLEEP script is loaded before the WAKEUP script. That should trigger a warning. -- ------------------------------------------------------------------------- Amit Kucheria, Kernel Developer, Verdurent ------------------------------------------------------------------------- ^ permalink raw reply [flat|nested] 11+ messages in thread
* [PATCH 3/3] MFD: TWL4030: OMAP: Board-specifc DPS scripts for RX51 board [not found] <1247050046-31104-1-git-send-email-./0000-cover-letter.patch> 2009-07-08 10:49 ` [PATCH 1/3] MFD: TWL4030: Add support for TWL4030/5030 dynamic power switching Amit Kucheria 2009-07-08 10:49 ` [PATCH 2/3] MFD: TWL4030: print warning for out-of-order script loading Amit Kucheria @ 2009-07-08 10:49 ` Amit Kucheria 2009-08-04 18:03 ` Samuel Ortiz 2 siblings, 1 reply; 11+ messages in thread From: Amit Kucheria @ 2009-07-08 10:49 UTC (permalink / raw) To: sameo; +Cc: dbrownell, linux-kernel, linux-omap Add support for Dynamic Power Switching (DPS) for the RX51 board. Signed-off-by: Amit Kucheria <amit.kucheria@verdurent.com> --- arch/arm/mach-omap2/board-rx51-peripherals.c | 122 +++++++++++++++++++++++++- 1 files changed, 120 insertions(+), 2 deletions(-) diff --git a/arch/arm/mach-omap2/board-rx51-peripherals.c b/arch/arm/mach-omap2/board-rx51-peripherals.c index 9a0bf67..f7ebec8 100644 --- a/arch/arm/mach-omap2/board-rx51-peripherals.c +++ b/arch/arm/mach-omap2/board-rx51-peripherals.c @@ -1,5 +1,5 @@ /* - * linux/arch/arm/mach-omap2/board-rx51-flash.c + * linux/arch/arm/mach-omap2/board-rx51-peripherals.c * * Copyright (C) 2008-2009 Nokia * @@ -278,7 +278,124 @@ static struct twl4030_gpio_platform_data rx51_gpio_data = { .setup = rx51_twlgpio_setup, }; -static struct twl4030_platform_data rx51_twldata = { +static struct twl4030_ins sleep_on_seq[] __initdata = { +/* + * Turn off VDD1 and VDD2. + */ + {MSG_SINGULAR(DEV_GRP_P1, 0xf, RES_STATE_OFF), 4}, + {MSG_SINGULAR(DEV_GRP_P1, 0x10, RES_STATE_OFF), 2}, +/* + * And also turn off the OMAP3 PLLs and the sysclk output. + */ + {MSG_SINGULAR(DEV_GRP_P1, 0x7, RES_STATE_OFF), 3}, + {MSG_SINGULAR(DEV_GRP_P1, 0x17, RES_STATE_OFF), 3}, +}; + +static struct twl4030_script sleep_on_script __initdata = { + .script = sleep_on_seq, + .size = ARRAY_SIZE(sleep_on_seq), + .flags = TRITON_SLEEP_SCRIPT, +}; + +static struct twl4030_ins wakeup_seq[] __initdata = { +/* + * Reenable the OMAP3 PLLs. + * Wakeup VDD1 and VDD2. + * Reenable sysclk output. + */ + {MSG_SINGULAR(DEV_GRP_P1, 0x7, RES_STATE_ACTIVE), 0x30}, + {MSG_SINGULAR(DEV_GRP_P1, 0xf, RES_STATE_ACTIVE), 0x30}, + {MSG_SINGULAR(DEV_GRP_P1, 0x10, RES_STATE_ACTIVE), 0x37}, + {MSG_SINGULAR(DEV_GRP_P1, 0x19, RES_STATE_ACTIVE), 3}, +}; + +static struct twl4030_script wakeup_script __initdata = { + .script = wakeup_seq, + .size = ARRAY_SIZE(wakeup_seq), + .flags = TRITON_WAKEUP12_SCRIPT, +}; + +static struct twl4030_ins wakeup_p3_seq[] __initdata = { +/* + * Wakeup VDD1 (dummy to be able to insert a delay) + * Enable CLKEN + */ + {MSG_SINGULAR(DEV_GRP_P1, 0x17, RES_STATE_ACTIVE), 3}, +}; + +static struct twl4030_script wakeup_p3_script __initdata = { + .script = wakeup_p3_seq, + .size = ARRAY_SIZE(wakeup_p3_seq), + .flags = TRITON_WAKEUP3_SCRIPT, +}; + +static struct twl4030_ins wrst_seq[] __initdata = { +/* + * Reset twl4030. + * Reset VDD1 regulator. + * Reset VDD2 regulator. + * Reset VPLL1 regulator. + * Enable sysclk output. + * Reenable twl4030. + */ + {MSG_SINGULAR(DEV_GRP_NULL, RES_RESET, RES_STATE_OFF), 2}, + {MSG_BROADCAST(DEV_GRP_NULL, RES_GRP_ALL, 0, 1, RES_STATE_ACTIVE), + 0x13}, + {MSG_BROADCAST(DEV_GRP_NULL, RES_GRP_PP, 0, 2, RES_STATE_WRST), 0x13}, + {MSG_BROADCAST(DEV_GRP_NULL, RES_GRP_PP, 0, 3, RES_STATE_OFF), 0x13}, + {MSG_SINGULAR(DEV_GRP_NULL, RES_VDD1, RES_STATE_WRST), 0x13}, + {MSG_SINGULAR(DEV_GRP_NULL, RES_VDD2, RES_STATE_WRST), 0x13}, + {MSG_SINGULAR(DEV_GRP_NULL, RES_VPLL1, RES_STATE_WRST), 0x35}, + {MSG_SINGULAR(DEV_GRP_P1, RES_HFCLKOUT, RES_STATE_ACTIVE), 2}, + {MSG_SINGULAR(DEV_GRP_NULL, RES_RESET, RES_STATE_ACTIVE), 2}, +}; + +static struct twl4030_script wrst_script __initdata = { + .script = wrst_seq, + .size = ARRAY_SIZE(wrst_seq), + .flags = TRITON_WRST_SCRIPT, +}; + +static struct twl4030_script *twl4030_scripts[] __initdata = { + /* wakeup12 script should be loaded before sleep script, otherwise a + board might hit retention before loading of wakeup script is + completed. This can cause boot failures depending on timing issues. + */ + &wakeup_script, + &sleep_on_script, + &wakeup_p3_script, + &wrst_script, +}; + +static struct twl4030_resconfig twl4030_rconfig[] __initdata = { + { .resource = RES_VINTANA1, .devgroup = -1, .type = -1, .type2 = 1 }, + { .resource = RES_VINTANA2, .devgroup = -1, .type = -1, .type2 = 1 }, + { .resource = RES_VINTDIG, .devgroup = -1, .type = -1, .type2 = 1 }, + { .resource = RES_VMMC1, .devgroup = -1, .type = -1, .type2 = 3}, + { .resource = RES_VMMC2, .devgroup = DEV_GRP_NULL, .type = -1, + .type2 = 3}, + { .resource = RES_VAUX1, .devgroup = -1, .type = -1, .type2 = 3}, + { .resource = RES_VAUX2, .devgroup = -1, .type = -1, .type2 = 3}, + { .resource = RES_VAUX3, .devgroup = -1, .type = -1, .type2 = 3}, + { .resource = RES_VAUX4, .devgroup = -1, .type = -1, .type2 = 3}, + { .resource = RES_VPLL2, .devgroup = -1, .type = -1, .type2 = 3}, + { .resource = RES_VDAC, .devgroup = -1, .type = -1, .type2 = 3}, + { .resource = RES_VSIM, .devgroup = DEV_GRP_NULL, .type = -1, + .type2 = 3}, + { .resource = RES_CLKEN, .devgroup = DEV_GRP_P3, .type = -1, + .type2 = 1 }, + { 0, 0}, +}; + +static struct twl4030_power_data rx51_t2scripts_data __initdata = { + .scripts = twl4030_scripts, + .size = ARRAY_SIZE(twl4030_scripts), + .resource_config = twl4030_rconfig, +}; + + + +static struct twl4030_platform_data rx51_twldata __initdata = { .irq_base = TWL4030_IRQ_BASE, .irq_end = TWL4030_IRQ_END, @@ -286,6 +403,7 @@ static struct twl4030_platform_data rx51_twldata = { .gpio = &rx51_gpio_data, .keypad = &rx51_kp_data, .madc = &rx51_madc_data, + .power = &rx51_t2scripts_data, .vaux1 = &rx51_vaux1, .vaux2 = &rx51_vaux2, -- 1.6.3.3 ^ permalink raw reply related [flat|nested] 11+ messages in thread
* Re: [PATCH 3/3] MFD: TWL4030: OMAP: Board-specifc DPS scripts for RX51 board 2009-07-08 10:49 ` [PATCH 3/3] MFD: TWL4030: OMAP: Board-specifc DPS scripts for RX51 board Amit Kucheria @ 2009-08-04 18:03 ` Samuel Ortiz 2009-08-17 11:46 ` Amit Kucheria 0 siblings, 1 reply; 11+ messages in thread From: Samuel Ortiz @ 2009-08-04 18:03 UTC (permalink / raw) To: Amit Kucheria; +Cc: dbrownell, linux-kernel, linux-omap Hi Amit, On Wed, Jul 08, 2009 at 01:49:39PM +0300, Amit Kucheria wrote: > Add support for Dynamic Power Switching (DPS) for the RX51 board. Could I have an Acked-by from Tony or someone from Nokia maintaining this board (e.g. Lauri) ? Cheers, Samuel. > Signed-off-by: Amit Kucheria <amit.kucheria@verdurent.com> > --- > arch/arm/mach-omap2/board-rx51-peripherals.c | 122 +++++++++++++++++++++++++- > 1 files changed, 120 insertions(+), 2 deletions(-) > > diff --git a/arch/arm/mach-omap2/board-rx51-peripherals.c b/arch/arm/mach-omap2/board-rx51-peripherals.c > index 9a0bf67..f7ebec8 100644 > --- a/arch/arm/mach-omap2/board-rx51-peripherals.c > +++ b/arch/arm/mach-omap2/board-rx51-peripherals.c > @@ -1,5 +1,5 @@ > /* > - * linux/arch/arm/mach-omap2/board-rx51-flash.c > + * linux/arch/arm/mach-omap2/board-rx51-peripherals.c > * > * Copyright (C) 2008-2009 Nokia > * > @@ -278,7 +278,124 @@ static struct twl4030_gpio_platform_data rx51_gpio_data = { > .setup = rx51_twlgpio_setup, > }; > > -static struct twl4030_platform_data rx51_twldata = { > +static struct twl4030_ins sleep_on_seq[] __initdata = { > +/* > + * Turn off VDD1 and VDD2. > + */ > + {MSG_SINGULAR(DEV_GRP_P1, 0xf, RES_STATE_OFF), 4}, > + {MSG_SINGULAR(DEV_GRP_P1, 0x10, RES_STATE_OFF), 2}, > +/* > + * And also turn off the OMAP3 PLLs and the sysclk output. > + */ > + {MSG_SINGULAR(DEV_GRP_P1, 0x7, RES_STATE_OFF), 3}, > + {MSG_SINGULAR(DEV_GRP_P1, 0x17, RES_STATE_OFF), 3}, > +}; > + > +static struct twl4030_script sleep_on_script __initdata = { > + .script = sleep_on_seq, > + .size = ARRAY_SIZE(sleep_on_seq), > + .flags = TRITON_SLEEP_SCRIPT, > +}; > + > +static struct twl4030_ins wakeup_seq[] __initdata = { > +/* > + * Reenable the OMAP3 PLLs. > + * Wakeup VDD1 and VDD2. > + * Reenable sysclk output. > + */ > + {MSG_SINGULAR(DEV_GRP_P1, 0x7, RES_STATE_ACTIVE), 0x30}, > + {MSG_SINGULAR(DEV_GRP_P1, 0xf, RES_STATE_ACTIVE), 0x30}, > + {MSG_SINGULAR(DEV_GRP_P1, 0x10, RES_STATE_ACTIVE), 0x37}, > + {MSG_SINGULAR(DEV_GRP_P1, 0x19, RES_STATE_ACTIVE), 3}, > +}; > + > +static struct twl4030_script wakeup_script __initdata = { > + .script = wakeup_seq, > + .size = ARRAY_SIZE(wakeup_seq), > + .flags = TRITON_WAKEUP12_SCRIPT, > +}; > + > +static struct twl4030_ins wakeup_p3_seq[] __initdata = { > +/* > + * Wakeup VDD1 (dummy to be able to insert a delay) > + * Enable CLKEN > + */ > + {MSG_SINGULAR(DEV_GRP_P1, 0x17, RES_STATE_ACTIVE), 3}, > +}; > + > +static struct twl4030_script wakeup_p3_script __initdata = { > + .script = wakeup_p3_seq, > + .size = ARRAY_SIZE(wakeup_p3_seq), > + .flags = TRITON_WAKEUP3_SCRIPT, > +}; > + > +static struct twl4030_ins wrst_seq[] __initdata = { > +/* > + * Reset twl4030. > + * Reset VDD1 regulator. > + * Reset VDD2 regulator. > + * Reset VPLL1 regulator. > + * Enable sysclk output. > + * Reenable twl4030. > + */ > + {MSG_SINGULAR(DEV_GRP_NULL, RES_RESET, RES_STATE_OFF), 2}, > + {MSG_BROADCAST(DEV_GRP_NULL, RES_GRP_ALL, 0, 1, RES_STATE_ACTIVE), > + 0x13}, > + {MSG_BROADCAST(DEV_GRP_NULL, RES_GRP_PP, 0, 2, RES_STATE_WRST), 0x13}, > + {MSG_BROADCAST(DEV_GRP_NULL, RES_GRP_PP, 0, 3, RES_STATE_OFF), 0x13}, > + {MSG_SINGULAR(DEV_GRP_NULL, RES_VDD1, RES_STATE_WRST), 0x13}, > + {MSG_SINGULAR(DEV_GRP_NULL, RES_VDD2, RES_STATE_WRST), 0x13}, > + {MSG_SINGULAR(DEV_GRP_NULL, RES_VPLL1, RES_STATE_WRST), 0x35}, > + {MSG_SINGULAR(DEV_GRP_P1, RES_HFCLKOUT, RES_STATE_ACTIVE), 2}, > + {MSG_SINGULAR(DEV_GRP_NULL, RES_RESET, RES_STATE_ACTIVE), 2}, > +}; > + > +static struct twl4030_script wrst_script __initdata = { > + .script = wrst_seq, > + .size = ARRAY_SIZE(wrst_seq), > + .flags = TRITON_WRST_SCRIPT, > +}; > + > +static struct twl4030_script *twl4030_scripts[] __initdata = { > + /* wakeup12 script should be loaded before sleep script, otherwise a > + board might hit retention before loading of wakeup script is > + completed. This can cause boot failures depending on timing issues. > + */ > + &wakeup_script, > + &sleep_on_script, > + &wakeup_p3_script, > + &wrst_script, > +}; > + > +static struct twl4030_resconfig twl4030_rconfig[] __initdata = { > + { .resource = RES_VINTANA1, .devgroup = -1, .type = -1, .type2 = 1 }, > + { .resource = RES_VINTANA2, .devgroup = -1, .type = -1, .type2 = 1 }, > + { .resource = RES_VINTDIG, .devgroup = -1, .type = -1, .type2 = 1 }, > + { .resource = RES_VMMC1, .devgroup = -1, .type = -1, .type2 = 3}, > + { .resource = RES_VMMC2, .devgroup = DEV_GRP_NULL, .type = -1, > + .type2 = 3}, > + { .resource = RES_VAUX1, .devgroup = -1, .type = -1, .type2 = 3}, > + { .resource = RES_VAUX2, .devgroup = -1, .type = -1, .type2 = 3}, > + { .resource = RES_VAUX3, .devgroup = -1, .type = -1, .type2 = 3}, > + { .resource = RES_VAUX4, .devgroup = -1, .type = -1, .type2 = 3}, > + { .resource = RES_VPLL2, .devgroup = -1, .type = -1, .type2 = 3}, > + { .resource = RES_VDAC, .devgroup = -1, .type = -1, .type2 = 3}, > + { .resource = RES_VSIM, .devgroup = DEV_GRP_NULL, .type = -1, > + .type2 = 3}, > + { .resource = RES_CLKEN, .devgroup = DEV_GRP_P3, .type = -1, > + .type2 = 1 }, > + { 0, 0}, > +}; > + > +static struct twl4030_power_data rx51_t2scripts_data __initdata = { > + .scripts = twl4030_scripts, > + .size = ARRAY_SIZE(twl4030_scripts), > + .resource_config = twl4030_rconfig, > +}; > + > + > + > +static struct twl4030_platform_data rx51_twldata __initdata = { > .irq_base = TWL4030_IRQ_BASE, > .irq_end = TWL4030_IRQ_END, > > @@ -286,6 +403,7 @@ static struct twl4030_platform_data rx51_twldata = { > .gpio = &rx51_gpio_data, > .keypad = &rx51_kp_data, > .madc = &rx51_madc_data, > + .power = &rx51_t2scripts_data, > > .vaux1 = &rx51_vaux1, > .vaux2 = &rx51_vaux2, > -- > 1.6.3.3 > -- Intel Open Source Technology Centre http://oss.intel.com/ ^ permalink raw reply [flat|nested] 11+ messages in thread
* Re: [PATCH 3/3] MFD: TWL4030: OMAP: Board-specifc DPS scripts for RX51 board 2009-08-04 18:03 ` Samuel Ortiz @ 2009-08-17 11:46 ` Amit Kucheria 2009-08-24 12:33 ` Tony Lindgren 0 siblings, 1 reply; 11+ messages in thread From: Amit Kucheria @ 2009-08-17 11:46 UTC (permalink / raw) To: Samuel Ortiz; +Cc: dbrownell, linux-kernel, linux-omap On 09 Aug 04, Samuel Ortiz wrote: > Hi Amit, > > On Wed, Jul 08, 2009 at 01:49:39PM +0300, Amit Kucheria wrote: > > Add support for Dynamic Power Switching (DPS) for the RX51 board. > Could I have an Acked-by from Tony or someone from Nokia maintaining this > board (e.g. Lauri) ? > I'll cc Tony in the refreshed patches. Please note that this is not the final version of the script. As the scripts are optimised, I'll send in fixes. Regards, Amit > > Signed-off-by: Amit Kucheria <amit.kucheria@verdurent.com> > > --- > > arch/arm/mach-omap2/board-rx51-peripherals.c | 122 +++++++++++++++++++++++++- > > 1 files changed, 120 insertions(+), 2 deletions(-) > > > > diff --git a/arch/arm/mach-omap2/board-rx51-peripherals.c b/arch/arm/mach-omap2/board-rx51-peripherals.c > > index 9a0bf67..f7ebec8 100644 > > --- a/arch/arm/mach-omap2/board-rx51-peripherals.c > > +++ b/arch/arm/mach-omap2/board-rx51-peripherals.c > > @@ -1,5 +1,5 @@ > > /* > > - * linux/arch/arm/mach-omap2/board-rx51-flash.c > > + * linux/arch/arm/mach-omap2/board-rx51-peripherals.c > > * > > * Copyright (C) 2008-2009 Nokia > > * > > @@ -278,7 +278,124 @@ static struct twl4030_gpio_platform_data rx51_gpio_data = { > > .setup = rx51_twlgpio_setup, > > }; > > > > -static struct twl4030_platform_data rx51_twldata = { > > +static struct twl4030_ins sleep_on_seq[] __initdata = { > > +/* > > + * Turn off VDD1 and VDD2. > > + */ > > + {MSG_SINGULAR(DEV_GRP_P1, 0xf, RES_STATE_OFF), 4}, > > + {MSG_SINGULAR(DEV_GRP_P1, 0x10, RES_STATE_OFF), 2}, > > +/* > > + * And also turn off the OMAP3 PLLs and the sysclk output. > > + */ > > + {MSG_SINGULAR(DEV_GRP_P1, 0x7, RES_STATE_OFF), 3}, > > + {MSG_SINGULAR(DEV_GRP_P1, 0x17, RES_STATE_OFF), 3}, > > +}; > > + > > +static struct twl4030_script sleep_on_script __initdata = { > > + .script = sleep_on_seq, > > + .size = ARRAY_SIZE(sleep_on_seq), > > + .flags = TRITON_SLEEP_SCRIPT, > > +}; > > + > > +static struct twl4030_ins wakeup_seq[] __initdata = { > > +/* > > + * Reenable the OMAP3 PLLs. > > + * Wakeup VDD1 and VDD2. > > + * Reenable sysclk output. > > + */ > > + {MSG_SINGULAR(DEV_GRP_P1, 0x7, RES_STATE_ACTIVE), 0x30}, > > + {MSG_SINGULAR(DEV_GRP_P1, 0xf, RES_STATE_ACTIVE), 0x30}, > > + {MSG_SINGULAR(DEV_GRP_P1, 0x10, RES_STATE_ACTIVE), 0x37}, > > + {MSG_SINGULAR(DEV_GRP_P1, 0x19, RES_STATE_ACTIVE), 3}, > > +}; > > + > > +static struct twl4030_script wakeup_script __initdata = { > > + .script = wakeup_seq, > > + .size = ARRAY_SIZE(wakeup_seq), > > + .flags = TRITON_WAKEUP12_SCRIPT, > > +}; > > + > > +static struct twl4030_ins wakeup_p3_seq[] __initdata = { > > +/* > > + * Wakeup VDD1 (dummy to be able to insert a delay) > > + * Enable CLKEN > > + */ > > + {MSG_SINGULAR(DEV_GRP_P1, 0x17, RES_STATE_ACTIVE), 3}, > > +}; > > + > > +static struct twl4030_script wakeup_p3_script __initdata = { > > + .script = wakeup_p3_seq, > > + .size = ARRAY_SIZE(wakeup_p3_seq), > > + .flags = TRITON_WAKEUP3_SCRIPT, > > +}; > > + > > +static struct twl4030_ins wrst_seq[] __initdata = { > > +/* > > + * Reset twl4030. > > + * Reset VDD1 regulator. > > + * Reset VDD2 regulator. > > + * Reset VPLL1 regulator. > > + * Enable sysclk output. > > + * Reenable twl4030. > > + */ > > + {MSG_SINGULAR(DEV_GRP_NULL, RES_RESET, RES_STATE_OFF), 2}, > > + {MSG_BROADCAST(DEV_GRP_NULL, RES_GRP_ALL, 0, 1, RES_STATE_ACTIVE), > > + 0x13}, > > + {MSG_BROADCAST(DEV_GRP_NULL, RES_GRP_PP, 0, 2, RES_STATE_WRST), 0x13}, > > + {MSG_BROADCAST(DEV_GRP_NULL, RES_GRP_PP, 0, 3, RES_STATE_OFF), 0x13}, > > + {MSG_SINGULAR(DEV_GRP_NULL, RES_VDD1, RES_STATE_WRST), 0x13}, > > + {MSG_SINGULAR(DEV_GRP_NULL, RES_VDD2, RES_STATE_WRST), 0x13}, > > + {MSG_SINGULAR(DEV_GRP_NULL, RES_VPLL1, RES_STATE_WRST), 0x35}, > > + {MSG_SINGULAR(DEV_GRP_P1, RES_HFCLKOUT, RES_STATE_ACTIVE), 2}, > > + {MSG_SINGULAR(DEV_GRP_NULL, RES_RESET, RES_STATE_ACTIVE), 2}, > > +}; > > + > > +static struct twl4030_script wrst_script __initdata = { > > + .script = wrst_seq, > > + .size = ARRAY_SIZE(wrst_seq), > > + .flags = TRITON_WRST_SCRIPT, > > +}; > > + > > +static struct twl4030_script *twl4030_scripts[] __initdata = { > > + /* wakeup12 script should be loaded before sleep script, otherwise a > > + board might hit retention before loading of wakeup script is > > + completed. This can cause boot failures depending on timing issues. > > + */ > > + &wakeup_script, > > + &sleep_on_script, > > + &wakeup_p3_script, > > + &wrst_script, > > +}; > > + > > +static struct twl4030_resconfig twl4030_rconfig[] __initdata = { > > + { .resource = RES_VINTANA1, .devgroup = -1, .type = -1, .type2 = 1 }, > > + { .resource = RES_VINTANA2, .devgroup = -1, .type = -1, .type2 = 1 }, > > + { .resource = RES_VINTDIG, .devgroup = -1, .type = -1, .type2 = 1 }, > > + { .resource = RES_VMMC1, .devgroup = -1, .type = -1, .type2 = 3}, > > + { .resource = RES_VMMC2, .devgroup = DEV_GRP_NULL, .type = -1, > > + .type2 = 3}, > > + { .resource = RES_VAUX1, .devgroup = -1, .type = -1, .type2 = 3}, > > + { .resource = RES_VAUX2, .devgroup = -1, .type = -1, .type2 = 3}, > > + { .resource = RES_VAUX3, .devgroup = -1, .type = -1, .type2 = 3}, > > + { .resource = RES_VAUX4, .devgroup = -1, .type = -1, .type2 = 3}, > > + { .resource = RES_VPLL2, .devgroup = -1, .type = -1, .type2 = 3}, > > + { .resource = RES_VDAC, .devgroup = -1, .type = -1, .type2 = 3}, > > + { .resource = RES_VSIM, .devgroup = DEV_GRP_NULL, .type = -1, > > + .type2 = 3}, > > + { .resource = RES_CLKEN, .devgroup = DEV_GRP_P3, .type = -1, > > + .type2 = 1 }, > > + { 0, 0}, > > +}; > > + > > +static struct twl4030_power_data rx51_t2scripts_data __initdata = { > > + .scripts = twl4030_scripts, > > + .size = ARRAY_SIZE(twl4030_scripts), > > + .resource_config = twl4030_rconfig, > > +}; > > + > > + > > + > > +static struct twl4030_platform_data rx51_twldata __initdata = { > > .irq_base = TWL4030_IRQ_BASE, > > .irq_end = TWL4030_IRQ_END, > > > > @@ -286,6 +403,7 @@ static struct twl4030_platform_data rx51_twldata = { > > .gpio = &rx51_gpio_data, > > .keypad = &rx51_kp_data, > > .madc = &rx51_madc_data, > > + .power = &rx51_t2scripts_data, > > > > .vaux1 = &rx51_vaux1, > > .vaux2 = &rx51_vaux2, > > -- > > 1.6.3.3 > > > > -- > Intel Open Source Technology Centre > http://oss.intel.com/ -- ------------------------------------------------------------------------- Amit Kucheria, Kernel Developer, Verdurent ------------------------------------------------------------------------- ^ permalink raw reply [flat|nested] 11+ messages in thread
* Re: [PATCH 3/3] MFD: TWL4030: OMAP: Board-specifc DPS scripts for RX51 board 2009-08-17 11:46 ` Amit Kucheria @ 2009-08-24 12:33 ` Tony Lindgren 0 siblings, 0 replies; 11+ messages in thread From: Tony Lindgren @ 2009-08-24 12:33 UTC (permalink / raw) To: Samuel Ortiz, dbrownell, linux-kernel, linux-omap, Peter 'p2' De Schrijver * Amit Kucheria <amit.kucheria@verdurent.com> [090817 14:47]: > On 09 Aug 04, Samuel Ortiz wrote: > > Hi Amit, > > > > On Wed, Jul 08, 2009 at 01:49:39PM +0300, Amit Kucheria wrote: > > > Add support for Dynamic Power Switching (DPS) for the RX51 board. > > Could I have an Acked-by from Tony or someone from Nokia maintaining this > > board (e.g. Lauri) ? > > > > I'll cc Tony in the refreshed patches. Please note that this is not the > final version of the script. As the scripts are optimised, I'll send in > fixes. I'm fine with these, for the board specific configuration for twl4030, Amit and Peter are the people to ack them. Acked-by: Tony Lindgren <tony@atomide.com> > > Regards, > Amit > > > > Signed-off-by: Amit Kucheria <amit.kucheria@verdurent.com> > > > --- > > > arch/arm/mach-omap2/board-rx51-peripherals.c | 122 +++++++++++++++++++++++++- > > > 1 files changed, 120 insertions(+), 2 deletions(-) > > > > > > diff --git a/arch/arm/mach-omap2/board-rx51-peripherals.c b/arch/arm/mach-omap2/board-rx51-peripherals.c > > > index 9a0bf67..f7ebec8 100644 > > > --- a/arch/arm/mach-omap2/board-rx51-peripherals.c > > > +++ b/arch/arm/mach-omap2/board-rx51-peripherals.c > > > @@ -1,5 +1,5 @@ > > > /* > > > - * linux/arch/arm/mach-omap2/board-rx51-flash.c > > > + * linux/arch/arm/mach-omap2/board-rx51-peripherals.c > > > * > > > * Copyright (C) 2008-2009 Nokia > > > * > > > @@ -278,7 +278,124 @@ static struct twl4030_gpio_platform_data rx51_gpio_data = { > > > .setup = rx51_twlgpio_setup, > > > }; > > > > > > -static struct twl4030_platform_data rx51_twldata = { > > > +static struct twl4030_ins sleep_on_seq[] __initdata = { > > > +/* > > > + * Turn off VDD1 and VDD2. > > > + */ > > > + {MSG_SINGULAR(DEV_GRP_P1, 0xf, RES_STATE_OFF), 4}, > > > + {MSG_SINGULAR(DEV_GRP_P1, 0x10, RES_STATE_OFF), 2}, > > > +/* > > > + * And also turn off the OMAP3 PLLs and the sysclk output. > > > + */ > > > + {MSG_SINGULAR(DEV_GRP_P1, 0x7, RES_STATE_OFF), 3}, > > > + {MSG_SINGULAR(DEV_GRP_P1, 0x17, RES_STATE_OFF), 3}, > > > +}; > > > + > > > +static struct twl4030_script sleep_on_script __initdata = { > > > + .script = sleep_on_seq, > > > + .size = ARRAY_SIZE(sleep_on_seq), > > > + .flags = TRITON_SLEEP_SCRIPT, > > > +}; > > > + > > > +static struct twl4030_ins wakeup_seq[] __initdata = { > > > +/* > > > + * Reenable the OMAP3 PLLs. > > > + * Wakeup VDD1 and VDD2. > > > + * Reenable sysclk output. > > > + */ > > > + {MSG_SINGULAR(DEV_GRP_P1, 0x7, RES_STATE_ACTIVE), 0x30}, > > > + {MSG_SINGULAR(DEV_GRP_P1, 0xf, RES_STATE_ACTIVE), 0x30}, > > > + {MSG_SINGULAR(DEV_GRP_P1, 0x10, RES_STATE_ACTIVE), 0x37}, > > > + {MSG_SINGULAR(DEV_GRP_P1, 0x19, RES_STATE_ACTIVE), 3}, > > > +}; > > > + > > > +static struct twl4030_script wakeup_script __initdata = { > > > + .script = wakeup_seq, > > > + .size = ARRAY_SIZE(wakeup_seq), > > > + .flags = TRITON_WAKEUP12_SCRIPT, > > > +}; > > > + > > > +static struct twl4030_ins wakeup_p3_seq[] __initdata = { > > > +/* > > > + * Wakeup VDD1 (dummy to be able to insert a delay) > > > + * Enable CLKEN > > > + */ > > > + {MSG_SINGULAR(DEV_GRP_P1, 0x17, RES_STATE_ACTIVE), 3}, > > > +}; > > > + > > > +static struct twl4030_script wakeup_p3_script __initdata = { > > > + .script = wakeup_p3_seq, > > > + .size = ARRAY_SIZE(wakeup_p3_seq), > > > + .flags = TRITON_WAKEUP3_SCRIPT, > > > +}; > > > + > > > +static struct twl4030_ins wrst_seq[] __initdata = { > > > +/* > > > + * Reset twl4030. > > > + * Reset VDD1 regulator. > > > + * Reset VDD2 regulator. > > > + * Reset VPLL1 regulator. > > > + * Enable sysclk output. > > > + * Reenable twl4030. > > > + */ > > > + {MSG_SINGULAR(DEV_GRP_NULL, RES_RESET, RES_STATE_OFF), 2}, > > > + {MSG_BROADCAST(DEV_GRP_NULL, RES_GRP_ALL, 0, 1, RES_STATE_ACTIVE), > > > + 0x13}, > > > + {MSG_BROADCAST(DEV_GRP_NULL, RES_GRP_PP, 0, 2, RES_STATE_WRST), 0x13}, > > > + {MSG_BROADCAST(DEV_GRP_NULL, RES_GRP_PP, 0, 3, RES_STATE_OFF), 0x13}, > > > + {MSG_SINGULAR(DEV_GRP_NULL, RES_VDD1, RES_STATE_WRST), 0x13}, > > > + {MSG_SINGULAR(DEV_GRP_NULL, RES_VDD2, RES_STATE_WRST), 0x13}, > > > + {MSG_SINGULAR(DEV_GRP_NULL, RES_VPLL1, RES_STATE_WRST), 0x35}, > > > + {MSG_SINGULAR(DEV_GRP_P1, RES_HFCLKOUT, RES_STATE_ACTIVE), 2}, > > > + {MSG_SINGULAR(DEV_GRP_NULL, RES_RESET, RES_STATE_ACTIVE), 2}, > > > +}; > > > + > > > +static struct twl4030_script wrst_script __initdata = { > > > + .script = wrst_seq, > > > + .size = ARRAY_SIZE(wrst_seq), > > > + .flags = TRITON_WRST_SCRIPT, > > > +}; > > > + > > > +static struct twl4030_script *twl4030_scripts[] __initdata = { > > > + /* wakeup12 script should be loaded before sleep script, otherwise a > > > + board might hit retention before loading of wakeup script is > > > + completed. This can cause boot failures depending on timing issues. > > > + */ > > > + &wakeup_script, > > > + &sleep_on_script, > > > + &wakeup_p3_script, > > > + &wrst_script, > > > +}; > > > + > > > +static struct twl4030_resconfig twl4030_rconfig[] __initdata = { > > > + { .resource = RES_VINTANA1, .devgroup = -1, .type = -1, .type2 = 1 }, > > > + { .resource = RES_VINTANA2, .devgroup = -1, .type = -1, .type2 = 1 }, > > > + { .resource = RES_VINTDIG, .devgroup = -1, .type = -1, .type2 = 1 }, > > > + { .resource = RES_VMMC1, .devgroup = -1, .type = -1, .type2 = 3}, > > > + { .resource = RES_VMMC2, .devgroup = DEV_GRP_NULL, .type = -1, > > > + .type2 = 3}, > > > + { .resource = RES_VAUX1, .devgroup = -1, .type = -1, .type2 = 3}, > > > + { .resource = RES_VAUX2, .devgroup = -1, .type = -1, .type2 = 3}, > > > + { .resource = RES_VAUX3, .devgroup = -1, .type = -1, .type2 = 3}, > > > + { .resource = RES_VAUX4, .devgroup = -1, .type = -1, .type2 = 3}, > > > + { .resource = RES_VPLL2, .devgroup = -1, .type = -1, .type2 = 3}, > > > + { .resource = RES_VDAC, .devgroup = -1, .type = -1, .type2 = 3}, > > > + { .resource = RES_VSIM, .devgroup = DEV_GRP_NULL, .type = -1, > > > + .type2 = 3}, > > > + { .resource = RES_CLKEN, .devgroup = DEV_GRP_P3, .type = -1, > > > + .type2 = 1 }, > > > + { 0, 0}, > > > +}; > > > + > > > +static struct twl4030_power_data rx51_t2scripts_data __initdata = { > > > + .scripts = twl4030_scripts, > > > + .size = ARRAY_SIZE(twl4030_scripts), > > > + .resource_config = twl4030_rconfig, > > > +}; > > > + > > > + > > > + > > > +static struct twl4030_platform_data rx51_twldata __initdata = { > > > .irq_base = TWL4030_IRQ_BASE, > > > .irq_end = TWL4030_IRQ_END, > > > > > > @@ -286,6 +403,7 @@ static struct twl4030_platform_data rx51_twldata = { > > > .gpio = &rx51_gpio_data, > > > .keypad = &rx51_kp_data, > > > .madc = &rx51_madc_data, > > > + .power = &rx51_t2scripts_data, > > > > > > .vaux1 = &rx51_vaux1, > > > .vaux2 = &rx51_vaux2, > > > -- > > > 1.6.3.3 > > > > > > > -- > > Intel Open Source Technology Centre > > http://oss.intel.com/ > > -- > ------------------------------------------------------------------------- > Amit Kucheria, Kernel Developer, Verdurent > ------------------------------------------------------------------------- > -- > To unsubscribe from this list: send the line "unsubscribe linux-omap" in > the body of a message to majordomo@vger.kernel.org > More majordomo info at http://vger.kernel.org/majordomo-info.html ^ permalink raw reply [flat|nested] 11+ messages in thread
[parent not found: <1250517469-15899-1-git-send-email-amit.kucheria@verdurent.com>]
* [PATCH 2/3] MFD: TWL4030: print warning for out-of-order script loading [not found] <1250517469-15899-1-git-send-email-amit.kucheria@verdurent.com> @ 2009-08-17 14:01 ` Amit Kucheria 0 siblings, 0 replies; 11+ messages in thread From: Amit Kucheria @ 2009-08-17 14:01 UTC (permalink / raw) To: linux-kernel; +Cc: sameo, dbrownell, linux-omap When the sleep script is loaded before the wakeup script, there is a chance that the system might go to sleep before the wakeup script loading is completed. This will lead to a system that does not wakeup and has been observed to cause non-booting boards. Various options were considered to solve this problem, including modification of the core twl4030 power code to be smart enough to reorder the loading of the scripts. But it felt too over-engineered. Hence this patch just warns the DPS script developer so that they may be reordered in the board-code itself. Signed-off-by: Amit Kucheria <amit.kucheria@verdurent.com> Cc: sameo@linux.intel.com Cc: dbrownell@users.sourceforge.net Cc: linux-omap@vger.kernel.org --- drivers/mfd/twl4030-power.c | 6 ++++++ 1 files changed, 6 insertions(+), 0 deletions(-) diff --git a/drivers/mfd/twl4030-power.c b/drivers/mfd/twl4030-power.c index e7688b0..5284c7c 100644 --- a/drivers/mfd/twl4030-power.c +++ b/drivers/mfd/twl4030-power.c @@ -379,6 +379,7 @@ static int __init load_twl4030_script(struct twl4030_script *tscript, u8 address) { int err; + static int order; /* Make sure the script isn't going beyond last valid address (0x3f) */ if ((address + tscript->size) > END_OF_SCRIPT) { @@ -399,6 +400,7 @@ static int __init load_twl4030_script(struct twl4030_script *tscript, err = twl4030_config_wakeup12_sequence(address); if (err) goto out; + order = 1; } if (tscript->flags & TWL4030_WAKEUP3_SCRIPT) { err = twl4030_config_wakeup3_sequence(address); @@ -406,6 +408,10 @@ static int __init load_twl4030_script(struct twl4030_script *tscript, goto out; } if (tscript->flags & TWL4030_SLEEP_SCRIPT) + if (order) + pr_warning("TWL4030: Bad order of scripts (sleep "\ + "script before wakeup) Leads to boot"\ + "failure on some boards\n"); err = twl4030_config_sleep_sequence(address); out: return err; -- 1.6.3.3 ^ permalink raw reply related [flat|nested] 11+ messages in thread
end of thread, other threads:[~2009-08-24 12:33 UTC | newest]
Thread overview: 11+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
[not found] <1247050046-31104-1-git-send-email-./0000-cover-letter.patch>
2009-07-08 10:49 ` [PATCH 1/3] MFD: TWL4030: Add support for TWL4030/5030 dynamic power switching Amit Kucheria
2009-08-04 17:55 ` Samuel Ortiz
2009-08-17 11:42 ` Amit Kucheria
2009-07-08 10:49 ` [PATCH 2/3] MFD: TWL4030: print warning for out-of-order script loading Amit Kucheria
2009-08-04 18:00 ` Samuel Ortiz
2009-08-17 11:43 ` Amit Kucheria
2009-07-08 10:49 ` [PATCH 3/3] MFD: TWL4030: OMAP: Board-specifc DPS scripts for RX51 board Amit Kucheria
2009-08-04 18:03 ` Samuel Ortiz
2009-08-17 11:46 ` Amit Kucheria
2009-08-24 12:33 ` Tony Lindgren
[not found] <1250517469-15899-1-git-send-email-amit.kucheria@verdurent.com>
2009-08-17 14:01 ` [PATCH 2/3] MFD: TWL4030: print warning for out-of-order script loading Amit Kucheria
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox