123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220 |
- /*
- * Copyright (C) 2016 Maxime Ripard <maxime.ripard@free-electrons.com>
- * Copyright (C) 2017 Jonathan Liu <net147@gmail.com>
- *
- * 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; either version 2 of
- * the License, or (at your option) any later version.
- */
- #include <linux/clk.h>
- #include <linux/i2c.h>
- #include <linux/iopoll.h>
- #include "sun4i_hdmi.h"
- #define SUN4I_HDMI_DDC_INT_STATUS_ERROR_MASK ( \
- SUN4I_HDMI_DDC_INT_STATUS_ILLEGAL_FIFO_OPERATION | \
- SUN4I_HDMI_DDC_INT_STATUS_DDC_RX_FIFO_UNDERFLOW | \
- SUN4I_HDMI_DDC_INT_STATUS_DDC_TX_FIFO_OVERFLOW | \
- SUN4I_HDMI_DDC_INT_STATUS_ARBITRATION_ERROR | \
- SUN4I_HDMI_DDC_INT_STATUS_ACK_ERROR | \
- SUN4I_HDMI_DDC_INT_STATUS_BUS_ERROR \
- )
- /* FIFO request bit is set when FIFO level is above RX_THRESHOLD during read */
- #define RX_THRESHOLD SUN4I_HDMI_DDC_FIFO_CTRL_RX_THRES_MAX
- /* FIFO request bit is set when FIFO level is below TX_THRESHOLD during write */
- #define TX_THRESHOLD 1
- static int fifo_transfer(struct sun4i_hdmi *hdmi, u8 *buf, int len, bool read)
- {
- /*
- * 1 byte takes 9 clock cycles (8 bits + 1 ACK) = 90 us for 100 kHz
- * clock. As clock rate is fixed, just round it up to 100 us.
- */
- const unsigned long byte_time_ns = 100;
- const u32 mask = SUN4I_HDMI_DDC_INT_STATUS_ERROR_MASK |
- SUN4I_HDMI_DDC_INT_STATUS_FIFO_REQUEST |
- SUN4I_HDMI_DDC_INT_STATUS_TRANSFER_COMPLETE;
- u32 reg;
- /* Limit transfer length by FIFO threshold */
- len = min_t(int, len, read ? (RX_THRESHOLD + 1) :
- (SUN4I_HDMI_DDC_FIFO_SIZE - TX_THRESHOLD + 1));
- /* Wait until error, FIFO request bit set or transfer complete */
- if (readl_poll_timeout(hdmi->base + SUN4I_HDMI_DDC_INT_STATUS_REG, reg,
- reg & mask, len * byte_time_ns, 100000))
- return -ETIMEDOUT;
- if (reg & SUN4I_HDMI_DDC_INT_STATUS_ERROR_MASK)
- return -EIO;
- if (read)
- readsb(hdmi->base + SUN4I_HDMI_DDC_FIFO_DATA_REG, buf, len);
- else
- writesb(hdmi->base + SUN4I_HDMI_DDC_FIFO_DATA_REG, buf, len);
- /* Clear FIFO request bit */
- writel(SUN4I_HDMI_DDC_INT_STATUS_FIFO_REQUEST,
- hdmi->base + SUN4I_HDMI_DDC_INT_STATUS_REG);
- return len;
- }
- static int xfer_msg(struct sun4i_hdmi *hdmi, struct i2c_msg *msg)
- {
- int i, len;
- u32 reg;
- /* Set FIFO direction */
- reg = readl(hdmi->base + SUN4I_HDMI_DDC_CTRL_REG);
- reg &= ~SUN4I_HDMI_DDC_CTRL_FIFO_DIR_MASK;
- reg |= (msg->flags & I2C_M_RD) ?
- SUN4I_HDMI_DDC_CTRL_FIFO_DIR_READ :
- SUN4I_HDMI_DDC_CTRL_FIFO_DIR_WRITE;
- writel(reg, hdmi->base + SUN4I_HDMI_DDC_CTRL_REG);
- /* Set I2C address */
- writel(SUN4I_HDMI_DDC_ADDR_SLAVE(msg->addr),
- hdmi->base + SUN4I_HDMI_DDC_ADDR_REG);
- /* Set FIFO RX/TX thresholds and clear FIFO */
- reg = readl(hdmi->base + SUN4I_HDMI_DDC_FIFO_CTRL_REG);
- reg |= SUN4I_HDMI_DDC_FIFO_CTRL_CLEAR;
- reg &= ~SUN4I_HDMI_DDC_FIFO_CTRL_RX_THRES_MASK;
- reg |= SUN4I_HDMI_DDC_FIFO_CTRL_RX_THRES(RX_THRESHOLD);
- reg &= ~SUN4I_HDMI_DDC_FIFO_CTRL_TX_THRES_MASK;
- reg |= SUN4I_HDMI_DDC_FIFO_CTRL_TX_THRES(TX_THRESHOLD);
- writel(reg, hdmi->base + SUN4I_HDMI_DDC_FIFO_CTRL_REG);
- if (readl_poll_timeout(hdmi->base + SUN4I_HDMI_DDC_FIFO_CTRL_REG,
- reg,
- !(reg & SUN4I_HDMI_DDC_FIFO_CTRL_CLEAR),
- 100, 2000))
- return -EIO;
- /* Set transfer length */
- writel(msg->len, hdmi->base + SUN4I_HDMI_DDC_BYTE_COUNT_REG);
- /* Set command */
- writel(msg->flags & I2C_M_RD ?
- SUN4I_HDMI_DDC_CMD_IMPLICIT_READ :
- SUN4I_HDMI_DDC_CMD_IMPLICIT_WRITE,
- hdmi->base + SUN4I_HDMI_DDC_CMD_REG);
- /* Clear interrupt status bits */
- writel(SUN4I_HDMI_DDC_INT_STATUS_ERROR_MASK |
- SUN4I_HDMI_DDC_INT_STATUS_FIFO_REQUEST |
- SUN4I_HDMI_DDC_INT_STATUS_TRANSFER_COMPLETE,
- hdmi->base + SUN4I_HDMI_DDC_INT_STATUS_REG);
- /* Start command */
- reg = readl(hdmi->base + SUN4I_HDMI_DDC_CTRL_REG);
- writel(reg | SUN4I_HDMI_DDC_CTRL_START_CMD,
- hdmi->base + SUN4I_HDMI_DDC_CTRL_REG);
- /* Transfer bytes */
- for (i = 0; i < msg->len; i += len) {
- len = fifo_transfer(hdmi, msg->buf + i, msg->len - i,
- msg->flags & I2C_M_RD);
- if (len <= 0)
- return len;
- }
- /* Wait for command to finish */
- if (readl_poll_timeout(hdmi->base + SUN4I_HDMI_DDC_CTRL_REG,
- reg,
- !(reg & SUN4I_HDMI_DDC_CTRL_START_CMD),
- 100, 100000))
- return -EIO;
- /* Check for errors */
- reg = readl(hdmi->base + SUN4I_HDMI_DDC_INT_STATUS_REG);
- if ((reg & SUN4I_HDMI_DDC_INT_STATUS_ERROR_MASK) ||
- !(reg & SUN4I_HDMI_DDC_INT_STATUS_TRANSFER_COMPLETE)) {
- return -EIO;
- }
- return 0;
- }
- static int sun4i_hdmi_i2c_xfer(struct i2c_adapter *adap,
- struct i2c_msg *msgs, int num)
- {
- struct sun4i_hdmi *hdmi = i2c_get_adapdata(adap);
- u32 reg;
- int err, i, ret = num;
- for (i = 0; i < num; i++) {
- if (!msgs[i].len)
- return -EINVAL;
- if (msgs[i].len > SUN4I_HDMI_DDC_BYTE_COUNT_MAX)
- return -EINVAL;
- }
- /* Reset I2C controller */
- writel(SUN4I_HDMI_DDC_CTRL_ENABLE | SUN4I_HDMI_DDC_CTRL_RESET,
- hdmi->base + SUN4I_HDMI_DDC_CTRL_REG);
- if (readl_poll_timeout(hdmi->base + SUN4I_HDMI_DDC_CTRL_REG, reg,
- !(reg & SUN4I_HDMI_DDC_CTRL_RESET),
- 100, 2000))
- return -EIO;
- writel(SUN4I_HDMI_DDC_LINE_CTRL_SDA_ENABLE |
- SUN4I_HDMI_DDC_LINE_CTRL_SCL_ENABLE,
- hdmi->base + SUN4I_HDMI_DDC_LINE_CTRL_REG);
- clk_prepare_enable(hdmi->ddc_clk);
- clk_set_rate(hdmi->ddc_clk, 100000);
- for (i = 0; i < num; i++) {
- err = xfer_msg(hdmi, &msgs[i]);
- if (err) {
- ret = err;
- break;
- }
- }
- clk_disable_unprepare(hdmi->ddc_clk);
- return ret;
- }
- static u32 sun4i_hdmi_i2c_func(struct i2c_adapter *adap)
- {
- return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL;
- }
- static const struct i2c_algorithm sun4i_hdmi_i2c_algorithm = {
- .master_xfer = sun4i_hdmi_i2c_xfer,
- .functionality = sun4i_hdmi_i2c_func,
- };
- int sun4i_hdmi_i2c_create(struct device *dev, struct sun4i_hdmi *hdmi)
- {
- struct i2c_adapter *adap;
- int ret = 0;
- ret = sun4i_ddc_create(hdmi, hdmi->tmds_clk);
- if (ret)
- return ret;
- adap = devm_kzalloc(dev, sizeof(*adap), GFP_KERNEL);
- if (!adap)
- return -ENOMEM;
- adap->owner = THIS_MODULE;
- adap->class = I2C_CLASS_DDC;
- adap->algo = &sun4i_hdmi_i2c_algorithm;
- strlcpy(adap->name, "sun4i_hdmi_i2c adapter", sizeof(adap->name));
- i2c_set_adapdata(adap, hdmi);
- ret = i2c_add_adapter(adap);
- if (ret)
- return ret;
- hdmi->i2c = adap;
- return ret;
- }
|