2020-08-09 12:59:38 +02:00
// SPDX-License-Identifier: GPL-2.0
/*
* Copyright ( C ) 2020 Marek Vasut < marex @ denx . de >
*
* Based on rpi_touchscreen . c by Eric Anholt < eric @ anholt . net >
*/
# include <linux/backlight.h>
# include <linux/err.h>
# include <linux/gpio.h>
2022-01-24 17:01:26 -05:00
# include <linux/gpio/driver.h>
2020-08-09 12:59:38 +02:00
# include <linux/i2c.h>
# include <linux/init.h>
# include <linux/interrupt.h>
# include <linux/module.h>
# include <linux/regmap.h>
# include <linux/regulator/driver.h>
# include <linux/regulator/machine.h>
# include <linux/regulator/of_regulator.h>
# include <linux/slab.h>
/* I2C registers of the Atmel microcontroller. */
# define REG_ID 0x80
# define REG_PORTA 0x81
# define REG_PORTB 0x82
2022-01-24 17:01:25 -05:00
# define REG_PORTC 0x83
2020-08-09 12:59:38 +02:00
# define REG_POWERON 0x85
# define REG_PWM 0x86
2022-01-24 17:01:25 -05:00
# define REG_ADDR_L 0x8c
# define REG_ADDR_H 0x8d
# define REG_WRITE_DATA_H 0x90
# define REG_WRITE_DATA_L 0x91
# define PA_LCD_DITHB BIT(0)
# define PA_LCD_MODE BIT(1)
# define PA_LCD_LR BIT(2)
# define PA_LCD_UD BIT(3)
# define PB_BRIDGE_PWRDNX_N BIT(0)
# define PB_LCD_VCC_N BIT(1)
# define PB_LCD_MAIN BIT(7)
# define PC_LED_EN BIT(0)
# define PC_RST_TP_N BIT(1)
# define PC_RST_LCD_N BIT(2)
# define PC_RST_BRIDGE_N BIT(3)
2020-08-09 12:59:38 +02:00
2022-01-24 17:01:26 -05:00
enum gpio_signals {
RST_BRIDGE_N , /* TC358762 bridge reset */
RST_TP_N , /* Touch controller reset */
NUM_GPIO
} ;
struct gpio_signal_mappings {
unsigned int reg ;
unsigned int mask ;
} ;
static const struct gpio_signal_mappings mappings [ NUM_GPIO ] = {
[ RST_BRIDGE_N ] = { REG_PORTC , PC_RST_BRIDGE_N | PC_RST_LCD_N } ,
[ RST_TP_N ] = { REG_PORTC , PC_RST_TP_N } ,
} ;
2022-01-24 17:01:23 -05:00
struct attiny_lcd {
/* lock to serialise overall accesses to the Atmel */
struct mutex lock ;
struct regmap * regmap ;
2022-01-24 17:01:26 -05:00
bool gpio_states [ NUM_GPIO ] ;
u8 port_states [ 3 ] ;
struct gpio_chip gc ;
2022-01-24 17:01:23 -05:00
} ;
2020-08-09 12:59:38 +02:00
static const struct regmap_config attiny_regmap_config = {
. reg_bits = 8 ,
. val_bits = 8 ,
2022-01-24 17:01:23 -05:00
. disable_locking = 1 ,
2022-01-24 17:01:25 -05:00
. max_register = REG_WRITE_DATA_L ,
2020-08-09 12:59:38 +02:00
. cache_type = REGCACHE_NONE ,
} ;
2022-01-24 17:01:26 -05:00
static int attiny_set_port_state ( struct attiny_lcd * state , int reg , u8 val )
{
state - > port_states [ reg - REG_PORTA ] = val ;
return regmap_write ( state - > regmap , reg , val ) ;
} ;
static u8 attiny_get_port_state ( struct attiny_lcd * state , int reg )
{
return state - > port_states [ reg - REG_PORTA ] ;
} ;
2020-08-09 12:59:38 +02:00
static int attiny_lcd_power_enable ( struct regulator_dev * rdev )
{
2022-01-24 17:01:25 -05:00
struct attiny_lcd * state = rdev_get_drvdata ( rdev ) ;
2022-01-24 17:01:22 -05:00
2022-01-24 17:01:25 -05:00
mutex_lock ( & state - > lock ) ;
2022-01-24 17:01:22 -05:00
2022-01-24 17:01:25 -05:00
/* Ensure bridge, and tp stay in reset */
2022-01-24 17:01:26 -05:00
attiny_set_port_state ( state , REG_PORTC , 0 ) ;
2022-01-24 17:01:25 -05:00
usleep_range ( 5000 , 10000 ) ;
2020-08-09 12:59:38 +02:00
/* Default to the same orientation as the closed source
* firmware used for the panel . Runtime rotation
* configuration will be supported using VC4 ' s plane
* orientation bits .
*/
2022-01-24 17:01:26 -05:00
attiny_set_port_state ( state , REG_PORTA , PA_LCD_LR ) ;
2022-01-24 17:01:25 -05:00
usleep_range ( 5000 , 10000 ) ;
2022-01-24 17:01:26 -05:00
/* Main regulator on, and power to the panel (LCD_VCC_N) */
attiny_set_port_state ( state , REG_PORTB , PB_LCD_MAIN ) ;
2022-01-24 17:01:25 -05:00
usleep_range ( 5000 , 10000 ) ;
/* Bring controllers out of reset */
2022-01-24 17:01:26 -05:00
attiny_set_port_state ( state , REG_PORTC , PC_LED_EN ) ;
2022-01-24 17:01:25 -05:00
msleep ( 80 ) ;
mutex_unlock ( & state - > lock ) ;
2022-01-24 17:01:23 -05:00
2020-08-09 12:59:38 +02:00
return 0 ;
}
static int attiny_lcd_power_disable ( struct regulator_dev * rdev )
{
2022-01-24 17:01:25 -05:00
struct attiny_lcd * state = rdev_get_drvdata ( rdev ) ;
2022-01-24 17:01:23 -05:00
2022-01-24 17:01:25 -05:00
mutex_lock ( & state - > lock ) ;
2022-01-24 17:01:23 -05:00
2020-08-09 12:59:38 +02:00
regmap_write ( rdev - > regmap , REG_PWM , 0 ) ;
2022-01-24 17:01:25 -05:00
usleep_range ( 5000 , 10000 ) ;
2022-01-24 17:01:26 -05:00
attiny_set_port_state ( state , REG_PORTA , 0 ) ;
2022-01-24 17:01:25 -05:00
usleep_range ( 5000 , 10000 ) ;
2022-01-24 17:01:26 -05:00
attiny_set_port_state ( state , REG_PORTB , PB_LCD_VCC_N ) ;
2022-01-24 17:01:25 -05:00
usleep_range ( 5000 , 10000 ) ;
2022-01-24 17:01:26 -05:00
attiny_set_port_state ( state , REG_PORTC , 0 ) ;
2022-01-24 17:01:22 -05:00
msleep ( 30 ) ;
2022-01-24 17:01:23 -05:00
2022-01-24 17:01:25 -05:00
mutex_unlock ( & state - > lock ) ;
2022-01-24 17:01:23 -05:00
2020-08-09 12:59:38 +02:00
return 0 ;
}
static int attiny_lcd_power_is_enabled ( struct regulator_dev * rdev )
{
2022-01-24 17:01:25 -05:00
struct attiny_lcd * state = rdev_get_drvdata ( rdev ) ;
2020-08-09 12:59:38 +02:00
unsigned int data ;
2022-01-24 17:01:22 -05:00
int ret , i ;
2020-08-09 12:59:38 +02:00
2022-01-24 17:01:25 -05:00
mutex_lock ( & state - > lock ) ;
2020-08-09 12:59:38 +02:00
2022-01-24 17:01:22 -05:00
for ( i = 0 ; i < 10 ; i + + ) {
2022-01-24 17:01:25 -05:00
ret = regmap_read ( rdev - > regmap , REG_PORTC , & data ) ;
2022-01-24 17:01:22 -05:00
if ( ! ret )
break ;
usleep_range ( 10000 , 12000 ) ;
}
2022-01-24 17:01:25 -05:00
mutex_unlock ( & state - > lock ) ;
2022-01-24 17:01:23 -05:00
2020-08-09 12:59:38 +02:00
if ( ret < 0 )
return ret ;
2022-01-24 17:01:25 -05:00
return data & PC_RST_BRIDGE_N ;
2020-08-09 12:59:38 +02:00
}
static const struct regulator_init_data attiny_regulator_default = {
. constraints = {
. valid_ops_mask = REGULATOR_CHANGE_STATUS ,
} ,
} ;
static const struct regulator_ops attiny_regulator_ops = {
. enable = attiny_lcd_power_enable ,
. disable = attiny_lcd_power_disable ,
. is_enabled = attiny_lcd_power_is_enabled ,
} ;
static const struct regulator_desc attiny_regulator = {
. name = " tc358762-power " ,
. ops = & attiny_regulator_ops ,
. type = REGULATOR_VOLTAGE ,
. owner = THIS_MODULE ,
} ;
static int attiny_update_status ( struct backlight_device * bl )
{
2022-01-24 17:01:23 -05:00
struct attiny_lcd * state = bl_get_data ( bl ) ;
struct regmap * regmap = state - > regmap ;
2020-08-09 12:59:38 +02:00
int brightness = bl - > props . brightness ;
2022-01-24 17:01:22 -05:00
int ret , i ;
2020-08-09 12:59:38 +02:00
2022-01-24 17:01:23 -05:00
mutex_lock ( & state - > lock ) ;
2020-08-09 12:59:38 +02:00
if ( bl - > props . power ! = FB_BLANK_UNBLANK | |
bl - > props . fb_blank ! = FB_BLANK_UNBLANK )
brightness = 0 ;
2022-01-24 17:01:22 -05:00
for ( i = 0 ; i < 10 ; i + + ) {
ret = regmap_write ( regmap , REG_PWM , brightness ) ;
if ( ! ret )
break ;
}
2022-01-24 17:01:23 -05:00
mutex_unlock ( & state - > lock ) ;
2022-01-24 17:01:22 -05:00
return ret ;
2020-08-09 12:59:38 +02:00
}
static const struct backlight_ops attiny_bl = {
. update_status = attiny_update_status ,
} ;
2022-01-24 17:01:26 -05:00
static int attiny_gpio_get_direction ( struct gpio_chip * gc , unsigned int off )
{
return GPIO_LINE_DIRECTION_OUT ;
}
static void attiny_gpio_set ( struct gpio_chip * gc , unsigned int off , int val )
{
struct attiny_lcd * state = gpiochip_get_data ( gc ) ;
u8 last_val ;
if ( off > = NUM_GPIO )
return ;
mutex_lock ( & state - > lock ) ;
last_val = attiny_get_port_state ( state , mappings [ off ] . reg ) ;
if ( val )
last_val | = mappings [ off ] . mask ;
else
last_val & = ~ mappings [ off ] . mask ;
attiny_set_port_state ( state , mappings [ off ] . reg , last_val ) ;
if ( off = = RST_BRIDGE_N & & val ) {
usleep_range ( 5000 , 8000 ) ;
regmap_write ( state - > regmap , REG_ADDR_H , 0x04 ) ;
usleep_range ( 5000 , 8000 ) ;
regmap_write ( state - > regmap , REG_ADDR_L , 0x7c ) ;
usleep_range ( 5000 , 8000 ) ;
regmap_write ( state - > regmap , REG_WRITE_DATA_H , 0x00 ) ;
usleep_range ( 5000 , 8000 ) ;
regmap_write ( state - > regmap , REG_WRITE_DATA_L , 0x00 ) ;
msleep ( 100 ) ;
}
mutex_unlock ( & state - > lock ) ;
}
2020-08-09 12:59:38 +02:00
/*
* I2C driver interface functions
*/
static int attiny_i2c_probe ( struct i2c_client * i2c ,
const struct i2c_device_id * id )
{
struct backlight_properties props = { } ;
struct regulator_config config = { } ;
struct backlight_device * bl ;
struct regulator_dev * rdev ;
2022-01-24 17:01:23 -05:00
struct attiny_lcd * state ;
2020-08-09 12:59:38 +02:00
struct regmap * regmap ;
unsigned int data ;
int ret ;
2022-01-24 17:01:23 -05:00
state = devm_kzalloc ( & i2c - > dev , sizeof ( * state ) , GFP_KERNEL ) ;
if ( ! state )
return - ENOMEM ;
mutex_init ( & state - > lock ) ;
i2c_set_clientdata ( i2c , state ) ;
2020-08-09 12:59:38 +02:00
regmap = devm_regmap_init_i2c ( i2c , & attiny_regmap_config ) ;
if ( IS_ERR ( regmap ) ) {
ret = PTR_ERR ( regmap ) ;
dev_err ( & i2c - > dev , " Failed to allocate register map: %d \n " ,
ret ) ;
2022-01-24 17:01:23 -05:00
goto error ;
2020-08-09 12:59:38 +02:00
}
ret = regmap_read ( regmap , REG_ID , & data ) ;
if ( ret < 0 ) {
dev_err ( & i2c - > dev , " Failed to read REG_ID reg: %d \n " , ret ) ;
2022-01-24 17:01:23 -05:00
goto error ;
2020-08-09 12:59:38 +02:00
}
switch ( data ) {
case 0xde : /* ver 1 */
case 0xc3 : /* ver 2 */
break ;
default :
dev_err ( & i2c - > dev , " Unknown Atmel firmware revision: 0x%02x \n " , data ) ;
2022-01-24 17:01:23 -05:00
ret = - ENODEV ;
goto error ;
2020-08-09 12:59:38 +02:00
}
regmap_write ( regmap , REG_POWERON , 0 ) ;
2022-01-24 17:01:22 -05:00
msleep ( 30 ) ;
2022-01-24 17:01:24 -05:00
regmap_write ( regmap , REG_PWM , 0 ) ;
2020-08-09 12:59:38 +02:00
config . dev = & i2c - > dev ;
config . regmap = regmap ;
config . of_node = i2c - > dev . of_node ;
config . init_data = & attiny_regulator_default ;
2022-01-24 17:01:25 -05:00
config . driver_data = state ;
2020-08-09 12:59:38 +02:00
rdev = devm_regulator_register ( & i2c - > dev , & attiny_regulator , & config ) ;
if ( IS_ERR ( rdev ) ) {
dev_err ( & i2c - > dev , " Failed to register ATTINY regulator \n " ) ;
2022-01-24 17:01:23 -05:00
ret = PTR_ERR ( rdev ) ;
goto error ;
2020-08-09 12:59:38 +02:00
}
props . type = BACKLIGHT_RAW ;
props . max_brightness = 0xff ;
2022-01-24 17:01:23 -05:00
state - > regmap = regmap ;
2022-01-24 17:01:21 -05:00
bl = devm_backlight_device_register ( & i2c - > dev , dev_name ( & i2c - > dev ) ,
2022-01-24 17:01:23 -05:00
& i2c - > dev , state , & attiny_bl ,
2020-08-09 12:59:38 +02:00
& props ) ;
2022-01-24 17:01:23 -05:00
if ( IS_ERR ( bl ) ) {
ret = PTR_ERR ( bl ) ;
goto error ;
}
2020-08-09 12:59:38 +02:00
bl - > props . brightness = 0xff ;
2022-01-24 17:01:26 -05:00
state - > gc . parent = & i2c - > dev ;
state - > gc . label = i2c - > name ;
state - > gc . owner = THIS_MODULE ;
state - > gc . of_node = i2c - > dev . of_node ;
state - > gc . base = - 1 ;
state - > gc . ngpio = NUM_GPIO ;
state - > gc . set = attiny_gpio_set ;
state - > gc . get_direction = attiny_gpio_get_direction ;
state - > gc . can_sleep = true ;
ret = devm_gpiochip_add_data ( & i2c - > dev , & state - > gc , state ) ;
if ( ret ) {
dev_err ( & i2c - > dev , " Failed to create gpiochip: %d \n " , ret ) ;
goto error ;
}
2022-01-24 17:01:23 -05:00
return 0 ;
error :
mutex_destroy ( & state - > lock ) ;
return ret ;
}
static int attiny_i2c_remove ( struct i2c_client * client )
{
struct attiny_lcd * state = i2c_get_clientdata ( client ) ;
mutex_destroy ( & state - > lock ) ;
2020-08-09 12:59:38 +02:00
return 0 ;
}
static const struct of_device_id attiny_dt_ids [ ] = {
{ . compatible = " raspberrypi,7inch-touchscreen-panel-regulator " } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , attiny_dt_ids ) ;
static struct i2c_driver attiny_regulator_driver = {
. driver = {
. name = " rpi_touchscreen_attiny " ,
. of_match_table = of_match_ptr ( attiny_dt_ids ) ,
} ,
. probe = attiny_i2c_probe ,
2022-01-24 17:01:23 -05:00
. remove = attiny_i2c_remove ,
2020-08-09 12:59:38 +02:00
} ;
module_i2c_driver ( attiny_regulator_driver ) ;
MODULE_AUTHOR ( " Marek Vasut <marex@denx.de> " ) ;
MODULE_DESCRIPTION ( " Regulator device driver for Raspberry Pi 7-inch touchscreen " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;