59616c830f
* Reimplement dsi85 as DRM bridge (squashed). - blunt import of preliminary bridge driver - make dsi85 a kernel extension instead of a module to circumvent problems with DRM init order - make i2c matchstring match our DTS - find and use associated drm_panel - add hardcoded DSI85 init in bridge-enable - Do not drm_connector_register yet at bridge_attach time (unlike other bridge drivers) - Add a mipi_dsi_device to configure #lanes The DSI host wants to know the number of lanes, and guesses 1 lane if not given. The old attempt using just panel-simple was easy; a panel declared as DSI panel in panel-simple gets a mipi_dsi_device which it can configure. This driver follows the example of the adv7533 driver and creates its own mipi_dsi_device, where it places a hard-coded number of lanes. - cleanup * Compute DSI85 register values from Panel At bridge_mode_set time, fill all logical DSI85 registers from panel parameters and constant assumptions, then write logical registers to physical registers. Still needs override for a few logical registers: related to DSI clock, and related to horizontal parameters which our panel_simple contains with factor 2 to force proper DSI timings. * compute DSI input clock from DT attributes and mode * add debugging mode_fixup handler to bridge * add DT attrs for more DSI85 registers sync delay, DSI clock divider. And cleanup. * remove bridge_mode_fixup debug helper * use DRM logging macros uniformly * Derive sync polarity from panel's flags but make sure that DSI always uses standard polarity; fix that in the mode_fixup callback. * fix computation of dsi_clk_divider Code assumed wrong bit position in 0x0B, but with the right bit position, the divider is trivial to compute from existing params. There goes one DT attribute. * make LVDS regs 0x19..0x1B configurable by DT * add documentation
922 lines
27 KiB
C
Executable file
922 lines
27 KiB
C
Executable file
/*
|
|
* Copyright (C) 2017, Hella-Gutmann Solutions GmbH
|
|
*
|
|
* Partly based on panel-dsi85.c from:
|
|
*
|
|
* Copyright 2011 Texas Instruments, Inc.
|
|
* Author: Archit Taneja <archit@ti.com>
|
|
* based on d2l panel driver by Jerry Alexander <x0135174@ti.com>
|
|
*
|
|
* Also partly based on the ptn3460 and adv7533 kernel drivers.
|
|
*
|
|
* This program iss free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 as published by
|
|
* the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
* more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along with
|
|
* this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <linux/module.h>
|
|
#include <linux/debugfs.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/err.h>
|
|
#include <linux/jiffies.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/seq_file.h>
|
|
#include <linux/backlight.h>
|
|
#include <linux/fb.h>
|
|
#include <linux/interrupt.h>
|
|
#include <linux/gpio.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/regulator/consumer.h>
|
|
#include <linux/mutex.h>
|
|
#include <linux/i2c.h>
|
|
#include <linux/gpio/consumer.h>
|
|
#include <linux/module.h>
|
|
#include <linux/of_platform.h>
|
|
#include <linux/of_graph.h>
|
|
#include <linux/platform_device.h>
|
|
|
|
#include <drm/drmP.h>
|
|
#include <drm/drm_crtc.h>
|
|
#include <drm/drm_mipi_dsi.h>
|
|
#include <drm/drm_panel.h>
|
|
#include <drm/drm_edid.h>
|
|
#include <drm/drm_atomic_helper.h>
|
|
#include <drm/drm_crtc_helper.h>
|
|
|
|
#include <video/mipi_display.h>
|
|
|
|
#include "dsi85_registers.h"
|
|
|
|
#define SN65DSI85_MODULE_NAME "bridge-sn65dsi85"
|
|
|
|
/* Config struct
|
|
* (to be filled from DT. See dsi85.txt.)
|
|
*/
|
|
struct sn65dsi85_config {
|
|
u32 lanes;
|
|
u32 lvds_channels;
|
|
u32 sync_delay;
|
|
|
|
u32 de_neg_polarity;
|
|
u32 force_24bpp_mode;
|
|
u32 force_24bpp_format1;
|
|
u32 lvds_vocm;
|
|
u32 lvds_vod_swing_cha;
|
|
u32 lvds_vod_swing_chb;
|
|
u32 lvds_even_odd_swap;
|
|
u32 lvds_reverse;
|
|
u32 lvds_term;
|
|
u32 lvds_cm_adjust_cha;
|
|
u32 lvds_cm_adjust_chb;
|
|
};
|
|
|
|
/*
|
|
* Device state
|
|
*/
|
|
struct sn65dsi85_device {
|
|
struct sn65dsi85_config *config;
|
|
struct i2c_client *i2c;
|
|
const struct i2c_device_id *i2c_id;
|
|
struct drm_bridge bridge;
|
|
struct drm_connector connector;
|
|
struct drm_panel *panel;
|
|
struct device_node *dsi_host_node;
|
|
struct mipi_dsi_device *dsi;
|
|
};
|
|
|
|
/* Registers of the DSI85, named like in the data sheet.
|
|
* except for _LO/_HI pairs, which appear as one u16,
|
|
* and TPG-only registers, which are omitted. */
|
|
struct sn65dsi85_regs {
|
|
u8 pll_en_stat;
|
|
u8 lvds_clk_range;
|
|
u8 hs_clk_src;
|
|
u8 dsi_clk_divider;
|
|
u8 refclk_multiplier;
|
|
|
|
u8 left_right_pixels;
|
|
u8 dsi_channel_mode;
|
|
u8 cha_dsi_lanes;
|
|
u8 chb_dsi_lanes;
|
|
u8 sot_err_tol_dis;
|
|
u8 cha_dsi_data_eq;
|
|
u8 chb_dsi_data_eq;
|
|
u8 cha_dsi_clk_eq;
|
|
u8 chb_dsi_clk_eq;
|
|
|
|
u8 cha_dsi_clk_rng;
|
|
u8 chb_dsi_clk_rng;
|
|
|
|
u8 de_neg_polarity;
|
|
u8 hs_neg_polarity;
|
|
u8 vs_neg_polarity;
|
|
u8 lvds_link_cfg;
|
|
u8 cha_24bpp_mode;
|
|
u8 chb_24bpp_mode;
|
|
u8 cha_24bpp_format1;
|
|
u8 chb_24bpp_format1;
|
|
|
|
u8 cha_lvds_vocm;
|
|
u8 chb_lvds_vocm;
|
|
u8 cha_lvds_vod_swing;
|
|
u8 chb_lvds_vod_swing;
|
|
|
|
u8 even_odd_swap;
|
|
u8 cha_reverse_lvds;
|
|
u8 chb_reverse_lvds;
|
|
u8 cha_lvds_term;
|
|
u8 chb_lvds_term;
|
|
|
|
u8 cha_lvds_cm_adjust;
|
|
u8 chb_lvds_cm_adjust;
|
|
|
|
u16 cha_active_line_length;
|
|
u16 chb_active_line_length;
|
|
|
|
u16 cha_sync_delay;
|
|
u16 chb_sync_delay;
|
|
|
|
u16 cha_hsync_pulse_width;
|
|
u16 chb_hsync_pulse_width;
|
|
|
|
u16 cha_vsync_pulse_width;
|
|
u16 chb_vsync_pulse_width;
|
|
|
|
u8 cha_horizontal_back_porch;
|
|
u8 chb_horizontal_back_porch;
|
|
|
|
/* omitted: TPG-related */
|
|
};
|
|
|
|
static inline struct sn65dsi85_device *bridge_to_sn65dsi85(struct drm_bridge *bridge)
|
|
{
|
|
return container_of(bridge, struct sn65dsi85_device, bridge);
|
|
}
|
|
static inline struct sn65dsi85_device *connector_to_sn65dsi85(struct drm_connector *connector)
|
|
{
|
|
return container_of(connector, struct sn65dsi85_device, connector);
|
|
}
|
|
|
|
static int dsi85_hardcoded_deinit(struct i2c_client* client)
|
|
{
|
|
/* disable pll */
|
|
i2c_smbus_write_byte_data(client, 0x0D, 0x00);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Read configuration from metadata
|
|
*/
|
|
static struct sn65dsi85_config*
|
|
sn65dsi85_get_pdata(struct i2c_client *client)
|
|
{
|
|
struct device_node *dsi85_node = client->dev.of_node;
|
|
struct device_node *endpoint = NULL;
|
|
struct sn65dsi85_config *pdata = NULL;
|
|
|
|
DRM_DEV_DEBUG(&client->dev, "Consulting DT node %s", dsi85_node->full_name);
|
|
|
|
endpoint = of_graph_get_next_endpoint(dsi85_node, NULL);
|
|
if(!endpoint) {
|
|
DRM_DEV_ERROR(&client->dev, "Cannot find OF endpoint\n");
|
|
return NULL;
|
|
}
|
|
|
|
pdata = devm_kzalloc(&client->dev, sizeof(*pdata), GFP_KERNEL);
|
|
if(!pdata) {
|
|
DRM_DEV_ERROR(&client->dev, "Failed to kzalloc the config object");
|
|
goto out_put_endpoint;
|
|
}
|
|
|
|
/* read DT attributes into Config */
|
|
if(of_property_read_u32(dsi85_node, "dsi-lanes", &pdata->lanes) < 0) {
|
|
DRM_DEV_INFO(&client->dev, "DT: dsi-lanes property not found, using default\n");
|
|
pdata->lanes = 4;
|
|
}
|
|
if(of_property_read_u32(dsi85_node, "lvds-channels", &pdata->lvds_channels) < 0) {
|
|
DRM_DEV_INFO(&client->dev, "DT: lvds-channels property not found, using default\n");
|
|
pdata->lvds_channels = 1;
|
|
} else {
|
|
if( pdata->lvds_channels < 1 || pdata->lvds_channels > 2 ) {
|
|
DRM_DEV_ERROR(&client->dev, "DT: lvds-channels must be 1 or 2, not %u", pdata->lvds_channels);
|
|
goto out_free;
|
|
}
|
|
}
|
|
of_property_read_u32(dsi85_node, "sync-delay", &pdata->sync_delay);
|
|
|
|
of_property_read_u32(dsi85_node, "de-neg-polarity", &pdata->de_neg_polarity);
|
|
of_property_read_u32(dsi85_node, "force-24bpp-mode", &pdata->force_24bpp_mode);
|
|
of_property_read_u32(dsi85_node, "force-24bpp-format1", &pdata->force_24bpp_format1);
|
|
of_property_read_u32(dsi85_node, "lvds-vocm", &pdata->lvds_vocm);
|
|
of_property_read_u32(dsi85_node, "lvds-vod-swing-cha", &pdata->lvds_vod_swing_cha);
|
|
of_property_read_u32(dsi85_node, "lvds-vod-swing-chb", &pdata->lvds_vod_swing_chb);
|
|
of_property_read_u32(dsi85_node, "lvds-even-odd-swap", &pdata->lvds_even_odd_swap);
|
|
of_property_read_u32(dsi85_node, "lvds-reverse", &pdata->lvds_reverse);
|
|
of_property_read_u32(dsi85_node, "lvds-term", &pdata->lvds_term);
|
|
of_property_read_u32(dsi85_node, "lvds-cm-adjust-cha", &pdata->lvds_cm_adjust_cha);
|
|
of_property_read_u32(dsi85_node, "lvds-cm-adjust-chb", &pdata->lvds_cm_adjust_chb);
|
|
|
|
DRM_DEV_INFO(&client->dev, "DT attribute result: dsi-lanes=%d lvds-channels=%d sync-delay=%u",
|
|
pdata->lanes, pdata->lvds_channels, pdata->sync_delay);
|
|
of_node_put(endpoint);
|
|
return pdata;
|
|
|
|
out_free: /* Sorry, DT node is invalid, bailing out. */
|
|
devm_kfree(&client->dev, pdata);
|
|
pdata = NULL;
|
|
|
|
out_put_endpoint:
|
|
of_node_put(endpoint);
|
|
return pdata;
|
|
|
|
}
|
|
|
|
static enum drm_connector_status sn65dsi85_connector_detect(struct drm_connector *connector,
|
|
bool force)
|
|
{
|
|
DRM_DEV_DEBUG(connector->dev->dev, "%s entry", __func__);
|
|
return connector_status_connected;
|
|
}
|
|
|
|
static void sn65dsi85_connector_destroy(struct drm_connector* connector)
|
|
{
|
|
DRM_DEV_DEBUG(connector->dev->dev, "%s entry", __func__);
|
|
drm_connector_cleanup(connector);
|
|
}
|
|
|
|
static struct drm_connector_funcs sn65dsi85_connector_funcs = {
|
|
.dpms = drm_atomic_helper_connector_dpms,
|
|
.fill_modes = drm_helper_probe_single_connector_modes,
|
|
.detect = sn65dsi85_connector_detect,
|
|
.destroy = sn65dsi85_connector_destroy,
|
|
.reset = drm_atomic_helper_connector_reset,
|
|
.atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state,
|
|
.atomic_destroy_state = drm_atomic_helper_connector_destroy_state,
|
|
};
|
|
|
|
static struct drm_encoder *sn65dsi85_best_encoder(struct drm_connector* connector)
|
|
{
|
|
struct sn65dsi85_device *self = connector_to_sn65dsi85(connector);
|
|
/* The bridge's encoder is the best encoder */
|
|
return self->bridge.encoder;
|
|
}
|
|
|
|
/* Get modes (connector_helper callback), delegated to Panel */
|
|
static int sn65dsi85_get_modes(struct drm_connector* connector)
|
|
{
|
|
struct sn65dsi85_device *self;
|
|
int num_modes = 0;
|
|
DRM_DEV_DEBUG(connector->dev->dev, "%s entry connector=%p", __func__, connector);
|
|
|
|
if(!connector) {
|
|
pr_err("Null connector in sn65dsi85_get_modes");
|
|
return num_modes;
|
|
}
|
|
self = connector_to_sn65dsi85(connector);
|
|
|
|
if (self->panel && self->panel->funcs && self->panel->funcs->get_modes) {
|
|
num_modes = self->panel->funcs->get_modes(self->panel);
|
|
if (num_modes > 0) {
|
|
DRM_DEV_DEBUG(connector->dev->dev, "%s got %d modes from panel, good enough", __func__, num_modes);
|
|
}
|
|
else {
|
|
DRM_WARN("got 0 modes from panel");
|
|
}
|
|
}
|
|
else {
|
|
DRM_WARN("could not interrogate drm_panel for modes");
|
|
}
|
|
|
|
return num_modes;
|
|
}
|
|
|
|
static struct drm_connector_helper_funcs sn65dsi85_connector_helper_funcs = {
|
|
.get_modes = sn65dsi85_get_modes,
|
|
.best_encoder = sn65dsi85_best_encoder,
|
|
};
|
|
|
|
static void sn65dsi85_bridge_pre_enable(struct drm_bridge* bridge)
|
|
{
|
|
struct sn65dsi85_device *self = bridge_to_sn65dsi85(bridge);
|
|
int res;
|
|
DRM_DEV_DEBUG(bridge->dev->dev, "%s entry", __func__);
|
|
|
|
res = drm_panel_prepare(self->panel);
|
|
if (res) {
|
|
DRM_ERROR("failed to prepare my panel: %d\n", res);
|
|
return;
|
|
}
|
|
|
|
}
|
|
static void sn65dsi85_bridge_enable(struct drm_bridge* bridge)
|
|
{
|
|
struct sn65dsi85_device *self = bridge_to_sn65dsi85(bridge);
|
|
int res;
|
|
DRM_DEV_DEBUG(bridge->dev->dev, "%s entry", __func__);
|
|
|
|
DRM_DEV_INFO(bridge->dev->dev, "%s: time for PLL-Enable", __func__);
|
|
i2c_smbus_write_byte_data(self->i2c, 0x0D, 0x01);
|
|
i2c_smbus_write_byte_data(self->i2c, 0x09, 0x01);
|
|
DRM_DEV_INFO(bridge->dev->dev, "%s: PLL-Enable done", __func__);
|
|
|
|
res = drm_panel_enable(self->panel);
|
|
if (res) {
|
|
DRM_ERROR("failed to enable my panel: %d\n", res);
|
|
return;
|
|
}
|
|
}
|
|
static void sn65dsi85_bridge_disable(struct drm_bridge* bridge)
|
|
{
|
|
struct sn65dsi85_device *self = bridge_to_sn65dsi85(bridge);
|
|
int res;
|
|
DRM_DEV_DEBUG(bridge->dev->dev, "%s entry", __func__);
|
|
|
|
DRM_DEV_INFO(bridge->dev->dev, "Time for hardcoded deinit");
|
|
dsi85_hardcoded_deinit(self->i2c);
|
|
DRM_DEV_INFO(bridge->dev->dev, "Done hardcoded deinit");
|
|
|
|
res = drm_panel_disable(self->panel);
|
|
if (res) {
|
|
DRM_ERROR("failed to disable my panel: %d\n", res);
|
|
return;
|
|
}
|
|
}
|
|
static void sn65dsi85_bridge_post_disable(struct drm_bridge* bridge)
|
|
{
|
|
struct sn65dsi85_device *self = bridge_to_sn65dsi85(bridge);
|
|
int res;
|
|
DRM_DEV_DEBUG(bridge->dev->dev, "%s entry", __func__);
|
|
|
|
res = drm_panel_unprepare(self->panel);
|
|
if (res) {
|
|
DRM_ERROR("failed to unprepare my panel: %d\n", res);
|
|
return;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Configure DSI link properties by creating a mipi_dsi_device
|
|
*/
|
|
int sn65dsi85_attach_dsi(struct sn65dsi85_device *self)
|
|
{
|
|
struct device *dev = &self->i2c->dev;
|
|
struct mipi_dsi_host *host;
|
|
struct mipi_dsi_device *dsi;
|
|
int ret = 0;
|
|
const struct mipi_dsi_device_info info = { .type = "sn65dsi85_dsi_client",
|
|
.channel = 0,
|
|
.node = NULL,
|
|
};
|
|
|
|
host = of_find_mipi_dsi_host_by_node(self->dsi_host_node);
|
|
if (!host) {
|
|
DRM_DEV_ERROR(dev, "%s failed to find dsi host by OF node %s\n", __func__,
|
|
self->dsi_host_node? self->dsi_host_node->full_name : "null-OF-node");
|
|
return -EPROBE_DEFER;
|
|
}
|
|
|
|
dsi = mipi_dsi_device_register_full(host, &info);
|
|
if (IS_ERR(dsi)) {
|
|
ret = PTR_ERR(dsi);
|
|
DRM_DEV_ERROR(dev, "%s failed to create dsi device: %d\n", __func__, ret);
|
|
goto err_dsi_device;
|
|
}
|
|
|
|
self->dsi = dsi;
|
|
|
|
/* In the simple driver, this was part of the panel_desc_dsi instance: */
|
|
dsi->lanes = self->config->lanes;
|
|
dsi->format = MIPI_DSI_FMT_RGB888; /* maybe from DT later */
|
|
dsi->mode_flags = MIPI_DSI_MODE_VIDEO; /* maybe from DT later */
|
|
|
|
ret = mipi_dsi_attach(dsi);
|
|
if (ret < 0) {
|
|
DRM_DEV_ERROR(dev, "%s failed to attach dsi to host: %d\n", __func__, ret);
|
|
goto err_dsi_attach;
|
|
}
|
|
|
|
return 0;
|
|
|
|
err_dsi_attach:
|
|
mipi_dsi_device_unregister(dsi);
|
|
err_dsi_device:
|
|
return ret;
|
|
}
|
|
|
|
void sn65dsi85_detach_dsi(struct sn65dsi85_device *self)
|
|
{
|
|
mipi_dsi_detach(self->dsi);
|
|
mipi_dsi_device_unregister(self->dsi);
|
|
}
|
|
|
|
/*
|
|
* Main Bridge config callback.
|
|
*
|
|
* Create a DRM Connector which answers queries for supported modes
|
|
* (by delegating to the Panel, in our case).
|
|
* Create a DSI Device which holds the DSI protocol details.
|
|
*/
|
|
static int sn65dsi85_bridge_attach(struct drm_bridge* bridge)
|
|
{
|
|
struct sn65dsi85_device* self = bridge_to_sn65dsi85(bridge);
|
|
int ret = 0;
|
|
|
|
DRM_DEV_DEBUG(bridge->dev->dev, "%s entry", __func__);
|
|
|
|
if(!bridge->encoder) {
|
|
DRM_ERROR("Bridge has no Encoder");
|
|
return -ENODEV;
|
|
}
|
|
|
|
self->connector.polled = 0; // no hotplugging here
|
|
ret = drm_connector_init(bridge->dev, &self->connector,
|
|
&sn65dsi85_connector_funcs,
|
|
DRM_MODE_CONNECTOR_LVDS);
|
|
if(ret) {
|
|
DRM_ERROR("Could not drm_connector_init: %d", ret);
|
|
return ret;
|
|
}
|
|
drm_connector_helper_add(&self->connector, &sn65dsi85_connector_helper_funcs);
|
|
|
|
/* Tempting to do right now, but not a good idea:
|
|
* drm_connector_register(&self->connector);
|
|
* Wants to create kobject nodes, but the connector has no kobject parent as of yet.
|
|
* The kobject parent will be assigned later,
|
|
* but drm_connector_init has already added our connector
|
|
* to its internal list, and there will be a call to drm_connector_register_all later.
|
|
*/
|
|
|
|
ret = drm_mode_connector_attach_encoder(&self->connector, self->bridge.encoder);
|
|
if(ret) {
|
|
DRM_ERROR("Could not drm_mode_connector_attach_encoder: %d", ret);
|
|
return ret;
|
|
}
|
|
|
|
/* In order to configure the DSI parameters, create a mipi_dsi_device
|
|
*/
|
|
ret = sn65dsi85_attach_dsi(self);
|
|
if(ret) {
|
|
DRM_ERROR("Could not sn65dsi85_attach_dsi: %d", ret);
|
|
return ret;
|
|
}
|
|
|
|
|
|
if (self->panel)
|
|
drm_panel_attach(self->panel, &self->connector);
|
|
|
|
/* but no hotplug IRQ. Other drivers do that here. */
|
|
|
|
|
|
return ret;
|
|
}
|
|
|
|
static inline void write_dsi85(struct i2c_client *i2c, u8 reg, u8 value)
|
|
{
|
|
int res;
|
|
res = i2c_smbus_write_byte_data(i2c, reg, value);
|
|
DRM_DEV_DEBUG(&i2c->dev, "write reg 0x%X <- 0x%X (res=%d)", reg, value, res);
|
|
if(res < 0) {
|
|
DRM_DEV_ERROR(&i2c->dev, "could not write DSI85 register 0x%x, i2c error %d", reg, res);
|
|
}
|
|
}
|
|
|
|
static inline u8 compute_dsi_clock_range_code_from_mode(struct drm_display_mode *mode,
|
|
int bytes_per_pixel)
|
|
{
|
|
/*
|
|
* DSI85 wants to know the approximate DSI clock in 5MHz increments.
|
|
*/
|
|
|
|
int total_pixels_per_sec = mode->htotal * mode->vtotal * mode->vrefresh;
|
|
int dsi_clk = total_pixels_per_sec * bytes_per_pixel;
|
|
int dsi_clk_div5MHz = dsi_clk / 5000000;
|
|
|
|
if(8 <= dsi_clk_div5MHz && dsi_clk_div5MHz <= 0x64) {
|
|
return (u8) dsi_clk_div5MHz;
|
|
}
|
|
else {
|
|
DRM_ERROR("DSI clock out of range: %d pixels/s give clock %d, range %d out of range\n",
|
|
total_pixels_per_sec, dsi_clk, dsi_clk_div5MHz);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Translate config and mode attributes to a complete DSI85 configuration,
|
|
* write them to registers.
|
|
*/
|
|
static void sn65dsi85_apply_mode(struct sn65dsi85_device *self,
|
|
struct drm_display_mode *mode)
|
|
{
|
|
struct i2c_client *i2c = self->i2c;
|
|
int lvds_clock;
|
|
bool lvds_clock_is_half_dsi_clock = (self->config->lvds_channels == 2);
|
|
struct sn65dsi85_config* dt = self->config;
|
|
|
|
struct sn65dsi85_regs regs = {
|
|
.pll_en_stat = 0,
|
|
.lvds_clk_range = 0,
|
|
.hs_clk_src = 1, /* LVDS pixel clock derived from MIPI D-PHY channel A HS continuous clock */
|
|
.dsi_clk_divider = lvds_clock_is_half_dsi_clock? 5: 2, /* divide by 6 or 3 for 24bpp */
|
|
|
|
.refclk_multiplier = 0,
|
|
|
|
.left_right_pixels = 0,
|
|
.dsi_channel_mode = 1, /* single channel DSI receiver */
|
|
.cha_dsi_lanes = 0, /* four lanes A */
|
|
.chb_dsi_lanes = 0x3, /* one lane B (default) */
|
|
.sot_err_tol_dis = 0, /* tolerate single bit errors for SoT leader sequence */
|
|
|
|
.cha_dsi_data_eq = 0,
|
|
.chb_dsi_data_eq = 0,
|
|
|
|
.cha_dsi_clk_eq = 0,
|
|
.chb_dsi_clk_eq = 0,
|
|
.cha_dsi_clk_rng = compute_dsi_clock_range_code_from_mode(mode, 3),
|
|
.chb_dsi_clk_rng = 0,
|
|
|
|
.de_neg_polarity = !!dt->de_neg_polarity,
|
|
.hs_neg_polarity = 0, /* from Panel, see below */
|
|
.vs_neg_polarity = 0,
|
|
.lvds_link_cfg = (dt->lvds_channels==2)?0:1,
|
|
.cha_24bpp_mode = !!dt->force_24bpp_mode,
|
|
.chb_24bpp_mode = !!dt->force_24bpp_mode,
|
|
.cha_24bpp_format1 = !!dt->force_24bpp_format1,
|
|
.chb_24bpp_format1 = !!dt->force_24bpp_format1,
|
|
|
|
.cha_lvds_vocm = !!(dt->lvds_vocm & 0x02),
|
|
.chb_lvds_vocm = !!(dt->lvds_vocm & 0x01),
|
|
.cha_lvds_vod_swing = dt->lvds_vod_swing_cha,
|
|
.chb_lvds_vod_swing = dt->lvds_vod_swing_chb,
|
|
|
|
.even_odd_swap = !!(dt->lvds_even_odd_swap),
|
|
.cha_reverse_lvds = !!(dt->lvds_reverse & 0x02),
|
|
.chb_reverse_lvds = !!(dt->lvds_reverse & 0x01),
|
|
.cha_lvds_term = !!(dt->lvds_term & 0x02),
|
|
.chb_lvds_term = !!(dt->lvds_term & 0x01),
|
|
|
|
.cha_lvds_cm_adjust = dt->lvds_cm_adjust_cha,
|
|
.chb_lvds_cm_adjust = dt->lvds_cm_adjust_chb,
|
|
|
|
.cha_active_line_length = mode->hdisplay,
|
|
.chb_active_line_length = 0,
|
|
|
|
.cha_sync_delay = (u16)dt->sync_delay,
|
|
.chb_sync_delay = 0,
|
|
|
|
.cha_hsync_pulse_width = mode->hsync_end - mode->hsync_start,
|
|
.chb_hsync_pulse_width = 0,
|
|
|
|
.cha_vsync_pulse_width = mode->vsync_end - mode->vsync_start,
|
|
.chb_vsync_pulse_width = 0,
|
|
|
|
.cha_horizontal_back_porch = mode->hsync_start - mode->hdisplay,
|
|
.chb_horizontal_back_porch = 0,
|
|
};
|
|
|
|
/* ------- Decide about complicated register values... ------- */
|
|
|
|
/* LVDS panel may need half of everything horizontal if using two LVDS channels */
|
|
if (lvds_clock_is_half_dsi_clock) {
|
|
lvds_clock = mode->clock / 2;
|
|
regs.cha_hsync_pulse_width /= 2;
|
|
regs.cha_horizontal_back_porch /= 2;
|
|
} else {
|
|
lvds_clock = mode->clock;
|
|
}
|
|
|
|
/* Compute LVDS clock-range register; ranges from datasheet */
|
|
if (lvds_clock <= 37500) {
|
|
/* use 0 */
|
|
}
|
|
else if (lvds_clock <= 62500) {
|
|
regs.lvds_clk_range = 0x01;
|
|
}
|
|
else if (lvds_clock <= 87500){
|
|
regs.lvds_clk_range = 0x02;
|
|
}
|
|
else if (lvds_clock <= 112500) {
|
|
regs.lvds_clk_range = 0x03;
|
|
}
|
|
else if (lvds_clock <= 137500) {
|
|
regs.lvds_clk_range = 0x04;
|
|
}
|
|
else {
|
|
regs.lvds_clk_range = 0x05;
|
|
}
|
|
|
|
/* Sync polarities */
|
|
if( (mode->flags & DRM_MODE_FLAG_NHSYNC) != 0 ) {
|
|
regs.hs_neg_polarity = 1;
|
|
}
|
|
if( (mode->flags & DRM_MODE_FLAG_NVSYNC) != 0 ) {
|
|
regs.vs_neg_polarity = 1;
|
|
}
|
|
|
|
/* Soft reset and disable PLL */
|
|
write_dsi85(i2c, DSI85_SOFT_RESET, 1 );
|
|
write_dsi85(i2c, DSI85_PLL_EN, 0 );
|
|
|
|
/* ------- CORE_PLL register ------- */
|
|
write_dsi85(i2c, DSI85_CORE_PLL,
|
|
((regs.pll_en_stat & 0x01) << 7)
|
|
| ((regs.lvds_clk_range & 0x07) << 1)
|
|
| ((regs.hs_clk_src & 0x01) << 0 ));
|
|
|
|
/* ------- PLL_DIV register ------- */
|
|
write_dsi85(i2c, DSI85_PLL_DIV,
|
|
(regs.dsi_clk_divider << 3)
|
|
| (regs.refclk_multiplier & 0x3) );
|
|
|
|
/* ------- DSI ------- */
|
|
write_dsi85(i2c, DSI85_DSI_CFG,
|
|
((regs.left_right_pixels & 0x01) << 7)
|
|
| ((regs.dsi_channel_mode & 0x03) << 5)
|
|
| ((regs.cha_dsi_lanes & 0x03) << 3)
|
|
| ((regs.chb_dsi_lanes & 0x03) << 1)
|
|
| ((regs.sot_err_tol_dis) << 0));
|
|
write_dsi85(i2c, DSI85_DSI_EQ,
|
|
((regs.cha_dsi_data_eq & 0x03) << 6)
|
|
| ((regs.chb_dsi_data_eq & 0x03) << 4)
|
|
| ((regs.cha_dsi_clk_eq & 0x03) << 2)
|
|
| ((regs.chb_dsi_clk_eq & 0x03) << 0));
|
|
write_dsi85(i2c, DSI85_CHA_DSI_CLK_RNG,
|
|
regs.cha_dsi_clk_rng);
|
|
write_dsi85(i2c, DSI85_CHB_DSI_CLK_RNG,
|
|
regs.chb_dsi_clk_rng);
|
|
|
|
/* ------- LVDS ------- */
|
|
|
|
write_dsi85(i2c, DSI85_LVDS_MODE,
|
|
((regs.de_neg_polarity & 0x01) << 7)
|
|
| ((regs.hs_neg_polarity & 0x01) << 6)
|
|
| ((regs.vs_neg_polarity & 0x01) << 5)
|
|
| ((regs.lvds_link_cfg & 0x01) << 4)
|
|
| ((regs.cha_24bpp_mode & 0x01) << 3)
|
|
| ((regs.chb_24bpp_mode & 0x01) << 2)
|
|
| ((regs.cha_24bpp_format1 & 0x01) << 1)
|
|
| ((regs.chb_24bpp_format1 & 0x01) << 0));
|
|
write_dsi85(i2c, DSI85_LVDS_SIGN,
|
|
((regs.cha_lvds_vocm & 0x01) << 6)
|
|
| ((regs.chb_lvds_vocm & 0x01) << 4)
|
|
| ((regs.cha_lvds_vod_swing & 0x03) << 2)
|
|
| ((regs.chb_lvds_vod_swing & 0x03) << 0));
|
|
write_dsi85(i2c, DSI85_LVDS_TERM,
|
|
((regs.even_odd_swap & 0x01) << 6)
|
|
| ((regs.cha_reverse_lvds & 0x01) << 5)
|
|
| ((regs.chb_reverse_lvds & 0x01) << 4)
|
|
| ((regs.cha_lvds_term & 0x01) << 1)
|
|
| ((regs.chb_lvds_term & 0x01) << 0));
|
|
write_dsi85(i2c, DSI85_LVDS_ADJUST,
|
|
((regs.cha_lvds_cm_adjust & 0x03) << 4)
|
|
| ((regs.chb_lvds_cm_adjust & 0x03) << 0));
|
|
|
|
/* ------- Dimensions ------- */
|
|
write_dsi85(i2c, DSI85_CHA_LINE_LEN_LO,
|
|
(u8)(regs.cha_active_line_length & 0x00ffu));
|
|
write_dsi85(i2c, DSI85_CHA_LINE_LEN_HI,
|
|
(u8)((regs.cha_active_line_length & 0x0f00u) >> 8));
|
|
write_dsi85(i2c, DSI85_CHB_LINE_LEN_LO,
|
|
(u8)(regs.chb_active_line_length & 0x00ffu));
|
|
write_dsi85(i2c, DSI85_CHB_LINE_LEN_HI,
|
|
(u8)((regs.chb_active_line_length & 0x0f00u) >> 8));
|
|
|
|
write_dsi85(i2c, DSI85_CHA_VERT_LINES_LO, 0 ); /* TPG only */
|
|
write_dsi85(i2c, DSI85_CHA_VERT_LINES_HI, 0 );
|
|
write_dsi85(i2c, DSI85_CHB_VERT_LINES_LO, 0 );
|
|
write_dsi85(i2c, DSI85_CHB_VERT_LINES_HI, 0 );
|
|
|
|
write_dsi85(i2c, DSI85_CHA_SYNC_DELAY_LO,
|
|
(u8)(regs.cha_sync_delay & 0x00ffu));
|
|
write_dsi85(i2c, DSI85_CHA_SYNC_DELAY_HI,
|
|
(u8)((regs.cha_sync_delay & 0x0f00u) >> 8));
|
|
write_dsi85(i2c, DSI85_CHB_SYNC_DELAY_LO,
|
|
(u8)(regs.chb_sync_delay & 0x00ffu));
|
|
write_dsi85(i2c, DSI85_CHB_SYNC_DELAY_HI,
|
|
(u8)((regs.chb_sync_delay & 0x0f00u) >> 8));
|
|
|
|
write_dsi85(i2c, DSI85_CHA_HSYNC_WIDTH_LO,
|
|
(u8)(regs.cha_hsync_pulse_width & 0x00ffu));
|
|
write_dsi85(i2c, DSI85_CHA_HSYNC_WIDTH_HI,
|
|
(u8)((regs.cha_hsync_pulse_width & 0x0f00u) >> 8));
|
|
write_dsi85(i2c, DSI85_CHB_HSYNC_WIDTH_LO,
|
|
(u8)(regs.chb_hsync_pulse_width & 0x00ffu));
|
|
write_dsi85(i2c, DSI85_CHB_HSYNC_WIDTH_HI,
|
|
(u8)((regs.chb_hsync_pulse_width & 0x0f00u) >> 8));
|
|
|
|
write_dsi85(i2c, DSI85_CHA_VSYNC_WIDTH_LO,
|
|
(u8)(regs.cha_vsync_pulse_width & 0x00ffu));
|
|
write_dsi85(i2c, DSI85_CHA_VSYNC_WIDTH_HI,
|
|
(u8)((regs.cha_vsync_pulse_width & 0x0f00u) >> 8));
|
|
write_dsi85(i2c, DSI85_CHB_VSYNC_WIDTH_LO,
|
|
(u8)(regs.chb_vsync_pulse_width & 0x00ffu));
|
|
write_dsi85(i2c, DSI85_CHB_VSYNC_WIDTH_HI,
|
|
(u8)((regs.chb_vsync_pulse_width & 0x0f00u) >> 8));
|
|
|
|
write_dsi85(i2c, DSI85_CHA_HORZ_BACKPORCH, regs.cha_horizontal_back_porch );
|
|
write_dsi85(i2c, DSI85_CHB_HORZ_BACKPORCH, regs.chb_horizontal_back_porch );
|
|
|
|
write_dsi85(i2c, DSI85_CHA_VERT_BACKPORCH, 0 ); /* TPG only */
|
|
write_dsi85(i2c, DSI85_CHB_VERT_BACKPORCH, 0 ); /* TPG only */
|
|
write_dsi85(i2c, DSI85_CHA_HORZ_FRONTPORCH, 0 ); /* TPG only */
|
|
write_dsi85(i2c, DSI85_CHB_HORZ_FRONTPORCH, 0 ); /* TPG only */
|
|
write_dsi85(i2c, DSI85_CHA_HORZ_FRONTPORCH, 0 ); /* TPG only */
|
|
write_dsi85(i2c, DSI85_CHB_VERT_FRONTPORCH, 0 ); /* TPG only */
|
|
|
|
}
|
|
|
|
|
|
static void sn65dsi85_bridge_mode_set(struct drm_bridge *bridge,
|
|
struct drm_display_mode *mode,
|
|
struct drm_display_mode *adjusted_mode)
|
|
{
|
|
struct sn65dsi85_device *self = bridge_to_sn65dsi85(bridge);
|
|
|
|
DRM_DEV_DEBUG(bridge->dev->dev, "%s entry", __func__);
|
|
sn65dsi85_apply_mode(self, mode);
|
|
DRM_DEV_DEBUG(bridge->dev->dev, "%s exit", __func__);
|
|
}
|
|
|
|
static bool sn65dsi85_bridge_mode_fixup(struct drm_bridge *bridge,
|
|
const struct drm_display_mode *mode,
|
|
struct drm_display_mode *adjusted_mode)
|
|
{
|
|
/* The panel may request negative vsync and hsync, but please don't
|
|
* tell the DSI host; we want the DSI host to transmit uniform polarity,
|
|
* and sn65dsi85_apply_mode will tell the DSI85 what polarity to produce. */
|
|
if( mode->flags & (DRM_MODE_FLAG_NHSYNC | DRM_MODE_FLAG_NVSYNC) ) {
|
|
adjusted_mode->flags &= ~(DRM_MODE_FLAG_NHSYNC | DRM_MODE_FLAG_NVSYNC);
|
|
DRM_DEBUG("Clearing FLAG_NHSYNC and FLAG_NVSYNC for DSI");
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static const struct drm_bridge_funcs sn65dsi85_bridge_funcs = {
|
|
.pre_enable = sn65dsi85_bridge_pre_enable,
|
|
.enable = sn65dsi85_bridge_enable,
|
|
.disable = sn65dsi85_bridge_disable,
|
|
.post_disable = sn65dsi85_bridge_post_disable,
|
|
.attach = sn65dsi85_bridge_attach,
|
|
.mode_set = sn65dsi85_bridge_mode_set,
|
|
.mode_fixup = sn65dsi85_bridge_mode_fixup,
|
|
};
|
|
|
|
/*
|
|
* Probe an SN65DSI85 on an I2C bus.
|
|
*
|
|
* This installs a drm_bridge into the DRM system to do the actual mode-setting work.
|
|
*
|
|
* @c i2c_client pointer
|
|
* @id i2c_device_id pointer
|
|
*/
|
|
static int sn65dsi85_probe(struct i2c_client* c, const struct i2c_device_id *id)
|
|
{
|
|
struct sn65dsi85_config *pdata = NULL;
|
|
struct sn65dsi85_device *ctx = NULL;
|
|
struct device_node *endpoint, *panel_node, *host_node;
|
|
int error_value = 0;
|
|
|
|
DRM_DEV_INFO(&c->dev, "sn65dsi85_probe: welcome!\n");
|
|
|
|
pdata = sn65dsi85_get_pdata(c);
|
|
if(!pdata) {
|
|
DRM_DEV_ERROR(&c->dev, "Cannot read configuration. Probe failed.\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
ctx = devm_kzalloc(&c->dev, sizeof(struct sn65dsi85_device), GFP_KERNEL);
|
|
if(!ctx) {
|
|
DRM_DEV_ERROR(&c->dev, "Cannot allocate device state\n");
|
|
error_value = -ENOMEM;
|
|
goto fail_free_pdata;
|
|
}
|
|
ctx->config = pdata;
|
|
ctx->i2c = c;
|
|
ctx->i2c_id = id;
|
|
i2c_set_clientdata(c, ctx);
|
|
|
|
/* Find the associated DSI host. port@0 is input */
|
|
endpoint = of_graph_get_endpoint_by_regs(c->dev.of_node, 0, -1);
|
|
if (endpoint) {
|
|
DRM_DEV_DEBUG(&c->dev, "%s my DSI-side endpoint: %s", __func__, endpoint->full_name);
|
|
host_node = of_graph_get_remote_port_parent(endpoint);
|
|
if (host_node) {
|
|
DRM_DEV_DEBUG(&c->dev, "%s my DSI host node: %s", __func__, host_node->full_name);
|
|
/* Store a pointer to the DT node because the DSI host may not be up yet */
|
|
ctx->dsi_host_node = host_node;
|
|
}
|
|
else {
|
|
DRM_DEV_ERROR(&c->dev, "%s cannot find DSI host node", __func__);
|
|
}
|
|
}
|
|
else {
|
|
DRM_DEV_ERROR(&c->dev, "%s cannot find endpoint 0 (towards DSI)", __func__);
|
|
error_value = -ENOENT;
|
|
goto fail_free_pdata;
|
|
}
|
|
|
|
/* Find the associated panel. port@1 is output */
|
|
endpoint = of_graph_get_endpoint_by_regs(c->dev.of_node, 1, -1);
|
|
if (endpoint) {
|
|
DRM_DEV_DEBUG(&c->dev, "%s my panel-side endpoint: %s", __func__, endpoint->full_name);
|
|
panel_node = of_graph_get_remote_port_parent(endpoint);
|
|
if (panel_node) {
|
|
ctx->panel = of_drm_find_panel(panel_node);
|
|
if (!ctx->panel) {
|
|
DRM_WARN("Cannot find panel by panel node name=%s",
|
|
panel_node->full_name);
|
|
return -EPROBE_DEFER;
|
|
}
|
|
of_node_put(panel_node);
|
|
}
|
|
}
|
|
else {
|
|
DRM_DEV_ERROR(&c->dev, "%s cannot find endpoint 1 (towards panel)", __func__);
|
|
error_value = -ENOENT;
|
|
goto fail_free_pdata;
|
|
}
|
|
|
|
ctx->bridge.funcs = &sn65dsi85_bridge_funcs;
|
|
ctx->bridge.of_node = c->dev.of_node;
|
|
|
|
error_value = drm_bridge_add(&ctx->bridge);
|
|
(void)error_value; // "Unconditionally returns zero"
|
|
|
|
DRM_DEV_INFO(&c->dev, "sn65dsi85_probe: done!\n");
|
|
|
|
return 0;
|
|
|
|
fail_free_pdata:
|
|
kfree(pdata);
|
|
DRM_DEV_INFO(&c->dev, "sn65dsi85_probe: err=%d :(\n", error_value);
|
|
return error_value;
|
|
|
|
}
|
|
|
|
/*
|
|
* Remove SN65DSI85
|
|
* @c: pointer to the i2c_client
|
|
*/
|
|
static int sn65dsi85_remove(struct i2c_client *c)
|
|
{
|
|
struct sn65dsi85_device *ctx = i2c_get_clientdata(c);
|
|
DRM_DEV_DEBUG(&c->dev, "%s entry", __func__);
|
|
|
|
if(ctx->dsi) {
|
|
sn65dsi85_detach_dsi(ctx);
|
|
}
|
|
|
|
drm_bridge_remove(&ctx->bridge);
|
|
|
|
DRM_DEV_DEBUG(&c->dev, "%s exit", __func__);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Device-Tree declaration
|
|
*/
|
|
static const struct of_device_id sn65dsi85_of_match[] = {
|
|
{.compatible = "ti,dsi85" },
|
|
{}
|
|
};
|
|
MODULE_DEVICE_TABLE(of, sn65dsi85_of_match);
|
|
|
|
|
|
/*
|
|
* This is an I2C client driver module instead of a general module
|
|
*/
|
|
static const struct i2c_device_id sn65dsi85_i2c_id[] = {
|
|
{ "ti,dsi85", 0 },
|
|
{ }
|
|
};
|
|
MODULE_DEVICE_TABLE(i2c, sn65dsi85_i2c_id);
|
|
|
|
static struct i2c_driver sn65dsi85_i2c_driver = {
|
|
.driver = {
|
|
.of_match_table = of_match_ptr(sn65dsi85_of_match),
|
|
.owner = THIS_MODULE,
|
|
.name = SN65DSI85_MODULE_NAME,
|
|
},
|
|
.probe = sn65dsi85_probe,
|
|
.remove = sn65dsi85_remove,
|
|
.id_table = sn65dsi85_i2c_id,
|
|
};
|
|
|
|
module_i2c_driver(sn65dsi85_i2c_driver);
|
|
|
|
|
|
MODULE_AUTHOR("Konrad Anton <konrad.anton@awinia.de>");
|
|
MODULE_DESCRIPTION("DRM Driver for SN65DSI85 MIPI-DSI-to-LVDS converter");
|
|
MODULE_LICENSE("GPL");
|
|
|
|
/* EOF */
|