aboutsummaryrefslogtreecommitdiff
path: root/src/drivers/i2c/ptn3460/ptn3460.c
diff options
context:
space:
mode:
authorUwe Poeche <uwe.poeche@siemens.com>2019-10-28 11:28:50 +0100
committerPatrick Georgi <pgeorgi@google.com>2019-11-12 18:24:19 +0000
commit11a34ec4c230ab9a6a825eae57eb848aa052a6ed (patch)
tree9f6c0d3ac9002115836aab6c2b96df83d6ff428d /src/drivers/i2c/ptn3460/ptn3460.c
parentc484da1a98610d783131a3a3998c0a999b97f9f5 (diff)
drivers/i2c/ptn3460: Provide chip driver for PTN3460
This patch provides a chip driver for the DP-2-LVDS bridge PTN3460. The bridge is configured via I2C. As the mainboard has all the information regarding the attached LCD type, there are three hooks into mainboard code to get the information like EDID data and PTN config. TEST=Display is working on Siemens mainboards (e.g. mc_tcu3, mc_apl1, ...). Change-Id: Ie4c8176cd16836fa5b8fd2f72faf7a55723b82f6 Signed-off-by: Uwe Poeche <uwe.poeche@siemens.com> Reviewed-on: https://review.coreboot.org/c/coreboot/+/36642 Tested-by: build bot (Jenkins) <no-reply@coreboot.org> Reviewed-by: Werner Zeh <werner.zeh@siemens.com>
Diffstat (limited to 'src/drivers/i2c/ptn3460/ptn3460.c')
-rw-r--r--src/drivers/i2c/ptn3460/ptn3460.c157
1 files changed, 157 insertions, 0 deletions
diff --git a/src/drivers/i2c/ptn3460/ptn3460.c b/src/drivers/i2c/ptn3460/ptn3460.c
new file mode 100644
index 0000000000..ef25745ed1
--- /dev/null
+++ b/src/drivers/i2c/ptn3460/ptn3460.c
@@ -0,0 +1,157 @@
+/*
+ * This file is part of the coreboot project.
+ *
+ * Copyright (C) 2019 Siemens AG
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; version 2 of the License.
+ *
+ * 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.
+ */
+
+
+#include <console/console.h>
+#include <device/i2c_bus.h>
+#include <types.h>
+
+#include "ptn3460.h"
+
+/**
+ * \brief This function selects one of 7 EDID-tables inside PTN3460
+ * which should be emulated on display port and turn emulation ON
+ * @param *dev Pointer to the relevant I2C controller
+ * @param edid_num Number of EDID to emulate (0..6)
+ * @return PTN_SUCCESS or error code
+ */
+static int ptn_select_edid(struct device *dev, uint8_t edid_num)
+{
+ int status = 0;
+ u8 val;
+
+ if (edid_num > PTN_MAX_EDID_NUM)
+ return PTN_INVALID_EDID;
+ val = (edid_num << 1) | PTN_ENABLE_EMULATION;
+ status = i2c_dev_writeb_at(dev, PTN_CONFIG_OFF + 4, val);
+ return status ? (PTN_BUS_ERROR | status) : PTN_SUCCESS;
+}
+
+/**
+ * \brief This function writes one EDID data structure to PTN3460
+ * @param *dev Pointer to the relevant I2C controller
+ * @param edid_num Number of EDID that must be written (0..6)
+ * @param *data Pointer to a buffer where data to write is stored in
+ * @return PTN_SUCCESS on success or error code
+ */
+static int ptn3460_write_edid(struct device *dev, u8 edid_num, u8 *data)
+{
+ int status;
+ int i;
+
+ if (edid_num > PTN_MAX_EDID_NUM)
+ return PTN_INVALID_EDID;
+
+ /* First enable access to the desired EDID table */
+ status = i2c_dev_writeb_at(dev, PTN_CONFIG_OFF + 5, edid_num);
+ if (status)
+ return (PTN_BUS_ERROR | status);
+
+ /* Now we can simply write EDID data to ptn3460 */
+ for (i = 0; i < PTN_EDID_LEN; i++) {
+ status = i2c_dev_writeb_at(dev, PTN_EDID_OFF + i, data[i]);
+ if (status)
+ return (PTN_BUS_ERROR | status);
+ }
+ return PTN_SUCCESS;
+}
+
+/**
+ * \brief This function sets up the DP2LVDS-converter to be used with the
+ * appropriate EDID data
+ * @param *dev Pointer to the I2C controller where PTN3460 is attached
+ */
+static void ptn3460_init(struct device *dev)
+{
+ struct ptn_3460_config cfg;
+ uint8_t edid_data[PTN_EDID_LEN], edid_tab, *ptr = (uint8_t *) &cfg;
+ int i, val;
+
+ /* Mainboard provides EDID data. */
+ if (mb_get_edid(edid_data) != CB_SUCCESS) {
+ printk(BIOS_ERR, "PTN3460 error: Unable to get EDID data from mainboard.\n");
+ return;
+ }
+
+ /* Mainboard decides which EDID table has to be used. */
+ edid_tab = mb_select_edid_table();
+ if (edid_tab > PTN_MAX_EDID_NUM) {
+ printk(BIOS_ERR, "PTN3460 error: invalid EDID table (%d) selected.\n",
+ edid_tab);
+ return;
+ }
+ /* Write EDID data into PTN. */
+ val = ptn3460_write_edid(dev, edid_tab, edid_data);
+ if (val != PTN_SUCCESS) {
+ printk(BIOS_ERR, "PTN3460 error: writing EDID data into device failed.\n");
+ return;
+ }
+ /* Activate the selected EDID block. */
+ ptn_select_edid(dev, edid_tab);
+ /* Read out PTN configuration data. */
+ for (i = 0; i < sizeof(struct ptn_3460_config); i++) {
+ val = i2c_dev_readb_at(dev, PTN_CONFIG_OFF + i);
+ if (val < 0) {
+ printk(BIOS_ERR,
+ "PTN3460 error: Unable to read config data from device.\n");
+ return;
+ }
+ *ptr++ = (uint8_t)val; /* fill config structure via ptr */
+ }
+ /* Mainboard can modify the configuration data.
+ Write back configuration data to PTN3460 if modified by mainboard */
+ if (mb_adjust_cfg(&cfg) == PTN_CFG_MODIFIED) {
+ ptr = (uint8_t *) &cfg;
+ for (i = 0; i < sizeof(struct ptn_3460_config); i++) {
+ val = i2c_dev_writeb_at(dev, PTN_CONFIG_OFF + i, *ptr++);
+ if (val < 0) {
+ printk(BIOS_ERR,
+ "PTN3460 error: Unable to write config data.\n");
+ return;
+ }
+ }
+ }
+}
+
+__weak enum cb_err mb_get_edid(uint8_t edid_data[0x80])
+{
+ return CB_ERR;
+}
+__weak uint8_t mb_select_edid_table(void)
+{
+ return 0;
+}
+__weak int mb_adjust_cfg(struct ptn_3460_config *cfg_ptr)
+{
+ return 0;
+}
+
+static struct device_operations ptn3460_ops = {
+ .read_resources = DEVICE_NOOP,
+ .set_resources = DEVICE_NOOP,
+ .enable_resources = DEVICE_NOOP,
+ .init = ptn3460_init,
+ .final = DEVICE_NOOP
+};
+
+static void ptn3460_enable(struct device *dev)
+{
+ dev->ops = &ptn3460_ops;
+}
+
+struct chip_operations drivers_i2c_ptn3460_ops = {
+ CHIP_NAME("PTN3460")
+ .enable_dev = ptn3460_enable
+};