2019-12-30 09:04:19 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2016 MediaTek Inc.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/err.h>
|
|
|
|
#include <linux/gpio/consumer.h>
|
|
|
|
#include <linux/i2c.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/of_graph.h>
|
|
|
|
#include <linux/regulator/consumer.h>
|
|
|
|
|
|
|
|
#include <drm/drm_bridge.h>
|
|
|
|
#include <drm/drm_mipi_dsi.h>
|
|
|
|
#include <drm/drm_of.h>
|
|
|
|
#include <drm/drm_panel.h>
|
|
|
|
#include <drm/drm_print.h>
|
|
|
|
|
|
|
|
#define PAGE2_GPIO_H 0xa7
|
|
|
|
#define PS_GPIO9 BIT(1)
|
|
|
|
#define PAGE2_I2C_BYPASS 0xea
|
|
|
|
#define I2C_BYPASS_EN 0xd0
|
|
|
|
#define PAGE2_MCS_EN 0xf3
|
|
|
|
#define MCS_EN BIT(0)
|
|
|
|
#define PAGE3_SET_ADD 0xfe
|
|
|
|
#define VDO_CTL_ADD 0x13
|
|
|
|
#define VDO_DIS 0x18
|
|
|
|
#define VDO_EN 0x1c
|
|
|
|
#define DP_NUM_LANES 4
|
|
|
|
|
|
|
|
/*
|
|
|
|
* PS8640 uses multiple addresses:
|
|
|
|
* page[0]: for DP control
|
|
|
|
* page[1]: for VIDEO Bridge
|
|
|
|
* page[2]: for control top
|
|
|
|
* page[3]: for DSI Link Control1
|
|
|
|
* page[4]: for MIPI Phy
|
|
|
|
* page[5]: for VPLL
|
|
|
|
* page[6]: for DSI Link Control2
|
|
|
|
* page[7]: for SPI ROM mapping
|
|
|
|
*/
|
|
|
|
enum page_addr_offset {
|
|
|
|
PAGE0_DP_CNTL = 0,
|
|
|
|
PAGE1_VDO_BDG,
|
|
|
|
PAGE2_TOP_CNTL,
|
|
|
|
PAGE3_DSI_CNTL1,
|
|
|
|
PAGE4_MIPI_PHY,
|
|
|
|
PAGE5_VPLL,
|
|
|
|
PAGE6_DSI_CNTL2,
|
|
|
|
PAGE7_SPI_CNTL,
|
|
|
|
MAX_DEVS
|
|
|
|
};
|
|
|
|
|
|
|
|
enum ps8640_vdo_control {
|
|
|
|
DISABLE = VDO_DIS,
|
|
|
|
ENABLE = VDO_EN,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ps8640 {
|
|
|
|
struct drm_bridge bridge;
|
|
|
|
struct drm_bridge *panel_bridge;
|
|
|
|
struct mipi_dsi_device *dsi;
|
|
|
|
struct i2c_client *page[MAX_DEVS];
|
|
|
|
struct regulator_bulk_data supplies[2];
|
|
|
|
struct gpio_desc *gpio_reset;
|
|
|
|
struct gpio_desc *gpio_powerdown;
|
|
|
|
};
|
|
|
|
|
|
|
|
static inline struct ps8640 *bridge_to_ps8640(struct drm_bridge *e)
|
|
|
|
{
|
|
|
|
return container_of(e, struct ps8640, bridge);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ps8640_bridge_vdo_control(struct ps8640 *ps_bridge,
|
|
|
|
const enum ps8640_vdo_control ctrl)
|
|
|
|
{
|
|
|
|
struct i2c_client *client = ps_bridge->page[PAGE3_DSI_CNTL1];
|
|
|
|
u8 vdo_ctrl_buf[] = { VDO_CTL_ADD, ctrl };
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = i2c_smbus_write_i2c_block_data(client, PAGE3_SET_ADD,
|
|
|
|
sizeof(vdo_ctrl_buf),
|
|
|
|
vdo_ctrl_buf);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ps8640_pre_enable(struct drm_bridge *bridge)
|
|
|
|
{
|
|
|
|
struct ps8640 *ps_bridge = bridge_to_ps8640(bridge);
|
|
|
|
struct i2c_client *client = ps_bridge->page[PAGE2_TOP_CNTL];
|
|
|
|
unsigned long timeout;
|
|
|
|
int ret, status;
|
|
|
|
|
|
|
|
ret = regulator_bulk_enable(ARRAY_SIZE(ps_bridge->supplies),
|
|
|
|
ps_bridge->supplies);
|
|
|
|
if (ret < 0) {
|
|
|
|
DRM_ERROR("cannot enable regulators %d\n", ret);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
gpiod_set_value(ps_bridge->gpio_powerdown, 0);
|
|
|
|
gpiod_set_value(ps_bridge->gpio_reset, 1);
|
|
|
|
usleep_range(2000, 2500);
|
|
|
|
gpiod_set_value(ps_bridge->gpio_reset, 0);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Wait for the ps8640 embedded MCU to be ready
|
|
|
|
* First wait 200ms and then check the MCU ready flag every 20ms
|
|
|
|
*/
|
|
|
|
msleep(200);
|
|
|
|
|
|
|
|
timeout = jiffies + msecs_to_jiffies(200) + 1;
|
|
|
|
|
|
|
|
while (time_is_after_jiffies(timeout)) {
|
|
|
|
status = i2c_smbus_read_byte_data(client, PAGE2_GPIO_H);
|
|
|
|
if (status < 0) {
|
|
|
|
DRM_ERROR("failed read PAGE2_GPIO_H: %d\n", status);
|
|
|
|
goto err_regulators_disable;
|
|
|
|
}
|
|
|
|
if ((status & PS_GPIO9) == PS_GPIO9)
|
|
|
|
break;
|
|
|
|
|
|
|
|
msleep(20);
|
|
|
|
}
|
|
|
|
|
|
|
|
msleep(50);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The Manufacturer Command Set (MCS) is a device dependent interface
|
|
|
|
* intended for factory programming of the display module default
|
|
|
|
* parameters. Once the display module is configured, the MCS shall be
|
|
|
|
* disabled by the manufacturer. Once disabled, all MCS commands are
|
|
|
|
* ignored by the display interface.
|
|
|
|
*/
|
|
|
|
status = i2c_smbus_read_byte_data(client, PAGE2_MCS_EN);
|
|
|
|
if (status < 0) {
|
|
|
|
DRM_ERROR("failed read PAGE2_MCS_EN: %d\n", status);
|
|
|
|
goto err_regulators_disable;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = i2c_smbus_write_byte_data(client, PAGE2_MCS_EN,
|
|
|
|
status & ~MCS_EN);
|
|
|
|
if (ret < 0) {
|
|
|
|
DRM_ERROR("failed write PAGE2_MCS_EN: %d\n", ret);
|
|
|
|
goto err_regulators_disable;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = ps8640_bridge_vdo_control(ps_bridge, ENABLE);
|
|
|
|
if (ret) {
|
|
|
|
DRM_ERROR("failed to enable VDO: %d\n", ret);
|
|
|
|
goto err_regulators_disable;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Switch access edp panel's edid through i2c */
|
|
|
|
ret = i2c_smbus_write_byte_data(client, PAGE2_I2C_BYPASS,
|
|
|
|
I2C_BYPASS_EN);
|
|
|
|
if (ret < 0) {
|
|
|
|
DRM_ERROR("failed write PAGE2_I2C_BYPASS: %d\n", ret);
|
|
|
|
goto err_regulators_disable;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
err_regulators_disable:
|
|
|
|
regulator_bulk_disable(ARRAY_SIZE(ps_bridge->supplies),
|
|
|
|
ps_bridge->supplies);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ps8640_post_disable(struct drm_bridge *bridge)
|
|
|
|
{
|
|
|
|
struct ps8640 *ps_bridge = bridge_to_ps8640(bridge);
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = ps8640_bridge_vdo_control(ps_bridge, DISABLE);
|
|
|
|
if (ret < 0)
|
|
|
|
DRM_ERROR("failed to disable VDO: %d\n", ret);
|
|
|
|
|
|
|
|
gpiod_set_value(ps_bridge->gpio_reset, 1);
|
|
|
|
gpiod_set_value(ps_bridge->gpio_powerdown, 1);
|
|
|
|
ret = regulator_bulk_disable(ARRAY_SIZE(ps_bridge->supplies),
|
|
|
|
ps_bridge->supplies);
|
|
|
|
if (ret < 0)
|
|
|
|
DRM_ERROR("cannot disable regulators %d\n", ret);
|
|
|
|
}
|
|
|
|
|
drm/bridge: Extend bridge API to disable connector creation
Most bridge drivers create a DRM connector to model the connector at the
output of the bridge. This model is historical and has worked pretty
well so far, but causes several issues:
- It prevents supporting more complex display pipelines where DRM
connector operations are split over multiple components. For instance a
pipeline with a bridge connected to the DDC signals to read EDID data,
and another one connected to the HPD signal to detect connection and
disconnection, will not be possible to support through this model.
- It requires every bridge driver to implement similar connector
handling code, resulting in code duplication.
- It assumes that a bridge will either be wired to a connector or to
another bridge, but doesn't support bridges that can be used in both
positions very well (although there is some ad-hoc support for this in
the analogix_dp bridge driver).
In order to solve these issues, ownership of the connector should be
moved to the display controller driver (where it can be implemented
using helpers provided by the core).
Extend the bridge API to allow disabling connector creation in bridge
drivers as a first step towards the new model. The new flags argument to
the bridge .attach() operation allows instructing the bridge driver to
skip creating a connector. Unconditionally set the new flags argument to
0 for now to keep the existing behaviour, and modify all existing bridge
drivers to return an error when connector creation is not requested as
they don't support this feature yet.
The change is based on the following semantic patch, with manual review
and edits.
@ rule1 @
identifier funcs;
identifier fn;
@@
struct drm_bridge_funcs funcs = {
...,
.attach = fn
};
@ depends on rule1 @
identifier rule1.fn;
identifier bridge;
statement S, S1;
@@
int fn(
struct drm_bridge *bridge
+ , enum drm_bridge_attach_flags flags
)
{
... when != S
+ if (flags & DRM_BRIDGE_ATTACH_NO_CONNECTOR) {
+ DRM_ERROR("Fix bridge driver to make connector optional!");
+ return -EINVAL;
+ }
+
S1
...
}
@ depends on rule1 @
identifier rule1.fn;
identifier bridge, flags;
expression E1, E2, E3;
@@
int fn(
struct drm_bridge *bridge,
enum drm_bridge_attach_flags flags
) {
<...
drm_bridge_attach(E1, E2, E3
+ , flags
)
...>
}
@@
expression E1, E2, E3;
@@
drm_bridge_attach(E1, E2, E3
+ , 0
)
Signed-off-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
Reviewed-by: Boris Brezillon <boris.brezillon@collabora.com>
Acked-by: Sam Ravnborg <sam@ravnborg.org>
Reviewed-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
Tested-by: Sebastian Reichel <sebastian.reichel@collabora.com>
Reviewed-by: Sebastian Reichel <sebastian.reichel@collabora.com>
Acked-by: Daniel Vetter <daniel.vetter@ffwll.ch>
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200226112514.12455-10-laurent.pinchart@ideasonboard.com
2020-02-26 11:24:29 +00:00
|
|
|
static int ps8640_bridge_attach(struct drm_bridge *bridge,
|
|
|
|
enum drm_bridge_attach_flags flags)
|
2019-12-30 09:04:19 +00:00
|
|
|
{
|
|
|
|
struct ps8640 *ps_bridge = bridge_to_ps8640(bridge);
|
|
|
|
struct device *dev = &ps_bridge->page[0]->dev;
|
|
|
|
struct device_node *in_ep, *dsi_node;
|
|
|
|
struct mipi_dsi_device *dsi;
|
|
|
|
struct mipi_dsi_host *host;
|
|
|
|
int ret;
|
|
|
|
const struct mipi_dsi_device_info info = { .type = "ps8640",
|
|
|
|
.channel = 0,
|
|
|
|
.node = NULL,
|
|
|
|
};
|
|
|
|
/* port@0 is ps8640 dsi input port */
|
|
|
|
in_ep = of_graph_get_endpoint_by_regs(dev->of_node, 0, -1);
|
|
|
|
if (!in_ep)
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
dsi_node = of_graph_get_remote_port_parent(in_ep);
|
|
|
|
of_node_put(in_ep);
|
|
|
|
if (!dsi_node)
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
host = of_find_mipi_dsi_host_by_node(dsi_node);
|
|
|
|
of_node_put(dsi_node);
|
|
|
|
if (!host)
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
dsi = mipi_dsi_device_register_full(host, &info);
|
|
|
|
if (IS_ERR(dsi)) {
|
|
|
|
dev_err(dev, "failed to create dsi device\n");
|
|
|
|
ret = PTR_ERR(dsi);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ps_bridge->dsi = dsi;
|
|
|
|
|
|
|
|
dsi->host = host;
|
|
|
|
dsi->mode_flags = MIPI_DSI_MODE_VIDEO |
|
|
|
|
MIPI_DSI_MODE_VIDEO_SYNC_PULSE;
|
|
|
|
dsi->format = MIPI_DSI_FMT_RGB888;
|
|
|
|
dsi->lanes = DP_NUM_LANES;
|
|
|
|
ret = mipi_dsi_attach(dsi);
|
|
|
|
if (ret)
|
|
|
|
goto err_dsi_attach;
|
|
|
|
|
|
|
|
/* Attach the panel-bridge to the dsi bridge */
|
|
|
|
return drm_bridge_attach(bridge->encoder, ps_bridge->panel_bridge,
|
drm/bridge: Extend bridge API to disable connector creation
Most bridge drivers create a DRM connector to model the connector at the
output of the bridge. This model is historical and has worked pretty
well so far, but causes several issues:
- It prevents supporting more complex display pipelines where DRM
connector operations are split over multiple components. For instance a
pipeline with a bridge connected to the DDC signals to read EDID data,
and another one connected to the HPD signal to detect connection and
disconnection, will not be possible to support through this model.
- It requires every bridge driver to implement similar connector
handling code, resulting in code duplication.
- It assumes that a bridge will either be wired to a connector or to
another bridge, but doesn't support bridges that can be used in both
positions very well (although there is some ad-hoc support for this in
the analogix_dp bridge driver).
In order to solve these issues, ownership of the connector should be
moved to the display controller driver (where it can be implemented
using helpers provided by the core).
Extend the bridge API to allow disabling connector creation in bridge
drivers as a first step towards the new model. The new flags argument to
the bridge .attach() operation allows instructing the bridge driver to
skip creating a connector. Unconditionally set the new flags argument to
0 for now to keep the existing behaviour, and modify all existing bridge
drivers to return an error when connector creation is not requested as
they don't support this feature yet.
The change is based on the following semantic patch, with manual review
and edits.
@ rule1 @
identifier funcs;
identifier fn;
@@
struct drm_bridge_funcs funcs = {
...,
.attach = fn
};
@ depends on rule1 @
identifier rule1.fn;
identifier bridge;
statement S, S1;
@@
int fn(
struct drm_bridge *bridge
+ , enum drm_bridge_attach_flags flags
)
{
... when != S
+ if (flags & DRM_BRIDGE_ATTACH_NO_CONNECTOR) {
+ DRM_ERROR("Fix bridge driver to make connector optional!");
+ return -EINVAL;
+ }
+
S1
...
}
@ depends on rule1 @
identifier rule1.fn;
identifier bridge, flags;
expression E1, E2, E3;
@@
int fn(
struct drm_bridge *bridge,
enum drm_bridge_attach_flags flags
) {
<...
drm_bridge_attach(E1, E2, E3
+ , flags
)
...>
}
@@
expression E1, E2, E3;
@@
drm_bridge_attach(E1, E2, E3
+ , 0
)
Signed-off-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
Reviewed-by: Boris Brezillon <boris.brezillon@collabora.com>
Acked-by: Sam Ravnborg <sam@ravnborg.org>
Reviewed-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
Tested-by: Sebastian Reichel <sebastian.reichel@collabora.com>
Reviewed-by: Sebastian Reichel <sebastian.reichel@collabora.com>
Acked-by: Daniel Vetter <daniel.vetter@ffwll.ch>
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200226112514.12455-10-laurent.pinchart@ideasonboard.com
2020-02-26 11:24:29 +00:00
|
|
|
&ps_bridge->bridge, flags);
|
2019-12-30 09:04:19 +00:00
|
|
|
|
|
|
|
err_dsi_attach:
|
|
|
|
mipi_dsi_device_unregister(dsi);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct drm_bridge_funcs ps8640_bridge_funcs = {
|
|
|
|
.attach = ps8640_bridge_attach,
|
|
|
|
.post_disable = ps8640_post_disable,
|
|
|
|
.pre_enable = ps8640_pre_enable,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int ps8640_probe(struct i2c_client *client)
|
|
|
|
{
|
|
|
|
struct device *dev = &client->dev;
|
|
|
|
struct device_node *np = dev->of_node;
|
|
|
|
struct ps8640 *ps_bridge;
|
|
|
|
struct drm_panel *panel;
|
|
|
|
int ret;
|
|
|
|
u32 i;
|
|
|
|
|
|
|
|
ps_bridge = devm_kzalloc(dev, sizeof(*ps_bridge), GFP_KERNEL);
|
|
|
|
if (!ps_bridge)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
/* port@1 is ps8640 output port */
|
|
|
|
ret = drm_of_find_panel_or_bridge(np, 1, 0, &panel, NULL);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
if (!panel)
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
ps_bridge->panel_bridge = devm_drm_panel_bridge_add(dev, panel);
|
|
|
|
if (IS_ERR(ps_bridge->panel_bridge))
|
|
|
|
return PTR_ERR(ps_bridge->panel_bridge);
|
|
|
|
|
|
|
|
ps_bridge->supplies[0].supply = "vdd33";
|
|
|
|
ps_bridge->supplies[1].supply = "vdd12";
|
|
|
|
ret = devm_regulator_bulk_get(dev, ARRAY_SIZE(ps_bridge->supplies),
|
|
|
|
ps_bridge->supplies);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ps_bridge->gpio_powerdown = devm_gpiod_get(&client->dev, "powerdown",
|
|
|
|
GPIOD_OUT_HIGH);
|
|
|
|
if (IS_ERR(ps_bridge->gpio_powerdown))
|
|
|
|
return PTR_ERR(ps_bridge->gpio_powerdown);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Assert the reset to avoid the bridge being initialized prematurely
|
|
|
|
*/
|
|
|
|
ps_bridge->gpio_reset = devm_gpiod_get(&client->dev, "reset",
|
|
|
|
GPIOD_OUT_HIGH);
|
|
|
|
if (IS_ERR(ps_bridge->gpio_reset))
|
|
|
|
return PTR_ERR(ps_bridge->gpio_reset);
|
|
|
|
|
|
|
|
ps_bridge->bridge.funcs = &ps8640_bridge_funcs;
|
|
|
|
ps_bridge->bridge.of_node = dev->of_node;
|
|
|
|
|
|
|
|
ps_bridge->page[PAGE0_DP_CNTL] = client;
|
|
|
|
|
|
|
|
for (i = 1; i < ARRAY_SIZE(ps_bridge->page); i++) {
|
|
|
|
ps_bridge->page[i] = devm_i2c_new_dummy_device(&client->dev,
|
|
|
|
client->adapter,
|
|
|
|
client->addr + i);
|
|
|
|
if (IS_ERR(ps_bridge->page[i])) {
|
|
|
|
dev_err(dev, "failed i2c dummy device, address %02x\n",
|
|
|
|
client->addr + i);
|
|
|
|
return PTR_ERR(ps_bridge->page[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
i2c_set_clientdata(client, ps_bridge);
|
|
|
|
|
|
|
|
drm_bridge_add(&ps_bridge->bridge);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ps8640_remove(struct i2c_client *client)
|
|
|
|
{
|
|
|
|
struct ps8640 *ps_bridge = i2c_get_clientdata(client);
|
|
|
|
|
|
|
|
drm_bridge_remove(&ps_bridge->bridge);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct of_device_id ps8640_match[] = {
|
|
|
|
{ .compatible = "parade,ps8640" },
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(of, ps8640_match);
|
|
|
|
|
|
|
|
static struct i2c_driver ps8640_driver = {
|
|
|
|
.probe_new = ps8640_probe,
|
|
|
|
.remove = ps8640_remove,
|
|
|
|
.driver = {
|
|
|
|
.name = "ps8640",
|
|
|
|
.of_match_table = ps8640_match,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
module_i2c_driver(ps8640_driver);
|
|
|
|
|
|
|
|
MODULE_AUTHOR("Jitao Shi <jitao.shi@mediatek.com>");
|
|
|
|
MODULE_AUTHOR("CK Hu <ck.hu@mediatek.com>");
|
|
|
|
MODULE_AUTHOR("Enric Balletbo i Serra <enric.balletbo@collabora.com>");
|
|
|
|
MODULE_DESCRIPTION("PARADE ps8640 DSI-eDP converter driver");
|
|
|
|
MODULE_LICENSE("GPL v2");
|