123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * Chrontel CH7033 Video Encoder Driver
- *
- * Copyright (C) 2019,2020 Lubomir Rintel
- */
- #include <linux/gpio/consumer.h>
- #include <linux/i2c.h>
- #include <linux/module.h>
- #include <linux/regmap.h>
- #include <drm/drm_atomic_helper.h>
- #include <drm/drm_bridge.h>
- #include <drm/drm_edid.h>
- #include <drm/drm_of.h>
- #include <drm/drm_print.h>
- #include <drm/drm_probe_helper.h>
- /* Page 0, Register 0x07 */
- enum {
- DRI_PD = BIT(3),
- IO_PD = BIT(5),
- };
- /* Page 0, Register 0x08 */
- enum {
- DRI_PDDRI = GENMASK(7, 4),
- PDDAC = GENMASK(3, 1),
- PANEN = BIT(0),
- };
- /* Page 0, Register 0x09 */
- enum {
- DPD = BIT(7),
- GCKOFF = BIT(6),
- TV_BP = BIT(5),
- SCLPD = BIT(4),
- SDPD = BIT(3),
- VGA_PD = BIT(2),
- HDBKPD = BIT(1),
- HDMI_PD = BIT(0),
- };
- /* Page 0, Register 0x0a */
- enum {
- MEMINIT = BIT(7),
- MEMIDLE = BIT(6),
- MEMPD = BIT(5),
- STOP = BIT(4),
- LVDS_PD = BIT(3),
- HD_DVIB = BIT(2),
- HDCP_PD = BIT(1),
- MCU_PD = BIT(0),
- };
- /* Page 0, Register 0x18 */
- enum {
- IDF = GENMASK(7, 4),
- INTEN = BIT(3),
- SWAP = GENMASK(2, 0),
- };
- enum {
- BYTE_SWAP_RGB = 0,
- BYTE_SWAP_RBG = 1,
- BYTE_SWAP_GRB = 2,
- BYTE_SWAP_GBR = 3,
- BYTE_SWAP_BRG = 4,
- BYTE_SWAP_BGR = 5,
- };
- /* Page 0, Register 0x19 */
- enum {
- HPO_I = BIT(5),
- VPO_I = BIT(4),
- DEPO_I = BIT(3),
- CRYS_EN = BIT(2),
- GCLKFREQ = GENMASK(2, 0),
- };
- /* Page 0, Register 0x2e */
- enum {
- HFLIP = BIT(7),
- VFLIP = BIT(6),
- DEPO_O = BIT(5),
- HPO_O = BIT(4),
- VPO_O = BIT(3),
- TE = GENMASK(2, 0),
- };
- /* Page 0, Register 0x2b */
- enum {
- SWAPS = GENMASK(7, 4),
- VFMT = GENMASK(3, 0),
- };
- /* Page 0, Register 0x54 */
- enum {
- COMP_BP = BIT(7),
- DAC_EN_T = BIT(6),
- HWO_HDMI_HI = GENMASK(5, 3),
- HOO_HDMI_HI = GENMASK(2, 0),
- };
- /* Page 0, Register 0x57 */
- enum {
- FLDSEN = BIT(7),
- VWO_HDMI_HI = GENMASK(5, 3),
- VOO_HDMI_HI = GENMASK(2, 0),
- };
- /* Page 0, Register 0x7e */
- enum {
- HDMI_LVDS_SEL = BIT(7),
- DE_GEN = BIT(6),
- PWM_INDEX_HI = BIT(5),
- USE_DE = BIT(4),
- R_INT = GENMASK(3, 0),
- };
- /* Page 1, Register 0x07 */
- enum {
- BPCKSEL = BIT(7),
- DRI_CMFB_EN = BIT(6),
- CEC_PUEN = BIT(5),
- CEC_T = BIT(3),
- CKINV = BIT(2),
- CK_TVINV = BIT(1),
- DRI_CKS2 = BIT(0),
- };
- /* Page 1, Register 0x08 */
- enum {
- DACG = BIT(6),
- DACKTST = BIT(5),
- DEDGEB = BIT(4),
- SYO = BIT(3),
- DRI_IT_LVDS = GENMASK(2, 1),
- DISPON = BIT(0),
- };
- /* Page 1, Register 0x0c */
- enum {
- DRI_PLL_CP = GENMASK(7, 6),
- DRI_PLL_DIVSEL = BIT(5),
- DRI_PLL_N1_1 = BIT(4),
- DRI_PLL_N1_0 = BIT(3),
- DRI_PLL_N3_1 = BIT(2),
- DRI_PLL_N3_0 = BIT(1),
- DRI_PLL_CKTSTEN = BIT(0),
- };
- /* Page 1, Register 0x6b */
- enum {
- VCO3CS = GENMASK(7, 6),
- ICPGBK2_0 = GENMASK(5, 3),
- DRI_VCO357SC = BIT(2),
- PDPLL2 = BIT(1),
- DRI_PD_SER = BIT(0),
- };
- /* Page 1, Register 0x6c */
- enum {
- PLL2N11 = GENMASK(7, 4),
- PLL2N5_4 = BIT(3),
- PLL2N5_TOP = BIT(2),
- DRI_PLL_PD = BIT(1),
- PD_I2CM = BIT(0),
- };
- /* Page 3, Register 0x28 */
- enum {
- DIFF_EN = GENMASK(7, 6),
- CORREC_EN = GENMASK(5, 4),
- VGACLK_BP = BIT(3),
- HM_LV_SEL = BIT(2),
- HD_VGA_SEL = BIT(1),
- };
- /* Page 3, Register 0x2a */
- enum {
- LVDSCLK_BP = BIT(7),
- HDTVCLK_BP = BIT(6),
- HDMICLK_BP = BIT(5),
- HDTV_BP = BIT(4),
- HDMI_BP = BIT(3),
- THRWL = GENMASK(2, 0),
- };
- /* Page 4, Register 0x52 */
- enum {
- PGM_ARSTB = BIT(7),
- MCU_ARSTB = BIT(6),
- MCU_RETB = BIT(2),
- RESETIB = BIT(1),
- RESETDB = BIT(0),
- };
- struct ch7033_priv {
- struct regmap *regmap;
- struct drm_bridge *next_bridge;
- struct drm_bridge bridge;
- struct drm_connector connector;
- };
- #define conn_to_ch7033_priv(x) \
- container_of(x, struct ch7033_priv, connector)
- #define bridge_to_ch7033_priv(x) \
- container_of(x, struct ch7033_priv, bridge)
- static enum drm_connector_status ch7033_connector_detect(
- struct drm_connector *connector, bool force)
- {
- struct ch7033_priv *priv = conn_to_ch7033_priv(connector);
- return drm_bridge_detect(priv->next_bridge);
- }
- static const struct drm_connector_funcs ch7033_connector_funcs = {
- .reset = drm_atomic_helper_connector_reset,
- .fill_modes = drm_helper_probe_single_connector_modes,
- .detect = ch7033_connector_detect,
- .destroy = drm_connector_cleanup,
- .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state,
- .atomic_destroy_state = drm_atomic_helper_connector_destroy_state,
- };
- static int ch7033_connector_get_modes(struct drm_connector *connector)
- {
- struct ch7033_priv *priv = conn_to_ch7033_priv(connector);
- struct edid *edid;
- int ret;
- edid = drm_bridge_get_edid(priv->next_bridge, connector);
- drm_connector_update_edid_property(connector, edid);
- if (edid) {
- ret = drm_add_edid_modes(connector, edid);
- kfree(edid);
- } else {
- ret = drm_add_modes_noedid(connector, 1920, 1080);
- drm_set_preferred_mode(connector, 1024, 768);
- }
- return ret;
- }
- static struct drm_encoder *ch7033_connector_best_encoder(
- struct drm_connector *connector)
- {
- struct ch7033_priv *priv = conn_to_ch7033_priv(connector);
- return priv->bridge.encoder;
- }
- static const struct drm_connector_helper_funcs ch7033_connector_helper_funcs = {
- .get_modes = ch7033_connector_get_modes,
- .best_encoder = ch7033_connector_best_encoder,
- };
- static void ch7033_hpd_event(void *arg, enum drm_connector_status status)
- {
- struct ch7033_priv *priv = arg;
- if (priv->bridge.dev)
- drm_helper_hpd_irq_event(priv->connector.dev);
- }
- static int ch7033_bridge_attach(struct drm_bridge *bridge,
- enum drm_bridge_attach_flags flags)
- {
- struct ch7033_priv *priv = bridge_to_ch7033_priv(bridge);
- struct drm_connector *connector = &priv->connector;
- int ret;
- ret = drm_bridge_attach(bridge->encoder, priv->next_bridge, bridge,
- DRM_BRIDGE_ATTACH_NO_CONNECTOR);
- if (ret)
- return ret;
- if (flags & DRM_BRIDGE_ATTACH_NO_CONNECTOR)
- return 0;
- if (priv->next_bridge->ops & DRM_BRIDGE_OP_DETECT) {
- connector->polled = DRM_CONNECTOR_POLL_HPD;
- } else {
- connector->polled = DRM_CONNECTOR_POLL_CONNECT |
- DRM_CONNECTOR_POLL_DISCONNECT;
- }
- if (priv->next_bridge->ops & DRM_BRIDGE_OP_HPD) {
- drm_bridge_hpd_enable(priv->next_bridge, ch7033_hpd_event,
- priv);
- }
- drm_connector_helper_add(connector,
- &ch7033_connector_helper_funcs);
- ret = drm_connector_init_with_ddc(bridge->dev, &priv->connector,
- &ch7033_connector_funcs,
- priv->next_bridge->type,
- priv->next_bridge->ddc);
- if (ret) {
- DRM_ERROR("Failed to initialize connector\n");
- return ret;
- }
- return drm_connector_attach_encoder(&priv->connector, bridge->encoder);
- }
- static void ch7033_bridge_detach(struct drm_bridge *bridge)
- {
- struct ch7033_priv *priv = bridge_to_ch7033_priv(bridge);
- if (priv->next_bridge->ops & DRM_BRIDGE_OP_HPD)
- drm_bridge_hpd_disable(priv->next_bridge);
- drm_connector_cleanup(&priv->connector);
- }
- static enum drm_mode_status ch7033_bridge_mode_valid(struct drm_bridge *bridge,
- const struct drm_display_info *info,
- const struct drm_display_mode *mode)
- {
- if (mode->clock > 165000)
- return MODE_CLOCK_HIGH;
- if (mode->hdisplay >= 1920)
- return MODE_BAD_HVALUE;
- if (mode->vdisplay >= 1080)
- return MODE_BAD_VVALUE;
- return MODE_OK;
- }
- static void ch7033_bridge_disable(struct drm_bridge *bridge)
- {
- struct ch7033_priv *priv = bridge_to_ch7033_priv(bridge);
- regmap_write(priv->regmap, 0x03, 0x04);
- regmap_update_bits(priv->regmap, 0x52, RESETDB, 0x00);
- }
- static void ch7033_bridge_enable(struct drm_bridge *bridge)
- {
- struct ch7033_priv *priv = bridge_to_ch7033_priv(bridge);
- regmap_write(priv->regmap, 0x03, 0x04);
- regmap_update_bits(priv->regmap, 0x52, RESETDB, RESETDB);
- }
- static void ch7033_bridge_mode_set(struct drm_bridge *bridge,
- const struct drm_display_mode *mode,
- const struct drm_display_mode *adjusted_mode)
- {
- struct ch7033_priv *priv = bridge_to_ch7033_priv(bridge);
- int hbporch = mode->hsync_start - mode->hdisplay;
- int hsynclen = mode->hsync_end - mode->hsync_start;
- int vbporch = mode->vsync_start - mode->vdisplay;
- int vsynclen = mode->vsync_end - mode->vsync_start;
- /*
- * Page 4
- */
- regmap_write(priv->regmap, 0x03, 0x04);
- /* Turn everything off to set all the registers to their defaults. */
- regmap_write(priv->regmap, 0x52, 0x00);
- /* Bring I/O block up. */
- regmap_write(priv->regmap, 0x52, RESETIB);
- /*
- * Page 0
- */
- regmap_write(priv->regmap, 0x03, 0x00);
- /* Bring up parts we need from the power down. */
- regmap_update_bits(priv->regmap, 0x07, DRI_PD | IO_PD, 0);
- regmap_update_bits(priv->regmap, 0x08, DRI_PDDRI | PDDAC | PANEN, 0);
- regmap_update_bits(priv->regmap, 0x09, DPD | GCKOFF |
- HDMI_PD | VGA_PD, 0);
- regmap_update_bits(priv->regmap, 0x0a, HD_DVIB, 0);
- /* Horizontal input timing. */
- regmap_write(priv->regmap, 0x0b, (mode->htotal >> 8) << 3 |
- (mode->hdisplay >> 8));
- regmap_write(priv->regmap, 0x0c, mode->hdisplay);
- regmap_write(priv->regmap, 0x0d, mode->htotal);
- regmap_write(priv->regmap, 0x0e, (hsynclen >> 8) << 3 |
- (hbporch >> 8));
- regmap_write(priv->regmap, 0x0f, hbporch);
- regmap_write(priv->regmap, 0x10, hsynclen);
- /* Vertical input timing. */
- regmap_write(priv->regmap, 0x11, (mode->vtotal >> 8) << 3 |
- (mode->vdisplay >> 8));
- regmap_write(priv->regmap, 0x12, mode->vdisplay);
- regmap_write(priv->regmap, 0x13, mode->vtotal);
- regmap_write(priv->regmap, 0x14, ((vsynclen >> 8) << 3) |
- (vbporch >> 8));
- regmap_write(priv->regmap, 0x15, vbporch);
- regmap_write(priv->regmap, 0x16, vsynclen);
- /* Input color swap. */
- regmap_update_bits(priv->regmap, 0x18, SWAP, BYTE_SWAP_BGR);
- /* Input clock and sync polarity. */
- regmap_update_bits(priv->regmap, 0x19, 0x1, mode->clock >> 16);
- regmap_update_bits(priv->regmap, 0x19, HPO_I | VPO_I | GCLKFREQ,
- (mode->flags & DRM_MODE_FLAG_PHSYNC) ? HPO_I : 0 |
- (mode->flags & DRM_MODE_FLAG_PVSYNC) ? VPO_I : 0 |
- mode->clock >> 16);
- regmap_write(priv->regmap, 0x1a, mode->clock >> 8);
- regmap_write(priv->regmap, 0x1b, mode->clock);
- /* Horizontal output timing. */
- regmap_write(priv->regmap, 0x1f, (mode->htotal >> 8) << 3 |
- (mode->hdisplay >> 8));
- regmap_write(priv->regmap, 0x20, mode->hdisplay);
- regmap_write(priv->regmap, 0x21, mode->htotal);
- /* Vertical output timing. */
- regmap_write(priv->regmap, 0x25, (mode->vtotal >> 8) << 3 |
- (mode->vdisplay >> 8));
- regmap_write(priv->regmap, 0x26, mode->vdisplay);
- regmap_write(priv->regmap, 0x27, mode->vtotal);
- /* VGA channel bypass */
- regmap_update_bits(priv->regmap, 0x2b, VFMT, 9);
- /* Output sync polarity. */
- regmap_update_bits(priv->regmap, 0x2e, HPO_O | VPO_O,
- (mode->flags & DRM_MODE_FLAG_PHSYNC) ? HPO_O : 0 |
- (mode->flags & DRM_MODE_FLAG_PVSYNC) ? VPO_O : 0);
- /* HDMI horizontal output timing. */
- regmap_update_bits(priv->regmap, 0x54, HWO_HDMI_HI | HOO_HDMI_HI,
- (hsynclen >> 8) << 3 |
- (hbporch >> 8));
- regmap_write(priv->regmap, 0x55, hbporch);
- regmap_write(priv->regmap, 0x56, hsynclen);
- /* HDMI vertical output timing. */
- regmap_update_bits(priv->regmap, 0x57, VWO_HDMI_HI | VOO_HDMI_HI,
- (vsynclen >> 8) << 3 |
- (vbporch >> 8));
- regmap_write(priv->regmap, 0x58, vbporch);
- regmap_write(priv->regmap, 0x59, vsynclen);
- /* Pick HDMI, not LVDS. */
- regmap_update_bits(priv->regmap, 0x7e, HDMI_LVDS_SEL, HDMI_LVDS_SEL);
- /*
- * Page 1
- */
- regmap_write(priv->regmap, 0x03, 0x01);
- /* No idea what these do, but VGA is wobbly and blinky without them. */
- regmap_update_bits(priv->regmap, 0x07, CKINV, CKINV);
- regmap_update_bits(priv->regmap, 0x08, DISPON, DISPON);
- /* DRI PLL */
- regmap_update_bits(priv->regmap, 0x0c, DRI_PLL_DIVSEL, DRI_PLL_DIVSEL);
- if (mode->clock <= 40000) {
- regmap_update_bits(priv->regmap, 0x0c, DRI_PLL_N1_1 |
- DRI_PLL_N1_0 |
- DRI_PLL_N3_1 |
- DRI_PLL_N3_0,
- 0);
- } else if (mode->clock < 80000) {
- regmap_update_bits(priv->regmap, 0x0c, DRI_PLL_N1_1 |
- DRI_PLL_N1_0 |
- DRI_PLL_N3_1 |
- DRI_PLL_N3_0,
- DRI_PLL_N3_0 |
- DRI_PLL_N1_0);
- } else {
- regmap_update_bits(priv->regmap, 0x0c, DRI_PLL_N1_1 |
- DRI_PLL_N1_0 |
- DRI_PLL_N3_1 |
- DRI_PLL_N3_0,
- DRI_PLL_N3_1 |
- DRI_PLL_N1_1);
- }
- /* This seems to be color calibration for VGA. */
- regmap_write(priv->regmap, 0x64, 0x29); /* LSB Blue */
- regmap_write(priv->regmap, 0x65, 0x29); /* LSB Green */
- regmap_write(priv->regmap, 0x66, 0x29); /* LSB Red */
- regmap_write(priv->regmap, 0x67, 0x00); /* MSB Blue */
- regmap_write(priv->regmap, 0x68, 0x00); /* MSB Green */
- regmap_write(priv->regmap, 0x69, 0x00); /* MSB Red */
- regmap_update_bits(priv->regmap, 0x6b, DRI_PD_SER, 0x00);
- regmap_update_bits(priv->regmap, 0x6c, DRI_PLL_PD, 0x00);
- /*
- * Page 3
- */
- regmap_write(priv->regmap, 0x03, 0x03);
- /* More bypasses and apparently another HDMI/LVDS selector. */
- regmap_update_bits(priv->regmap, 0x28, VGACLK_BP | HM_LV_SEL,
- VGACLK_BP | HM_LV_SEL);
- regmap_update_bits(priv->regmap, 0x2a, HDMICLK_BP | HDMI_BP,
- HDMICLK_BP | HDMI_BP);
- /*
- * Page 4
- */
- regmap_write(priv->regmap, 0x03, 0x04);
- /* Output clock. */
- regmap_write(priv->regmap, 0x10, mode->clock >> 16);
- regmap_write(priv->regmap, 0x11, mode->clock >> 8);
- regmap_write(priv->regmap, 0x12, mode->clock);
- }
- static const struct drm_bridge_funcs ch7033_bridge_funcs = {
- .attach = ch7033_bridge_attach,
- .detach = ch7033_bridge_detach,
- .mode_valid = ch7033_bridge_mode_valid,
- .disable = ch7033_bridge_disable,
- .enable = ch7033_bridge_enable,
- .mode_set = ch7033_bridge_mode_set,
- };
- static const struct regmap_config ch7033_regmap_config = {
- .reg_bits = 8,
- .val_bits = 8,
- .max_register = 0x7f,
- };
- static int ch7033_probe(struct i2c_client *client,
- const struct i2c_device_id *id)
- {
- struct device *dev = &client->dev;
- struct ch7033_priv *priv;
- unsigned int val;
- int ret;
- priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
- if (!priv)
- return -ENOMEM;
- dev_set_drvdata(dev, priv);
- ret = drm_of_find_panel_or_bridge(dev->of_node, 1, -1, NULL,
- &priv->next_bridge);
- if (ret)
- return ret;
- priv->regmap = devm_regmap_init_i2c(client, &ch7033_regmap_config);
- if (IS_ERR(priv->regmap)) {
- dev_err(&client->dev, "regmap init failed\n");
- return PTR_ERR(priv->regmap);
- }
- ret = regmap_read(priv->regmap, 0x00, &val);
- if (ret < 0) {
- dev_err(&client->dev, "error reading the model id: %d\n", ret);
- return ret;
- }
- if ((val & 0xf7) != 0x56) {
- dev_err(&client->dev, "the device is not a ch7033\n");
- return -ENODEV;
- }
- regmap_write(priv->regmap, 0x03, 0x04);
- ret = regmap_read(priv->regmap, 0x51, &val);
- if (ret < 0) {
- dev_err(&client->dev, "error reading the model id: %d\n", ret);
- return ret;
- }
- if ((val & 0x0f) != 3) {
- dev_err(&client->dev, "unknown revision %u\n", val);
- return -ENODEV;
- }
- INIT_LIST_HEAD(&priv->bridge.list);
- priv->bridge.funcs = &ch7033_bridge_funcs;
- priv->bridge.of_node = dev->of_node;
- drm_bridge_add(&priv->bridge);
- dev_info(dev, "Chrontel CH7033 Video Encoder\n");
- return 0;
- }
- static void ch7033_remove(struct i2c_client *client)
- {
- struct device *dev = &client->dev;
- struct ch7033_priv *priv = dev_get_drvdata(dev);
- drm_bridge_remove(&priv->bridge);
- }
- static const struct of_device_id ch7033_dt_ids[] = {
- { .compatible = "chrontel,ch7033", },
- { }
- };
- MODULE_DEVICE_TABLE(of, ch7033_dt_ids);
- static const struct i2c_device_id ch7033_ids[] = {
- { "ch7033", 0 },
- { }
- };
- MODULE_DEVICE_TABLE(i2c, ch7033_ids);
- static struct i2c_driver ch7033_driver = {
- .probe = ch7033_probe,
- .remove = ch7033_remove,
- .driver = {
- .name = "ch7033",
- .of_match_table = of_match_ptr(ch7033_dt_ids),
- },
- .id_table = ch7033_ids,
- };
- module_i2c_driver(ch7033_driver);
- MODULE_AUTHOR("Lubomir Rintel <[email protected]>");
- MODULE_DESCRIPTION("Chrontel CH7033 Video Encoder Driver");
- MODULE_LICENSE("GPL v2");
|