Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 1 | /* |
| 2 | * This file is part of the coreboot project. |
| 3 | * |
| 4 | * Copyright (C) 2014 Siemens AG |
| 5 | * |
| 6 | * This program is free software; you can redistribute it and/or modify |
| 7 | * it under the terms of the GNU General Public License as published by |
| 8 | * the Free Software Foundation; version 2 of the License. |
| 9 | * |
| 10 | * This program is distributed in the hope that it will be useful, |
| 11 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 12 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 13 | * GNU General Public License for more details. |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 14 | */ |
| 15 | |
| 16 | #include <device/pci.h> |
Ben Gardner | fa6014a | 2015-12-08 21:20:25 -0600 | [diff] [blame] | 17 | #include <soc/baytrail.h> |
| 18 | #include <soc/pci_devs.h> |
| 19 | #include <soc/iosf.h> |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 20 | #include <delay.h> |
Ben Gardner | fa6014a | 2015-12-08 21:20:25 -0600 | [diff] [blame] | 21 | #include <soc/i2c.h> |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 22 | |
| 23 | /* Wait for the transmit FIFO till there is at least one slot empty. |
| 24 | * FIFO stall due to transmit abort will be checked and resolved |
| 25 | */ |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 26 | static int wait_tx_fifo(char *base_adr) |
| 27 | { |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 28 | int i; |
Ben Gardner | 77e351d | 2016-03-23 09:40:37 -0500 | [diff] [blame] | 29 | u32 as; |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 30 | |
Ben Gardner | 77e351d | 2016-03-23 09:40:37 -0500 | [diff] [blame] | 31 | as = read32(base_adr + I2C_ABORT_SOURCE) & 0x1ffff; |
| 32 | if (as) { |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 33 | /* Reading back I2C_CLR_TX_ABRT resets abort lock on TX FIFO */ |
Ben Gardner | 77e351d | 2016-03-23 09:40:37 -0500 | [diff] [blame] | 34 | i = read32(base_adr + I2C_CLR_TX_ABRT); |
| 35 | return I2C_ERR_ABORT | as; |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 36 | } |
| 37 | |
| 38 | /* Wait here for a free slot in TX-FIFO */ |
| 39 | i = I2C_TIMEOUT_US; |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 40 | while (!(read32(base_adr + I2C_STATUS) & I2C_TFNF)) { |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 41 | udelay(1); |
| 42 | if (!--i) |
| 43 | return I2C_ERR_TIMEOUT; |
| 44 | } |
| 45 | |
| 46 | return I2C_SUCCESS; |
| 47 | } |
| 48 | |
| 49 | /* Wait for the receive FIFO till there is at least one valid entry to read. |
| 50 | * FIFO stall due to transmit abort will be checked and resolved |
| 51 | */ |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 52 | static int wait_rx_fifo(char *base_adr) |
| 53 | { |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 54 | int i; |
Ben Gardner | 77e351d | 2016-03-23 09:40:37 -0500 | [diff] [blame] | 55 | u32 as; |
| 56 | |
| 57 | as = read32(base_adr + I2C_ABORT_SOURCE) & 0x1ffff; |
| 58 | if (as) { |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 59 | /* Reading back I2C_CLR_TX_ABRT resets abort lock on TX FIFO */ |
Ben Gardner | 77e351d | 2016-03-23 09:40:37 -0500 | [diff] [blame] | 60 | i = read32(base_adr + I2C_CLR_TX_ABRT); |
| 61 | return I2C_ERR_ABORT | as; |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 62 | } |
| 63 | |
| 64 | /* Wait here for a received entry in RX-FIFO */ |
| 65 | i = I2C_TIMEOUT_US; |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 66 | while (!(read32(base_adr + I2C_STATUS) & I2C_RFNE)) { |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 67 | udelay(1); |
| 68 | if (!--i) |
| 69 | return I2C_ERR_TIMEOUT; |
| 70 | } |
| 71 | |
| 72 | return I2C_SUCCESS; |
| 73 | } |
| 74 | |
| 75 | /* When there will be a fast switch between send and receive, one have |
| 76 | * to wait until the first operation is completely finished |
| 77 | * before starting the second operation |
| 78 | */ |
| 79 | static int wait_for_idle(char *base_adr) |
| 80 | { |
| 81 | int i; |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 82 | int status; |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 83 | |
| 84 | /* For IDLE, increase timeout by ten times */ |
| 85 | i = I2C_TIMEOUT_US * 10; |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 86 | status = read32(base_adr + I2C_STATUS); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 87 | while (((status & I2C_MST_ACTIVITY) || (!(status & I2C_TFE)))) { |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 88 | status = read32(base_adr + I2C_STATUS); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 89 | udelay(1); |
| 90 | if (!--i) |
| 91 | return I2C_ERR_TIMEOUT; |
| 92 | } |
| 93 | |
| 94 | return I2C_SUCCESS; |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 95 | } |
| 96 | |
| 97 | /** \brief Enables I2C-controller, sets up BAR and timing parameters |
| 98 | * @param bus Number of the I2C-controller to use (0...6) |
| 99 | * @return I2C_SUCCESS on success, otherwise error code |
| 100 | */ |
| 101 | int i2c_init(unsigned bus) |
| 102 | { |
Elyes HAOUAS | 509edac | 2018-05-27 17:48:32 +0200 | [diff] [blame] | 103 | struct device *dev; |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 104 | int base_adr[7] = {I2C0_MEM_BASE, I2C1_MEM_BASE, I2C2_MEM_BASE, |
| 105 | I2C3_MEM_BASE, I2C4_MEM_BASE, I2C5_MEM_BASE, |
| 106 | I2C6_MEM_BASE}; |
| 107 | char *base_ptr; |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 108 | |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 109 | /* Ensure the desired device is valid */ |
Martin Roth | c890996 | 2015-12-21 13:14:58 -0700 | [diff] [blame] | 110 | if (bus >= ARRAY_SIZE(base_adr)) { |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 111 | printk(BIOS_ERR, "I2C: Only I2C controllers 0...6 are available.\n"); |
| 112 | return I2C_ERR; |
| 113 | } |
| 114 | |
| 115 | base_ptr = (char*)base_adr[bus]; |
| 116 | /* Set the I2C-device the user wants to use */ |
Kyösti Mälkki | c70eed1 | 2018-05-22 02:18:00 +0300 | [diff] [blame] | 117 | dev = pcidev_on_root(I2C1_DEV, bus + 1); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 118 | |
| 119 | /* Ensure we have the right PCI device */ |
| 120 | if ((pci_read_config16(dev, 0x0) != I2C_PCI_VENDOR_ID) || |
| 121 | (pci_read_config16(dev, 0x2) != (I2C0_PCI_DEV_ID + bus))) { |
| 122 | printk(BIOS_ERR, "I2C: Controller %d not found!\n", bus); |
| 123 | return I2C_ERR; |
| 124 | } |
| 125 | |
| 126 | /* Set memory base */ |
| 127 | pci_write_config32(dev, PCI_BASE_ADDRESS_0, (int)base_ptr); |
| 128 | |
| 129 | /* Enable memory space */ |
| 130 | pci_write_config32(dev, PCI_COMMAND, |
| 131 | (pci_read_config32(dev, PCI_COMMAND) | 0x2)); |
| 132 | |
| 133 | /* Set up some settings of I2C controller */ |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 134 | write32(base_ptr + I2C_CTRL, |
| 135 | I2C_RESTART_EN | (I2C_STANDARD_MODE << 1) | I2C_MASTER_ENABLE); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 136 | /* Adjust frequency for standard mode to 100 kHz */ |
| 137 | /* The counter value can be computed by N=100MHz/2/I2C_CLK */ |
| 138 | /* Thus, for 100 kHz I2C_CLK, N is 0x1F4 */ |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 139 | write32(base_ptr + I2C_SS_SCL_HCNT, 0x1f4); |
| 140 | write32(base_ptr + I2C_SS_SCL_LCNT, 0x1f4); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 141 | /* For 400 kHz, the counter value is 0x7d */ |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 142 | write32(base_ptr + I2C_FS_SCL_HCNT, 0x7d); |
| 143 | write32(base_ptr + I2C_FS_SCL_LCNT, 0x7d); |
Ben Gardner | 44bb9bd | 2016-06-01 09:25:28 -0500 | [diff] [blame] | 144 | /* no interrupts in BIOS */ |
| 145 | write32(base_ptr + I2C_INTR_MASK, 0); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 146 | |
| 147 | /* Enable the I2C controller for operation */ |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 148 | write32(base_ptr + I2C_ENABLE, 0x1); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 149 | |
| 150 | printk(BIOS_INFO, "I2C: Controller %d enabled.\n", bus); |
| 151 | return I2C_SUCCESS; |
| 152 | } |
| 153 | |
| 154 | /** \brief Read bytes over I2C-Bus from a slave. This function tries only one |
| 155 | * time to transmit data. In case of an error (abort) error code is |
| 156 | * returned. Retransmission has to be done from caller! |
| 157 | * @param bus Number of the I2C-controller to use (0...6) |
| 158 | * @param chip 7 Bit of the slave address on I2C bus |
| 159 | * @param addr Address inside slave where to read from |
| 160 | * @param *buf Pointer to the buffer where to store read data |
| 161 | * @param len Number of bytes to read |
| 162 | * @return I2C_SUCCESS when read was successful, otherwise error code |
| 163 | */ |
| 164 | int i2c_read(unsigned bus, unsigned chip, unsigned addr, |
| 165 | uint8_t *buf, unsigned len) |
| 166 | { |
| 167 | int i = 0; |
| 168 | char *base_ptr = NULL; |
Elyes HAOUAS | 509edac | 2018-05-27 17:48:32 +0200 | [diff] [blame] | 169 | struct device *dev; |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 170 | unsigned int val; |
| 171 | int stat; |
| 172 | |
| 173 | /* Get base address of desired I2C-controller */ |
Kyösti Mälkki | c70eed1 | 2018-05-22 02:18:00 +0300 | [diff] [blame] | 174 | dev = pcidev_on_root(I2C1_DEV, bus + 1); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 175 | base_ptr = (char *)pci_read_config32(dev, PCI_BASE_ADDRESS_0); |
| 176 | if (base_ptr == NULL) { |
| 177 | printk(BIOS_INFO, "I2C: Invalid Base address\n"); |
| 178 | return I2C_ERR_INVALID_ADR; |
| 179 | } |
| 180 | |
| 181 | /* Ensure I2C controller is not active before setting slave address */ |
| 182 | stat = wait_for_idle(base_ptr); |
| 183 | if (stat != I2C_SUCCESS) |
| 184 | return stat; |
Ben Gardner | 77e351d | 2016-03-23 09:40:37 -0500 | [diff] [blame] | 185 | |
| 186 | /* clear any abort status from a previous transaction */ |
| 187 | read32(base_ptr + I2C_CLR_TX_ABRT); |
| 188 | |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 189 | /* Now we can program the desired slave address and start transfer */ |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 190 | write32(base_ptr + I2C_TARGET_ADR, chip & 0xff); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 191 | /* Send address inside slave to read from */ |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 192 | write32(base_ptr + I2C_DATA_CMD, addr & 0xff); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 193 | |
| 194 | /* For the next byte we need a repeated start condition */ |
| 195 | val = I2C_RW_CMD | I2C_RESTART; |
| 196 | /* Now we can read desired amount of data over I2C */ |
| 197 | for (i = 0; i < len; i++) { |
| 198 | /* A read is initiated by writing dummy data to the DATA-register */ |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 199 | write32(base_ptr + I2C_DATA_CMD, val); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 200 | stat = wait_rx_fifo(base_ptr); |
| 201 | if (stat) |
| 202 | return stat; |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 203 | buf[i] = read32(base_ptr + I2C_DATA_CMD) & 0xff; |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 204 | val = I2C_RW_CMD; |
| 205 | if (i == (len - 2)) { |
| 206 | /* For the last byte we need a stop condition to be generated */ |
| 207 | val |= I2C_STOP; |
| 208 | } |
| 209 | } |
| 210 | return I2C_SUCCESS; |
| 211 | } |
| 212 | |
| 213 | /** \brief Write bytes over I2C-Bus from a slave. This function tries only one |
| 214 | * time to transmit data. In case of an error (abort) error code is |
| 215 | * returned. Retransmission has to be done from caller! |
| 216 | * @param bus Number of the I2C-controller to use (0...6) |
| 217 | * @param chip 7 Bit of the slave address on I2C bus |
| 218 | * @param addr Address inside slave where to write to |
| 219 | * @param *buf Pointer to the buffer where data to write is stored |
| 220 | * @param len Number of bytes to write |
| 221 | * @return I2C_SUCCESS when read was successful, otherwise error code |
| 222 | */ |
| 223 | int i2c_write(unsigned bus, unsigned chip, unsigned addr, |
| 224 | const uint8_t *buf, unsigned len) |
| 225 | { |
| 226 | int i; |
| 227 | char *base_ptr; |
Elyes HAOUAS | 509edac | 2018-05-27 17:48:32 +0200 | [diff] [blame] | 228 | struct device *dev; |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 229 | unsigned int val; |
| 230 | int stat; |
| 231 | |
| 232 | /* Get base address of desired I2C-controller */ |
Kyösti Mälkki | c70eed1 | 2018-05-22 02:18:00 +0300 | [diff] [blame] | 233 | dev = pcidev_on_root(I2C1_DEV, bus + 1); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 234 | base_ptr = (char *)pci_read_config32(dev, PCI_BASE_ADDRESS_0); |
| 235 | if (base_ptr == NULL) { |
| 236 | return I2C_ERR_INVALID_ADR; |
| 237 | } |
| 238 | |
| 239 | /* Ensure I2C controller is not active yet */ |
| 240 | stat = wait_for_idle(base_ptr); |
| 241 | if (stat) { |
| 242 | return stat; |
| 243 | } |
Ben Gardner | 77e351d | 2016-03-23 09:40:37 -0500 | [diff] [blame] | 244 | |
| 245 | /* clear any abort status from a previous transaction */ |
| 246 | read32(base_ptr + I2C_CLR_TX_ABRT); |
| 247 | |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 248 | /* Program slave address to use for this transfer */ |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 249 | write32(base_ptr + I2C_TARGET_ADR, chip & 0xff); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 250 | |
| 251 | /* Send address inside slave to write data to */ |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 252 | write32(base_ptr + I2C_DATA_CMD, addr & 0xff); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 253 | |
| 254 | for (i = 0; i < len; i++) { |
| 255 | val = (unsigned int)(buf[i] & 0xff); /* Take only 8 bits */ |
| 256 | if (i == (len - 1)) { |
| 257 | /* For the last byte we need a stop condition */ |
| 258 | val |= I2C_STOP; |
| 259 | } |
| 260 | stat = wait_tx_fifo(base_ptr); |
| 261 | if (stat) { |
| 262 | return stat; |
| 263 | } |
Ben Gardner | 5aecd0e | 2016-03-23 10:11:24 -0500 | [diff] [blame] | 264 | write32(base_ptr + I2C_DATA_CMD, val); |
Werner Zeh | 0f9c9de | 2015-02-10 13:02:34 +0100 | [diff] [blame] | 265 | } |
| 266 | return I2C_SUCCESS; |
| 267 | } |