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