|  | /* | 
|  | * Copyright (C) 2014 Broadcom Corporation | 
|  | * | 
|  | * This program is free software; you can redistribute it and/or | 
|  | * modify it under the terms of the GNU General Public License as | 
|  | * published by the Free Software Foundation version 2. | 
|  | * | 
|  | * This program is distributed "as is" WITHOUT ANY WARRANTY of any | 
|  | * kind, whether express or implied; without even the implied warranty | 
|  | * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | * GNU General Public License for more details. | 
|  | */ | 
|  |  | 
|  | #include <linux/delay.h> | 
|  | #include <linux/i2c.h> | 
|  | #include <linux/interrupt.h> | 
|  | #include <linux/io.h> | 
|  | #include <linux/kernel.h> | 
|  | #include <linux/module.h> | 
|  | #include <linux/platform_device.h> | 
|  | #include <linux/slab.h> | 
|  |  | 
|  | #define CFG_OFFSET                   0x00 | 
|  | #define CFG_RESET_SHIFT              31 | 
|  | #define CFG_EN_SHIFT                 30 | 
|  | #define CFG_M_RETRY_CNT_SHIFT        16 | 
|  | #define CFG_M_RETRY_CNT_MASK         0x0f | 
|  |  | 
|  | #define TIM_CFG_OFFSET               0x04 | 
|  | #define TIM_CFG_MODE_400_SHIFT       31 | 
|  |  | 
|  | #define M_FIFO_CTRL_OFFSET           0x0c | 
|  | #define M_FIFO_RX_FLUSH_SHIFT        31 | 
|  | #define M_FIFO_TX_FLUSH_SHIFT        30 | 
|  | #define M_FIFO_RX_CNT_SHIFT          16 | 
|  | #define M_FIFO_RX_CNT_MASK           0x7f | 
|  | #define M_FIFO_RX_THLD_SHIFT         8 | 
|  | #define M_FIFO_RX_THLD_MASK          0x3f | 
|  |  | 
|  | #define M_CMD_OFFSET                 0x30 | 
|  | #define M_CMD_START_BUSY_SHIFT       31 | 
|  | #define M_CMD_STATUS_SHIFT           25 | 
|  | #define M_CMD_STATUS_MASK            0x07 | 
|  | #define M_CMD_STATUS_SUCCESS         0x0 | 
|  | #define M_CMD_STATUS_LOST_ARB        0x1 | 
|  | #define M_CMD_STATUS_NACK_ADDR       0x2 | 
|  | #define M_CMD_STATUS_NACK_DATA       0x3 | 
|  | #define M_CMD_STATUS_TIMEOUT         0x4 | 
|  | #define M_CMD_PROTOCOL_SHIFT         9 | 
|  | #define M_CMD_PROTOCOL_MASK          0xf | 
|  | #define M_CMD_PROTOCOL_BLK_WR        0x7 | 
|  | #define M_CMD_PROTOCOL_BLK_RD        0x8 | 
|  | #define M_CMD_PEC_SHIFT              8 | 
|  | #define M_CMD_RD_CNT_SHIFT           0 | 
|  | #define M_CMD_RD_CNT_MASK            0xff | 
|  |  | 
|  | #define IE_OFFSET                    0x38 | 
|  | #define IE_M_RX_FIFO_FULL_SHIFT      31 | 
|  | #define IE_M_RX_THLD_SHIFT           30 | 
|  | #define IE_M_START_BUSY_SHIFT        28 | 
|  | #define IE_M_TX_UNDERRUN_SHIFT       27 | 
|  |  | 
|  | #define IS_OFFSET                    0x3c | 
|  | #define IS_M_RX_FIFO_FULL_SHIFT      31 | 
|  | #define IS_M_RX_THLD_SHIFT           30 | 
|  | #define IS_M_START_BUSY_SHIFT        28 | 
|  | #define IS_M_TX_UNDERRUN_SHIFT       27 | 
|  |  | 
|  | #define M_TX_OFFSET                  0x40 | 
|  | #define M_TX_WR_STATUS_SHIFT         31 | 
|  | #define M_TX_DATA_SHIFT              0 | 
|  | #define M_TX_DATA_MASK               0xff | 
|  |  | 
|  | #define M_RX_OFFSET                  0x44 | 
|  | #define M_RX_STATUS_SHIFT            30 | 
|  | #define M_RX_STATUS_MASK             0x03 | 
|  | #define M_RX_PEC_ERR_SHIFT           29 | 
|  | #define M_RX_DATA_SHIFT              0 | 
|  | #define M_RX_DATA_MASK               0xff | 
|  |  | 
|  | #define I2C_TIMEOUT_MSEC             50000 | 
|  | #define M_TX_RX_FIFO_SIZE            64 | 
|  |  | 
|  | enum bus_speed_index { | 
|  | I2C_SPD_100K = 0, | 
|  | I2C_SPD_400K, | 
|  | }; | 
|  |  | 
|  | struct bcm_iproc_i2c_dev { | 
|  | struct device *device; | 
|  | int irq; | 
|  |  | 
|  | void __iomem *base; | 
|  |  | 
|  | struct i2c_adapter adapter; | 
|  | unsigned int bus_speed; | 
|  |  | 
|  | struct completion done; | 
|  | int xfer_is_done; | 
|  |  | 
|  | struct i2c_msg *msg; | 
|  |  | 
|  | /* bytes that have been transferred */ | 
|  | unsigned int tx_bytes; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * Can be expanded in the future if more interrupt status bits are utilized | 
|  | */ | 
|  | #define ISR_MASK (BIT(IS_M_START_BUSY_SHIFT) | BIT(IS_M_TX_UNDERRUN_SHIFT)) | 
|  |  | 
|  | static irqreturn_t bcm_iproc_i2c_isr(int irq, void *data) | 
|  | { | 
|  | struct bcm_iproc_i2c_dev *iproc_i2c = data; | 
|  | u32 status = readl(iproc_i2c->base + IS_OFFSET); | 
|  |  | 
|  | status &= ISR_MASK; | 
|  |  | 
|  | if (!status) | 
|  | return IRQ_NONE; | 
|  |  | 
|  | /* TX FIFO is empty and we have more data to send */ | 
|  | if (status & BIT(IS_M_TX_UNDERRUN_SHIFT)) { | 
|  | struct i2c_msg *msg = iproc_i2c->msg; | 
|  | unsigned int tx_bytes = msg->len - iproc_i2c->tx_bytes; | 
|  | unsigned int i; | 
|  | u32 val; | 
|  |  | 
|  | /* can only fill up to the FIFO size */ | 
|  | tx_bytes = min_t(unsigned int, tx_bytes, M_TX_RX_FIFO_SIZE); | 
|  | for (i = 0; i < tx_bytes; i++) { | 
|  | /* start from where we left over */ | 
|  | unsigned int idx = iproc_i2c->tx_bytes + i; | 
|  |  | 
|  | val = msg->buf[idx]; | 
|  |  | 
|  | /* mark the last byte */ | 
|  | if (idx == msg->len - 1) { | 
|  | u32 tmp; | 
|  |  | 
|  | val |= BIT(M_TX_WR_STATUS_SHIFT); | 
|  |  | 
|  | /* | 
|  | * Since this is the last byte, we should | 
|  | * now disable TX FIFO underrun interrupt | 
|  | */ | 
|  | tmp = readl(iproc_i2c->base + IE_OFFSET); | 
|  | tmp &= ~BIT(IE_M_TX_UNDERRUN_SHIFT); | 
|  | writel(tmp, iproc_i2c->base + IE_OFFSET); | 
|  | } | 
|  |  | 
|  | /* load data into TX FIFO */ | 
|  | writel(val, iproc_i2c->base + M_TX_OFFSET); | 
|  | } | 
|  | /* update number of transferred bytes */ | 
|  | iproc_i2c->tx_bytes += tx_bytes; | 
|  | } | 
|  |  | 
|  | if (status & BIT(IS_M_START_BUSY_SHIFT)) { | 
|  | iproc_i2c->xfer_is_done = 1; | 
|  | complete(&iproc_i2c->done); | 
|  | } | 
|  |  | 
|  | writel(status, iproc_i2c->base + IS_OFFSET); | 
|  |  | 
|  | return IRQ_HANDLED; | 
|  | } | 
|  |  | 
|  | static int bcm_iproc_i2c_init(struct bcm_iproc_i2c_dev *iproc_i2c) | 
|  | { | 
|  | u32 val; | 
|  |  | 
|  | /* put controller in reset */ | 
|  | val = readl(iproc_i2c->base + CFG_OFFSET); | 
|  | val |= 1 << CFG_RESET_SHIFT; | 
|  | val &= ~(1 << CFG_EN_SHIFT); | 
|  | writel(val, iproc_i2c->base + CFG_OFFSET); | 
|  |  | 
|  | /* wait 100 usec per spec */ | 
|  | udelay(100); | 
|  |  | 
|  | /* bring controller out of reset */ | 
|  | val &= ~(1 << CFG_RESET_SHIFT); | 
|  | writel(val, iproc_i2c->base + CFG_OFFSET); | 
|  |  | 
|  | /* flush TX/RX FIFOs and set RX FIFO threshold to zero */ | 
|  | val = (1 << M_FIFO_RX_FLUSH_SHIFT) | (1 << M_FIFO_TX_FLUSH_SHIFT); | 
|  | writel(val, iproc_i2c->base + M_FIFO_CTRL_OFFSET); | 
|  | /* disable all interrupts */ | 
|  | writel(0, iproc_i2c->base + IE_OFFSET); | 
|  |  | 
|  | /* clear all pending interrupts */ | 
|  | writel(0xffffffff, iproc_i2c->base + IS_OFFSET); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void bcm_iproc_i2c_enable_disable(struct bcm_iproc_i2c_dev *iproc_i2c, | 
|  | bool enable) | 
|  | { | 
|  | u32 val; | 
|  |  | 
|  | val = readl(iproc_i2c->base + CFG_OFFSET); | 
|  | if (enable) | 
|  | val |= BIT(CFG_EN_SHIFT); | 
|  | else | 
|  | val &= ~BIT(CFG_EN_SHIFT); | 
|  | writel(val, iproc_i2c->base + CFG_OFFSET); | 
|  | } | 
|  |  | 
|  | static int bcm_iproc_i2c_check_status(struct bcm_iproc_i2c_dev *iproc_i2c, | 
|  | struct i2c_msg *msg) | 
|  | { | 
|  | u32 val; | 
|  |  | 
|  | val = readl(iproc_i2c->base + M_CMD_OFFSET); | 
|  | val = (val >> M_CMD_STATUS_SHIFT) & M_CMD_STATUS_MASK; | 
|  |  | 
|  | switch (val) { | 
|  | case M_CMD_STATUS_SUCCESS: | 
|  | return 0; | 
|  |  | 
|  | case M_CMD_STATUS_LOST_ARB: | 
|  | dev_dbg(iproc_i2c->device, "lost bus arbitration\n"); | 
|  | return -EAGAIN; | 
|  |  | 
|  | case M_CMD_STATUS_NACK_ADDR: | 
|  | dev_dbg(iproc_i2c->device, "NAK addr:0x%02x\n", msg->addr); | 
|  | return -ENXIO; | 
|  |  | 
|  | case M_CMD_STATUS_NACK_DATA: | 
|  | dev_dbg(iproc_i2c->device, "NAK data\n"); | 
|  | return -ENXIO; | 
|  |  | 
|  | case M_CMD_STATUS_TIMEOUT: | 
|  | dev_dbg(iproc_i2c->device, "bus timeout\n"); | 
|  | return -ETIMEDOUT; | 
|  |  | 
|  | default: | 
|  | dev_dbg(iproc_i2c->device, "unknown error code=%d\n", val); | 
|  |  | 
|  | /* re-initialize i2c for recovery */ | 
|  | bcm_iproc_i2c_enable_disable(iproc_i2c, false); | 
|  | bcm_iproc_i2c_init(iproc_i2c); | 
|  | bcm_iproc_i2c_enable_disable(iproc_i2c, true); | 
|  |  | 
|  | return -EIO; | 
|  | } | 
|  | } | 
|  |  | 
|  | static int bcm_iproc_i2c_xfer_single_msg(struct bcm_iproc_i2c_dev *iproc_i2c, | 
|  | struct i2c_msg *msg) | 
|  | { | 
|  | int ret, i; | 
|  | u8 addr; | 
|  | u32 val; | 
|  | unsigned int tx_bytes; | 
|  | unsigned long time_left = msecs_to_jiffies(I2C_TIMEOUT_MSEC); | 
|  |  | 
|  | /* check if bus is busy */ | 
|  | if (!!(readl(iproc_i2c->base + M_CMD_OFFSET) & | 
|  | BIT(M_CMD_START_BUSY_SHIFT))) { | 
|  | dev_warn(iproc_i2c->device, "bus is busy\n"); | 
|  | return -EBUSY; | 
|  | } | 
|  |  | 
|  | iproc_i2c->msg = msg; | 
|  |  | 
|  | /* format and load slave address into the TX FIFO */ | 
|  | addr = i2c_8bit_addr_from_msg(msg); | 
|  | writel(addr, iproc_i2c->base + M_TX_OFFSET); | 
|  |  | 
|  | /* | 
|  | * For a write transaction, load data into the TX FIFO. Only allow | 
|  | * loading up to TX FIFO size - 1 bytes of data since the first byte | 
|  | * has been used up by the slave address | 
|  | */ | 
|  | tx_bytes = min_t(unsigned int, msg->len, M_TX_RX_FIFO_SIZE - 1); | 
|  | if (!(msg->flags & I2C_M_RD)) { | 
|  | for (i = 0; i < tx_bytes; i++) { | 
|  | val = msg->buf[i]; | 
|  |  | 
|  | /* mark the last byte */ | 
|  | if (i == msg->len - 1) | 
|  | val |= 1 << M_TX_WR_STATUS_SHIFT; | 
|  |  | 
|  | writel(val, iproc_i2c->base + M_TX_OFFSET); | 
|  | } | 
|  | iproc_i2c->tx_bytes = tx_bytes; | 
|  | } | 
|  |  | 
|  | /* mark as incomplete before starting the transaction */ | 
|  | reinit_completion(&iproc_i2c->done); | 
|  | iproc_i2c->xfer_is_done = 0; | 
|  |  | 
|  | /* | 
|  | * Enable the "start busy" interrupt, which will be triggered after the | 
|  | * transaction is done, i.e., the internal start_busy bit, transitions | 
|  | * from 1 to 0. | 
|  | */ | 
|  | val = BIT(IE_M_START_BUSY_SHIFT); | 
|  |  | 
|  | /* | 
|  | * If TX data size is larger than the TX FIFO, need to enable TX | 
|  | * underrun interrupt, which will be triggerred when the TX FIFO is | 
|  | * empty. When that happens we can then pump more data into the FIFO | 
|  | */ | 
|  | if (!(msg->flags & I2C_M_RD) && | 
|  | msg->len > iproc_i2c->tx_bytes) | 
|  | val |= BIT(IE_M_TX_UNDERRUN_SHIFT); | 
|  |  | 
|  | writel(val, iproc_i2c->base + IE_OFFSET); | 
|  |  | 
|  | /* | 
|  | * Now we can activate the transfer. For a read operation, specify the | 
|  | * number of bytes to read | 
|  | */ | 
|  | val = BIT(M_CMD_START_BUSY_SHIFT); | 
|  | if (msg->flags & I2C_M_RD) { | 
|  | val |= (M_CMD_PROTOCOL_BLK_RD << M_CMD_PROTOCOL_SHIFT) | | 
|  | (msg->len << M_CMD_RD_CNT_SHIFT); | 
|  | } else { | 
|  | val |= (M_CMD_PROTOCOL_BLK_WR << M_CMD_PROTOCOL_SHIFT); | 
|  | } | 
|  | writel(val, iproc_i2c->base + M_CMD_OFFSET); | 
|  |  | 
|  | time_left = wait_for_completion_timeout(&iproc_i2c->done, time_left); | 
|  |  | 
|  | /* disable all interrupts */ | 
|  | writel(0, iproc_i2c->base + IE_OFFSET); | 
|  | /* read it back to flush the write */ | 
|  | readl(iproc_i2c->base + IE_OFFSET); | 
|  |  | 
|  | /* make sure the interrupt handler isn't running */ | 
|  | synchronize_irq(iproc_i2c->irq); | 
|  |  | 
|  | if (!time_left && !iproc_i2c->xfer_is_done) { | 
|  | dev_err(iproc_i2c->device, "transaction timed out\n"); | 
|  |  | 
|  | /* flush FIFOs */ | 
|  | val = (1 << M_FIFO_RX_FLUSH_SHIFT) | | 
|  | (1 << M_FIFO_TX_FLUSH_SHIFT); | 
|  | writel(val, iproc_i2c->base + M_FIFO_CTRL_OFFSET); | 
|  | return -ETIMEDOUT; | 
|  | } | 
|  |  | 
|  | ret = bcm_iproc_i2c_check_status(iproc_i2c, msg); | 
|  | if (ret) { | 
|  | /* flush both TX/RX FIFOs */ | 
|  | val = (1 << M_FIFO_RX_FLUSH_SHIFT) | | 
|  | (1 << M_FIFO_TX_FLUSH_SHIFT); | 
|  | writel(val, iproc_i2c->base + M_FIFO_CTRL_OFFSET); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * For a read operation, we now need to load the data from FIFO | 
|  | * into the memory buffer | 
|  | */ | 
|  | if (msg->flags & I2C_M_RD) { | 
|  | for (i = 0; i < msg->len; i++) { | 
|  | msg->buf[i] = (readl(iproc_i2c->base + M_RX_OFFSET) >> | 
|  | M_RX_DATA_SHIFT) & M_RX_DATA_MASK; | 
|  | } | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int bcm_iproc_i2c_xfer(struct i2c_adapter *adapter, | 
|  | struct i2c_msg msgs[], int num) | 
|  | { | 
|  | struct bcm_iproc_i2c_dev *iproc_i2c = i2c_get_adapdata(adapter); | 
|  | int ret, i; | 
|  |  | 
|  | /* go through all messages */ | 
|  | for (i = 0; i < num; i++) { | 
|  | ret = bcm_iproc_i2c_xfer_single_msg(iproc_i2c, &msgs[i]); | 
|  | if (ret) { | 
|  | dev_dbg(iproc_i2c->device, "xfer failed\n"); | 
|  | return ret; | 
|  | } | 
|  | } | 
|  |  | 
|  | return num; | 
|  | } | 
|  |  | 
|  | static uint32_t bcm_iproc_i2c_functionality(struct i2c_adapter *adap) | 
|  | { | 
|  | return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL; | 
|  | } | 
|  |  | 
|  | static const struct i2c_algorithm bcm_iproc_algo = { | 
|  | .master_xfer = bcm_iproc_i2c_xfer, | 
|  | .functionality = bcm_iproc_i2c_functionality, | 
|  | }; | 
|  |  | 
|  | static const struct i2c_adapter_quirks bcm_iproc_i2c_quirks = { | 
|  | /* need to reserve one byte in the FIFO for the slave address */ | 
|  | .max_read_len = M_TX_RX_FIFO_SIZE - 1, | 
|  | }; | 
|  |  | 
|  | static int bcm_iproc_i2c_cfg_speed(struct bcm_iproc_i2c_dev *iproc_i2c) | 
|  | { | 
|  | unsigned int bus_speed; | 
|  | u32 val; | 
|  | int ret = of_property_read_u32(iproc_i2c->device->of_node, | 
|  | "clock-frequency", &bus_speed); | 
|  | if (ret < 0) { | 
|  | dev_info(iproc_i2c->device, | 
|  | "unable to interpret clock-frequency DT property\n"); | 
|  | bus_speed = 100000; | 
|  | } | 
|  |  | 
|  | if (bus_speed < 100000) { | 
|  | dev_err(iproc_i2c->device, "%d Hz bus speed not supported\n", | 
|  | bus_speed); | 
|  | dev_err(iproc_i2c->device, | 
|  | "valid speeds are 100khz and 400khz\n"); | 
|  | return -EINVAL; | 
|  | } else if (bus_speed < 400000) { | 
|  | bus_speed = 100000; | 
|  | } else { | 
|  | bus_speed = 400000; | 
|  | } | 
|  |  | 
|  | iproc_i2c->bus_speed = bus_speed; | 
|  | val = readl(iproc_i2c->base + TIM_CFG_OFFSET); | 
|  | val &= ~(1 << TIM_CFG_MODE_400_SHIFT); | 
|  | val |= (bus_speed == 400000) << TIM_CFG_MODE_400_SHIFT; | 
|  | writel(val, iproc_i2c->base + TIM_CFG_OFFSET); | 
|  |  | 
|  | dev_info(iproc_i2c->device, "bus set to %u Hz\n", bus_speed); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int bcm_iproc_i2c_probe(struct platform_device *pdev) | 
|  | { | 
|  | int irq, ret = 0; | 
|  | struct bcm_iproc_i2c_dev *iproc_i2c; | 
|  | struct i2c_adapter *adap; | 
|  | struct resource *res; | 
|  |  | 
|  | iproc_i2c = devm_kzalloc(&pdev->dev, sizeof(*iproc_i2c), | 
|  | GFP_KERNEL); | 
|  | if (!iproc_i2c) | 
|  | return -ENOMEM; | 
|  |  | 
|  | platform_set_drvdata(pdev, iproc_i2c); | 
|  | iproc_i2c->device = &pdev->dev; | 
|  | init_completion(&iproc_i2c->done); | 
|  |  | 
|  | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 
|  | iproc_i2c->base = devm_ioremap_resource(iproc_i2c->device, res); | 
|  | if (IS_ERR(iproc_i2c->base)) | 
|  | return PTR_ERR(iproc_i2c->base); | 
|  |  | 
|  | ret = bcm_iproc_i2c_init(iproc_i2c); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | ret = bcm_iproc_i2c_cfg_speed(iproc_i2c); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | irq = platform_get_irq(pdev, 0); | 
|  | if (irq <= 0) { | 
|  | dev_err(iproc_i2c->device, "no irq resource\n"); | 
|  | return irq; | 
|  | } | 
|  | iproc_i2c->irq = irq; | 
|  |  | 
|  | ret = devm_request_irq(iproc_i2c->device, irq, bcm_iproc_i2c_isr, 0, | 
|  | pdev->name, iproc_i2c); | 
|  | if (ret < 0) { | 
|  | dev_err(iproc_i2c->device, "unable to request irq %i\n", irq); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | bcm_iproc_i2c_enable_disable(iproc_i2c, true); | 
|  |  | 
|  | adap = &iproc_i2c->adapter; | 
|  | i2c_set_adapdata(adap, iproc_i2c); | 
|  | strlcpy(adap->name, "Broadcom iProc I2C adapter", sizeof(adap->name)); | 
|  | adap->algo = &bcm_iproc_algo; | 
|  | adap->quirks = &bcm_iproc_i2c_quirks; | 
|  | adap->dev.parent = &pdev->dev; | 
|  | adap->dev.of_node = pdev->dev.of_node; | 
|  |  | 
|  | return i2c_add_adapter(adap); | 
|  | } | 
|  |  | 
|  | static int bcm_iproc_i2c_remove(struct platform_device *pdev) | 
|  | { | 
|  | struct bcm_iproc_i2c_dev *iproc_i2c = platform_get_drvdata(pdev); | 
|  |  | 
|  | /* make sure there's no pending interrupt when we remove the adapter */ | 
|  | writel(0, iproc_i2c->base + IE_OFFSET); | 
|  | readl(iproc_i2c->base + IE_OFFSET); | 
|  | synchronize_irq(iproc_i2c->irq); | 
|  |  | 
|  | i2c_del_adapter(&iproc_i2c->adapter); | 
|  | bcm_iproc_i2c_enable_disable(iproc_i2c, false); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #ifdef CONFIG_PM_SLEEP | 
|  |  | 
|  | static int bcm_iproc_i2c_suspend(struct device *dev) | 
|  | { | 
|  | struct bcm_iproc_i2c_dev *iproc_i2c = dev_get_drvdata(dev); | 
|  |  | 
|  | /* make sure there's no pending interrupt when we go into suspend */ | 
|  | writel(0, iproc_i2c->base + IE_OFFSET); | 
|  | readl(iproc_i2c->base + IE_OFFSET); | 
|  | synchronize_irq(iproc_i2c->irq); | 
|  |  | 
|  | /* now disable the controller */ | 
|  | bcm_iproc_i2c_enable_disable(iproc_i2c, false); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int bcm_iproc_i2c_resume(struct device *dev) | 
|  | { | 
|  | struct bcm_iproc_i2c_dev *iproc_i2c = dev_get_drvdata(dev); | 
|  | int ret; | 
|  | u32 val; | 
|  |  | 
|  | /* | 
|  | * Power domain could have been shut off completely in system deep | 
|  | * sleep, so re-initialize the block here | 
|  | */ | 
|  | ret = bcm_iproc_i2c_init(iproc_i2c); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | /* configure to the desired bus speed */ | 
|  | val = readl(iproc_i2c->base + TIM_CFG_OFFSET); | 
|  | val &= ~(1 << TIM_CFG_MODE_400_SHIFT); | 
|  | val |= (iproc_i2c->bus_speed == 400000) << TIM_CFG_MODE_400_SHIFT; | 
|  | writel(val, iproc_i2c->base + TIM_CFG_OFFSET); | 
|  |  | 
|  | bcm_iproc_i2c_enable_disable(iproc_i2c, true); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static const struct dev_pm_ops bcm_iproc_i2c_pm_ops = { | 
|  | .suspend_late = &bcm_iproc_i2c_suspend, | 
|  | .resume_early = &bcm_iproc_i2c_resume | 
|  | }; | 
|  |  | 
|  | #define BCM_IPROC_I2C_PM_OPS (&bcm_iproc_i2c_pm_ops) | 
|  | #else | 
|  | #define BCM_IPROC_I2C_PM_OPS NULL | 
|  | #endif /* CONFIG_PM_SLEEP */ | 
|  |  | 
|  | static const struct of_device_id bcm_iproc_i2c_of_match[] = { | 
|  | { .compatible = "brcm,iproc-i2c" }, | 
|  | { /* sentinel */ } | 
|  | }; | 
|  | MODULE_DEVICE_TABLE(of, bcm_iproc_i2c_of_match); | 
|  |  | 
|  | static struct platform_driver bcm_iproc_i2c_driver = { | 
|  | .driver = { | 
|  | .name = "bcm-iproc-i2c", | 
|  | .of_match_table = bcm_iproc_i2c_of_match, | 
|  | .pm = BCM_IPROC_I2C_PM_OPS, | 
|  | }, | 
|  | .probe = bcm_iproc_i2c_probe, | 
|  | .remove = bcm_iproc_i2c_remove, | 
|  | }; | 
|  | module_platform_driver(bcm_iproc_i2c_driver); | 
|  |  | 
|  | MODULE_AUTHOR("Ray Jui <rjui@broadcom.com>"); | 
|  | MODULE_DESCRIPTION("Broadcom iProc I2C Driver"); | 
|  | MODULE_LICENSE("GPL v2"); |