1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Copyright (c) 2009,2010       One Laptop per Child
4  */
5 
6 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
7 
8 #include <linux/acpi.h>
9 #include <linux/delay.h>
10 #include <linux/i2c.h>
11 #include <linux/gpio/consumer.h>
12 #include <linux/gpio/machine.h>
13 #include <asm/olpc.h>
14 
15 /* TODO: this eventually belongs in linux/vx855.h */
16 #define NR_VX855_GPI    14
17 #define NR_VX855_GPO    13
18 #define NR_VX855_GPIO   15
19 
20 #define VX855_GPI(n)    (n)
21 #define VX855_GPO(n)    (NR_VX855_GPI + (n))
22 #define VX855_GPIO(n)   (NR_VX855_GPI + NR_VX855_GPO + (n))
23 
24 #include "olpc_dcon.h"
25 
26 /* Hardware setup on the XO 1.5:
27  *	DCONLOAD connects to VX855_GPIO1 (not SMBCK2)
28  *	DCONBLANK connects to VX855_GPIO8 (not SSPICLK)  unused in driver
29  *	DCONSTAT0 connects to VX855_GPI10 (not SSPISDI)
30  *	DCONSTAT1 connects to VX855_GPI11 (not nSSPISS)
31  *	DCONIRQ connects to VX855_GPIO12
32  *	DCONSMBDATA connects to VX855 graphics CRTSPD
33  *	DCONSMBCLK connects to VX855 graphics CRTSPCLK
34  */
35 
36 #define VX855_GENL_PURPOSE_OUTPUT 0x44c /* PMIO_Rx4c-4f */
37 #define VX855_GPI_STATUS_CHG 0x450  /* PMIO_Rx50 */
38 #define VX855_GPI_SCI_SMI 0x452  /* PMIO_Rx52 */
39 #define BIT_GPIO12 0x40
40 
41 #define PREFIX "OLPC DCON:"
42 
43 enum dcon_gpios {
44 	OLPC_DCON_STAT0,
45 	OLPC_DCON_STAT1,
46 	OLPC_DCON_LOAD,
47 };
48 
49 struct gpiod_lookup_table gpios_table = {
50 	.dev_id = NULL,
51 	.table = {
52 		GPIO_LOOKUP("VX855 South Bridge", VX855_GPIO(1), "dcon_load",
53 			    GPIO_ACTIVE_LOW),
54 		GPIO_LOOKUP("VX855 South Bridge", VX855_GPI(10), "dcon_stat0",
55 			    GPIO_ACTIVE_LOW),
56 		GPIO_LOOKUP("VX855 South Bridge", VX855_GPI(11), "dcon_stat1",
57 			    GPIO_ACTIVE_LOW),
58 		{ },
59 	},
60 };
61 
62 static const struct dcon_gpio gpios_asis[] = {
63 	[OLPC_DCON_STAT0] = { .name = "dcon_stat0", .flags = GPIOD_ASIS },
64 	[OLPC_DCON_STAT1] = { .name = "dcon_stat1", .flags = GPIOD_ASIS },
65 	[OLPC_DCON_LOAD] = { .name = "dcon_load", .flags = GPIOD_ASIS },
66 };
67 
68 static struct gpio_desc *gpios[3];
69 
70 static void dcon_clear_irq(void)
71 {
72 	/* irq status will appear in PMIO_Rx50[6] (RW1C) on gpio12 */
73 	outb(BIT_GPIO12, VX855_GPI_STATUS_CHG);
74 }
75 
76 static int dcon_was_irq(void)
77 {
78 	u8 tmp;
79 
80 	/* irq status will appear in PMIO_Rx50[6] on gpio12 */
81 	tmp = inb(VX855_GPI_STATUS_CHG);
82 
83 	return !!(tmp & BIT_GPIO12);
84 }
85 
86 static int dcon_init_xo_1_5(struct dcon_priv *dcon)
87 {
88 	unsigned int irq;
89 	const struct dcon_gpio *pin = &gpios_asis[0];
90 	int i;
91 	int ret;
92 
93 	/* Add GPIO look up table */
94 	gpios_table.dev_id = dev_name(&dcon->client->dev);
95 	gpiod_add_lookup_table(&gpios_table);
96 
97 	/* Get GPIO descriptor */
98 	for (i = 0; i < ARRAY_SIZE(gpios_asis); i++) {
99 		gpios[i] = devm_gpiod_get(&dcon->client->dev, pin[i].name,
100 					  pin[i].flags);
101 		if (IS_ERR(gpios[i])) {
102 			ret = PTR_ERR(gpios[i]);
103 			pr_err("failed to request %s GPIO: %d\n", pin[i].name,
104 			       ret);
105 			return ret;
106 		}
107 	}
108 
109 	dcon_clear_irq();
110 
111 	/* set   PMIO_Rx52[6] to enable SCI/SMI on gpio12 */
112 	outb(inb(VX855_GPI_SCI_SMI) | BIT_GPIO12, VX855_GPI_SCI_SMI);
113 
114 	/* Determine the current state of DCONLOAD, likely set by firmware */
115 	/* GPIO1 */
116 	dcon->curr_src = (inl(VX855_GENL_PURPOSE_OUTPUT) & 0x1000) ?
117 			DCON_SOURCE_CPU : DCON_SOURCE_DCON;
118 	dcon->pending_src = dcon->curr_src;
119 
120 	/* we're sharing the IRQ with ACPI */
121 	irq = acpi_gbl_FADT.sci_interrupt;
122 	if (request_irq(irq, &dcon_interrupt, IRQF_SHARED, "DCON", dcon)) {
123 		pr_err("DCON (IRQ%d) allocation failed\n", irq);
124 		return 1;
125 	}
126 
127 	return 0;
128 }
129 
130 static void set_i2c_line(int sda, int scl)
131 {
132 	unsigned char tmp;
133 	unsigned int port = 0x26;
134 
135 	/* FIXME: This directly accesses the CRT GPIO controller !!! */
136 	outb(port, 0x3c4);
137 	tmp = inb(0x3c5);
138 
139 	if (scl)
140 		tmp |= 0x20;
141 	else
142 		tmp &= ~0x20;
143 
144 	if (sda)
145 		tmp |= 0x10;
146 	else
147 		tmp &= ~0x10;
148 
149 	tmp |= 0x01;
150 
151 	outb(port, 0x3c4);
152 	outb(tmp, 0x3c5);
153 }
154 
155 static void dcon_wiggle_xo_1_5(void)
156 {
157 	int x;
158 
159 	/*
160 	 * According to HiMax, when powering the DCON up we should hold
161 	 * SMB_DATA high for 8 SMB_CLK cycles.  This will force the DCON
162 	 * state machine to reset to a (sane) initial state.  Mitch Bradley
163 	 * did some testing and discovered that holding for 16 SMB_CLK cycles
164 	 * worked a lot more reliably, so that's what we do here.
165 	 */
166 	set_i2c_line(1, 1);
167 
168 	for (x = 0; x < 16; x++) {
169 		udelay(5);
170 		set_i2c_line(1, 0);
171 		udelay(5);
172 		set_i2c_line(1, 1);
173 	}
174 	udelay(5);
175 
176 	/* set   PMIO_Rx52[6] to enable SCI/SMI on gpio12 */
177 	outb(inb(VX855_GPI_SCI_SMI) | BIT_GPIO12, VX855_GPI_SCI_SMI);
178 }
179 
180 static void dcon_set_dconload_xo_1_5(int val)
181 {
182 	gpiod_set_value(gpios[OLPC_DCON_LOAD], val);
183 }
184 
185 static int dcon_read_status_xo_1_5(u8 *status)
186 {
187 	if (!dcon_was_irq())
188 		return -1;
189 
190 	/* i believe this is the same as "inb(0x44b) & 3" */
191 	*status = gpiod_get_value(gpios[OLPC_DCON_STAT0]);
192 	*status |= gpiod_get_value(gpios[OLPC_DCON_STAT1]) << 1;
193 
194 	dcon_clear_irq();
195 
196 	return 0;
197 }
198 
199 struct dcon_platform_data dcon_pdata_xo_1_5 = {
200 	.init = dcon_init_xo_1_5,
201 	.bus_stabilize_wiggle = dcon_wiggle_xo_1_5,
202 	.set_dconload = dcon_set_dconload_xo_1_5,
203 	.read_status = dcon_read_status_xo_1_5,
204 };
205