mirror of
https://github.com/torvalds/linux.git
synced 2024-12-18 17:12:55 +00:00
e4a7e3f741
The I2C to the Atmel is very fussy, and locks up easily on Pi0-3 particularly on reads. If running at 100kHz on Pi3, reading the ID register generally locks up the Atmel, but splitting the register select write and read into two transactions is reliable. Signed-off-by: Dave Stevenson <dave.stevenson@raspberrypi.com> Signed-off-by: Detlev Casanova <detlev.casanova@collabora.com> Link: https://lore.kernel.org/r/20220124220129.158891-10-detlev.casanova@collabora.com Signed-off-by: Mark Brown <broonie@kernel.org>
418 lines
9.5 KiB
C
418 lines
9.5 KiB
C
// 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>
|
|
#include <linux/gpio/driver.h>
|
|
#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
|
|
#define REG_PORTC 0x83
|
|
#define REG_POWERON 0x85
|
|
#define REG_PWM 0x86
|
|
#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)
|
|
|
|
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 },
|
|
};
|
|
|
|
struct attiny_lcd {
|
|
/* lock to serialise overall accesses to the Atmel */
|
|
struct mutex lock;
|
|
struct regmap *regmap;
|
|
bool gpio_states[NUM_GPIO];
|
|
u8 port_states[3];
|
|
|
|
struct gpio_chip gc;
|
|
};
|
|
|
|
static const struct regmap_config attiny_regmap_config = {
|
|
.reg_bits = 8,
|
|
.val_bits = 8,
|
|
.disable_locking = 1,
|
|
.max_register = REG_WRITE_DATA_L,
|
|
.cache_type = REGCACHE_RBTREE,
|
|
};
|
|
|
|
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];
|
|
};
|
|
|
|
static int attiny_lcd_power_enable(struct regulator_dev *rdev)
|
|
{
|
|
struct attiny_lcd *state = rdev_get_drvdata(rdev);
|
|
|
|
mutex_lock(&state->lock);
|
|
|
|
/* Ensure bridge, and tp stay in reset */
|
|
attiny_set_port_state(state, REG_PORTC, 0);
|
|
usleep_range(5000, 10000);
|
|
|
|
/* 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.
|
|
*/
|
|
attiny_set_port_state(state, REG_PORTA, PA_LCD_LR);
|
|
usleep_range(5000, 10000);
|
|
/* Main regulator on, and power to the panel (LCD_VCC_N) */
|
|
attiny_set_port_state(state, REG_PORTB, PB_LCD_MAIN);
|
|
usleep_range(5000, 10000);
|
|
/* Bring controllers out of reset */
|
|
attiny_set_port_state(state, REG_PORTC, PC_LED_EN);
|
|
|
|
msleep(80);
|
|
|
|
mutex_unlock(&state->lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int attiny_lcd_power_disable(struct regulator_dev *rdev)
|
|
{
|
|
struct attiny_lcd *state = rdev_get_drvdata(rdev);
|
|
|
|
mutex_lock(&state->lock);
|
|
|
|
regmap_write(rdev->regmap, REG_PWM, 0);
|
|
usleep_range(5000, 10000);
|
|
|
|
attiny_set_port_state(state, REG_PORTA, 0);
|
|
usleep_range(5000, 10000);
|
|
attiny_set_port_state(state, REG_PORTB, PB_LCD_VCC_N);
|
|
usleep_range(5000, 10000);
|
|
attiny_set_port_state(state, REG_PORTC, 0);
|
|
msleep(30);
|
|
|
|
mutex_unlock(&state->lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int attiny_lcd_power_is_enabled(struct regulator_dev *rdev)
|
|
{
|
|
struct attiny_lcd *state = rdev_get_drvdata(rdev);
|
|
unsigned int data;
|
|
int ret, i;
|
|
|
|
mutex_lock(&state->lock);
|
|
|
|
for (i = 0; i < 10; i++) {
|
|
ret = regmap_read(rdev->regmap, REG_PORTC, &data);
|
|
if (!ret)
|
|
break;
|
|
usleep_range(10000, 12000);
|
|
}
|
|
|
|
mutex_unlock(&state->lock);
|
|
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
return data & PC_RST_BRIDGE_N;
|
|
}
|
|
|
|
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)
|
|
{
|
|
struct attiny_lcd *state = bl_get_data(bl);
|
|
struct regmap *regmap = state->regmap;
|
|
int brightness = bl->props.brightness;
|
|
int ret, i;
|
|
|
|
mutex_lock(&state->lock);
|
|
|
|
if (bl->props.power != FB_BLANK_UNBLANK ||
|
|
bl->props.fb_blank != FB_BLANK_UNBLANK)
|
|
brightness = 0;
|
|
|
|
for (i = 0; i < 10; i++) {
|
|
ret = regmap_write(regmap, REG_PWM, brightness);
|
|
if (!ret)
|
|
break;
|
|
}
|
|
|
|
mutex_unlock(&state->lock);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static const struct backlight_ops attiny_bl = {
|
|
.update_status = attiny_update_status,
|
|
};
|
|
|
|
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);
|
|
}
|
|
|
|
static int attiny_i2c_read(struct i2c_client *client, u8 reg, unsigned int *buf)
|
|
{
|
|
struct i2c_msg msgs[1];
|
|
u8 addr_buf[1] = { reg };
|
|
u8 data_buf[1] = { 0, };
|
|
int ret;
|
|
|
|
/* Write register address */
|
|
msgs[0].addr = client->addr;
|
|
msgs[0].flags = 0;
|
|
msgs[0].len = ARRAY_SIZE(addr_buf);
|
|
msgs[0].buf = addr_buf;
|
|
|
|
ret = i2c_transfer(client->adapter, msgs, ARRAY_SIZE(msgs));
|
|
if (ret != ARRAY_SIZE(msgs))
|
|
return -EIO;
|
|
|
|
usleep_range(5000, 10000);
|
|
|
|
/* Read data from register */
|
|
msgs[0].addr = client->addr;
|
|
msgs[0].flags = I2C_M_RD;
|
|
msgs[0].len = 1;
|
|
msgs[0].buf = data_buf;
|
|
|
|
ret = i2c_transfer(client->adapter, msgs, ARRAY_SIZE(msgs));
|
|
if (ret != ARRAY_SIZE(msgs))
|
|
return -EIO;
|
|
|
|
*buf = data_buf[0];
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* 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;
|
|
struct attiny_lcd *state;
|
|
struct regmap *regmap;
|
|
unsigned int data;
|
|
int ret;
|
|
|
|
state = devm_kzalloc(&i2c->dev, sizeof(*state), GFP_KERNEL);
|
|
if (!state)
|
|
return -ENOMEM;
|
|
|
|
mutex_init(&state->lock);
|
|
i2c_set_clientdata(i2c, state);
|
|
|
|
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);
|
|
goto error;
|
|
}
|
|
|
|
ret = attiny_i2c_read(i2c, REG_ID, &data);
|
|
if (ret < 0) {
|
|
dev_err(&i2c->dev, "Failed to read REG_ID reg: %d\n", ret);
|
|
goto error;
|
|
}
|
|
|
|
switch (data) {
|
|
case 0xde: /* ver 1 */
|
|
case 0xc3: /* ver 2 */
|
|
break;
|
|
default:
|
|
dev_err(&i2c->dev, "Unknown Atmel firmware revision: 0x%02x\n", data);
|
|
ret = -ENODEV;
|
|
goto error;
|
|
}
|
|
|
|
regmap_write(regmap, REG_POWERON, 0);
|
|
msleep(30);
|
|
regmap_write(regmap, REG_PWM, 0);
|
|
|
|
config.dev = &i2c->dev;
|
|
config.regmap = regmap;
|
|
config.of_node = i2c->dev.of_node;
|
|
config.init_data = &attiny_regulator_default;
|
|
config.driver_data = state;
|
|
|
|
rdev = devm_regulator_register(&i2c->dev, &attiny_regulator, &config);
|
|
if (IS_ERR(rdev)) {
|
|
dev_err(&i2c->dev, "Failed to register ATTINY regulator\n");
|
|
ret = PTR_ERR(rdev);
|
|
goto error;
|
|
}
|
|
|
|
props.type = BACKLIGHT_RAW;
|
|
props.max_brightness = 0xff;
|
|
|
|
state->regmap = regmap;
|
|
|
|
bl = devm_backlight_device_register(&i2c->dev, dev_name(&i2c->dev),
|
|
&i2c->dev, state, &attiny_bl,
|
|
&props);
|
|
if (IS_ERR(bl)) {
|
|
ret = PTR_ERR(bl);
|
|
goto error;
|
|
}
|
|
|
|
bl->props.brightness = 0xff;
|
|
|
|
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;
|
|
}
|
|
|
|
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);
|
|
|
|
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,
|
|
.remove = attiny_i2c_remove,
|
|
};
|
|
|
|
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");
|