1*f39db26cSSui Jingfeng // SPDX-License-Identifier: GPL-2.0+
2*f39db26cSSui Jingfeng /*
3*f39db26cSSui Jingfeng * Copyright (C) 2023 Loongson Technology Corporation Limited
4*f39db26cSSui Jingfeng */
5*f39db26cSSui Jingfeng
6*f39db26cSSui Jingfeng #include <drm/drm_managed.h>
7*f39db26cSSui Jingfeng
8*f39db26cSSui Jingfeng #include "lsdc_drv.h"
9*f39db26cSSui Jingfeng #include "lsdc_output.h"
10*f39db26cSSui Jingfeng
11*f39db26cSSui Jingfeng /*
12*f39db26cSSui Jingfeng * __lsdc_gpio_i2c_set - set the state of a gpio pin indicated by mask
13*f39db26cSSui Jingfeng * @mask: gpio pin mask
14*f39db26cSSui Jingfeng * @state: "0" for low, "1" for high
15*f39db26cSSui Jingfeng */
__lsdc_gpio_i2c_set(struct lsdc_i2c * const li2c,int mask,int state)16*f39db26cSSui Jingfeng static void __lsdc_gpio_i2c_set(struct lsdc_i2c * const li2c, int mask, int state)
17*f39db26cSSui Jingfeng {
18*f39db26cSSui Jingfeng struct lsdc_device *ldev = to_lsdc(li2c->ddev);
19*f39db26cSSui Jingfeng unsigned long flags;
20*f39db26cSSui Jingfeng u8 val;
21*f39db26cSSui Jingfeng
22*f39db26cSSui Jingfeng spin_lock_irqsave(&ldev->reglock, flags);
23*f39db26cSSui Jingfeng
24*f39db26cSSui Jingfeng if (state) {
25*f39db26cSSui Jingfeng /*
26*f39db26cSSui Jingfeng * Setting this pin as input directly, write 1 for input.
27*f39db26cSSui Jingfeng * The external pull-up resistor will pull the level up
28*f39db26cSSui Jingfeng */
29*f39db26cSSui Jingfeng val = readb(li2c->dir_reg);
30*f39db26cSSui Jingfeng val |= mask;
31*f39db26cSSui Jingfeng writeb(val, li2c->dir_reg);
32*f39db26cSSui Jingfeng } else {
33*f39db26cSSui Jingfeng /* First set this pin as output, write 0 for output */
34*f39db26cSSui Jingfeng val = readb(li2c->dir_reg);
35*f39db26cSSui Jingfeng val &= ~mask;
36*f39db26cSSui Jingfeng writeb(val, li2c->dir_reg);
37*f39db26cSSui Jingfeng
38*f39db26cSSui Jingfeng /* Then, make this pin output 0 */
39*f39db26cSSui Jingfeng val = readb(li2c->dat_reg);
40*f39db26cSSui Jingfeng val &= ~mask;
41*f39db26cSSui Jingfeng writeb(val, li2c->dat_reg);
42*f39db26cSSui Jingfeng }
43*f39db26cSSui Jingfeng
44*f39db26cSSui Jingfeng spin_unlock_irqrestore(&ldev->reglock, flags);
45*f39db26cSSui Jingfeng }
46*f39db26cSSui Jingfeng
47*f39db26cSSui Jingfeng /*
48*f39db26cSSui Jingfeng * __lsdc_gpio_i2c_get - read value back from the gpio pin indicated by mask
49*f39db26cSSui Jingfeng * @mask: gpio pin mask
50*f39db26cSSui Jingfeng * return "0" for low, "1" for high
51*f39db26cSSui Jingfeng */
__lsdc_gpio_i2c_get(struct lsdc_i2c * const li2c,int mask)52*f39db26cSSui Jingfeng static int __lsdc_gpio_i2c_get(struct lsdc_i2c * const li2c, int mask)
53*f39db26cSSui Jingfeng {
54*f39db26cSSui Jingfeng struct lsdc_device *ldev = to_lsdc(li2c->ddev);
55*f39db26cSSui Jingfeng unsigned long flags;
56*f39db26cSSui Jingfeng u8 val;
57*f39db26cSSui Jingfeng
58*f39db26cSSui Jingfeng spin_lock_irqsave(&ldev->reglock, flags);
59*f39db26cSSui Jingfeng
60*f39db26cSSui Jingfeng /* First set this pin as input */
61*f39db26cSSui Jingfeng val = readb(li2c->dir_reg);
62*f39db26cSSui Jingfeng val |= mask;
63*f39db26cSSui Jingfeng writeb(val, li2c->dir_reg);
64*f39db26cSSui Jingfeng
65*f39db26cSSui Jingfeng /* Then get level state from this pin */
66*f39db26cSSui Jingfeng val = readb(li2c->dat_reg);
67*f39db26cSSui Jingfeng
68*f39db26cSSui Jingfeng spin_unlock_irqrestore(&ldev->reglock, flags);
69*f39db26cSSui Jingfeng
70*f39db26cSSui Jingfeng return (val & mask) ? 1 : 0;
71*f39db26cSSui Jingfeng }
72*f39db26cSSui Jingfeng
lsdc_gpio_i2c_set_sda(void * i2c,int state)73*f39db26cSSui Jingfeng static void lsdc_gpio_i2c_set_sda(void *i2c, int state)
74*f39db26cSSui Jingfeng {
75*f39db26cSSui Jingfeng struct lsdc_i2c * const li2c = (struct lsdc_i2c *)i2c;
76*f39db26cSSui Jingfeng /* set state on the li2c->sda pin */
77*f39db26cSSui Jingfeng return __lsdc_gpio_i2c_set(li2c, li2c->sda, state);
78*f39db26cSSui Jingfeng }
79*f39db26cSSui Jingfeng
lsdc_gpio_i2c_set_scl(void * i2c,int state)80*f39db26cSSui Jingfeng static void lsdc_gpio_i2c_set_scl(void *i2c, int state)
81*f39db26cSSui Jingfeng {
82*f39db26cSSui Jingfeng struct lsdc_i2c * const li2c = (struct lsdc_i2c *)i2c;
83*f39db26cSSui Jingfeng /* set state on the li2c->scl pin */
84*f39db26cSSui Jingfeng return __lsdc_gpio_i2c_set(li2c, li2c->scl, state);
85*f39db26cSSui Jingfeng }
86*f39db26cSSui Jingfeng
lsdc_gpio_i2c_get_sda(void * i2c)87*f39db26cSSui Jingfeng static int lsdc_gpio_i2c_get_sda(void *i2c)
88*f39db26cSSui Jingfeng {
89*f39db26cSSui Jingfeng struct lsdc_i2c * const li2c = (struct lsdc_i2c *)i2c;
90*f39db26cSSui Jingfeng /* read value from the li2c->sda pin */
91*f39db26cSSui Jingfeng return __lsdc_gpio_i2c_get(li2c, li2c->sda);
92*f39db26cSSui Jingfeng }
93*f39db26cSSui Jingfeng
lsdc_gpio_i2c_get_scl(void * i2c)94*f39db26cSSui Jingfeng static int lsdc_gpio_i2c_get_scl(void *i2c)
95*f39db26cSSui Jingfeng {
96*f39db26cSSui Jingfeng struct lsdc_i2c * const li2c = (struct lsdc_i2c *)i2c;
97*f39db26cSSui Jingfeng /* read the value from the li2c->scl pin */
98*f39db26cSSui Jingfeng return __lsdc_gpio_i2c_get(li2c, li2c->scl);
99*f39db26cSSui Jingfeng }
100*f39db26cSSui Jingfeng
lsdc_destroy_i2c(struct drm_device * ddev,void * data)101*f39db26cSSui Jingfeng static void lsdc_destroy_i2c(struct drm_device *ddev, void *data)
102*f39db26cSSui Jingfeng {
103*f39db26cSSui Jingfeng struct lsdc_i2c *li2c = (struct lsdc_i2c *)data;
104*f39db26cSSui Jingfeng
105*f39db26cSSui Jingfeng if (li2c) {
106*f39db26cSSui Jingfeng i2c_del_adapter(&li2c->adapter);
107*f39db26cSSui Jingfeng kfree(li2c);
108*f39db26cSSui Jingfeng }
109*f39db26cSSui Jingfeng }
110*f39db26cSSui Jingfeng
111*f39db26cSSui Jingfeng /*
112*f39db26cSSui Jingfeng * The DC in ls7a1000/ls7a2000/ls2k2000 has builtin gpio hardware
113*f39db26cSSui Jingfeng *
114*f39db26cSSui Jingfeng * @reg_base: gpio reg base
115*f39db26cSSui Jingfeng * @index: output channel index, 0 for PIPE0, 1 for PIPE1
116*f39db26cSSui Jingfeng */
lsdc_create_i2c_chan(struct drm_device * ddev,struct lsdc_display_pipe * dispipe,unsigned int index)117*f39db26cSSui Jingfeng int lsdc_create_i2c_chan(struct drm_device *ddev,
118*f39db26cSSui Jingfeng struct lsdc_display_pipe *dispipe,
119*f39db26cSSui Jingfeng unsigned int index)
120*f39db26cSSui Jingfeng {
121*f39db26cSSui Jingfeng struct lsdc_device *ldev = to_lsdc(ddev);
122*f39db26cSSui Jingfeng struct i2c_adapter *adapter;
123*f39db26cSSui Jingfeng struct lsdc_i2c *li2c;
124*f39db26cSSui Jingfeng int ret;
125*f39db26cSSui Jingfeng
126*f39db26cSSui Jingfeng li2c = kzalloc(sizeof(*li2c), GFP_KERNEL);
127*f39db26cSSui Jingfeng if (!li2c)
128*f39db26cSSui Jingfeng return -ENOMEM;
129*f39db26cSSui Jingfeng
130*f39db26cSSui Jingfeng dispipe->li2c = li2c;
131*f39db26cSSui Jingfeng
132*f39db26cSSui Jingfeng if (index == 0) {
133*f39db26cSSui Jingfeng li2c->sda = 0x01; /* pin 0 */
134*f39db26cSSui Jingfeng li2c->scl = 0x02; /* pin 1 */
135*f39db26cSSui Jingfeng } else if (index == 1) {
136*f39db26cSSui Jingfeng li2c->sda = 0x04; /* pin 2 */
137*f39db26cSSui Jingfeng li2c->scl = 0x08; /* pin 3 */
138*f39db26cSSui Jingfeng } else {
139*f39db26cSSui Jingfeng return -ENOENT;
140*f39db26cSSui Jingfeng }
141*f39db26cSSui Jingfeng
142*f39db26cSSui Jingfeng li2c->ddev = ddev;
143*f39db26cSSui Jingfeng li2c->dir_reg = ldev->reg_base + LS7A_DC_GPIO_DIR_REG;
144*f39db26cSSui Jingfeng li2c->dat_reg = ldev->reg_base + LS7A_DC_GPIO_DAT_REG;
145*f39db26cSSui Jingfeng
146*f39db26cSSui Jingfeng li2c->bit.setsda = lsdc_gpio_i2c_set_sda;
147*f39db26cSSui Jingfeng li2c->bit.setscl = lsdc_gpio_i2c_set_scl;
148*f39db26cSSui Jingfeng li2c->bit.getsda = lsdc_gpio_i2c_get_sda;
149*f39db26cSSui Jingfeng li2c->bit.getscl = lsdc_gpio_i2c_get_scl;
150*f39db26cSSui Jingfeng li2c->bit.udelay = 5;
151*f39db26cSSui Jingfeng li2c->bit.timeout = usecs_to_jiffies(2200);
152*f39db26cSSui Jingfeng li2c->bit.data = li2c;
153*f39db26cSSui Jingfeng
154*f39db26cSSui Jingfeng adapter = &li2c->adapter;
155*f39db26cSSui Jingfeng adapter->algo_data = &li2c->bit;
156*f39db26cSSui Jingfeng adapter->owner = THIS_MODULE;
157*f39db26cSSui Jingfeng adapter->class = I2C_CLASS_DDC;
158*f39db26cSSui Jingfeng adapter->dev.parent = ddev->dev;
159*f39db26cSSui Jingfeng adapter->nr = -1;
160*f39db26cSSui Jingfeng
161*f39db26cSSui Jingfeng snprintf(adapter->name, sizeof(adapter->name), "lsdc-i2c%u", index);
162*f39db26cSSui Jingfeng
163*f39db26cSSui Jingfeng i2c_set_adapdata(adapter, li2c);
164*f39db26cSSui Jingfeng
165*f39db26cSSui Jingfeng ret = i2c_bit_add_bus(adapter);
166*f39db26cSSui Jingfeng if (ret) {
167*f39db26cSSui Jingfeng kfree(li2c);
168*f39db26cSSui Jingfeng return ret;
169*f39db26cSSui Jingfeng }
170*f39db26cSSui Jingfeng
171*f39db26cSSui Jingfeng ret = drmm_add_action_or_reset(ddev, lsdc_destroy_i2c, li2c);
172*f39db26cSSui Jingfeng if (ret)
173*f39db26cSSui Jingfeng return ret;
174*f39db26cSSui Jingfeng
175*f39db26cSSui Jingfeng drm_info(ddev, "%s(sda pin mask=%u, scl pin mask=%u) created\n",
176*f39db26cSSui Jingfeng adapter->name, li2c->sda, li2c->scl);
177*f39db26cSSui Jingfeng
178*f39db26cSSui Jingfeng return 0;
179*f39db26cSSui Jingfeng }
180