2008-10-15 12:15:39 +02:00
/*
2009-12-13 21:23:33 +01:00
* twl_core . c - driver for TWL4030 / TWL5030 / TWL60X0 / TPS659x0 PM
* and audio CODEC devices
2008-10-15 12:15:39 +02:00
*
* Copyright ( C ) 2005 - 2006 Texas Instruments , Inc .
*
* Modifications to defer interrupt handling to a kernel thread :
* Copyright ( C ) 2006 MontaVista Software , Inc .
*
* Based on tlv320aic23 . c :
* Copyright ( c ) by Kai Svahn < kai . svahn @ nokia . com >
*
* Code cleanup and modifications to IRQ handler .
* by syed khasim < x0khasim @ ti . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* 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/init.h>
# include <linux/mutex.h>
# include <linux/platform_device.h>
# include <linux/clk.h>
2008-10-20 23:46:28 +02:00
# include <linux/err.h>
2008-10-15 12:15:39 +02:00
2008-12-01 00:43:58 +01:00
# include <linux/regulator/machine.h>
2008-10-15 12:15:39 +02:00
# include <linux/i2c.h>
2009-12-13 20:05:51 +01:00
# include <linux/i2c/twl.h>
2008-10-15 12:15:39 +02:00
2009-02-04 21:26:07 +01:00
# if defined(CONFIG_ARCH_OMAP2) || defined(CONFIG_ARCH_OMAP3)
2009-10-20 09:40:47 -07:00
# include <plat/cpu.h>
2009-01-11 01:21:48 +01:00
# endif
2008-10-15 12:15:39 +02:00
/*
* The TWL4030 " Triton 2 " is one of a family of a multi - function " Power
* Management and System Companion Device " chips originally designed for
* use in OMAP2 and OMAP 3 based systems . Its control interfaces use I2C ,
* often at around 3 Mbit / sec , including for interrupt handling .
*
* This driver core provides genirq support for the interrupts emitted ,
* by the various modules , and exports register access primitives .
*
* FIXME this driver currently requires use of the first interrupt line
* ( and associated registers ) .
*/
2009-12-13 21:23:33 +01:00
# define DRIVER_NAME "twl"
2008-10-15 12:15:39 +02:00
# if defined(CONFIG_KEYBOARD_TWL4030) || defined(CONFIG_KEYBOARD_TWL4030_MODULE)
# define twl_has_keypad() true
# else
# define twl_has_keypad() false
# endif
# if defined(CONFIG_GPIO_TWL4030) || defined(CONFIG_GPIO_TWL4030_MODULE)
# define twl_has_gpio() true
# else
# define twl_has_gpio() false
# endif
2008-12-01 00:43:58 +01:00
# if defined(CONFIG_REGULATOR_TWL4030) \
| | defined ( CONFIG_REGULATOR_TWL4030_MODULE )
# define twl_has_regulator() true
# else
# define twl_has_regulator() false
# endif
2008-10-15 12:15:39 +02:00
# if defined(CONFIG_TWL4030_MADC) || defined(CONFIG_TWL4030_MADC_MODULE)
# define twl_has_madc() true
# else
# define twl_has_madc() false
# endif
2009-08-31 18:32:18 +02:00
# ifdef CONFIG_TWL4030_POWER
# define twl_has_power() true
# else
# define twl_has_power() false
# endif
2008-10-15 12:15:39 +02:00
# if defined(CONFIG_RTC_DRV_TWL4030) || defined(CONFIG_RTC_DRV_TWL4030_MODULE)
# define twl_has_rtc() true
# else
# define twl_has_rtc() false
# endif
2010-12-10 18:09:52 +05:30
# if defined(CONFIG_TWL4030_USB) || defined(CONFIG_TWL4030_USB_MODULE) ||\
defined ( CONFIG_TWL6030_USB ) | | defined ( CONFIG_TWL6030_USB_MODULE )
2008-10-15 12:15:39 +02:00
# define twl_has_usb() true
# else
# define twl_has_usb() false
# endif
2009-03-27 16:42:17 +02:00
# if defined(CONFIG_TWL4030_WATCHDOG) || \
defined ( CONFIG_TWL4030_WATCHDOG_MODULE )
# define twl_has_watchdog() true
# else
# define twl_has_watchdog() false
# endif
2008-10-15 12:15:39 +02:00
2010-02-23 18:10:19 -06:00
# if defined(CONFIG_TWL4030_CODEC) || defined(CONFIG_TWL4030_CODEC_MODULE) ||\
2011-04-27 02:14:07 -05:00
defined ( CONFIG_TWL6040_CORE ) | | defined ( CONFIG_TWL6040_CORE_MODULE )
2009-10-22 13:26:45 +03:00
# define twl_has_codec() true
# else
# define twl_has_codec() false
# endif
2010-10-01 13:17:50 +03:00
# if defined(CONFIG_CHARGER_TWL4030) || defined(CONFIG_CHARGER_TWL4030_MODULE)
# define twl_has_bci() true
# else
# define twl_has_bci() false
# endif
2008-10-15 12:15:39 +02:00
/* Triton Core internal information (BEGIN) */
/* Last - for index max*/
# define TWL4030_MODULE_LAST TWL4030_MODULE_SECURED_REG
2009-12-13 21:23:33 +01:00
# define TWL_NUM_SLAVES 4
2008-10-15 12:15:39 +02:00
2009-08-03 18:16:38 +02:00
# if defined(CONFIG_INPUT_TWL4030_PWRBUTTON) \
2010-02-11 13:47:09 +01:00
| | defined ( CONFIG_INPUT_TWL4030_PWRBUTTON_MODULE )
2009-08-03 18:16:38 +02:00
# define twl_has_pwrbutton() true
# else
# define twl_has_pwrbutton() false
# endif
2008-10-15 12:15:39 +02:00
2009-12-13 21:23:33 +01:00
# define SUB_CHIP_ID0 0
# define SUB_CHIP_ID1 1
# define SUB_CHIP_ID2 2
# define SUB_CHIP_ID3 3
# define TWL_MODULE_LAST TWL4030_MODULE_LAST
2008-10-15 12:15:39 +02:00
/* Base Address defns for twl4030_map[] */
/* subchip/slave 0 - USB ID */
# define TWL4030_BASEADD_USB 0x0000
/* subchip/slave 1 - AUD ID */
# define TWL4030_BASEADD_AUDIO_VOICE 0x0000
# define TWL4030_BASEADD_GPIO 0x0098
# define TWL4030_BASEADD_INTBR 0x0085
# define TWL4030_BASEADD_PIH 0x0080
# define TWL4030_BASEADD_TEST 0x004C
/* subchip/slave 2 - AUX ID */
# define TWL4030_BASEADD_INTERRUPTS 0x00B9
# define TWL4030_BASEADD_LED 0x00EE
# define TWL4030_BASEADD_MADC 0x0000
# define TWL4030_BASEADD_MAIN_CHARGE 0x0074
# define TWL4030_BASEADD_PRECHARGE 0x00AA
# define TWL4030_BASEADD_PWM0 0x00F8
# define TWL4030_BASEADD_PWM1 0x00FB
# define TWL4030_BASEADD_PWMA 0x00EF
# define TWL4030_BASEADD_PWMB 0x00F1
# define TWL4030_BASEADD_KEYPAD 0x00D2
2009-11-10 17:26:15 +02:00
# define TWL5031_BASEADD_ACCESSORY 0x0074 /* Replaces Main Charge */
# define TWL5031_BASEADD_INTERRUPTS 0x00B9 / * Different than TWL4030's
one */
2008-10-15 12:15:39 +02:00
/* subchip/slave 3 - POWER ID */
# define TWL4030_BASEADD_BACKUP 0x0014
# define TWL4030_BASEADD_INT 0x002E
# define TWL4030_BASEADD_PM_MASTER 0x0036
# define TWL4030_BASEADD_PM_RECEIVER 0x005B
# define TWL4030_BASEADD_RTC 0x001C
# define TWL4030_BASEADD_SECURED_REG 0x0000
/* Triton Core internal information (END) */
2009-12-14 00:25:31 +01:00
/* subchip/slave 0 0x48 - POWER */
# define TWL6030_BASEADD_RTC 0x0000
# define TWL6030_BASEADD_MEM 0x0017
# define TWL6030_BASEADD_PM_MASTER 0x001F
# define TWL6030_BASEADD_PM_SLAVE_MISC 0x0030 /* PM_RECEIVER */
# define TWL6030_BASEADD_PM_MISC 0x00E2
# define TWL6030_BASEADD_PM_PUPD 0x00F0
/* subchip/slave 1 0x49 - FEATURE */
# define TWL6030_BASEADD_USB 0x0000
# define TWL6030_BASEADD_GPADC_CTRL 0x002E
# define TWL6030_BASEADD_AUX 0x0090
# define TWL6030_BASEADD_PWM 0x00BA
# define TWL6030_BASEADD_GASGAUGE 0x00C0
# define TWL6030_BASEADD_PIH 0x00D0
# define TWL6030_BASEADD_CHARGER 0x00E0
2011-05-12 14:27:55 +01:00
# define TWL6025_BASEADD_CHARGER 0x00DA
2009-12-14 00:25:31 +01:00
/* subchip/slave 2 0x4A - DFT */
# define TWL6030_BASEADD_DIEID 0x00C0
/* subchip/slave 3 0x4B - AUDIO */
# define TWL6030_BASEADD_AUDIO 0x0000
# define TWL6030_BASEADD_RSV 0x0000
2010-02-19 12:39:38 +01:00
# define TWL6030_BASEADD_ZERO 0x0000
2009-12-14 00:25:31 +01:00
2008-10-15 12:15:39 +02:00
/* Few power values */
# define R_CFG_BOOT 0x05
/* some fields in R_CFG_BOOT */
# define HFCLK_FREQ_19p2_MHZ (1 << 0)
# define HFCLK_FREQ_26_MHZ (2 << 0)
# define HFCLK_FREQ_38p4_MHZ (3 << 0)
# define HIGH_PERF_SQ (1 << 3)
2009-10-22 14:14:09 +03:00
# define CK32K_LOWPWR_EN (1 << 7)
2008-10-15 12:15:39 +02:00
2008-12-01 00:43:58 +01:00
/* chip-specific feature flags, for i2c_device_id.driver_data */
# define TWL4030_VAUX2 BIT(0) /* pre-5030 voltage ranges */
# define TPS_SUBSET BIT(1) /* tps659[23]0 have fewer LDOs */
2009-11-10 17:26:15 +02:00
# define TWL5031 BIT(2) /* twl5031 has different registers */
2009-12-14 00:25:31 +01:00
# define TWL6030_CLASS BIT(3) /* TWL6030 class */
2008-12-01 00:43:58 +01:00
2008-10-15 12:15:39 +02:00
/*----------------------------------------------------------------------*/
/* is driver active, bound to a chip? */
static bool inuse ;
2011-04-14 17:57:53 +05:30
/* TWL IDCODE Register value */
static u32 twl_idcode ;
2009-12-14 00:25:31 +01:00
static unsigned int twl_id ;
unsigned int twl_rev ( void )
{
return twl_id ;
}
EXPORT_SYMBOL ( twl_rev ) ;
/* Structure for each TWL4030/TWL6030 Slave */
2009-12-13 21:23:33 +01:00
struct twl_client {
2008-10-15 12:15:39 +02:00
struct i2c_client * client ;
u8 address ;
/* max numb of i2c_msg required is for read =2 */
struct i2c_msg xfer_msg [ 2 ] ;
/* To lock access to xfer_msg */
struct mutex xfer_lock ;
} ;
2009-12-13 21:23:33 +01:00
static struct twl_client twl_modules [ TWL_NUM_SLAVES ] ;
2008-10-15 12:15:39 +02:00
/* mapping the module id to slave id and base address */
2009-12-13 21:23:33 +01:00
struct twl_mapping {
2008-10-15 12:15:39 +02:00
unsigned char sid ; /* Slave ID */
unsigned char base ; /* base address */
} ;
2010-09-21 15:31:17 +05:30
static struct twl_mapping * twl_map ;
2008-10-15 12:15:39 +02:00
2009-12-13 21:23:33 +01:00
static struct twl_mapping twl4030_map [ TWL4030_MODULE_LAST + 1 ] = {
2008-10-15 12:15:39 +02:00
/*
* NOTE : don ' t change this table without updating the
2009-12-14 00:25:31 +01:00
* < linux / i2c / twl . h > defines for TWL4030_MODULE_ *
2008-10-15 12:15:39 +02:00
* so they continue to match the order in this table .
*/
{ 0 , TWL4030_BASEADD_USB } ,
{ 1 , TWL4030_BASEADD_AUDIO_VOICE } ,
{ 1 , TWL4030_BASEADD_GPIO } ,
{ 1 , TWL4030_BASEADD_INTBR } ,
{ 1 , TWL4030_BASEADD_PIH } ,
{ 1 , TWL4030_BASEADD_TEST } ,
{ 2 , TWL4030_BASEADD_KEYPAD } ,
{ 2 , TWL4030_BASEADD_MADC } ,
{ 2 , TWL4030_BASEADD_INTERRUPTS } ,
{ 2 , TWL4030_BASEADD_LED } ,
{ 2 , TWL4030_BASEADD_MAIN_CHARGE } ,
{ 2 , TWL4030_BASEADD_PRECHARGE } ,
{ 2 , TWL4030_BASEADD_PWM0 } ,
{ 2 , TWL4030_BASEADD_PWM1 } ,
{ 2 , TWL4030_BASEADD_PWMA } ,
{ 2 , TWL4030_BASEADD_PWMB } ,
2009-11-10 17:26:15 +02:00
{ 2 , TWL5031_BASEADD_ACCESSORY } ,
{ 2 , TWL5031_BASEADD_INTERRUPTS } ,
2008-10-15 12:15:39 +02:00
{ 3 , TWL4030_BASEADD_BACKUP } ,
{ 3 , TWL4030_BASEADD_INT } ,
{ 3 , TWL4030_BASEADD_PM_MASTER } ,
{ 3 , TWL4030_BASEADD_PM_RECEIVER } ,
{ 3 , TWL4030_BASEADD_RTC } ,
{ 3 , TWL4030_BASEADD_SECURED_REG } ,
} ;
2009-12-14 00:25:31 +01:00
static struct twl_mapping twl6030_map [ ] = {
/*
* NOTE : don ' t change this table without updating the
* < linux / i2c / twl . h > defines for TWL4030_MODULE_ *
* so they continue to match the order in this table .
*/
{ SUB_CHIP_ID1 , TWL6030_BASEADD_USB } ,
{ SUB_CHIP_ID3 , TWL6030_BASEADD_AUDIO } ,
{ SUB_CHIP_ID2 , TWL6030_BASEADD_DIEID } ,
{ SUB_CHIP_ID2 , TWL6030_BASEADD_RSV } ,
{ SUB_CHIP_ID1 , TWL6030_BASEADD_PIH } ,
{ SUB_CHIP_ID2 , TWL6030_BASEADD_RSV } ,
{ SUB_CHIP_ID2 , TWL6030_BASEADD_RSV } ,
{ SUB_CHIP_ID1 , TWL6030_BASEADD_GPADC_CTRL } ,
{ SUB_CHIP_ID2 , TWL6030_BASEADD_RSV } ,
{ SUB_CHIP_ID2 , TWL6030_BASEADD_RSV } ,
{ SUB_CHIP_ID1 , TWL6030_BASEADD_CHARGER } ,
{ SUB_CHIP_ID1 , TWL6030_BASEADD_GASGAUGE } ,
{ SUB_CHIP_ID1 , TWL6030_BASEADD_PWM } ,
2010-02-19 12:39:38 +01:00
{ SUB_CHIP_ID0 , TWL6030_BASEADD_ZERO } ,
{ SUB_CHIP_ID1 , TWL6030_BASEADD_ZERO } ,
2009-12-14 00:25:31 +01:00
2010-02-19 12:39:38 +01:00
{ SUB_CHIP_ID2 , TWL6030_BASEADD_ZERO } ,
{ SUB_CHIP_ID2 , TWL6030_BASEADD_ZERO } ,
2009-12-14 00:25:31 +01:00
{ SUB_CHIP_ID2 , TWL6030_BASEADD_RSV } ,
{ SUB_CHIP_ID2 , TWL6030_BASEADD_RSV } ,
{ SUB_CHIP_ID2 , TWL6030_BASEADD_RSV } ,
{ SUB_CHIP_ID0 , TWL6030_BASEADD_PM_MASTER } ,
{ SUB_CHIP_ID0 , TWL6030_BASEADD_PM_SLAVE_MISC } ,
{ SUB_CHIP_ID0 , TWL6030_BASEADD_RTC } ,
{ SUB_CHIP_ID0 , TWL6030_BASEADD_MEM } ,
2011-05-12 14:27:55 +01:00
{ SUB_CHIP_ID1 , TWL6025_BASEADD_CHARGER } ,
2009-12-14 00:25:31 +01:00
} ;
2008-10-15 12:15:39 +02:00
/*----------------------------------------------------------------------*/
/* Exported Functions */
/**
2009-12-13 21:23:33 +01:00
* twl_i2c_write - Writes a n bit register in TWL4030 / TWL5030 / TWL60X0
2008-10-15 12:15:39 +02:00
* @ mod_no : module number
* @ value : an array of num_bytes + 1 containing data to write
* @ reg : register address ( just offset will do )
* @ num_bytes : number of bytes to transfer
*
* IMPORTANT : for ' value ' parameter : Allocate value num_bytes + 1 and
* valid data starts at Offset 1.
*
* Returns the result of operation - 0 is success
*/
2009-12-13 21:23:33 +01:00
int twl_i2c_write ( u8 mod_no , u8 * value , u8 reg , unsigned num_bytes )
2008-10-15 12:15:39 +02:00
{
int ret ;
int sid ;
2009-12-13 21:23:33 +01:00
struct twl_client * twl ;
2008-10-15 12:15:39 +02:00
struct i2c_msg * msg ;
2009-12-13 21:23:33 +01:00
if ( unlikely ( mod_no > TWL_MODULE_LAST ) ) {
2008-10-15 12:15:39 +02:00
pr_err ( " %s: invalid module number %d \n " , DRIVER_NAME , mod_no ) ;
return - EPERM ;
}
2009-12-14 00:25:31 +01:00
sid = twl_map [ mod_no ] . sid ;
2009-12-13 21:23:33 +01:00
twl = & twl_modules [ sid ] ;
2008-10-15 12:15:39 +02:00
if ( unlikely ( ! inuse ) ) {
pr_err ( " %s: client %d is not initialized \n " , DRIVER_NAME , sid ) ;
return - EPERM ;
}
mutex_lock ( & twl - > xfer_lock ) ;
/*
* [ MSG1 ] : fill the register address data
* fill the data Tx buffer
*/
msg = & twl - > xfer_msg [ 0 ] ;
msg - > addr = twl - > address ;
msg - > len = num_bytes + 1 ;
msg - > flags = 0 ;
msg - > buf = value ;
/* over write the first byte of buffer with the register address */
2009-12-14 00:25:31 +01:00
* value = twl_map [ mod_no ] . base + reg ;
2008-10-15 12:15:39 +02:00
ret = i2c_transfer ( twl - > client - > adapter , twl - > xfer_msg , 1 ) ;
mutex_unlock ( & twl - > xfer_lock ) ;
2009-12-11 13:21:45 +01:00
/* i2c_transfer returns number of messages transferred */
if ( ret ! = 1 ) {
pr_err ( " %s: i2c_write failed to transfer all messages \n " ,
DRIVER_NAME ) ;
if ( ret < 0 )
return ret ;
else
return - EIO ;
} else {
return 0 ;
}
2008-10-15 12:15:39 +02:00
}
2009-12-13 21:23:33 +01:00
EXPORT_SYMBOL ( twl_i2c_write ) ;
2008-10-15 12:15:39 +02:00
/**
2009-12-13 21:23:33 +01:00
* twl_i2c_read - Reads a n bit register in TWL4030 / TWL5030 / TWL60X0
2008-10-15 12:15:39 +02:00
* @ mod_no : module number
* @ value : an array of num_bytes containing data to be read
* @ reg : register address ( just offset will do )
* @ num_bytes : number of bytes to transfer
*
* Returns result of operation - num_bytes is success else failure .
*/
2009-12-13 21:23:33 +01:00
int twl_i2c_read ( u8 mod_no , u8 * value , u8 reg , unsigned num_bytes )
2008-10-15 12:15:39 +02:00
{
int ret ;
u8 val ;
int sid ;
2009-12-13 21:23:33 +01:00
struct twl_client * twl ;
2008-10-15 12:15:39 +02:00
struct i2c_msg * msg ;
2009-12-13 21:23:33 +01:00
if ( unlikely ( mod_no > TWL_MODULE_LAST ) ) {
2008-10-15 12:15:39 +02:00
pr_err ( " %s: invalid module number %d \n " , DRIVER_NAME , mod_no ) ;
return - EPERM ;
}
2009-12-14 00:25:31 +01:00
sid = twl_map [ mod_no ] . sid ;
2009-12-13 21:23:33 +01:00
twl = & twl_modules [ sid ] ;
2008-10-15 12:15:39 +02:00
if ( unlikely ( ! inuse ) ) {
pr_err ( " %s: client %d is not initialized \n " , DRIVER_NAME , sid ) ;
return - EPERM ;
}
mutex_lock ( & twl - > xfer_lock ) ;
/* [MSG1] fill the register address data */
msg = & twl - > xfer_msg [ 0 ] ;
msg - > addr = twl - > address ;
msg - > len = 1 ;
msg - > flags = 0 ; /* Read the register value */
2009-12-14 00:25:31 +01:00
val = twl_map [ mod_no ] . base + reg ;
2008-10-15 12:15:39 +02:00
msg - > buf = & val ;
/* [MSG2] fill the data rx buffer */
msg = & twl - > xfer_msg [ 1 ] ;
msg - > addr = twl - > address ;
msg - > flags = I2C_M_RD ; /* Read the register value */
msg - > len = num_bytes ; /* only n bytes */
msg - > buf = value ;
ret = i2c_transfer ( twl - > client - > adapter , twl - > xfer_msg , 2 ) ;
mutex_unlock ( & twl - > xfer_lock ) ;
2009-12-11 13:21:45 +01:00
/* i2c_transfer returns number of messages transferred */
if ( ret ! = 2 ) {
pr_err ( " %s: i2c_read failed to transfer all messages \n " ,
DRIVER_NAME ) ;
if ( ret < 0 )
return ret ;
else
return - EIO ;
} else {
return 0 ;
}
2008-10-15 12:15:39 +02:00
}
2009-12-13 21:23:33 +01:00
EXPORT_SYMBOL ( twl_i2c_read ) ;
2008-10-15 12:15:39 +02:00
/**
2009-12-13 21:23:33 +01:00
* twl_i2c_write_u8 - Writes a 8 bit register in TWL4030 / TWL5030 / TWL60X0
2008-10-15 12:15:39 +02:00
* @ mod_no : module number
* @ value : the value to be written 8 bit
* @ reg : register address ( just offset will do )
*
* Returns result of operation - 0 is success
*/
2009-12-13 21:23:33 +01:00
int twl_i2c_write_u8 ( u8 mod_no , u8 value , u8 reg )
2008-10-15 12:15:39 +02:00
{
/* 2 bytes offset 1 contains the data offset 0 is used by i2c_write */
u8 temp_buffer [ 2 ] = { 0 } ;
/* offset 1 contains the data */
temp_buffer [ 1 ] = value ;
2009-12-13 21:23:33 +01:00
return twl_i2c_write ( mod_no , temp_buffer , reg , 1 ) ;
2008-10-15 12:15:39 +02:00
}
2009-12-13 21:23:33 +01:00
EXPORT_SYMBOL ( twl_i2c_write_u8 ) ;
2008-10-15 12:15:39 +02:00
/**
2009-12-13 21:23:33 +01:00
* twl_i2c_read_u8 - Reads a 8 bit register from TWL4030 / TWL5030 / TWL60X0
2008-10-15 12:15:39 +02:00
* @ mod_no : module number
* @ value : the value read 8 bit
* @ reg : register address ( just offset will do )
*
* Returns result of operation - 0 is success
*/
2009-12-13 21:23:33 +01:00
int twl_i2c_read_u8 ( u8 mod_no , u8 * value , u8 reg )
2008-10-15 12:15:39 +02:00
{
2009-12-13 21:23:33 +01:00
return twl_i2c_read ( mod_no , value , reg , 1 ) ;
2008-10-15 12:15:39 +02:00
}
2009-12-13 21:23:33 +01:00
EXPORT_SYMBOL ( twl_i2c_read_u8 ) ;
2008-10-15 12:15:39 +02:00
/*----------------------------------------------------------------------*/
2011-04-14 17:57:53 +05:30
/**
* twl_read_idcode_register - API to read the IDCODE register .
*
* Unlocks the IDCODE register and read the 32 bit value .
*/
static int twl_read_idcode_register ( void )
{
int err ;
err = twl_i2c_write_u8 ( TWL4030_MODULE_INTBR , TWL_EEPROM_R_UNLOCK ,
REG_UNLOCK_TEST_REG ) ;
if ( err ) {
pr_err ( " TWL4030 Unable to unlock IDCODE registers -%d \n " , err ) ;
goto fail ;
}
err = twl_i2c_read ( TWL4030_MODULE_INTBR , ( u8 * ) ( & twl_idcode ) ,
REG_IDCODE_7_0 , 4 ) ;
if ( err ) {
pr_err ( " TWL4030: unable to read IDCODE -%d \n " , err ) ;
goto fail ;
}
err = twl_i2c_write_u8 ( TWL4030_MODULE_INTBR , 0x0 , REG_UNLOCK_TEST_REG ) ;
if ( err )
pr_err ( " TWL4030 Unable to relock IDCODE registers -%d \n " , err ) ;
fail :
return err ;
}
/**
* twl_get_type - API to get TWL Si type .
*
* Api to get the TWL Si type from IDCODE value .
*/
int twl_get_type ( void )
{
return TWL_SIL_TYPE ( twl_idcode ) ;
}
EXPORT_SYMBOL_GPL ( twl_get_type ) ;
/**
* twl_get_version - API to get TWL Si version .
*
* Api to get the TWL Si version from IDCODE value .
*/
int twl_get_version ( void )
{
return TWL_SIL_REV ( twl_idcode ) ;
}
EXPORT_SYMBOL_GPL ( twl_get_version ) ;
2008-12-01 00:43:58 +01:00
static struct device *
add_numbered_child ( unsigned chip , const char * name , int num ,
2008-12-01 00:31:04 +01:00
void * pdata , unsigned pdata_len ,
bool can_wakeup , int irq0 , int irq1 )
2008-10-15 12:15:39 +02:00
{
2008-12-01 00:31:04 +01:00
struct platform_device * pdev ;
2009-12-13 21:23:33 +01:00
struct twl_client * twl = & twl_modules [ chip ] ;
2008-12-01 00:31:04 +01:00
int status ;
2008-12-01 00:43:58 +01:00
pdev = platform_device_alloc ( name , num ) ;
2008-12-01 00:31:04 +01:00
if ( ! pdev ) {
dev_dbg ( & twl - > client - > dev , " can't alloc dev \n " ) ;
status = - ENOMEM ;
goto err ;
}
2008-10-15 12:15:39 +02:00
2008-12-01 00:31:04 +01:00
device_init_wakeup ( & pdev - > dev , can_wakeup ) ;
pdev - > dev . parent = & twl - > client - > dev ;
2008-10-15 12:15:39 +02:00
2008-12-01 00:31:04 +01:00
if ( pdata ) {
status = platform_device_add_data ( pdev , pdata , pdata_len ) ;
if ( status < 0 ) {
dev_dbg ( & pdev - > dev , " can't add platform_data \n " ) ;
2008-10-15 12:15:39 +02:00
goto err ;
}
2008-12-01 00:31:04 +01:00
}
2008-10-15 12:15:39 +02:00
2008-12-01 00:31:04 +01:00
if ( irq0 ) {
struct resource r [ 2 ] = {
{ . start = irq0 , . flags = IORESOURCE_IRQ , } ,
{ . start = irq1 , . flags = IORESOURCE_IRQ , } ,
} ;
2008-10-15 12:15:39 +02:00
2008-12-01 00:31:04 +01:00
status = platform_device_add_resources ( pdev , r , irq1 ? 2 : 1 ) ;
2008-10-15 12:15:39 +02:00
if ( status < 0 ) {
2008-12-01 00:31:04 +01:00
dev_dbg ( & pdev - > dev , " can't add irqs \n " ) ;
2008-10-15 12:15:39 +02:00
goto err ;
}
}
2008-12-01 00:31:04 +01:00
status = platform_device_add ( pdev ) ;
2008-10-15 12:15:39 +02:00
2008-12-01 00:31:04 +01:00
err :
if ( status < 0 ) {
platform_device_put ( pdev ) ;
dev_err ( & twl - > client - > dev , " can't add %s dev \n " , name ) ;
return ERR_PTR ( status ) ;
}
return & pdev - > dev ;
}
2008-10-15 12:15:39 +02:00
2008-12-01 00:43:58 +01:00
static inline struct device * add_child ( unsigned chip , const char * name ,
void * pdata , unsigned pdata_len ,
bool can_wakeup , int irq0 , int irq1 )
{
return add_numbered_child ( chip , name , - 1 , pdata , pdata_len ,
can_wakeup , irq0 , irq1 ) ;
}
static struct device *
add_regulator_linked ( int num , struct regulator_init_data * pdata ,
struct regulator_consumer_supply * consumers ,
2011-05-12 14:27:55 +01:00
unsigned num_consumers , unsigned long features )
2008-12-01 00:43:58 +01:00
{
2009-12-14 00:25:31 +01:00
unsigned sub_chip_id ;
2008-12-01 00:43:58 +01:00
/* regulator framework demands init_data ... */
if ( ! pdata )
return NULL ;
2008-12-07 19:10:58 +01:00
if ( consumers ) {
2008-12-01 00:43:58 +01:00
pdata - > consumer_supplies = consumers ;
pdata - > num_consumer_supplies = num_consumers ;
}
2011-05-12 14:27:55 +01:00
pdata - > driver_data = ( void * ) features ;
2008-12-01 00:43:58 +01:00
/* NOTE: we currently ignore regulator IRQs, e.g. for short circuits */
2009-12-14 00:25:31 +01:00
sub_chip_id = twl_map [ TWL_MODULE_PM_MASTER ] . sid ;
return add_numbered_child ( sub_chip_id , " twl_reg " , num ,
2008-12-01 00:43:58 +01:00
pdata , sizeof ( * pdata ) , false , 0 , 0 ) ;
}
static struct device *
2011-05-12 14:27:55 +01:00
add_regulator ( int num , struct regulator_init_data * pdata ,
unsigned long features )
2008-12-01 00:43:58 +01:00
{
2011-05-12 14:27:55 +01:00
return add_regulator_linked ( num , pdata , NULL , 0 , features ) ;
2008-12-01 00:43:58 +01:00
}
2008-12-01 00:31:04 +01:00
/*
* NOTE : We know the first 8 IRQs after pdata - > base_irq are
* for the PIH , and the next are for the PWR_INT SIH , since
* that ' s how twl_init_irq ( ) sets things up .
*/
2008-10-15 12:15:39 +02:00
2008-12-01 00:43:58 +01:00
static int
add_children ( struct twl4030_platform_data * pdata , unsigned long features )
2008-12-01 00:31:04 +01:00
{
struct device * child ;
2009-12-14 00:25:31 +01:00
unsigned sub_chip_id ;
2008-10-15 12:15:39 +02:00
2008-12-01 00:31:04 +01:00
if ( twl_has_gpio ( ) & & pdata - > gpio ) {
2009-12-13 21:23:33 +01:00
child = add_child ( SUB_CHIP_ID1 , " twl4030_gpio " ,
2008-12-01 00:31:04 +01:00
pdata - > gpio , sizeof ( * pdata - > gpio ) ,
2009-12-13 21:23:33 +01:00
false , pdata - > irq_base + GPIO_INTR_OFFSET , 0 ) ;
2008-12-01 00:31:04 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2008-10-15 12:15:39 +02:00
}
if ( twl_has_keypad ( ) & & pdata - > keypad ) {
2009-12-13 21:23:33 +01:00
child = add_child ( SUB_CHIP_ID2 , " twl4030_keypad " ,
2008-12-01 00:31:04 +01:00
pdata - > keypad , sizeof ( * pdata - > keypad ) ,
2009-12-13 21:23:33 +01:00
true , pdata - > irq_base + KEYPAD_INTR_OFFSET , 0 ) ;
2008-12-01 00:31:04 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2008-10-15 12:15:39 +02:00
}
if ( twl_has_madc ( ) & & pdata - > madc ) {
2008-12-01 00:31:04 +01:00
child = add_child ( 2 , " twl4030_madc " ,
pdata - > madc , sizeof ( * pdata - > madc ) ,
2009-12-13 21:23:33 +01:00
true , pdata - > irq_base + MADC_INTR_OFFSET , 0 ) ;
2008-12-01 00:31:04 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2008-10-15 12:15:39 +02:00
}
if ( twl_has_rtc ( ) ) {
/*
2008-12-01 00:31:04 +01:00
* REVISIT platform_data here currently might expose the
2008-10-15 12:15:39 +02:00
* " msecure " line . . . but for now we just expect board
2008-12-01 00:31:04 +01:00
* setup to tell the chip " it's always ok to SET_TIME " .
2008-10-15 12:15:39 +02:00
* Eventually , Linux might become more aware of such
* HW security concerns , and " least privilege " .
*/
2009-12-14 00:25:31 +01:00
sub_chip_id = twl_map [ TWL_MODULE_RTC ] . sid ;
child = add_child ( sub_chip_id , " twl_rtc " ,
2008-12-01 00:31:04 +01:00
NULL , 0 ,
2009-12-13 21:23:33 +01:00
true , pdata - > irq_base + RTC_INTR_OFFSET , 0 ) ;
2008-12-01 00:31:04 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2008-10-15 12:15:39 +02:00
}
2009-12-13 22:29:47 +01:00
if ( twl_has_usb ( ) & & pdata - > usb & & twl_class_is_4030 ( ) ) {
2009-09-30 18:26:37 +03:00
static struct regulator_consumer_supply usb1v5 = {
. supply = " usb1v5 " ,
} ;
static struct regulator_consumer_supply usb1v8 = {
. supply = " usb1v8 " ,
} ;
static struct regulator_consumer_supply usb3v1 = {
. supply = " usb3v1 " ,
} ;
/* First add the regulators so that they can be used by transceiver */
if ( twl_has_regulator ( ) ) {
/* this is a template that gets copied */
struct regulator_init_data usb_fixed = {
. constraints . valid_modes_mask =
REGULATOR_MODE_NORMAL
| REGULATOR_MODE_STANDBY ,
. constraints . valid_ops_mask =
REGULATOR_CHANGE_MODE
| REGULATOR_CHANGE_STATUS ,
} ;
child = add_regulator_linked ( TWL4030_REG_VUSB1V5 ,
2011-05-12 14:27:55 +01:00
& usb_fixed , & usb1v5 , 1 ,
features ) ;
2009-09-30 18:26:37 +03:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
child = add_regulator_linked ( TWL4030_REG_VUSB1V8 ,
2011-05-12 14:27:55 +01:00
& usb_fixed , & usb1v8 , 1 ,
features ) ;
2009-09-30 18:26:37 +03:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
child = add_regulator_linked ( TWL4030_REG_VUSB3V1 ,
2011-05-12 14:27:55 +01:00
& usb_fixed , & usb3v1 , 1 ,
features ) ;
2009-09-30 18:26:37 +03:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
}
2008-12-01 00:31:04 +01:00
child = add_child ( 0 , " twl4030_usb " ,
pdata - > usb , sizeof ( * pdata - > usb ) ,
true ,
/* irq0 = USB_PRES, irq1 = USB */
2009-12-13 21:23:33 +01:00
pdata - > irq_base + USB_PRES_INTR_OFFSET ,
pdata - > irq_base + USB_INTR_OFFSET ) ;
2009-09-30 18:26:37 +03:00
2008-12-01 00:31:04 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2008-12-01 00:43:58 +01:00
/* we need to connect regulators to this transceiver */
2009-09-30 18:26:37 +03:00
if ( twl_has_regulator ( ) & & child ) {
usb1v5 . dev = child ;
usb1v8 . dev = child ;
usb3v1 . dev = child ;
}
2008-12-01 00:43:58 +01:00
}
2010-12-10 18:09:52 +05:30
if ( twl_has_usb ( ) & & pdata - > usb & & twl_class_is_6030 ( ) ) {
2011-05-12 14:27:55 +01:00
static struct regulator_consumer_supply usb3v3 ;
int regulator ;
2010-12-10 18:09:52 +05:30
if ( twl_has_regulator ( ) ) {
/* this is a template that gets copied */
struct regulator_init_data usb_fixed = {
. constraints . valid_modes_mask =
REGULATOR_MODE_NORMAL
| REGULATOR_MODE_STANDBY ,
. constraints . valid_ops_mask =
REGULATOR_CHANGE_MODE
| REGULATOR_CHANGE_STATUS ,
} ;
2011-05-12 14:27:55 +01:00
if ( features & TWL6025_SUBCLASS ) {
usb3v3 . supply = " ldousb " ;
regulator = TWL6025_REG_LDOUSB ;
} else {
usb3v3 . supply = " vusb " ;
regulator = TWL6030_REG_VUSB ;
}
child = add_regulator_linked ( regulator , & usb_fixed ,
& usb3v3 , 1 ,
features ) ;
2010-12-10 18:09:52 +05:30
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
}
2011-05-12 14:27:55 +01:00
pdata - > usb - > features = features ;
2010-12-10 18:09:52 +05:30
child = add_child ( 0 , " twl6030_usb " ,
pdata - > usb , sizeof ( * pdata - > usb ) ,
true ,
/* irq1 = VBUS_PRES, irq0 = USB ID */
pdata - > irq_base + USBOTG_INTR_OFFSET ,
pdata - > irq_base + USB_PRES_INTR_OFFSET ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
/* we need to connect regulators to this transceiver */
if ( twl_has_regulator ( ) & & child )
usb3v3 . dev = child ;
2011-05-12 14:27:55 +01:00
} else if ( twl_has_regulator ( ) & & twl_class_is_6030 ( ) ) {
if ( features & TWL6025_SUBCLASS )
child = add_regulator ( TWL6025_REG_LDOUSB ,
pdata - > ldousb , features ) ;
else
child = add_regulator ( TWL6030_REG_VUSB ,
pdata - > vusb , features ) ;
2010-12-10 18:09:52 +05:30
2011-05-12 14:27:55 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2010-12-10 18:09:52 +05:30
}
2008-12-01 00:43:58 +01:00
2011-02-23 16:26:54 +05:30
if ( twl_has_watchdog ( ) & & twl_class_is_4030 ( ) ) {
2009-03-27 16:42:17 +02:00
child = add_child ( 0 , " twl4030_wdt " , NULL , 0 , false , 0 , 0 ) ;
if ( IS_ERR ( child ) )
2009-08-03 18:16:38 +02:00
return PTR_ERR ( child ) ;
}
2011-02-23 16:26:54 +05:30
if ( twl_has_pwrbutton ( ) & & twl_class_is_4030 ( ) ) {
2009-08-03 18:16:38 +02:00
child = add_child ( 1 , " twl4030_pwrbutton " ,
NULL , 0 , true , pdata - > irq_base + 8 + 0 , 0 ) ;
if ( IS_ERR ( child ) )
2009-03-27 16:42:17 +02:00
return PTR_ERR ( child ) ;
}
2011-05-31 15:21:13 +03:00
if ( twl_has_codec ( ) & & pdata - > audio & & twl_class_is_4030 ( ) ) {
2010-02-23 18:10:19 -06:00
sub_chip_id = twl_map [ TWL_MODULE_AUDIO_VOICE ] . sid ;
2010-03-17 20:15:21 +00:00
child = add_child ( sub_chip_id , " twl4030-audio " ,
2011-05-31 15:21:13 +03:00
pdata - > audio , sizeof ( * pdata - > audio ) ,
2010-02-23 18:10:19 -06:00
false , 0 , 0 ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
}
2011-05-31 15:21:13 +03:00
if ( twl_has_codec ( ) & & pdata - > audio & & twl_class_is_6030 ( ) ) {
2010-02-23 18:10:19 -06:00
sub_chip_id = twl_map [ TWL_MODULE_AUDIO_VOICE ] . sid ;
2011-04-27 02:14:07 -05:00
child = add_child ( sub_chip_id , " twl6040 " ,
2011-05-31 15:21:13 +03:00
pdata - > audio , sizeof ( * pdata - > audio ) ,
2009-10-22 13:26:45 +03:00
false , 0 , 0 ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
}
2009-12-13 22:29:47 +01:00
/* twl4030 regulators */
if ( twl_has_regulator ( ) & & twl_class_is_4030 ( ) ) {
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VPLL1 , pdata - > vpll1 ,
features ) ;
2008-12-01 00:43:58 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2009-12-11 11:12:15 +01:00
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VIO , pdata - > vio ,
features ) ;
2009-12-11 11:12:15 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VDD1 , pdata - > vdd1 ,
features ) ;
2009-12-11 11:12:15 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VDD2 , pdata - > vdd2 ,
features ) ;
2009-12-11 11:12:15 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2008-12-01 00:43:58 +01:00
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VMMC1 , pdata - > vmmc1 ,
features ) ;
2008-12-01 00:43:58 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VDAC , pdata - > vdac ,
features ) ;
2008-12-01 00:43:58 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
child = add_regulator ( ( features & TWL4030_VAUX2 )
? TWL4030_REG_VAUX2_4030
: TWL4030_REG_VAUX2 ,
2011-05-12 14:27:55 +01:00
pdata - > vaux2 , features ) ;
2008-12-01 00:43:58 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2009-12-11 11:12:15 +01:00
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VINTANA1 , pdata - > vintana1 ,
features ) ;
2009-12-11 11:12:15 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VINTANA2 , pdata - > vintana2 ,
features ) ;
2009-12-11 11:12:15 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VINTDIG , pdata - > vintdig ,
features ) ;
2009-12-11 11:12:15 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2008-12-01 00:43:58 +01:00
}
/* maybe add LDOs that are omitted on cost-reduced parts */
2009-12-13 22:29:47 +01:00
if ( twl_has_regulator ( ) & & ! ( features & TPS_SUBSET )
& & twl_class_is_4030 ( ) ) {
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VPLL2 , pdata - > vpll2 ,
features ) ;
2008-12-01 00:43:58 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VMMC2 , pdata - > vmmc2 ,
features ) ;
2008-12-01 00:43:58 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VSIM , pdata - > vsim ,
features ) ;
2008-12-01 00:43:58 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VAUX1 , pdata - > vaux1 ,
features ) ;
2008-12-01 00:43:58 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VAUX3 , pdata - > vaux3 ,
features ) ;
2008-12-01 00:43:58 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL4030_REG_VAUX4 , pdata - > vaux4 ,
features ) ;
2008-12-01 00:43:58 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2008-10-15 12:15:39 +02:00
}
2009-12-13 22:29:47 +01:00
/* twl6030 regulators */
2011-05-12 14:27:55 +01:00
if ( twl_has_regulator ( ) & & twl_class_is_6030 ( ) & &
! ( features & TWL6025_SUBCLASS ) ) {
child = add_regulator ( TWL6030_REG_VMMC , pdata - > vmmc ,
features ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
child = add_regulator ( TWL6030_REG_VPP , pdata - > vpp ,
features ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
child = add_regulator ( TWL6030_REG_VUSIM , pdata - > vusim ,
features ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
child = add_regulator ( TWL6030_REG_VCXIO , pdata - > vcxio ,
features ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
child = add_regulator ( TWL6030_REG_VDAC , pdata - > vdac ,
features ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
child = add_regulator ( TWL6030_REG_VAUX1_6030 , pdata - > vaux1 ,
features ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
child = add_regulator ( TWL6030_REG_VAUX2_6030 , pdata - > vaux2 ,
features ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
child = add_regulator ( TWL6030_REG_VAUX3_6030 , pdata - > vaux3 ,
features ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
child = add_regulator ( TWL6030_REG_CLK32KG , pdata - > clk32kg ,
features ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
}
/* 6030 and 6025 share this regulator */
2009-12-13 22:29:47 +01:00
if ( twl_has_regulator ( ) & & twl_class_is_6030 ( ) ) {
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL6030_REG_VANA , pdata - > vana ,
features ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
}
/* twl6025 regulators */
if ( twl_has_regulator ( ) & & twl_class_is_6030 ( ) & &
( features & TWL6025_SUBCLASS ) ) {
child = add_regulator ( TWL6025_REG_LDO5 , pdata - > ldo5 ,
features ) ;
2009-12-13 22:29:47 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL6025_REG_LDO1 , pdata - > ldo1 ,
features ) ;
2009-12-13 22:29:47 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL6025_REG_LDO7 , pdata - > ldo7 ,
features ) ;
2009-12-13 22:29:47 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL6025_REG_LDO6 , pdata - > ldo6 ,
features ) ;
2009-12-13 22:29:47 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL6025_REG_LDOLN , pdata - > ldoln ,
features ) ;
2009-12-13 22:29:47 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL6025_REG_LDO2 , pdata - > ldo2 ,
features ) ;
2009-12-13 22:29:47 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL6025_REG_LDO4 , pdata - > ldo4 ,
features ) ;
2009-12-13 22:29:47 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL6025_REG_LDO3 , pdata - > ldo3 ,
features ) ;
2009-12-13 22:29:47 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL6025_REG_SMPS3 , pdata - > smps3 ,
features ) ;
2009-12-13 22:29:47 +01:00
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-02-10 18:44:50 +05:30
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL6025_REG_SMPS4 , pdata - > smps4 ,
features ) ;
2011-02-10 18:44:50 +05:30
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2011-05-12 14:27:55 +01:00
child = add_regulator ( TWL6025_REG_VIO , pdata - > vio6025 ,
features ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
2009-12-13 22:29:47 +01:00
}
2010-10-01 13:17:50 +03:00
if ( twl_has_bci ( ) & & pdata - > bci & &
! ( features & ( TPS_SUBSET | TWL5031 ) ) ) {
child = add_child ( 3 , " twl4030_bci " ,
pdata - > bci , sizeof ( * pdata - > bci ) , false ,
/* irq0 = CHG_PRES, irq1 = BCI */
pdata - > irq_base + BCI_PRES_INTR_OFFSET ,
pdata - > irq_base + BCI_INTR_OFFSET ) ;
if ( IS_ERR ( child ) )
return PTR_ERR ( child ) ;
}
2008-12-01 00:31:04 +01:00
return 0 ;
2008-10-15 12:15:39 +02:00
}
/*----------------------------------------------------------------------*/
/*
* These three functions initialize the on - chip clock framework ,
* letting it generate the right frequencies for USB , MADC , and
* other purposes .
*/
static inline int __init protect_pm_master ( void )
{
int e = 0 ;
2010-08-18 09:19:35 +03:00
e = twl_i2c_write_u8 ( TWL4030_MODULE_PM_MASTER , 0 ,
TWL4030_PM_MASTER_PROTECT_KEY ) ;
2008-10-15 12:15:39 +02:00
return e ;
}
static inline int __init unprotect_pm_master ( void )
{
int e = 0 ;
2010-08-18 09:19:35 +03:00
e | = twl_i2c_write_u8 ( TWL4030_MODULE_PM_MASTER ,
TWL4030_PM_MASTER_KEY_CFG1 ,
TWL4030_PM_MASTER_PROTECT_KEY ) ;
e | = twl_i2c_write_u8 ( TWL4030_MODULE_PM_MASTER ,
TWL4030_PM_MASTER_KEY_CFG2 ,
TWL4030_PM_MASTER_PROTECT_KEY ) ;
2008-10-15 12:15:39 +02:00
return e ;
}
2009-10-22 14:14:09 +03:00
static void clocks_init ( struct device * dev ,
struct twl4030_clock_init_data * clock )
2008-10-15 12:15:39 +02:00
{
int e = 0 ;
struct clk * osc ;
u32 rate ;
u8 ctrl = HFCLK_FREQ_26_MHZ ;
# if defined(CONFIG_ARCH_OMAP2) || defined(CONFIG_ARCH_OMAP3)
if ( cpu_is_omap2430 ( ) )
2009-01-10 10:40:42 +00:00
osc = clk_get ( dev , " osc_ck " ) ;
2008-10-15 12:15:39 +02:00
else
2009-01-10 10:40:42 +00:00
osc = clk_get ( dev , " osc_sys_ck " ) ;
2008-12-07 20:50:25 +01:00
2008-10-15 12:15:39 +02:00
if ( IS_ERR ( osc ) ) {
2009-12-13 21:23:33 +01:00
printk ( KERN_WARNING " Skipping twl internal clock init and "
2008-10-15 12:15:39 +02:00
" using bootloader value (unknown osc rate) \n " ) ;
return ;
}
rate = clk_get_rate ( osc ) ;
clk_put ( osc ) ;
2008-12-07 20:50:25 +01:00
# else
/* REVISIT for non-OMAP systems, pass the clock rate from
* board init code , using platform_data .
*/
osc = ERR_PTR ( - EIO ) ;
2009-12-13 21:23:33 +01:00
printk ( KERN_WARNING " Skipping twl internal clock init and "
2008-12-07 20:50:25 +01:00
" using bootloader value (unknown osc rate) \n " ) ;
return ;
# endif
2008-10-15 12:15:39 +02:00
switch ( rate ) {
case 19200000 :
ctrl = HFCLK_FREQ_19p2_MHZ ;
break ;
case 26000000 :
ctrl = HFCLK_FREQ_26_MHZ ;
break ;
case 38400000 :
ctrl = HFCLK_FREQ_38p4_MHZ ;
break ;
}
ctrl | = HIGH_PERF_SQ ;
2009-10-22 14:14:09 +03:00
if ( clock & & clock - > ck32k_lowpwr_enable )
ctrl | = CK32K_LOWPWR_EN ;
2008-10-15 12:15:39 +02:00
e | = unprotect_pm_master ( ) ;
/* effect->MADC+USB ck en */
2009-12-13 21:23:33 +01:00
e | = twl_i2c_write_u8 ( TWL_MODULE_PM_MASTER , ctrl , R_CFG_BOOT ) ;
2008-10-15 12:15:39 +02:00
e | = protect_pm_master ( ) ;
if ( e < 0 )
pr_err ( " %s: clock init err [%d] \n " , DRIVER_NAME , e ) ;
}
/*----------------------------------------------------------------------*/
2009-12-14 00:25:31 +01:00
int twl4030_init_irq ( int irq_num , unsigned irq_base , unsigned irq_end ) ;
int twl4030_exit_irq ( void ) ;
int twl4030_init_chip_irq ( const char * chip ) ;
int twl6030_init_irq ( int irq_num , unsigned irq_base , unsigned irq_end ) ;
int twl6030_exit_irq ( void ) ;
2008-10-15 12:15:39 +02:00
2009-12-13 21:23:33 +01:00
static int twl_remove ( struct i2c_client * client )
2008-10-15 12:15:39 +02:00
{
unsigned i ;
2008-10-20 23:46:28 +02:00
int status ;
2008-10-15 12:15:39 +02:00
2009-12-14 00:25:31 +01:00
if ( twl_class_is_4030 ( ) )
status = twl4030_exit_irq ( ) ;
else
status = twl6030_exit_irq ( ) ;
2008-10-20 23:46:28 +02:00
if ( status < 0 )
return status ;
2008-10-15 12:15:39 +02:00
2009-12-13 21:23:33 +01:00
for ( i = 0 ; i < TWL_NUM_SLAVES ; i + + ) {
struct twl_client * twl = & twl_modules [ i ] ;
2008-10-15 12:15:39 +02:00
if ( twl - > client & & twl - > client ! = client )
i2c_unregister_device ( twl - > client ) ;
2009-12-13 21:23:33 +01:00
twl_modules [ i ] . client = NULL ;
2008-10-15 12:15:39 +02:00
}
inuse = false ;
return 0 ;
}
/* NOTE: this driver only handles a single twl4030/tps659x0 chip */
2010-12-08 10:42:04 +08:00
static int __devinit
2009-12-13 21:23:33 +01:00
twl_probe ( struct i2c_client * client , const struct i2c_device_id * id )
2008-10-15 12:15:39 +02:00
{
int status ;
unsigned i ;
struct twl4030_platform_data * pdata = client - > dev . platform_data ;
2010-02-16 18:57:21 -06:00
u8 temp ;
2011-04-14 17:57:53 +05:30
int ret = 0 ;
2008-10-15 12:15:39 +02:00
if ( ! pdata ) {
dev_dbg ( & client - > dev , " no platform data? \n " ) ;
return - EINVAL ;
}
if ( i2c_check_functionality ( client - > adapter , I2C_FUNC_I2C ) = = 0 ) {
dev_dbg ( & client - > dev , " can't talk I2C? \n " ) ;
return - EIO ;
}
2008-10-20 23:46:28 +02:00
if ( inuse ) {
2008-10-15 12:15:39 +02:00
dev_dbg ( & client - > dev , " driver is already in use \n " ) ;
return - EBUSY ;
}
2009-12-13 21:23:33 +01:00
for ( i = 0 ; i < TWL_NUM_SLAVES ; i + + ) {
struct twl_client * twl = & twl_modules [ i ] ;
2008-10-15 12:15:39 +02:00
twl - > address = client - > addr + i ;
if ( i = = 0 )
twl - > client = client ;
else {
twl - > client = i2c_new_dummy ( client - > adapter ,
twl - > address ) ;
if ( ! twl - > client ) {
2009-10-20 16:22:52 +03:00
dev_err ( & client - > dev ,
2008-10-15 12:15:39 +02:00
" can't attach client %d \n " , i ) ;
status = - ENOMEM ;
goto fail ;
}
}
mutex_init ( & twl - > xfer_lock ) ;
}
inuse = true ;
2009-12-14 00:25:31 +01:00
if ( ( id - > driver_data ) & TWL6030_CLASS ) {
twl_id = TWL6030_CLASS_ID ;
twl_map = & twl6030_map [ 0 ] ;
} else {
twl_id = TWL4030_CLASS_ID ;
twl_map = & twl4030_map [ 0 ] ;
}
2008-10-15 12:15:39 +02:00
/* setup clock framework */
2009-10-22 14:14:09 +03:00
clocks_init ( & client - > dev , pdata - > clock ) ;
2008-10-15 12:15:39 +02:00
2011-04-14 17:57:53 +05:30
/* read TWL IDCODE Register */
if ( twl_id = = TWL4030_CLASS_ID ) {
ret = twl_read_idcode_register ( ) ;
WARN ( ret < 0 , " Error: reading twl_idcode register value \n " ) ;
}
2009-08-31 18:32:18 +02:00
/* load power event scripts */
if ( twl_has_power ( ) & & pdata - > power )
twl4030_power_init ( pdata - > power ) ;
2008-10-15 12:15:39 +02:00
/* Maybe init the T2 Interrupt subsystem */
if ( client - > irq
& & pdata - > irq_base
& & pdata - > irq_end > pdata - > irq_base ) {
2009-12-14 00:25:31 +01:00
if ( twl_class_is_4030 ( ) ) {
twl4030_init_chip_irq ( id - > name ) ;
status = twl4030_init_irq ( client - > irq , pdata - > irq_base ,
pdata - > irq_end ) ;
} else {
status = twl6030_init_irq ( client - > irq , pdata - > irq_base ,
pdata - > irq_end ) ;
}
2008-10-20 23:46:28 +02:00
if ( status < 0 )
goto fail ;
2008-10-15 12:15:39 +02:00
}
2010-02-16 18:57:21 -06:00
/* Disable TWL4030/TWL5030 I2C Pull-up on I2C1 and I2C4(SR) interface.
* Program I2C_SCL_CTRL_PU ( bit 0 ) = 0 , I2C_SDA_CTRL_PU ( bit 2 ) = 0 ,
* SR_I2C_SCL_CTRL_PU ( bit 4 ) = 0 and SR_I2C_SDA_CTRL_PU ( bit 6 ) = 0.
*/
if ( twl_class_is_4030 ( ) ) {
twl_i2c_read_u8 ( TWL4030_MODULE_INTBR , & temp , REG_GPPUPDCTR1 ) ;
temp & = ~ ( SR_I2C_SDA_CTRL_PU | SR_I2C_SCL_CTRL_PU | \
I2C_SDA_CTRL_PU | I2C_SCL_CTRL_PU ) ;
twl_i2c_write_u8 ( TWL4030_MODULE_INTBR , temp , REG_GPPUPDCTR1 ) ;
}
2008-12-01 00:43:58 +01:00
status = add_children ( pdata , id - > driver_data ) ;
2008-10-15 12:15:39 +02:00
fail :
if ( status < 0 )
2009-12-13 21:23:33 +01:00
twl_remove ( client ) ;
2008-10-15 12:15:39 +02:00
return status ;
}
2009-12-13 21:23:33 +01:00
static const struct i2c_device_id twl_ids [ ] = {
2008-12-01 00:43:58 +01:00
{ " twl4030 " , TWL4030_VAUX2 } , /* "Triton 2" */
{ " twl5030 " , 0 } , /* T2 updated */
2009-11-10 17:26:15 +02:00
{ " twl5031 " , TWL5031 } , /* TWL5030 updated */
2008-12-01 00:43:58 +01:00
{ " tps65950 " , 0 } , /* catalog version of twl5030 */
{ " tps65930 " , TPS_SUBSET } , /* fewer LDOs and DACs; no charger */
{ " tps65920 " , TPS_SUBSET } , /* fewer LDOs; no codec or charger */
2011-07-01 13:34:13 +02:00
{ " tps65921 " , TPS_SUBSET } , /* fewer LDOs; no codec, no LED
and vibrator . Charger in USB module */
2009-12-14 00:25:31 +01:00
{ " twl6030 " , TWL6030_CLASS } , /* "Phoenix power chip" */
2011-05-12 14:27:55 +01:00
{ " twl6025 " , TWL6030_CLASS | TWL6025_SUBCLASS } , /* "Phoenix lite" */
2008-10-15 12:15:39 +02:00
{ /* end of list */ } ,
} ;
2009-12-13 21:23:33 +01:00
MODULE_DEVICE_TABLE ( i2c , twl_ids ) ;
2008-10-15 12:15:39 +02:00
/* One Client Driver , 4 Clients */
2009-12-13 21:23:33 +01:00
static struct i2c_driver twl_driver = {
2008-10-15 12:15:39 +02:00
. driver . name = DRIVER_NAME ,
2009-12-13 21:23:33 +01:00
. id_table = twl_ids ,
. probe = twl_probe ,
. remove = twl_remove ,
2008-10-15 12:15:39 +02:00
} ;
2009-12-13 21:23:33 +01:00
static int __init twl_init ( void )
2008-10-15 12:15:39 +02:00
{
2009-12-13 21:23:33 +01:00
return i2c_add_driver ( & twl_driver ) ;
2008-10-15 12:15:39 +02:00
}
2009-12-13 21:23:33 +01:00
subsys_initcall ( twl_init ) ;
2008-10-15 12:15:39 +02:00
2009-12-13 21:23:33 +01:00
static void __exit twl_exit ( void )
2008-10-15 12:15:39 +02:00
{
2009-12-13 21:23:33 +01:00
i2c_del_driver ( & twl_driver ) ;
2008-10-15 12:15:39 +02:00
}
2009-12-13 21:23:33 +01:00
module_exit ( twl_exit ) ;
2008-10-15 12:15:39 +02:00
MODULE_AUTHOR ( " Texas Instruments, Inc. " ) ;
2009-12-13 21:23:33 +01:00
MODULE_DESCRIPTION ( " I2C Core interface for TWL " ) ;
2008-10-15 12:15:39 +02:00
MODULE_LICENSE ( " GPL " ) ;