xref: /rk3399_rockchip-uboot/drivers/i2c/intel_i2c.c (revision 5ebd27d860ec0c6e36f1b0f973653fe66a7360be)
1abb0b01eSSimon Glass /*
2abb0b01eSSimon Glass  * Copyright (c) 2015 Google, Inc
3abb0b01eSSimon Glass  * Written by Simon Glass <sjg@chromium.org>
4abb0b01eSSimon Glass  *
5ca6c5e03SStefan Roese  * SMBus block read/write support added by Stefan Roese:
6ca6c5e03SStefan Roese  * Copyright (C) 2016 Stefan Roese <sr@denx.de>
7ca6c5e03SStefan Roese  *
8abb0b01eSSimon Glass  * SPDX-License-Identifier:     GPL-2.0+
9abb0b01eSSimon Glass  */
10abb0b01eSSimon Glass 
11abb0b01eSSimon Glass #include <common.h>
12abb0b01eSSimon Glass #include <dm.h>
13abb0b01eSSimon Glass #include <i2c.h>
14ca6c5e03SStefan Roese #include <pci.h>
15abb0b01eSSimon Glass #include <asm/io.h>
16abb0b01eSSimon Glass 
17ca6c5e03SStefan Roese /* PCI Configuration Space (D31:F3): SMBus */
18ca6c5e03SStefan Roese #define SMB_BASE		0x20
19ca6c5e03SStefan Roese #define HOSTC			0x40
20ca6c5e03SStefan Roese #define  HST_EN			(1 << 0)
21ca6c5e03SStefan Roese #define SMB_RCV_SLVA		0x09
22ca6c5e03SStefan Roese 
23ca6c5e03SStefan Roese /* SMBus I/O bits. */
24ca6c5e03SStefan Roese #define SMBHSTSTAT		0x0
25ca6c5e03SStefan Roese #define SMBHSTCTL		0x2
26ca6c5e03SStefan Roese #define SMBHSTCMD		0x3
27ca6c5e03SStefan Roese #define SMBXMITADD		0x4
28ca6c5e03SStefan Roese #define SMBHSTDAT0		0x5
29ca6c5e03SStefan Roese #define SMBHSTDAT1		0x6
30ca6c5e03SStefan Roese #define SMBBLKDAT		0x7
31ca6c5e03SStefan Roese #define SMBTRNSADD		0x9
32ca6c5e03SStefan Roese #define SMBSLVDATA		0xa
33ca6c5e03SStefan Roese #define SMBAUXCTL		0xd
34ca6c5e03SStefan Roese #define SMLINK_PIN_CTL		0xe
35ca6c5e03SStefan Roese #define SMBUS_PIN_CTL		0xf
36ca6c5e03SStefan Roese 
37ca6c5e03SStefan Roese /* I801 Hosts Status register bits */
38ca6c5e03SStefan Roese #define SMBHSTSTS_BYTE_DONE	0x80
39ca6c5e03SStefan Roese #define SMBHSTSTS_INUSE_STS	0x40
40ca6c5e03SStefan Roese #define SMBHSTSTS_SMBALERT_STS	0x20
41ca6c5e03SStefan Roese #define SMBHSTSTS_FAILED	0x10
42ca6c5e03SStefan Roese #define SMBHSTSTS_BUS_ERR	0x08
43ca6c5e03SStefan Roese #define SMBHSTSTS_DEV_ERR	0x04
44ca6c5e03SStefan Roese #define SMBHSTSTS_INTR		0x02
45ca6c5e03SStefan Roese #define SMBHSTSTS_HOST_BUSY	0x01
46ca6c5e03SStefan Roese 
47ca6c5e03SStefan Roese /* I801 Host Control register bits */
48ca6c5e03SStefan Roese #define SMBHSTCNT_INTREN	0x01
49ca6c5e03SStefan Roese #define SMBHSTCNT_KILL		0x02
50ca6c5e03SStefan Roese #define SMBHSTCNT_LAST_BYTE	0x20
51ca6c5e03SStefan Roese #define SMBHSTCNT_START		0x40
52ca6c5e03SStefan Roese #define SMBHSTCNT_PEC_EN	0x80	/* ICH3 and later */
53ca6c5e03SStefan Roese 
54ca6c5e03SStefan Roese /* Auxiliary control register bits, ICH4+ only */
55ca6c5e03SStefan Roese #define SMBAUXCTL_CRC		1
56ca6c5e03SStefan Roese #define SMBAUXCTL_E32B		2
57ca6c5e03SStefan Roese 
58ca6c5e03SStefan Roese #define SMBUS_TIMEOUT	100	/* 100 ms */
59ca6c5e03SStefan Roese 
60ca6c5e03SStefan Roese struct intel_i2c {
61ca6c5e03SStefan Roese 	u32 base;
62ca6c5e03SStefan Roese 	int running;
63ca6c5e03SStefan Roese };
64ca6c5e03SStefan Roese 
smbus_wait_until_ready(u32 base)65ca6c5e03SStefan Roese static int smbus_wait_until_ready(u32 base)
66abb0b01eSSimon Glass {
67ca6c5e03SStefan Roese 	unsigned long ts;
68ca6c5e03SStefan Roese 	u8 byte;
69ca6c5e03SStefan Roese 
70ca6c5e03SStefan Roese 	ts = get_timer(0);
71ca6c5e03SStefan Roese 	do {
72ca6c5e03SStefan Roese 		byte = inb(base + SMBHSTSTAT);
73ca6c5e03SStefan Roese 		if (!(byte & 1))
74ca6c5e03SStefan Roese 			return 0;
75ca6c5e03SStefan Roese 	} while (get_timer(ts) < SMBUS_TIMEOUT);
76ca6c5e03SStefan Roese 
77ca6c5e03SStefan Roese 	return -ETIMEDOUT;
78abb0b01eSSimon Glass }
79abb0b01eSSimon Glass 
smbus_wait_until_done(u32 base)80ca6c5e03SStefan Roese static int smbus_wait_until_done(u32 base)
81abb0b01eSSimon Glass {
82ca6c5e03SStefan Roese 	unsigned long ts;
83ca6c5e03SStefan Roese 	u8 byte;
84ca6c5e03SStefan Roese 
85ca6c5e03SStefan Roese 	ts = get_timer(0);
86ca6c5e03SStefan Roese 	do {
87ca6c5e03SStefan Roese 		byte = inb(base + SMBHSTSTAT);
88ca6c5e03SStefan Roese 		if (!((byte & 1) || (byte & ~((1 << 6) | (1 << 0))) == 0))
89ca6c5e03SStefan Roese 			return 0;
90ca6c5e03SStefan Roese 	} while (get_timer(ts) < SMBUS_TIMEOUT);
91ca6c5e03SStefan Roese 
92ca6c5e03SStefan Roese 	return -ETIMEDOUT;
93abb0b01eSSimon Glass }
94abb0b01eSSimon Glass 
smbus_block_read(u32 base,u8 dev,u8 * buffer,int offset,int len)95ca6c5e03SStefan Roese static int smbus_block_read(u32 base, u8 dev, u8 *buffer,
96ca6c5e03SStefan Roese 			    int offset, int len)
97ca6c5e03SStefan Roese {
98ca6c5e03SStefan Roese 	u8 buf_temp[32];
99ca6c5e03SStefan Roese 	int count;
100ca6c5e03SStefan Roese 	int i;
101ca6c5e03SStefan Roese 
102ca6c5e03SStefan Roese 	debug("%s (%d): dev=0x%x offs=0x%x len=0x%x\n",
103ca6c5e03SStefan Roese 	      __func__, __LINE__, dev, offset, len);
104ca6c5e03SStefan Roese 	if (smbus_wait_until_ready(base) < 0)
105ca6c5e03SStefan Roese 		return -ETIMEDOUT;
106ca6c5e03SStefan Roese 
107ca6c5e03SStefan Roese 	/* Setup transaction */
108ca6c5e03SStefan Roese 
109ca6c5e03SStefan Roese 	/* Reset the data buffer index */
110ca6c5e03SStefan Roese 	inb(base + SMBHSTCTL);
111ca6c5e03SStefan Roese 
112ca6c5e03SStefan Roese 	/* Set the device I'm talking too */
113ca6c5e03SStefan Roese 	outb(((dev & 0x7f) << 1) | 1, base + SMBXMITADD);
114ca6c5e03SStefan Roese 	/* Set the command/address... */
115ca6c5e03SStefan Roese 	outb(offset & 0xff, base + SMBHSTCMD);
116ca6c5e03SStefan Roese 	/* Set up for a block read */
117ca6c5e03SStefan Roese 	outb((inb(base + SMBHSTCTL) & (~(0x7) << 2)) | (0x5 << 2),
118ca6c5e03SStefan Roese 	     (base + SMBHSTCTL));
119ca6c5e03SStefan Roese 	/* Clear any lingering errors, so the transaction will run */
120ca6c5e03SStefan Roese 	outb(inb(base + SMBHSTSTAT), base + SMBHSTSTAT);
121ca6c5e03SStefan Roese 
122ca6c5e03SStefan Roese 	/* Start the command */
123ca6c5e03SStefan Roese 	outb((inb(base + SMBHSTCTL) | SMBHSTCNT_START), base + SMBHSTCTL);
124ca6c5e03SStefan Roese 
125ca6c5e03SStefan Roese 	/* Poll for transaction completion */
126ca6c5e03SStefan Roese 	if (smbus_wait_until_done(base) < 0) {
127ca6c5e03SStefan Roese 		printf("SMBUS read transaction timeout (dev=0x%x)\n", dev);
128ca6c5e03SStefan Roese 		return -ETIMEDOUT;
129ca6c5e03SStefan Roese 	}
130ca6c5e03SStefan Roese 
131ca6c5e03SStefan Roese 	count = inb(base + SMBHSTDAT0);
132ca6c5e03SStefan Roese 	debug("%s (%d): count=%d (len=%d)\n", __func__, __LINE__, count, len);
133ca6c5e03SStefan Roese 	if (count == 0) {
134ca6c5e03SStefan Roese 		debug("ERROR: len=0 on read\n");
135ca6c5e03SStefan Roese 		return -EIO;
136ca6c5e03SStefan Roese 	}
137ca6c5e03SStefan Roese 
138ca6c5e03SStefan Roese 	if (count < len) {
139ca6c5e03SStefan Roese 		debug("ERROR: too few bytes read\n");
140ca6c5e03SStefan Roese 		return -EIO;
141ca6c5e03SStefan Roese 	}
142ca6c5e03SStefan Roese 
143ca6c5e03SStefan Roese 	if (count > 32) {
144ca6c5e03SStefan Roese 		debug("ERROR: count=%d too high\n", count);
145ca6c5e03SStefan Roese 		return -EIO;
146ca6c5e03SStefan Roese 	}
147ca6c5e03SStefan Roese 
148ca6c5e03SStefan Roese 	/* Read all available bytes from buffer */
149ca6c5e03SStefan Roese 	for (i = 0; i < count; i++)
150ca6c5e03SStefan Roese 		buf_temp[i] = inb(base + SMBBLKDAT);
151ca6c5e03SStefan Roese 
152ca6c5e03SStefan Roese 	memcpy(buffer, buf_temp, len);
153ca6c5e03SStefan Roese 
154ca6c5e03SStefan Roese 	/* Return results of transaction */
155ca6c5e03SStefan Roese 	if (!(inb(base + SMBHSTSTAT) & SMBHSTSTS_INTR))
156ca6c5e03SStefan Roese 		return -EIO;
157ca6c5e03SStefan Roese 
158ca6c5e03SStefan Roese 	return 0;
159ca6c5e03SStefan Roese }
160ca6c5e03SStefan Roese 
smbus_block_write(u32 base,u8 dev,u8 * buffer,int offset,int len)161ca6c5e03SStefan Roese static int smbus_block_write(u32 base, u8 dev, u8 *buffer,
162ca6c5e03SStefan Roese 			     int offset, int len)
163ca6c5e03SStefan Roese {
164ca6c5e03SStefan Roese 	int i;
165ca6c5e03SStefan Roese 
166ca6c5e03SStefan Roese 	debug("%s (%d): dev=0x%x offs=0x%x len=0x%x\n",
167ca6c5e03SStefan Roese 	      __func__, __LINE__, dev, offset, len);
168ca6c5e03SStefan Roese 	if (smbus_wait_until_ready(base) < 0)
169ca6c5e03SStefan Roese 		return -ETIMEDOUT;
170ca6c5e03SStefan Roese 
171ca6c5e03SStefan Roese 	/* Setup transaction */
172ca6c5e03SStefan Roese 	/* Set the device I'm talking too */
173ca6c5e03SStefan Roese 	outb(((dev & 0x7f) << 1) & ~0x01, base + SMBXMITADD);
174ca6c5e03SStefan Roese 	/* Set the command/address... */
175ca6c5e03SStefan Roese 	outb(offset, base + SMBHSTCMD);
176ca6c5e03SStefan Roese 	/* Set up for a block write */
177ca6c5e03SStefan Roese 	outb((inb(base + SMBHSTCTL) & (~(0x7) << 2)) | (0x5 << 2),
178ca6c5e03SStefan Roese 	     (base + SMBHSTCTL));
179ca6c5e03SStefan Roese 	/* Clear any lingering errors, so the transaction will run */
180ca6c5e03SStefan Roese 	outb(inb(base + SMBHSTSTAT), base + SMBHSTSTAT);
181ca6c5e03SStefan Roese 
182ca6c5e03SStefan Roese 	/* Write count in DAT0 register */
183ca6c5e03SStefan Roese 	outb(len, base + SMBHSTDAT0);
184ca6c5e03SStefan Roese 
185ca6c5e03SStefan Roese 	/* Write data bytes... */
186ca6c5e03SStefan Roese 	for (i = 0; i < len; i++)
187ca6c5e03SStefan Roese 		outb(*buffer++, base + SMBBLKDAT);
188ca6c5e03SStefan Roese 
189ca6c5e03SStefan Roese 	/* Start the command */
190ca6c5e03SStefan Roese 	outb((inb(base + SMBHSTCTL) | SMBHSTCNT_START), base + SMBHSTCTL);
191ca6c5e03SStefan Roese 
192ca6c5e03SStefan Roese 	/* Poll for transaction completion */
193ca6c5e03SStefan Roese 	if (smbus_wait_until_done(base) < 0) {
194ca6c5e03SStefan Roese 		printf("SMBUS write transaction timeout (dev=0x%x)\n", dev);
195ca6c5e03SStefan Roese 		return -ETIMEDOUT;
196ca6c5e03SStefan Roese 	}
197ca6c5e03SStefan Roese 
198ca6c5e03SStefan Roese 	/* Return results of transaction */
199ca6c5e03SStefan Roese 	if (!(inb(base + SMBHSTSTAT) & SMBHSTSTS_INTR))
200ca6c5e03SStefan Roese 		return -EIO;
201ca6c5e03SStefan Roese 
202ca6c5e03SStefan Roese 	return 0;
203ca6c5e03SStefan Roese }
204ca6c5e03SStefan Roese 
intel_i2c_xfer(struct udevice * bus,struct i2c_msg * msg,int nmsgs)205ca6c5e03SStefan Roese static int intel_i2c_xfer(struct udevice *bus, struct i2c_msg *msg, int nmsgs)
206ca6c5e03SStefan Roese {
207ca6c5e03SStefan Roese 	struct intel_i2c *i2c = dev_get_priv(bus);
208ca6c5e03SStefan Roese 	struct i2c_msg *dmsg, *omsg, dummy;
209ca6c5e03SStefan Roese 
210ca6c5e03SStefan Roese 	debug("i2c_xfer: %d messages\n", nmsgs);
211ca6c5e03SStefan Roese 
212ca6c5e03SStefan Roese 	memset(&dummy, 0, sizeof(struct i2c_msg));
213ca6c5e03SStefan Roese 
214ca6c5e03SStefan Roese 	/*
215ca6c5e03SStefan Roese 	 * We expect either two messages (one with an offset and one with the
216ca6c5e03SStefan Roese 	 * actucal data) or one message (just data)
217ca6c5e03SStefan Roese 	 */
218ca6c5e03SStefan Roese 	if (nmsgs > 2 || nmsgs == 0) {
219ca6c5e03SStefan Roese 		debug("%s: Only one or two messages are supported", __func__);
220ca6c5e03SStefan Roese 		return -EIO;
221ca6c5e03SStefan Roese 	}
222ca6c5e03SStefan Roese 
223ca6c5e03SStefan Roese 	omsg = nmsgs == 1 ? &dummy : msg;
224ca6c5e03SStefan Roese 	dmsg = nmsgs == 1 ? msg : msg + 1;
225ca6c5e03SStefan Roese 
226ca6c5e03SStefan Roese 	if (dmsg->flags & I2C_M_RD)
227ca6c5e03SStefan Roese 		return smbus_block_read(i2c->base, dmsg->addr, &dmsg->buf[0],
228ca6c5e03SStefan Roese 					omsg->buf[0], dmsg->len);
229ca6c5e03SStefan Roese 	else
230ca6c5e03SStefan Roese 		return smbus_block_write(i2c->base, dmsg->addr, &dmsg->buf[1],
231ca6c5e03SStefan Roese 					 dmsg->buf[0], dmsg->len - 1);
232ca6c5e03SStefan Roese }
233ca6c5e03SStefan Roese 
intel_i2c_probe_chip(struct udevice * bus,uint chip_addr,uint chip_flags)234ca6c5e03SStefan Roese static int intel_i2c_probe_chip(struct udevice *bus, uint chip_addr,
235ca6c5e03SStefan Roese 				uint chip_flags)
236ca6c5e03SStefan Roese {
237ca6c5e03SStefan Roese 	struct intel_i2c *i2c = dev_get_priv(bus);
238ca6c5e03SStefan Roese 	u8 buf[4];
239ca6c5e03SStefan Roese 
240ca6c5e03SStefan Roese 	return smbus_block_read(i2c->base, chip_addr, buf, 0, 1);
241ca6c5e03SStefan Roese }
242ca6c5e03SStefan Roese 
intel_i2c_set_bus_speed(struct udevice * bus,unsigned int speed)243ca6c5e03SStefan Roese static int intel_i2c_set_bus_speed(struct udevice *bus, unsigned int speed)
244abb0b01eSSimon Glass {
245abb0b01eSSimon Glass 	return 0;
246abb0b01eSSimon Glass }
247abb0b01eSSimon Glass 
intel_i2c_probe(struct udevice * dev)248abb0b01eSSimon Glass static int intel_i2c_probe(struct udevice *dev)
249abb0b01eSSimon Glass {
250ca6c5e03SStefan Roese 	struct intel_i2c *priv = dev_get_priv(dev);
251*fac3e796SSimon Glass 	ulong base;
252ca6c5e03SStefan Roese 
253ca6c5e03SStefan Roese 	/* Save base address from PCI BAR */
254*fac3e796SSimon Glass 	priv->base = (ulong)dm_pci_map_bar(dev, PCI_BASE_ADDRESS_4,
255ca6c5e03SStefan Roese 					   PCI_REGION_IO);
256ca6c5e03SStefan Roese 	base = priv->base;
2570c7645bdSSimon Glass 
2580c7645bdSSimon Glass 	/* Set SMBus enable. */
2590c7645bdSSimon Glass 	dm_pci_write_config8(dev, HOSTC, HST_EN);
2600c7645bdSSimon Glass 
261ca6c5e03SStefan Roese 	/* Disable interrupts */
262ca6c5e03SStefan Roese 	outb(inb(base + SMBHSTCTL) & ~SMBHSTCNT_INTREN, base + SMBHSTCTL);
2630c7645bdSSimon Glass 
264ca6c5e03SStefan Roese 	/* Set 32-byte data buffer mode */
265ca6c5e03SStefan Roese 	outb(inb(base + SMBAUXCTL) | SMBAUXCTL_E32B, base + SMBAUXCTL);
2660c7645bdSSimon Glass 
267ca6c5e03SStefan Roese 	return 0;
268ca6c5e03SStefan Roese }
269ca6c5e03SStefan Roese 
intel_i2c_bind(struct udevice * dev)270ca6c5e03SStefan Roese static int intel_i2c_bind(struct udevice *dev)
271ca6c5e03SStefan Roese {
272ca6c5e03SStefan Roese 	static int num_cards __attribute__ ((section(".data")));
273ca6c5e03SStefan Roese 	char name[20];
274ca6c5e03SStefan Roese 
275ca6c5e03SStefan Roese 	/* Create a unique device name for PCI type devices */
276ca6c5e03SStefan Roese 	if (device_is_on_pci_bus(dev)) {
277ca6c5e03SStefan Roese 		/*
278ca6c5e03SStefan Roese 		 * ToDo:
279ca6c5e03SStefan Roese 		 * Setting req_seq in the driver is probably not recommended.
280ca6c5e03SStefan Roese 		 * But without a DT alias the number is not configured. And
281ca6c5e03SStefan Roese 		 * using this driver is impossible for PCIe I2C devices.
282ca6c5e03SStefan Roese 		 * This can be removed, once a better (correct) way for this
283ca6c5e03SStefan Roese 		 * is found and implemented.
284ca6c5e03SStefan Roese 		 */
285ca6c5e03SStefan Roese 		dev->req_seq = num_cards;
286ca6c5e03SStefan Roese 		sprintf(name, "intel_i2c#%u", num_cards++);
287ca6c5e03SStefan Roese 		device_set_name(dev, name);
288ca6c5e03SStefan Roese 	}
2890c7645bdSSimon Glass 
290abb0b01eSSimon Glass 	return 0;
291abb0b01eSSimon Glass }
292abb0b01eSSimon Glass 
293abb0b01eSSimon Glass static const struct dm_i2c_ops intel_i2c_ops = {
294abb0b01eSSimon Glass 	.xfer		= intel_i2c_xfer,
295abb0b01eSSimon Glass 	.probe_chip	= intel_i2c_probe_chip,
296abb0b01eSSimon Glass 	.set_bus_speed	= intel_i2c_set_bus_speed,
297abb0b01eSSimon Glass };
298abb0b01eSSimon Glass 
299abb0b01eSSimon Glass static const struct udevice_id intel_i2c_ids[] = {
300abb0b01eSSimon Glass 	{ .compatible = "intel,ich-i2c" },
301abb0b01eSSimon Glass 	{ }
302abb0b01eSSimon Glass };
303abb0b01eSSimon Glass 
304abb0b01eSSimon Glass U_BOOT_DRIVER(intel_i2c) = {
305abb0b01eSSimon Glass 	.name	= "i2c_intel",
306abb0b01eSSimon Glass 	.id	= UCLASS_I2C,
307abb0b01eSSimon Glass 	.of_match = intel_i2c_ids,
308abb0b01eSSimon Glass 	.ops	= &intel_i2c_ops,
309ca6c5e03SStefan Roese 	.priv_auto_alloc_size = sizeof(struct intel_i2c),
310ca6c5e03SStefan Roese 	.bind	= intel_i2c_bind,
311abb0b01eSSimon Glass 	.probe	= intel_i2c_probe,
312abb0b01eSSimon Glass };
313ca6c5e03SStefan Roese 
314ca6c5e03SStefan Roese static struct pci_device_id intel_smbus_pci_supported[] = {
315ca6c5e03SStefan Roese 	/* Intel BayTrail SMBus on the PCI bus */
316ca6c5e03SStefan Roese 	{ PCI_VDEVICE(INTEL, 0x0f12) },
317ca6c5e03SStefan Roese 	/* Intel IvyBridge (Panther Point PCH) SMBus on the PCI bus */
318ca6c5e03SStefan Roese 	{ PCI_VDEVICE(INTEL, 0x1e22) },
319ca6c5e03SStefan Roese 	{},
320ca6c5e03SStefan Roese };
321ca6c5e03SStefan Roese 
322ca6c5e03SStefan Roese U_BOOT_PCI_DEVICE(intel_i2c, intel_smbus_pci_supported);
323