drm/panel: simple: add Multi-Inno Technology MI0700A2T-30
[drm/drm-misc.git] / drivers / i2c / busses / i2c-viai2c-zhaoxin.c
blob95dc64902b7cdb01beea04acf7bad8f20133d965
1 // SPDX-License-Identifier: GPL-2.0-or-later
2 /*
3 * Copyright(c) 2024 Shanghai Zhaoxin Semiconductor Corporation.
4 * All rights reserved.
5 */
7 #include <linux/acpi.h>
8 #include "i2c-viai2c-common.h"
11 * registers
13 /* Zhaoxin specific register bit fields */
14 /* REG_CR Bit fields */
15 #define ZXI2C_CR_MST_RST BIT(7)
16 #define ZXI2C_CR_FIFO_MODE BIT(14)
17 /* REG_ISR/IMR Bit fields */
18 #define ZXI2C_IRQ_FIFONACK BIT(4)
19 #define ZXI2C_IRQ_FIFOEND BIT(3)
20 #define ZXI2C_IRQ_MASK (VIAI2C_ISR_MASK_ALL \
21 | ZXI2C_IRQ_FIFOEND \
22 | ZXI2C_IRQ_FIFONACK)
23 /* Zhaoxin specific registers */
24 #define ZXI2C_REG_CLK 0x10
25 #define ZXI2C_CLK_50M BIT(0)
26 #define ZXI2C_REG_REV 0x11
27 #define ZXI2C_REG_HCR 0x12
28 #define ZXI2C_HCR_RST_FIFO GENMASK(1, 0)
29 #define ZXI2C_REG_HTDR 0x13
30 #define ZXI2C_REG_HRDR 0x14
31 #define ZXI2C_REG_HTLR 0x15
32 #define ZXI2C_REG_HRLR 0x16
33 #define ZXI2C_REG_HWCNTR 0x18
34 #define ZXI2C_REG_HRCNTR 0x19
36 /* parameters Constants */
37 #define ZXI2C_GOLD_FSTP_100K 0xF3
38 #define ZXI2C_GOLD_FSTP_400K 0x38
39 #define ZXI2C_GOLD_FSTP_1M 0x13
40 #define ZXI2C_GOLD_FSTP_3400K 0x37
41 #define ZXI2C_HS_CTRL_CODE (0x08 << 8)
43 #define ZXI2C_FIFO_SIZE 32
45 struct viai2c_zhaoxin {
46 u8 hrv;
47 u16 tr;
48 u16 mcr;
49 u16 xfer_len;
52 static int viai2c_fifo_xfer(struct viai2c *i2c)
54 u16 i;
55 u8 tmp;
56 struct i2c_msg *msg = i2c->msg;
57 void __iomem *base = i2c->base;
58 bool read = !!(msg->flags & I2C_M_RD);
59 struct viai2c_zhaoxin *priv = i2c->pltfm_priv;
61 /* reset fifo buffer */
62 tmp = ioread8(base + ZXI2C_REG_HCR);
63 iowrite8(tmp | ZXI2C_HCR_RST_FIFO, base + ZXI2C_REG_HCR);
65 /* set xfer len */
66 priv->xfer_len = min_t(u16, msg->len - i2c->xfered_len, ZXI2C_FIFO_SIZE);
67 if (read) {
68 iowrite8(priv->xfer_len - 1, base + ZXI2C_REG_HRLR);
69 } else {
70 iowrite8(priv->xfer_len - 1, base + ZXI2C_REG_HTLR);
71 /* set write data */
72 for (i = 0; i < priv->xfer_len; i++)
73 iowrite8(msg->buf[i2c->xfered_len + i], base + ZXI2C_REG_HTDR);
76 /* prepare to stop transmission */
77 if (priv->hrv && msg->len == (i2c->xfered_len + priv->xfer_len)) {
78 tmp = ioread8(base + VIAI2C_REG_CR);
79 tmp |= read ? VIAI2C_CR_RX_END : VIAI2C_CR_TX_END;
80 iowrite8(tmp, base + VIAI2C_REG_CR);
83 u16 tcr_val = i2c->tcr;
85 /* start transmission */
86 tcr_val |= read ? VIAI2C_TCR_READ : 0;
87 writew(tcr_val | msg->addr, base + VIAI2C_REG_TCR);
89 return 0;
92 static int viai2c_fifo_irq_xfer(struct viai2c *i2c)
94 u16 i;
95 u8 tmp;
96 struct i2c_msg *msg = i2c->msg;
97 void __iomem *base = i2c->base;
98 bool read = !!(msg->flags & I2C_M_RD);
99 struct viai2c_zhaoxin *priv = i2c->pltfm_priv;
101 /* get the received data */
102 if (read)
103 for (i = 0; i < priv->xfer_len; i++)
104 msg->buf[i2c->xfered_len + i] = ioread8(base + ZXI2C_REG_HRDR);
106 i2c->xfered_len += priv->xfer_len;
107 if (i2c->xfered_len == msg->len)
108 return 1;
110 /* reset fifo buffer */
111 tmp = ioread8(base + ZXI2C_REG_HCR);
112 iowrite8(tmp | ZXI2C_HCR_RST_FIFO, base + ZXI2C_REG_HCR);
114 /* set xfer len */
115 priv->xfer_len = min_t(u16, msg->len - i2c->xfered_len, ZXI2C_FIFO_SIZE);
116 if (read) {
117 iowrite8(priv->xfer_len - 1, base + ZXI2C_REG_HRLR);
118 } else {
119 iowrite8(priv->xfer_len - 1, base + ZXI2C_REG_HTLR);
120 /* set write data */
121 for (i = 0; i < priv->xfer_len; i++)
122 iowrite8(msg->buf[i2c->xfered_len + i], base + ZXI2C_REG_HTDR);
125 /* prepare to stop transmission */
126 if (priv->hrv && msg->len == (i2c->xfered_len + priv->xfer_len)) {
127 tmp = ioread8(base + VIAI2C_REG_CR);
128 tmp |= read ? VIAI2C_CR_RX_END : VIAI2C_CR_TX_END;
129 iowrite8(tmp, base + VIAI2C_REG_CR);
132 /* continue transmission */
133 tmp = ioread8(base + VIAI2C_REG_CR);
134 iowrite8(tmp |= VIAI2C_CR_CPU_RDY, base + VIAI2C_REG_CR);
136 return 0;
139 static int zxi2c_xfer(struct i2c_adapter *adap, struct i2c_msg *msgs, int num)
141 u8 tmp;
142 int ret;
143 struct viai2c *i2c = (struct viai2c *)i2c_get_adapdata(adap);
144 struct viai2c_zhaoxin *priv = i2c->pltfm_priv;
145 void __iomem *base = i2c->base;
147 ret = viai2c_wait_bus_not_busy(i2c);
148 if (ret)
149 return ret;
151 tmp = ioread8(base + VIAI2C_REG_CR);
152 tmp &= ~(VIAI2C_CR_RX_END | VIAI2C_CR_TX_END);
154 if (num == 1 && msgs->len >= 2 && (priv->hrv || msgs->len <= ZXI2C_FIFO_SIZE)) {
155 /* enable fifo mode */
156 iowrite16(ZXI2C_CR_FIFO_MODE | tmp, base + VIAI2C_REG_CR);
157 /* clear irq status */
158 iowrite8(ZXI2C_IRQ_MASK, base + VIAI2C_REG_ISR);
159 /* enable fifo irq */
160 iowrite8(VIAI2C_ISR_NACK_ADDR | ZXI2C_IRQ_FIFOEND, base + VIAI2C_REG_IMR);
162 i2c->msg = msgs;
163 i2c->mode = VIAI2C_FIFO_MODE;
164 priv->xfer_len = 0;
165 i2c->xfered_len = 0;
167 viai2c_fifo_xfer(i2c);
169 if (!wait_for_completion_timeout(&i2c->complete, VIAI2C_TIMEOUT))
170 return -ETIMEDOUT;
172 ret = i2c->ret;
173 } else {
174 /* enable byte mode */
175 iowrite16(tmp, base + VIAI2C_REG_CR);
176 /* clear irq status */
177 iowrite8(ZXI2C_IRQ_MASK, base + VIAI2C_REG_ISR);
178 /* enable byte irq */
179 iowrite8(VIAI2C_ISR_NACK_ADDR | VIAI2C_IMR_BYTE, base + VIAI2C_REG_IMR);
181 ret = viai2c_xfer(adap, msgs, num);
182 if (ret == -ETIMEDOUT)
183 iowrite16(tmp | VIAI2C_CR_END_MASK, base + VIAI2C_REG_CR);
185 /* dis interrupt */
186 iowrite8(0, base + VIAI2C_REG_IMR);
188 return ret;
191 static u32 zxi2c_func(struct i2c_adapter *adap)
193 return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL;
196 static const struct i2c_algorithm zxi2c_algorithm = {
197 .xfer = zxi2c_xfer,
198 .functionality = zxi2c_func,
201 static const struct i2c_adapter_quirks zxi2c_quirks = {
202 .flags = I2C_AQ_NO_ZERO_LEN | I2C_AQ_COMB_WRITE_THEN_READ,
205 static const u32 zxi2c_speed_params_table[][3] = {
206 /* speed, ZXI2C_TCR, ZXI2C_FSTP */
207 { I2C_MAX_STANDARD_MODE_FREQ, 0, ZXI2C_GOLD_FSTP_100K },
208 { I2C_MAX_FAST_MODE_FREQ, VIAI2C_TCR_FAST, ZXI2C_GOLD_FSTP_400K },
209 { I2C_MAX_FAST_MODE_PLUS_FREQ, VIAI2C_TCR_FAST, ZXI2C_GOLD_FSTP_1M },
210 { I2C_MAX_HIGH_SPEED_MODE_FREQ, VIAI2C_TCR_HS_MODE | VIAI2C_TCR_FAST,
211 ZXI2C_GOLD_FSTP_3400K },
214 static void zxi2c_set_bus_speed(struct viai2c *i2c)
216 struct viai2c_zhaoxin *priv = i2c->pltfm_priv;
218 iowrite16(priv->tr, i2c->base + VIAI2C_REG_TR);
219 iowrite8(ZXI2C_CLK_50M, i2c->base + ZXI2C_REG_CLK);
220 iowrite16(priv->mcr, i2c->base + VIAI2C_REG_MCR);
223 static void zxi2c_get_bus_speed(struct viai2c *i2c)
225 u8 i, count;
226 u8 fstp;
227 const u32 *params;
228 struct viai2c_zhaoxin *priv = i2c->pltfm_priv;
229 u32 acpi_speed = i2c_acpi_find_bus_speed(i2c->dev);
231 count = ARRAY_SIZE(zxi2c_speed_params_table);
232 for (i = 0; i < count; i++)
233 if (acpi_speed == zxi2c_speed_params_table[i][0])
234 break;
235 /* if not found, use 400k as default */
236 i = i < count ? i : 1;
238 params = zxi2c_speed_params_table[i];
239 fstp = ioread8(i2c->base + VIAI2C_REG_TR);
240 if (abs(fstp - params[2]) > 0x10) {
242 * if BIOS setting value far from golden value,
243 * use golden value and warn user
245 dev_warn(i2c->dev, "FW FSTP[%x] might cause wrong timings, dropped\n", fstp);
246 priv->tr = params[2] | 0xff00;
247 } else {
248 priv->tr = fstp | 0xff00;
251 i2c->tcr = params[1];
252 priv->mcr = ioread16(i2c->base + VIAI2C_REG_MCR);
253 /* for Hs-mode, use 0x80 as controller code */
254 if (params[0] == I2C_MAX_HIGH_SPEED_MODE_FREQ)
255 priv->mcr |= ZXI2C_HS_CTRL_CODE;
257 dev_info(i2c->dev, "speed mode is %s\n", i2c_freq_mode_string(params[0]));
260 static irqreturn_t zxi2c_isr(int irq, void *data)
262 struct viai2c *i2c = data;
263 u8 status;
265 /* save the status and write-clear it */
266 status = readw(i2c->base + VIAI2C_REG_ISR);
267 if (!status)
268 return IRQ_NONE;
270 writew(status, i2c->base + VIAI2C_REG_ISR);
272 i2c->ret = 0;
273 if (status & VIAI2C_ISR_NACK_ADDR)
274 i2c->ret = -EIO;
276 if (!i2c->ret) {
277 if (i2c->mode == VIAI2C_BYTE_MODE)
278 i2c->ret = viai2c_irq_xfer(i2c);
279 else
280 i2c->ret = viai2c_fifo_irq_xfer(i2c);
283 /* All the data has been successfully transferred or error occurred */
284 if (i2c->ret)
285 complete(&i2c->complete);
287 return IRQ_HANDLED;
290 static int zxi2c_probe(struct platform_device *pdev)
292 int error;
293 struct viai2c *i2c;
294 struct i2c_adapter *adap;
295 struct viai2c_zhaoxin *priv;
297 error = viai2c_init(pdev, &i2c, VIAI2C_PLAT_ZHAOXIN);
298 if (error)
299 return error;
301 i2c->irq = platform_get_irq(pdev, 0);
302 if (i2c->irq < 0)
303 return i2c->irq;
305 error = devm_request_irq(&pdev->dev, i2c->irq, zxi2c_isr,
306 IRQF_SHARED, pdev->name, i2c);
307 if (error)
308 return dev_err_probe(&pdev->dev, error,
309 "failed to request irq %i\n", i2c->irq);
311 priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL);
312 if (!priv)
313 return -ENOMEM;
314 i2c->pltfm_priv = priv;
316 zxi2c_get_bus_speed(i2c);
317 zxi2c_set_bus_speed(i2c);
319 priv->hrv = ioread8(i2c->base + ZXI2C_REG_REV);
321 adap = &i2c->adapter;
322 adap->owner = THIS_MODULE;
323 adap->algo = &zxi2c_algorithm;
324 adap->quirks = &zxi2c_quirks;
325 adap->dev.parent = &pdev->dev;
326 ACPI_COMPANION_SET(&adap->dev, ACPI_COMPANION(&pdev->dev));
327 snprintf(adap->name, sizeof(adap->name), "zhaoxin-%s-%s",
328 dev_name(pdev->dev.parent), dev_name(i2c->dev));
329 i2c_set_adapdata(adap, i2c);
331 return devm_i2c_add_adapter(&pdev->dev, adap);
334 static int __maybe_unused zxi2c_resume(struct device *dev)
336 struct viai2c *i2c = dev_get_drvdata(dev);
338 iowrite8(ZXI2C_CR_MST_RST, i2c->base + VIAI2C_REG_CR);
339 zxi2c_set_bus_speed(i2c);
341 return 0;
344 static const struct dev_pm_ops zxi2c_pm = {
345 SET_SYSTEM_SLEEP_PM_OPS(NULL, zxi2c_resume)
348 static const struct acpi_device_id zxi2c_acpi_match[] = {
349 {"IIC1D17", 0 },
352 MODULE_DEVICE_TABLE(acpi, zxi2c_acpi_match);
354 static struct platform_driver zxi2c_driver = {
355 .probe = zxi2c_probe,
356 .driver = {
357 .name = "i2c_zhaoxin",
358 .acpi_match_table = zxi2c_acpi_match,
359 .pm = &zxi2c_pm,
363 module_platform_driver(zxi2c_driver);
365 MODULE_AUTHOR("HansHu@zhaoxin.com");
366 MODULE_DESCRIPTION("Shanghai Zhaoxin IIC driver");
367 MODULE_LICENSE("GPL");