2016-06-16 12:36:17 +03:00
# include <linux/amba/bus.h>
# include <linux/amba/clcd.h>
# include <linux/gpio/consumer.h>
# include <linux/of.h>
# include <linux/of_graph.h>
# include <linux/delay.h>
# include <linux/bitops.h>
# include <linux/mfd/syscon.h>
# include <linux/regmap.h>
# include "amba-clcd-nomadik.h"
static struct gpio_desc * grestb ;
static struct gpio_desc * scen ;
static struct gpio_desc * scl ;
static struct gpio_desc * sda ;
static u8 tpg110_readwrite_reg ( bool write , u8 address , u8 outval )
{
int i ;
u8 inval = 0 ;
/* Assert SCEN */
gpiod_set_value_cansleep ( scen , 1 ) ;
ndelay ( 150 ) ;
/* Hammer out the address */
for ( i = 5 ; i > = 0 ; i - - ) {
if ( address & BIT ( i ) )
gpiod_set_value_cansleep ( sda , 1 ) ;
else
gpiod_set_value_cansleep ( sda , 0 ) ;
ndelay ( 150 ) ;
/* Send an SCL pulse */
gpiod_set_value_cansleep ( scl , 1 ) ;
ndelay ( 160 ) ;
gpiod_set_value_cansleep ( scl , 0 ) ;
ndelay ( 160 ) ;
}
if ( write ) {
/* WRITE */
gpiod_set_value_cansleep ( sda , 0 ) ;
} else {
/* READ */
gpiod_set_value_cansleep ( sda , 1 ) ;
}
ndelay ( 150 ) ;
/* Send an SCL pulse */
gpiod_set_value_cansleep ( scl , 1 ) ;
ndelay ( 160 ) ;
gpiod_set_value_cansleep ( scl , 0 ) ;
ndelay ( 160 ) ;
if ( ! write )
/* HiZ turn-around cycle */
gpiod_direction_input ( sda ) ;
ndelay ( 150 ) ;
/* Send an SCL pulse */
gpiod_set_value_cansleep ( scl , 1 ) ;
ndelay ( 160 ) ;
gpiod_set_value_cansleep ( scl , 0 ) ;
ndelay ( 160 ) ;
/* Hammer in/out the data */
for ( i = 7 ; i > = 0 ; i - - ) {
int value ;
if ( write ) {
value = ! ! ( outval & BIT ( i ) ) ;
gpiod_set_value_cansleep ( sda , value ) ;
} else {
value = gpiod_get_value ( sda ) ;
if ( value )
inval | = BIT ( i ) ;
}
ndelay ( 150 ) ;
/* Send an SCL pulse */
gpiod_set_value_cansleep ( scl , 1 ) ;
ndelay ( 160 ) ;
gpiod_set_value_cansleep ( scl , 0 ) ;
ndelay ( 160 ) ;
}
gpiod_direction_output ( sda , 0 ) ;
/* Deassert SCEN */
gpiod_set_value_cansleep ( scen , 0 ) ;
/* Satisfies SCEN pulse width */
udelay ( 1 ) ;
return inval ;
}
static u8 tpg110_read_reg ( u8 address )
{
return tpg110_readwrite_reg ( false , address , 0 ) ;
}
static void tpg110_write_reg ( u8 address , u8 outval )
{
tpg110_readwrite_reg ( true , address , outval ) ;
}
static void tpg110_startup ( struct device * dev )
{
u8 val ;
dev_info ( dev , " TPG110 display enable \n " ) ;
/* De-assert the reset signal */
gpiod_set_value_cansleep ( grestb , 0 ) ;
mdelay ( 1 ) ;
dev_info ( dev , " de-asserted GRESTB \n " ) ;
/* Test display communication */
tpg110_write_reg ( 0x00 , 0x55 ) ;
val = tpg110_read_reg ( 0x00 ) ;
if ( val = = 0x55 )
dev_info ( dev , " passed communication test \n " ) ;
val = tpg110_read_reg ( 0x01 ) ;
dev_info ( dev , " TPG110 chip ID: %d version: %d \n " ,
val > > 4 , val & 0x0f ) ;
/* Show display resolution */
val = tpg110_read_reg ( 0x02 ) ;
val & = 7 ;
switch ( val ) {
case 0x0 :
dev_info ( dev , " IN 400x240 RGB -> OUT 800x480 RGB (dual scan) " ) ;
break ;
case 0x1 :
dev_info ( dev , " IN 480x272 RGB -> OUT 800x480 RGB (dual scan) " ) ;
break ;
case 0x4 :
dev_info ( dev , " 480x640 RGB " ) ;
break ;
case 0x5 :
dev_info ( dev , " 480x272 RGB " ) ;
break ;
case 0x6 :
dev_info ( dev , " 640x480 RGB " ) ;
break ;
case 0x7 :
dev_info ( dev , " 800x480 RGB " ) ;
break ;
default :
dev_info ( dev , " ILLEGAL RESOLUTION " ) ;
break ;
}
val = tpg110_read_reg ( 0x03 ) ;
dev_info ( dev , " resolution is controlled by %s \n " ,
( val & BIT ( 7 ) ) ? " software " : " hardware " ) ;
}
static void tpg110_enable ( struct clcd_fb * fb )
{
struct device * dev = & fb - > dev - > dev ;
static bool startup ;
u8 val ;
if ( ! startup ) {
tpg110_startup ( dev ) ;
startup = true ;
}
/* Take chip out of standby */
val = tpg110_read_reg ( 0x03 ) ;
val | = BIT ( 0 ) ;
tpg110_write_reg ( 0x03 , val ) ;
}
static void tpg110_disable ( struct clcd_fb * fb )
{
u8 val ;
dev_info ( & fb - > dev - > dev , " TPG110 display disable \n " ) ;
val = tpg110_read_reg ( 0x03 ) ;
/* Put into standby */
val & = ~ BIT ( 0 ) ;
tpg110_write_reg ( 0x03 , val ) ;
}
static void tpg110_init ( struct device * dev , struct device_node * np ,
struct clcd_board * board )
{
dev_info ( dev , " TPG110 display init \n " ) ;
2017-01-09 17:02:28 +03:00
/* This asserts the GRESTB signal, putting the display into reset */
2017-01-12 19:39:24 +03:00
grestb = devm_get_gpiod_from_child ( dev , " grestb " , & np - > fwnode ,
GPIOD_OUT_HIGH , " grestb " ) ;
2016-06-16 12:36:17 +03:00
if ( IS_ERR ( grestb ) ) {
dev_err ( dev , " no GRESTB GPIO \n " ) ;
return ;
}
2017-01-12 19:39:24 +03:00
scen = devm_get_gpiod_from_child ( dev , " scen " , & np - > fwnode ,
GPIOD_OUT_LOW , " scen " ) ;
2016-06-16 12:36:17 +03:00
if ( IS_ERR ( scen ) ) {
dev_err ( dev , " no SCEN GPIO \n " ) ;
return ;
}
2017-01-12 19:39:24 +03:00
scl = devm_get_gpiod_from_child ( dev , " scl " , & np - > fwnode , GPIOD_OUT_LOW ,
" scl " ) ;
2016-06-16 12:36:17 +03:00
if ( IS_ERR ( scl ) ) {
dev_err ( dev , " no SCL GPIO \n " ) ;
return ;
}
2017-01-12 19:39:24 +03:00
sda = devm_get_gpiod_from_child ( dev , " sda " , & np - > fwnode , GPIOD_OUT_LOW ,
" sda " ) ;
2016-06-16 12:36:17 +03:00
if ( IS_ERR ( sda ) ) {
dev_err ( dev , " no SDA GPIO \n " ) ;
return ;
}
board - > enable = tpg110_enable ;
board - > disable = tpg110_disable ;
}
int nomadik_clcd_init_panel ( struct clcd_fb * fb ,
struct device_node * endpoint )
{
struct device_node * panel ;
panel = of_graph_get_remote_port_parent ( endpoint ) ;
if ( ! panel )
return - ENODEV ;
if ( of_device_is_compatible ( panel , " tpo,tpg110 " ) )
tpg110_init ( & fb - > dev - > dev , panel , fb - > board ) ;
else
dev_info ( & fb - > dev - > dev , " unknown panel \n " ) ;
/* Unknown panel, fall through */
return 0 ;
}
2016-08-26 18:34:29 +03:00
EXPORT_SYMBOL_GPL ( nomadik_clcd_init_panel ) ;
2016-06-16 12:36:17 +03:00
# define PMU_CTRL_OFFSET 0x0000
# define PMU_CTRL_LCDNDIF BIT(26)
int nomadik_clcd_init_board ( struct amba_device * adev ,
struct clcd_board * board )
{
struct regmap * pmu_regmap ;
dev_info ( & adev - > dev , " Nomadik CLCD board init \n " ) ;
pmu_regmap =
syscon_regmap_lookup_by_compatible ( " stericsson,nomadik-pmu " ) ;
if ( IS_ERR ( pmu_regmap ) ) {
dev_err ( & adev - > dev , " could not find PMU syscon regmap \n " ) ;
return PTR_ERR ( pmu_regmap ) ;
}
regmap_update_bits ( pmu_regmap ,
PMU_CTRL_OFFSET ,
PMU_CTRL_LCDNDIF ,
0 ) ;
dev_info ( & adev - > dev , " set PMU mux to CLCD mode \n " ) ;
return 0 ;
}
2016-08-26 18:34:29 +03:00
EXPORT_SYMBOL_GPL ( nomadik_clcd_init_board ) ;