cd6a1ca386
The GMA500 driver is using the legacy GPIO API to fetch three optional display control GPIO lines from the SFI description used by the Medfield platform. Switch this over to use GPIO descriptors and delete the custom platform data. We create three new static locals in the tc35876x bridge code but it is hardly any worse than the I2C client static local already there: I tried first to move it to the DRM driver state container but there are workarounds for probe order in the code so I just stayed off it, as the result is unpredictable. People wanting to do a more throrugh and proper cleanup of the GMA500 driver can work on top of this, I can't solve much more since I don't have access to the hardware, I can only attempt to tidy up my GPIO corner. Cc: Daniel Stone <daniels@collabora.com> Cc: Daniel Vetter <daniel.vetter@ffwll.ch> Reviewed-by: Andy Shevchenko <andriy.shevchenko@linux.intel.com> Acked-by: Patrik Jakobsson <patrik.r.jakobsson@gmail.com> Link: https://patchwork.freedesktop.org/patch/msgid/20191206094301.76368-1-linus.walleij@linaro.org Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
811 lines
21 KiB
C
811 lines
21 KiB
C
/*
|
|
* Copyright © 2011 Intel Corporation
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
* to deal in the Software without restriction, including without limitation
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice (including the next
|
|
* paragraph) shall be included in all copies or substantial portions of the
|
|
* Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
* DEALINGS IN THE SOFTWARE.
|
|
*
|
|
*/
|
|
|
|
#include <linux/delay.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/module.h>
|
|
#include <linux/gpio/consumer.h>
|
|
|
|
#include <asm/intel_scu_ipc.h>
|
|
|
|
#include "mdfld_dsi_dpi.h"
|
|
#include "mdfld_dsi_pkg_sender.h"
|
|
#include "mdfld_output.h"
|
|
#include "tc35876x-dsi-lvds.h"
|
|
|
|
static struct i2c_client *tc35876x_client;
|
|
static struct i2c_client *cmi_lcd_i2c_client;
|
|
/* Panel GPIOs */
|
|
static struct gpio_desc *bridge_reset;
|
|
static struct gpio_desc *bridge_bl_enable;
|
|
static struct gpio_desc *backlight_voltage;
|
|
|
|
|
|
#define FLD_MASK(start, end) (((1 << ((start) - (end) + 1)) - 1) << (end))
|
|
#define FLD_VAL(val, start, end) (((val) << (end)) & FLD_MASK(start, end))
|
|
|
|
/* DSI D-PHY Layer Registers */
|
|
#define D0W_DPHYCONTTX 0x0004
|
|
#define CLW_DPHYCONTRX 0x0020
|
|
#define D0W_DPHYCONTRX 0x0024
|
|
#define D1W_DPHYCONTRX 0x0028
|
|
#define D2W_DPHYCONTRX 0x002C
|
|
#define D3W_DPHYCONTRX 0x0030
|
|
#define COM_DPHYCONTRX 0x0038
|
|
#define CLW_CNTRL 0x0040
|
|
#define D0W_CNTRL 0x0044
|
|
#define D1W_CNTRL 0x0048
|
|
#define D2W_CNTRL 0x004C
|
|
#define D3W_CNTRL 0x0050
|
|
#define DFTMODE_CNTRL 0x0054
|
|
|
|
/* DSI PPI Layer Registers */
|
|
#define PPI_STARTPPI 0x0104
|
|
#define PPI_BUSYPPI 0x0108
|
|
#define PPI_LINEINITCNT 0x0110
|
|
#define PPI_LPTXTIMECNT 0x0114
|
|
#define PPI_LANEENABLE 0x0134
|
|
#define PPI_TX_RX_TA 0x013C
|
|
#define PPI_CLS_ATMR 0x0140
|
|
#define PPI_D0S_ATMR 0x0144
|
|
#define PPI_D1S_ATMR 0x0148
|
|
#define PPI_D2S_ATMR 0x014C
|
|
#define PPI_D3S_ATMR 0x0150
|
|
#define PPI_D0S_CLRSIPOCOUNT 0x0164
|
|
#define PPI_D1S_CLRSIPOCOUNT 0x0168
|
|
#define PPI_D2S_CLRSIPOCOUNT 0x016C
|
|
#define PPI_D3S_CLRSIPOCOUNT 0x0170
|
|
#define CLS_PRE 0x0180
|
|
#define D0S_PRE 0x0184
|
|
#define D1S_PRE 0x0188
|
|
#define D2S_PRE 0x018C
|
|
#define D3S_PRE 0x0190
|
|
#define CLS_PREP 0x01A0
|
|
#define D0S_PREP 0x01A4
|
|
#define D1S_PREP 0x01A8
|
|
#define D2S_PREP 0x01AC
|
|
#define D3S_PREP 0x01B0
|
|
#define CLS_ZERO 0x01C0
|
|
#define D0S_ZERO 0x01C4
|
|
#define D1S_ZERO 0x01C8
|
|
#define D2S_ZERO 0x01CC
|
|
#define D3S_ZERO 0x01D0
|
|
#define PPI_CLRFLG 0x01E0
|
|
#define PPI_CLRSIPO 0x01E4
|
|
#define HSTIMEOUT 0x01F0
|
|
#define HSTIMEOUTENABLE 0x01F4
|
|
|
|
/* DSI Protocol Layer Registers */
|
|
#define DSI_STARTDSI 0x0204
|
|
#define DSI_BUSYDSI 0x0208
|
|
#define DSI_LANEENABLE 0x0210
|
|
#define DSI_LANESTATUS0 0x0214
|
|
#define DSI_LANESTATUS1 0x0218
|
|
#define DSI_INTSTATUS 0x0220
|
|
#define DSI_INTMASK 0x0224
|
|
#define DSI_INTCLR 0x0228
|
|
#define DSI_LPTXTO 0x0230
|
|
|
|
/* DSI General Registers */
|
|
#define DSIERRCNT 0x0300
|
|
|
|
/* DSI Application Layer Registers */
|
|
#define APLCTRL 0x0400
|
|
#define RDPKTLN 0x0404
|
|
|
|
/* Video Path Registers */
|
|
#define VPCTRL 0x0450
|
|
#define HTIM1 0x0454
|
|
#define HTIM2 0x0458
|
|
#define VTIM1 0x045C
|
|
#define VTIM2 0x0460
|
|
#define VFUEN 0x0464
|
|
|
|
/* LVDS Registers */
|
|
#define LVMX0003 0x0480
|
|
#define LVMX0407 0x0484
|
|
#define LVMX0811 0x0488
|
|
#define LVMX1215 0x048C
|
|
#define LVMX1619 0x0490
|
|
#define LVMX2023 0x0494
|
|
#define LVMX2427 0x0498
|
|
#define LVCFG 0x049C
|
|
#define LVPHY0 0x04A0
|
|
#define LVPHY1 0x04A4
|
|
|
|
/* System Registers */
|
|
#define SYSSTAT 0x0500
|
|
#define SYSRST 0x0504
|
|
|
|
/* GPIO Registers */
|
|
/*#define GPIOC 0x0520*/
|
|
#define GPIOO 0x0524
|
|
#define GPIOI 0x0528
|
|
|
|
/* I2C Registers */
|
|
#define I2CTIMCTRL 0x0540
|
|
#define I2CMADDR 0x0544
|
|
#define WDATAQ 0x0548
|
|
#define RDATAQ 0x054C
|
|
|
|
/* Chip/Rev Registers */
|
|
#define IDREG 0x0580
|
|
|
|
/* Debug Registers */
|
|
#define DEBUG00 0x05A0
|
|
#define DEBUG01 0x05A4
|
|
|
|
/* Panel CABC registers */
|
|
#define PANEL_PWM_CONTROL 0x90
|
|
#define PANEL_FREQ_DIVIDER_HI 0x91
|
|
#define PANEL_FREQ_DIVIDER_LO 0x92
|
|
#define PANEL_DUTY_CONTROL 0x93
|
|
#define PANEL_MODIFY_RGB 0x94
|
|
#define PANEL_FRAMERATE_CONTROL 0x96
|
|
#define PANEL_PWM_MIN 0x97
|
|
#define PANEL_PWM_REF 0x98
|
|
#define PANEL_PWM_MAX 0x99
|
|
#define PANEL_ALLOW_DISTORT 0x9A
|
|
#define PANEL_BYPASS_PWMI 0x9B
|
|
|
|
/* Panel color management registers */
|
|
#define PANEL_CM_ENABLE 0x700
|
|
#define PANEL_CM_HUE 0x701
|
|
#define PANEL_CM_SATURATION 0x702
|
|
#define PANEL_CM_INTENSITY 0x703
|
|
#define PANEL_CM_BRIGHTNESS 0x704
|
|
#define PANEL_CM_CE_ENABLE 0x705
|
|
#define PANEL_CM_PEAK_EN 0x710
|
|
#define PANEL_CM_GAIN 0x711
|
|
#define PANEL_CM_HUETABLE_START 0x730
|
|
#define PANEL_CM_HUETABLE_END 0x747 /* inclusive */
|
|
|
|
/* Input muxing for registers LVMX0003...LVMX2427 */
|
|
enum {
|
|
INPUT_R0, /* 0 */
|
|
INPUT_R1,
|
|
INPUT_R2,
|
|
INPUT_R3,
|
|
INPUT_R4,
|
|
INPUT_R5,
|
|
INPUT_R6,
|
|
INPUT_R7,
|
|
INPUT_G0, /* 8 */
|
|
INPUT_G1,
|
|
INPUT_G2,
|
|
INPUT_G3,
|
|
INPUT_G4,
|
|
INPUT_G5,
|
|
INPUT_G6,
|
|
INPUT_G7,
|
|
INPUT_B0, /* 16 */
|
|
INPUT_B1,
|
|
INPUT_B2,
|
|
INPUT_B3,
|
|
INPUT_B4,
|
|
INPUT_B5,
|
|
INPUT_B6,
|
|
INPUT_B7,
|
|
INPUT_HSYNC, /* 24 */
|
|
INPUT_VSYNC,
|
|
INPUT_DE,
|
|
LOGIC_0,
|
|
/* 28...31 undefined */
|
|
};
|
|
|
|
#define INPUT_MUX(lvmx03, lvmx02, lvmx01, lvmx00) \
|
|
(FLD_VAL(lvmx03, 29, 24) | FLD_VAL(lvmx02, 20, 16) | \
|
|
FLD_VAL(lvmx01, 12, 8) | FLD_VAL(lvmx00, 4, 0))
|
|
|
|
/**
|
|
* tc35876x_regw - Write DSI-LVDS bridge register using I2C
|
|
* @client: struct i2c_client to use
|
|
* @reg: register address
|
|
* @value: value to write
|
|
*
|
|
* Returns 0 on success, or a negative error value.
|
|
*/
|
|
static int tc35876x_regw(struct i2c_client *client, u16 reg, u32 value)
|
|
{
|
|
int r;
|
|
u8 tx_data[] = {
|
|
/* NOTE: Register address big-endian, data little-endian. */
|
|
(reg >> 8) & 0xff,
|
|
reg & 0xff,
|
|
value & 0xff,
|
|
(value >> 8) & 0xff,
|
|
(value >> 16) & 0xff,
|
|
(value >> 24) & 0xff,
|
|
};
|
|
struct i2c_msg msgs[] = {
|
|
{
|
|
.addr = client->addr,
|
|
.flags = 0,
|
|
.buf = tx_data,
|
|
.len = ARRAY_SIZE(tx_data),
|
|
},
|
|
};
|
|
|
|
r = i2c_transfer(client->adapter, msgs, ARRAY_SIZE(msgs));
|
|
if (r < 0) {
|
|
dev_err(&client->dev, "%s: reg 0x%04x val 0x%08x error %d\n",
|
|
__func__, reg, value, r);
|
|
return r;
|
|
}
|
|
|
|
if (r < ARRAY_SIZE(msgs)) {
|
|
dev_err(&client->dev, "%s: reg 0x%04x val 0x%08x msgs %d\n",
|
|
__func__, reg, value, r);
|
|
return -EAGAIN;
|
|
}
|
|
|
|
dev_dbg(&client->dev, "%s: reg 0x%04x val 0x%08x\n",
|
|
__func__, reg, value);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* tc35876x_regr - Read DSI-LVDS bridge register using I2C
|
|
* @client: struct i2c_client to use
|
|
* @reg: register address
|
|
* @value: pointer for storing the value
|
|
*
|
|
* Returns 0 on success, or a negative error value.
|
|
*/
|
|
static int tc35876x_regr(struct i2c_client *client, u16 reg, u32 *value)
|
|
{
|
|
int r;
|
|
u8 tx_data[] = {
|
|
(reg >> 8) & 0xff,
|
|
reg & 0xff,
|
|
};
|
|
u8 rx_data[4];
|
|
struct i2c_msg msgs[] = {
|
|
{
|
|
.addr = client->addr,
|
|
.flags = 0,
|
|
.buf = tx_data,
|
|
.len = ARRAY_SIZE(tx_data),
|
|
},
|
|
{
|
|
.addr = client->addr,
|
|
.flags = I2C_M_RD,
|
|
.buf = rx_data,
|
|
.len = ARRAY_SIZE(rx_data),
|
|
},
|
|
};
|
|
|
|
r = i2c_transfer(client->adapter, msgs, ARRAY_SIZE(msgs));
|
|
if (r < 0) {
|
|
dev_err(&client->dev, "%s: reg 0x%04x error %d\n", __func__,
|
|
reg, r);
|
|
return r;
|
|
}
|
|
|
|
if (r < ARRAY_SIZE(msgs)) {
|
|
dev_err(&client->dev, "%s: reg 0x%04x msgs %d\n", __func__,
|
|
reg, r);
|
|
return -EAGAIN;
|
|
}
|
|
|
|
*value = rx_data[0] << 24 | rx_data[1] << 16 |
|
|
rx_data[2] << 8 | rx_data[3];
|
|
|
|
dev_dbg(&client->dev, "%s: reg 0x%04x value 0x%08x\n", __func__,
|
|
reg, *value);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void tc35876x_set_bridge_reset_state(struct drm_device *dev, int state)
|
|
{
|
|
if (WARN(!tc35876x_client, "%s called before probe", __func__))
|
|
return;
|
|
|
|
dev_dbg(&tc35876x_client->dev, "%s: state %d\n", __func__, state);
|
|
|
|
if (!bridge_reset)
|
|
return;
|
|
|
|
if (state) {
|
|
gpiod_set_value_cansleep(bridge_reset, 0);
|
|
mdelay(10);
|
|
} else {
|
|
/* Pull MIPI Bridge reset pin to Low */
|
|
gpiod_set_value_cansleep(bridge_reset, 0);
|
|
mdelay(20);
|
|
/* Pull MIPI Bridge reset pin to High */
|
|
gpiod_set_value_cansleep(bridge_reset, 1);
|
|
mdelay(40);
|
|
}
|
|
}
|
|
|
|
void tc35876x_configure_lvds_bridge(struct drm_device *dev)
|
|
{
|
|
struct i2c_client *i2c = tc35876x_client;
|
|
u32 ppi_lptxtimecnt;
|
|
u32 txtagocnt;
|
|
u32 txtasurecnt;
|
|
u32 id;
|
|
|
|
if (WARN(!tc35876x_client, "%s called before probe", __func__))
|
|
return;
|
|
|
|
dev_dbg(&tc35876x_client->dev, "%s\n", __func__);
|
|
|
|
if (!tc35876x_regr(i2c, IDREG, &id))
|
|
dev_info(&tc35876x_client->dev, "tc35876x ID 0x%08x\n", id);
|
|
else
|
|
dev_err(&tc35876x_client->dev, "Cannot read ID\n");
|
|
|
|
ppi_lptxtimecnt = 4;
|
|
txtagocnt = (5 * ppi_lptxtimecnt - 3) / 4;
|
|
txtasurecnt = 3 * ppi_lptxtimecnt / 2;
|
|
tc35876x_regw(i2c, PPI_TX_RX_TA, FLD_VAL(txtagocnt, 26, 16) |
|
|
FLD_VAL(txtasurecnt, 10, 0));
|
|
tc35876x_regw(i2c, PPI_LPTXTIMECNT, FLD_VAL(ppi_lptxtimecnt, 10, 0));
|
|
|
|
tc35876x_regw(i2c, PPI_D0S_CLRSIPOCOUNT, FLD_VAL(1, 5, 0));
|
|
tc35876x_regw(i2c, PPI_D1S_CLRSIPOCOUNT, FLD_VAL(1, 5, 0));
|
|
tc35876x_regw(i2c, PPI_D2S_CLRSIPOCOUNT, FLD_VAL(1, 5, 0));
|
|
tc35876x_regw(i2c, PPI_D3S_CLRSIPOCOUNT, FLD_VAL(1, 5, 0));
|
|
|
|
/* Enabling MIPI & PPI lanes, Enable 4 lanes */
|
|
tc35876x_regw(i2c, PPI_LANEENABLE,
|
|
BIT(4) | BIT(3) | BIT(2) | BIT(1) | BIT(0));
|
|
tc35876x_regw(i2c, DSI_LANEENABLE,
|
|
BIT(4) | BIT(3) | BIT(2) | BIT(1) | BIT(0));
|
|
tc35876x_regw(i2c, PPI_STARTPPI, BIT(0));
|
|
tc35876x_regw(i2c, DSI_STARTDSI, BIT(0));
|
|
|
|
/* Setting LVDS output frequency */
|
|
tc35876x_regw(i2c, LVPHY0, FLD_VAL(1, 20, 16) |
|
|
FLD_VAL(2, 15, 14) | FLD_VAL(6, 4, 0)); /* 0x00048006 */
|
|
|
|
/* Setting video panel control register,0x00000120 VTGen=ON ?!?!? */
|
|
tc35876x_regw(i2c, VPCTRL, BIT(8) | BIT(5));
|
|
|
|
/* Horizontal back porch and horizontal pulse width. 0x00280028 */
|
|
tc35876x_regw(i2c, HTIM1, FLD_VAL(40, 24, 16) | FLD_VAL(40, 8, 0));
|
|
|
|
/* Horizontal front porch and horizontal active video size. 0x00500500*/
|
|
tc35876x_regw(i2c, HTIM2, FLD_VAL(80, 24, 16) | FLD_VAL(1280, 10, 0));
|
|
|
|
/* Vertical back porch and vertical sync pulse width. 0x000e000a */
|
|
tc35876x_regw(i2c, VTIM1, FLD_VAL(14, 23, 16) | FLD_VAL(10, 7, 0));
|
|
|
|
/* Vertical front porch and vertical display size. 0x000e0320 */
|
|
tc35876x_regw(i2c, VTIM2, FLD_VAL(14, 23, 16) | FLD_VAL(800, 10, 0));
|
|
|
|
/* Set above HTIM1, HTIM2, VTIM1, and VTIM2 at next VSYNC. */
|
|
tc35876x_regw(i2c, VFUEN, BIT(0));
|
|
|
|
/* Soft reset LCD controller. */
|
|
tc35876x_regw(i2c, SYSRST, BIT(2));
|
|
|
|
/* LVDS-TX input muxing */
|
|
tc35876x_regw(i2c, LVMX0003,
|
|
INPUT_MUX(INPUT_R5, INPUT_R4, INPUT_R3, INPUT_R2));
|
|
tc35876x_regw(i2c, LVMX0407,
|
|
INPUT_MUX(INPUT_G2, INPUT_R7, INPUT_R1, INPUT_R6));
|
|
tc35876x_regw(i2c, LVMX0811,
|
|
INPUT_MUX(INPUT_G1, INPUT_G0, INPUT_G4, INPUT_G3));
|
|
tc35876x_regw(i2c, LVMX1215,
|
|
INPUT_MUX(INPUT_B2, INPUT_G7, INPUT_G6, INPUT_G5));
|
|
tc35876x_regw(i2c, LVMX1619,
|
|
INPUT_MUX(INPUT_B4, INPUT_B3, INPUT_B1, INPUT_B0));
|
|
tc35876x_regw(i2c, LVMX2023,
|
|
INPUT_MUX(LOGIC_0, INPUT_B7, INPUT_B6, INPUT_B5));
|
|
tc35876x_regw(i2c, LVMX2427,
|
|
INPUT_MUX(INPUT_R0, INPUT_DE, INPUT_VSYNC, INPUT_HSYNC));
|
|
|
|
/* Enable LVDS transmitter. */
|
|
tc35876x_regw(i2c, LVCFG, BIT(0));
|
|
|
|
/* Clear notifications. Don't write reserved bits. Was write 0xffffffff
|
|
* to 0x0288, must be in error?! */
|
|
tc35876x_regw(i2c, DSI_INTCLR, FLD_MASK(31, 30) | FLD_MASK(22, 0));
|
|
}
|
|
|
|
#define GPIOPWMCTRL 0x38F
|
|
#define PWM0CLKDIV0 0x62 /* low byte */
|
|
#define PWM0CLKDIV1 0x61 /* high byte */
|
|
|
|
#define SYSTEMCLK 19200000UL /* 19.2 MHz */
|
|
#define PWM_FREQUENCY 9600 /* Hz */
|
|
|
|
/* f = baseclk / (clkdiv + 1) => clkdiv = (baseclk - f) / f */
|
|
static inline u16 calc_clkdiv(unsigned long baseclk, unsigned int f)
|
|
{
|
|
return (baseclk - f) / f;
|
|
}
|
|
|
|
static void tc35876x_brightness_init(struct drm_device *dev)
|
|
{
|
|
int ret;
|
|
u8 pwmctrl;
|
|
u16 clkdiv;
|
|
|
|
/* Make sure the PWM reference is the 19.2 MHz system clock. Read first
|
|
* instead of setting directly to catch potential conflicts between PWM
|
|
* users. */
|
|
ret = intel_scu_ipc_ioread8(GPIOPWMCTRL, &pwmctrl);
|
|
if (ret || pwmctrl != 0x01) {
|
|
if (ret)
|
|
dev_err(&dev->pdev->dev, "GPIOPWMCTRL read failed\n");
|
|
else
|
|
dev_warn(&dev->pdev->dev, "GPIOPWMCTRL was not set to system clock (pwmctrl = 0x%02x)\n", pwmctrl);
|
|
|
|
ret = intel_scu_ipc_iowrite8(GPIOPWMCTRL, 0x01);
|
|
if (ret)
|
|
dev_err(&dev->pdev->dev, "GPIOPWMCTRL set failed\n");
|
|
}
|
|
|
|
clkdiv = calc_clkdiv(SYSTEMCLK, PWM_FREQUENCY);
|
|
|
|
ret = intel_scu_ipc_iowrite8(PWM0CLKDIV1, (clkdiv >> 8) & 0xff);
|
|
if (!ret)
|
|
ret = intel_scu_ipc_iowrite8(PWM0CLKDIV0, clkdiv & 0xff);
|
|
|
|
if (ret)
|
|
dev_err(&dev->pdev->dev, "PWM0CLKDIV set failed\n");
|
|
else
|
|
dev_dbg(&dev->pdev->dev, "PWM0CLKDIV set to 0x%04x (%d Hz)\n",
|
|
clkdiv, PWM_FREQUENCY);
|
|
}
|
|
|
|
#define PWM0DUTYCYCLE 0x67
|
|
|
|
void tc35876x_brightness_control(struct drm_device *dev, int level)
|
|
{
|
|
int ret;
|
|
u8 duty_val;
|
|
u8 panel_duty_val;
|
|
|
|
level = clamp(level, 0, MDFLD_DSI_BRIGHTNESS_MAX_LEVEL);
|
|
|
|
/* PWM duty cycle 0x00...0x63 corresponds to 0...99% */
|
|
duty_val = level * 0x63 / MDFLD_DSI_BRIGHTNESS_MAX_LEVEL;
|
|
|
|
/* I won't pretend to understand this formula. The panel spec is quite
|
|
* bad engrish.
|
|
*/
|
|
panel_duty_val = (2 * level - 100) * 0xA9 /
|
|
MDFLD_DSI_BRIGHTNESS_MAX_LEVEL + 0x56;
|
|
|
|
ret = intel_scu_ipc_iowrite8(PWM0DUTYCYCLE, duty_val);
|
|
if (ret)
|
|
dev_err(&tc35876x_client->dev, "%s: ipc write fail\n",
|
|
__func__);
|
|
|
|
if (cmi_lcd_i2c_client) {
|
|
ret = i2c_smbus_write_byte_data(cmi_lcd_i2c_client,
|
|
PANEL_PWM_MAX, panel_duty_val);
|
|
if (ret < 0)
|
|
dev_err(&cmi_lcd_i2c_client->dev, "%s: i2c write failed\n",
|
|
__func__);
|
|
}
|
|
}
|
|
|
|
void tc35876x_toshiba_bridge_panel_off(struct drm_device *dev)
|
|
{
|
|
if (WARN(!tc35876x_client, "%s called before probe", __func__))
|
|
return;
|
|
|
|
dev_dbg(&tc35876x_client->dev, "%s\n", __func__);
|
|
|
|
if (bridge_bl_enable)
|
|
gpiod_set_value_cansleep(bridge_bl_enable, 0);
|
|
|
|
if (backlight_voltage)
|
|
gpiod_set_value_cansleep(backlight_voltage, 0);
|
|
}
|
|
|
|
void tc35876x_toshiba_bridge_panel_on(struct drm_device *dev)
|
|
{
|
|
struct drm_psb_private *dev_priv = dev->dev_private;
|
|
|
|
if (WARN(!tc35876x_client, "%s called before probe", __func__))
|
|
return;
|
|
|
|
dev_dbg(&tc35876x_client->dev, "%s\n", __func__);
|
|
|
|
if (backlight_voltage) {
|
|
gpiod_set_value_cansleep(backlight_voltage, 1);
|
|
msleep(260);
|
|
}
|
|
|
|
if (cmi_lcd_i2c_client) {
|
|
int ret;
|
|
dev_dbg(&cmi_lcd_i2c_client->dev, "setting TCON\n");
|
|
/* Bit 4 is average_saving. Setting it to 1, the brightness is
|
|
* referenced to the average of the frame content. 0 means
|
|
* reference to the maximum of frame contents. Bits 3:0 are
|
|
* allow_distort. When set to a nonzero value, all color values
|
|
* between 255-allow_distort*2 and 255 are mapped to the
|
|
* 255-allow_distort*2 value.
|
|
*/
|
|
ret = i2c_smbus_write_byte_data(cmi_lcd_i2c_client,
|
|
PANEL_ALLOW_DISTORT, 0x10);
|
|
if (ret < 0)
|
|
dev_err(&cmi_lcd_i2c_client->dev,
|
|
"i2c write failed (%d)\n", ret);
|
|
ret = i2c_smbus_write_byte_data(cmi_lcd_i2c_client,
|
|
PANEL_BYPASS_PWMI, 0);
|
|
if (ret < 0)
|
|
dev_err(&cmi_lcd_i2c_client->dev,
|
|
"i2c write failed (%d)\n", ret);
|
|
/* Set minimum brightness value - this is tunable */
|
|
ret = i2c_smbus_write_byte_data(cmi_lcd_i2c_client,
|
|
PANEL_PWM_MIN, 0x35);
|
|
if (ret < 0)
|
|
dev_err(&cmi_lcd_i2c_client->dev,
|
|
"i2c write failed (%d)\n", ret);
|
|
}
|
|
|
|
if (bridge_bl_enable)
|
|
gpiod_set_value_cansleep(bridge_bl_enable, 1);
|
|
|
|
tc35876x_brightness_control(dev, dev_priv->brightness_adjusted);
|
|
}
|
|
|
|
static struct drm_display_mode *tc35876x_get_config_mode(struct drm_device *dev)
|
|
{
|
|
struct drm_display_mode *mode;
|
|
|
|
dev_dbg(&dev->pdev->dev, "%s\n", __func__);
|
|
|
|
mode = kzalloc(sizeof(*mode), GFP_KERNEL);
|
|
if (!mode)
|
|
return NULL;
|
|
|
|
/* FIXME: do this properly. */
|
|
mode->hdisplay = 1280;
|
|
mode->vdisplay = 800;
|
|
mode->hsync_start = 1360;
|
|
mode->hsync_end = 1400;
|
|
mode->htotal = 1440;
|
|
mode->vsync_start = 814;
|
|
mode->vsync_end = 824;
|
|
mode->vtotal = 838;
|
|
mode->clock = 33324 << 1;
|
|
|
|
dev_info(&dev->pdev->dev, "hdisplay(w) = %d\n", mode->hdisplay);
|
|
dev_info(&dev->pdev->dev, "vdisplay(h) = %d\n", mode->vdisplay);
|
|
dev_info(&dev->pdev->dev, "HSS = %d\n", mode->hsync_start);
|
|
dev_info(&dev->pdev->dev, "HSE = %d\n", mode->hsync_end);
|
|
dev_info(&dev->pdev->dev, "htotal = %d\n", mode->htotal);
|
|
dev_info(&dev->pdev->dev, "VSS = %d\n", mode->vsync_start);
|
|
dev_info(&dev->pdev->dev, "VSE = %d\n", mode->vsync_end);
|
|
dev_info(&dev->pdev->dev, "vtotal = %d\n", mode->vtotal);
|
|
dev_info(&dev->pdev->dev, "clock = %d\n", mode->clock);
|
|
|
|
drm_mode_set_name(mode);
|
|
drm_mode_set_crtcinfo(mode, 0);
|
|
|
|
mode->type |= DRM_MODE_TYPE_PREFERRED;
|
|
|
|
return mode;
|
|
}
|
|
|
|
/* DV1 Active area 216.96 x 135.6 mm */
|
|
#define DV1_PANEL_WIDTH 217
|
|
#define DV1_PANEL_HEIGHT 136
|
|
|
|
static int tc35876x_get_panel_info(struct drm_device *dev, int pipe,
|
|
struct panel_info *pi)
|
|
{
|
|
if (!dev || !pi)
|
|
return -EINVAL;
|
|
|
|
pi->width_mm = DV1_PANEL_WIDTH;
|
|
pi->height_mm = DV1_PANEL_HEIGHT;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tc35876x_bridge_probe(struct i2c_client *client,
|
|
const struct i2c_device_id *id)
|
|
{
|
|
dev_info(&client->dev, "%s\n", __func__);
|
|
|
|
if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) {
|
|
dev_err(&client->dev, "%s: i2c_check_functionality() failed\n",
|
|
__func__);
|
|
return -ENODEV;
|
|
}
|
|
|
|
bridge_reset = devm_gpiod_get_optional(&client->dev, "bridge-reset", GPIOD_OUT_LOW);
|
|
if (IS_ERR(bridge_reset))
|
|
return PTR_ERR(bridge_reset);
|
|
if (bridge_reset)
|
|
gpiod_set_consumer_name(bridge_reset, "tc35876x bridge reset");
|
|
|
|
bridge_bl_enable = devm_gpiod_get_optional(&client->dev, "bl-en", GPIOD_OUT_LOW);
|
|
if (IS_ERR(bridge_bl_enable))
|
|
return PTR_ERR(bridge_bl_enable);
|
|
if (bridge_bl_enable)
|
|
gpiod_set_consumer_name(bridge_bl_enable, "tc35876x panel bl en");
|
|
|
|
backlight_voltage = devm_gpiod_get_optional(&client->dev, "vadd", GPIOD_OUT_LOW);
|
|
if (IS_ERR(backlight_voltage))
|
|
return PTR_ERR(backlight_voltage);
|
|
if (backlight_voltage)
|
|
gpiod_set_consumer_name(backlight_voltage, "tc35876x panel vadd");
|
|
|
|
tc35876x_client = client;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tc35876x_bridge_remove(struct i2c_client *client)
|
|
{
|
|
dev_dbg(&client->dev, "%s\n", __func__);
|
|
|
|
tc35876x_client = NULL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct i2c_device_id tc35876x_bridge_id[] = {
|
|
{ "i2c_disp_brig", 0 },
|
|
{ }
|
|
};
|
|
MODULE_DEVICE_TABLE(i2c, tc35876x_bridge_id);
|
|
|
|
static struct i2c_driver tc35876x_bridge_i2c_driver = {
|
|
.driver = {
|
|
.name = "i2c_disp_brig",
|
|
},
|
|
.id_table = tc35876x_bridge_id,
|
|
.probe = tc35876x_bridge_probe,
|
|
.remove = tc35876x_bridge_remove,
|
|
};
|
|
|
|
/* LCD panel I2C */
|
|
static int cmi_lcd_i2c_probe(struct i2c_client *client,
|
|
const struct i2c_device_id *id)
|
|
{
|
|
dev_info(&client->dev, "%s\n", __func__);
|
|
|
|
if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) {
|
|
dev_err(&client->dev, "%s: i2c_check_functionality() failed\n",
|
|
__func__);
|
|
return -ENODEV;
|
|
}
|
|
|
|
cmi_lcd_i2c_client = client;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int cmi_lcd_i2c_remove(struct i2c_client *client)
|
|
{
|
|
dev_dbg(&client->dev, "%s\n", __func__);
|
|
|
|
cmi_lcd_i2c_client = NULL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct i2c_device_id cmi_lcd_i2c_id[] = {
|
|
{ "cmi-lcd", 0 },
|
|
{ }
|
|
};
|
|
MODULE_DEVICE_TABLE(i2c, cmi_lcd_i2c_id);
|
|
|
|
static struct i2c_driver cmi_lcd_i2c_driver = {
|
|
.driver = {
|
|
.name = "cmi-lcd",
|
|
},
|
|
.id_table = cmi_lcd_i2c_id,
|
|
.probe = cmi_lcd_i2c_probe,
|
|
.remove = cmi_lcd_i2c_remove,
|
|
};
|
|
|
|
/* HACK to create I2C device while it's not created by platform code */
|
|
#define CMI_LCD_I2C_ADAPTER 2
|
|
#define CMI_LCD_I2C_ADDR 0x60
|
|
|
|
static int cmi_lcd_hack_create_device(void)
|
|
{
|
|
struct i2c_adapter *adapter;
|
|
struct i2c_client *client;
|
|
struct i2c_board_info info = {
|
|
.type = "cmi-lcd",
|
|
.addr = CMI_LCD_I2C_ADDR,
|
|
};
|
|
|
|
pr_debug("%s\n", __func__);
|
|
|
|
adapter = i2c_get_adapter(CMI_LCD_I2C_ADAPTER);
|
|
if (!adapter) {
|
|
pr_err("%s: i2c_get_adapter(%d) failed\n", __func__,
|
|
CMI_LCD_I2C_ADAPTER);
|
|
return -EINVAL;
|
|
}
|
|
|
|
client = i2c_new_device(adapter, &info);
|
|
if (!client) {
|
|
pr_err("%s: i2c_new_device() failed\n", __func__);
|
|
i2c_put_adapter(adapter);
|
|
return -EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct drm_encoder_helper_funcs tc35876x_encoder_helper_funcs = {
|
|
.dpms = mdfld_dsi_dpi_dpms,
|
|
.mode_fixup = mdfld_dsi_dpi_mode_fixup,
|
|
.prepare = mdfld_dsi_dpi_prepare,
|
|
.mode_set = mdfld_dsi_dpi_mode_set,
|
|
.commit = mdfld_dsi_dpi_commit,
|
|
};
|
|
|
|
static const struct drm_encoder_funcs tc35876x_encoder_funcs = {
|
|
.destroy = drm_encoder_cleanup,
|
|
};
|
|
|
|
const struct panel_funcs mdfld_tc35876x_funcs = {
|
|
.encoder_funcs = &tc35876x_encoder_funcs,
|
|
.encoder_helper_funcs = &tc35876x_encoder_helper_funcs,
|
|
.get_config_mode = tc35876x_get_config_mode,
|
|
.get_panel_info = tc35876x_get_panel_info,
|
|
};
|
|
|
|
void tc35876x_init(struct drm_device *dev)
|
|
{
|
|
int r;
|
|
|
|
dev_dbg(&dev->pdev->dev, "%s\n", __func__);
|
|
|
|
cmi_lcd_hack_create_device();
|
|
|
|
r = i2c_add_driver(&cmi_lcd_i2c_driver);
|
|
if (r < 0)
|
|
dev_err(&dev->pdev->dev,
|
|
"%s: i2c_add_driver() for %s failed (%d)\n",
|
|
__func__, cmi_lcd_i2c_driver.driver.name, r);
|
|
|
|
r = i2c_add_driver(&tc35876x_bridge_i2c_driver);
|
|
if (r < 0)
|
|
dev_err(&dev->pdev->dev,
|
|
"%s: i2c_add_driver() for %s failed (%d)\n",
|
|
__func__, tc35876x_bridge_i2c_driver.driver.name, r);
|
|
|
|
tc35876x_brightness_init(dev);
|
|
}
|
|
|
|
void tc35876x_exit(void)
|
|
{
|
|
pr_debug("%s\n", __func__);
|
|
|
|
i2c_del_driver(&tc35876x_bridge_i2c_driver);
|
|
|
|
if (cmi_lcd_i2c_client)
|
|
i2c_del_driver(&cmi_lcd_i2c_driver);
|
|
}
|