root/drivers/media/usb/dvb-usb-v2/mxl111sf-i2c.c
// SPDX-License-Identifier: GPL-2.0-or-later
/*
 *  mxl111sf-i2c.c - driver for the MaxLinear MXL111SF
 *
 *  Copyright (C) 2010-2014 Michael Krufky <mkrufky@linuxtv.org>
 */

#include "mxl111sf-i2c.h"
#include "mxl111sf.h"

/* SW-I2C ----------------------------------------------------------------- */

#define SW_I2C_ADDR             0x1a
#define SW_I2C_EN               0x02
#define SW_SCL_OUT              0x04
#define SW_SDA_OUT              0x08
#define SW_SDA_IN               0x04

#define SW_I2C_BUSY_ADDR        0x2f
#define SW_I2C_BUSY             0x02

static int mxl111sf_i2c_bitbang_sendbyte(struct mxl111sf_state *state,
                                         u8 byte)
{
        int i, ret;
        u8 data = 0;

        mxl_i2c("(0x%02x)", byte);

        ret = mxl111sf_read_reg(state, SW_I2C_BUSY_ADDR, &data);
        if (mxl_fail(ret))
                goto fail;

        for (i = 0; i < 8; i++) {

                data = (byte & (0x80 >> i)) ? SW_SDA_OUT : 0;

                ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                         0x10 | SW_I2C_EN | data);
                if (mxl_fail(ret))
                        goto fail;

                ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                         0x10 | SW_I2C_EN | data | SW_SCL_OUT);
                if (mxl_fail(ret))
                        goto fail;

                ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                         0x10 | SW_I2C_EN | data);
                if (mxl_fail(ret))
                        goto fail;
        }

        /* last bit was 0 so we need to release SDA */
        if (!(byte & 1)) {
                ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                         0x10 | SW_I2C_EN | SW_SDA_OUT);
                if (mxl_fail(ret))
                        goto fail;
        }

        /* CLK high for ACK readback */
        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN | SW_SCL_OUT | SW_SDA_OUT);
        if (mxl_fail(ret))
                goto fail;

        ret = mxl111sf_read_reg(state, SW_I2C_BUSY_ADDR, &data);
        if (mxl_fail(ret))
                goto fail;

        /* drop the CLK after getting ACK, SDA will go high right away */
        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN | SW_SDA_OUT);
        if (mxl_fail(ret))
                goto fail;

        if (data & SW_SDA_IN)
                ret = -EIO;
fail:
        return ret;
}

static int mxl111sf_i2c_bitbang_recvbyte(struct mxl111sf_state *state,
                                         u8 *pbyte)
{
        int i, ret;
        u8 byte = 0;
        u8 data = 0;

        mxl_i2c("()");

        *pbyte = 0;

        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN | SW_SDA_OUT);
        if (mxl_fail(ret))
                goto fail;

        for (i = 0; i < 8; i++) {
                ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                         0x10 | SW_I2C_EN |
                                         SW_SCL_OUT | SW_SDA_OUT);
                if (mxl_fail(ret))
                        goto fail;

                ret = mxl111sf_read_reg(state, SW_I2C_BUSY_ADDR, &data);
                if (mxl_fail(ret))
                        goto fail;

                if (data & SW_SDA_IN)
                        byte |= (0x80 >> i);

                ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                         0x10 | SW_I2C_EN | SW_SDA_OUT);
                if (mxl_fail(ret))
                        goto fail;
        }
        *pbyte = byte;
fail:
        return ret;
}

static int mxl111sf_i2c_start(struct mxl111sf_state *state)
{
        int ret;

        mxl_i2c("()");

        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN | SW_SCL_OUT | SW_SDA_OUT);
        if (mxl_fail(ret))
                goto fail;

        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN | SW_SCL_OUT);
        if (mxl_fail(ret))
                goto fail;

        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN); /* start */
        mxl_fail(ret);
fail:
        return ret;
}

static int mxl111sf_i2c_stop(struct mxl111sf_state *state)
{
        int ret;

        mxl_i2c("()");

        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN); /* stop */
        if (mxl_fail(ret))
                goto fail;

        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN | SW_SCL_OUT);
        if (mxl_fail(ret))
                goto fail;

        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN | SW_SCL_OUT | SW_SDA_OUT);
        if (mxl_fail(ret))
                goto fail;

        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_SCL_OUT | SW_SDA_OUT);
        mxl_fail(ret);
fail:
        return ret;
}

static int mxl111sf_i2c_ack(struct mxl111sf_state *state)
{
        int ret;
        u8 b = 0;

        mxl_i2c("()");

        ret = mxl111sf_read_reg(state, SW_I2C_BUSY_ADDR, &b);
        if (mxl_fail(ret))
                goto fail;

        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN);
        if (mxl_fail(ret))
                goto fail;

        /* pull SDA low */
        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN | SW_SCL_OUT);
        if (mxl_fail(ret))
                goto fail;

        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN | SW_SDA_OUT);
        mxl_fail(ret);
fail:
        return ret;
}

static int mxl111sf_i2c_nack(struct mxl111sf_state *state)
{
        int ret;

        mxl_i2c("()");

        /* SDA high to signal last byte read from slave */
        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN | SW_SCL_OUT | SW_SDA_OUT);
        if (mxl_fail(ret))
                goto fail;

        ret = mxl111sf_write_reg(state, SW_I2C_ADDR,
                                 0x10 | SW_I2C_EN | SW_SDA_OUT);
        mxl_fail(ret);
fail:
        return ret;
}

/* ------------------------------------------------------------------------ */

static int mxl111sf_i2c_sw_xfer_msg(struct mxl111sf_state *state,
                                    struct i2c_msg *msg)
{
        int i, ret;

        mxl_i2c("()");

        if (msg->flags & I2C_M_RD) {

                ret = mxl111sf_i2c_start(state);
                if (mxl_fail(ret))
                        goto fail;

                ret = mxl111sf_i2c_bitbang_sendbyte(state,
                                                    (msg->addr << 1) | 0x01);
                if (mxl_fail(ret)) {
                        mxl111sf_i2c_stop(state);
                        goto fail;
                }

                for (i = 0; i < msg->len; i++) {
                        ret = mxl111sf_i2c_bitbang_recvbyte(state,
                                                            &msg->buf[i]);
                        if (mxl_fail(ret)) {
                                mxl111sf_i2c_stop(state);
                                goto fail;
                        }

                        if (i < msg->len - 1)
                                mxl111sf_i2c_ack(state);
                }

                mxl111sf_i2c_nack(state);

                ret = mxl111sf_i2c_stop(state);
                if (mxl_fail(ret))
                        goto fail;

        } else {

                ret = mxl111sf_i2c_start(state);
                if (mxl_fail(ret))
                        goto fail;

                ret = mxl111sf_i2c_bitbang_sendbyte(state,
                                                    (msg->addr << 1) & 0xfe);
                if (mxl_fail(ret)) {
                        mxl111sf_i2c_stop(state);
                        goto fail;
                }

                for (i = 0; i < msg->len; i++) {
                        ret = mxl111sf_i2c_bitbang_sendbyte(state,
                                                            msg->buf[i]);
                        if (mxl_fail(ret)) {
                                mxl111sf_i2c_stop(state);
                                goto fail;
                        }
                }

                /* FIXME: we only want to do this on the last transaction */
                mxl111sf_i2c_stop(state);
        }
fail:
        return ret;
}

/* HW-I2C ----------------------------------------------------------------- */

#define USB_WRITE_I2C_CMD     0x99
#define USB_READ_I2C_CMD      0xdd
#define USB_END_I2C_CMD       0xfe

#define USB_WRITE_I2C_CMD_LEN   26
#define USB_READ_I2C_CMD_LEN    24

#define I2C_MUX_REG           0x30
#define I2C_CONTROL_REG       0x00
#define I2C_SLAVE_ADDR_REG    0x08
#define I2C_DATA_REG          0x0c
#define I2C_INT_STATUS_REG    0x10

static int mxl111sf_i2c_send_data(struct mxl111sf_state *state,
                                  u8 index, u8 *wdata)
{
        int ret = mxl111sf_ctrl_msg(state, wdata[0],
                                    &wdata[1], 25, NULL, 0);
        mxl_fail(ret);

        return ret;
}

static int mxl111sf_i2c_get_data(struct mxl111sf_state *state,
                                 u8 index, u8 *wdata, u8 *rdata)
{
        int ret = mxl111sf_ctrl_msg(state, wdata[0],
                                    &wdata[1], 25, rdata, 24);
        mxl_fail(ret);

        return ret;
}

static u8 mxl111sf_i2c_check_status(struct mxl111sf_state *state)
{
        u8 status = 0;
        u8 buf[26];

        mxl_i2c_adv("()");

        buf[0] = USB_READ_I2C_CMD;
        buf[1] = 0x00;

        buf[2] = I2C_INT_STATUS_REG;
        buf[3] = 0x00;
        buf[4] = 0x00;

        buf[5] = USB_END_I2C_CMD;

        mxl111sf_i2c_get_data(state, 0, buf, buf);

        if (buf[1] & 0x04)
                status = 1;

        return status;
}

static u8 mxl111sf_i2c_check_fifo(struct mxl111sf_state *state)
{
        u8 status = 0;
        u8 buf[26];

        mxl_i2c("()");

        buf[0] = USB_READ_I2C_CMD;
        buf[1] = 0x00;

        buf[2] = I2C_MUX_REG;
        buf[3] = 0x00;
        buf[4] = 0x00;

        buf[5] = I2C_INT_STATUS_REG;
        buf[6] = 0x00;
        buf[7] = 0x00;
        buf[8] = USB_END_I2C_CMD;

        mxl111sf_i2c_get_data(state, 0, buf, buf);

        if (0x08 == (buf[1] & 0x08))
                status = 1;

        if ((buf[5] & 0x02) == 0x02)
                mxl_i2c("(buf[5] & 0x02) == 0x02"); /* FIXME */

        return status;
}

static int mxl111sf_i2c_readagain(struct mxl111sf_state *state,
                                  u8 count, u8 *rbuf)
{
        u8 i2c_w_data[26];
        u8 i2c_r_data[24];
        u8 i = 0;
        u8 fifo_status = 0;
        int status = 0;

        mxl_i2c("read %d bytes", count);

        while ((fifo_status == 0) && (i++ < 5))
                fifo_status = mxl111sf_i2c_check_fifo(state);

        i2c_w_data[0] = 0xDD;
        i2c_w_data[1] = 0x00;

        for (i = 2; i < 26; i++)
                i2c_w_data[i] = 0xFE;

        for (i = 0; i < count; i++) {
                i2c_w_data[2+(i*3)] = 0x0C;
                i2c_w_data[3+(i*3)] = 0x00;
                i2c_w_data[4+(i*3)] = 0x00;
        }

        mxl111sf_i2c_get_data(state, 0, i2c_w_data, i2c_r_data);

        /* Check for I2C NACK status */
        if (mxl111sf_i2c_check_status(state) == 1) {
                mxl_i2c("error!");
        } else {
                for (i = 0; i < count; i++) {
                        rbuf[i] = i2c_r_data[(i*3)+1];
                        mxl_i2c("%02x\t %02x",
                                i2c_r_data[(i*3)+1],
                                i2c_r_data[(i*3)+2]);
                }

                status = 1;
        }

        return status;
}

#define HWI2C400 1
static int mxl111sf_i2c_hw_xfer_msg(struct mxl111sf_state *state,
                                    struct i2c_msg *msg)
{
        int i, k, ret = 0;
        u16 index = 0;
        u8 buf[26];
        u8 i2c_r_data[24];
        u16 block_len;
        u16 left_over_len;
        u8 rd_status[8];
        u8 ret_status;
        u8 readbuff[26];

        mxl_i2c("addr: 0x%02x, read buff len: %d, write buff len: %d",
                msg->addr, (msg->flags & I2C_M_RD) ? msg->len : 0,
                (!(msg->flags & I2C_M_RD)) ? msg->len : 0);

        for (index = 0; index < 26; index++)
                buf[index] = USB_END_I2C_CMD;

        /* command to indicate data payload is destined for I2C interface */
        buf[0] = USB_WRITE_I2C_CMD;
        buf[1] = 0x00;

        /* enable I2C interface */
        buf[2] = I2C_MUX_REG;
        buf[3] = 0x80;
        buf[4] = 0x00;

        /* enable I2C interface */
        buf[5] = I2C_MUX_REG;
        buf[6] = 0x81;
        buf[7] = 0x00;

        /* set Timeout register on I2C interface */
        buf[8] = 0x14;
        buf[9] = 0xff;
        buf[10] = 0x00;
#if 0
        /* enable Interrupts on I2C interface */
        buf[8] = 0x24;
        buf[9] = 0xF7;
        buf[10] = 0x00;
#endif
        buf[11] = 0x24;
        buf[12] = 0xF7;
        buf[13] = 0x00;

        ret = mxl111sf_i2c_send_data(state, 0, buf);

        /* write data on I2C bus */
        if (!(msg->flags & I2C_M_RD) && (msg->len > 0)) {
                mxl_i2c("%d\t%02x", msg->len, msg->buf[0]);

                /* control register on I2C interface to initialize I2C bus */
                buf[2] = I2C_CONTROL_REG;
                buf[3] = 0x5E;
                buf[4] = (HWI2C400) ? 0x03 : 0x0D;

                /* I2C Slave device Address */
                buf[5] = I2C_SLAVE_ADDR_REG;
                buf[6] = (msg->addr);
                buf[7] = 0x00;
                buf[8] = USB_END_I2C_CMD;
                ret = mxl111sf_i2c_send_data(state, 0, buf);

                /* check for slave device status */
                if (mxl111sf_i2c_check_status(state) == 1) {
                        mxl_i2c("NACK writing slave address %02x",
                                msg->addr);
                        /* if NACK, stop I2C bus and exit */
                        buf[2] = I2C_CONTROL_REG;
                        buf[3] = 0x4E;
                        buf[4] = (HWI2C400) ? 0x03 : 0x0D;
                        ret = -EIO;
                        goto exit;
                }

                /* I2C interface can do I2C operations in block of 8 bytes of
                   I2C data. calculation to figure out number of blocks of i2c
                   data required to program */
                block_len = (msg->len / 8);
                left_over_len = (msg->len % 8);

                mxl_i2c("block_len %d, left_over_len %d",
                        block_len, left_over_len);

                for (index = 0; index < block_len; index++) {
                        for (i = 0; i < 8; i++) {
                                /* write data on I2C interface */
                                buf[2+(i*3)] = I2C_DATA_REG;
                                buf[3+(i*3)] = msg->buf[(index*8)+i];
                                buf[4+(i*3)] = 0x00;
                        }

                        ret = mxl111sf_i2c_send_data(state, 0, buf);

                        /* check for I2C NACK status */
                        if (mxl111sf_i2c_check_status(state) == 1) {
                                mxl_i2c("NACK writing slave address %02x",
                                        msg->addr);

                                /* if NACK, stop I2C bus and exit */
                                buf[2] = I2C_CONTROL_REG;
                                buf[3] = 0x4E;
                                buf[4] = (HWI2C400) ? 0x03 : 0x0D;
                                ret = -EIO;
                                goto exit;
                        }

                }

                if (left_over_len) {
                        for (k = 0; k < 26; k++)
                                buf[k] = USB_END_I2C_CMD;

                        buf[0] = 0x99;
                        buf[1] = 0x00;

                        for (i = 0; i < left_over_len; i++) {
                                buf[2+(i*3)] = I2C_DATA_REG;
                                buf[3+(i*3)] = msg->buf[(index*8)+i];
                                mxl_i2c("index = %d %d data %d",
                                        index, i, msg->buf[(index*8)+i]);
                                buf[4+(i*3)] = 0x00;
                        }
                        ret = mxl111sf_i2c_send_data(state, 0, buf);

                        /* check for I2C NACK status */
                        if (mxl111sf_i2c_check_status(state) == 1) {
                                mxl_i2c("NACK writing slave address %02x",
                                        msg->addr);

                                /* if NACK, stop I2C bus and exit */
                                buf[2] = I2C_CONTROL_REG;
                                buf[3] = 0x4E;
                                buf[4] = (HWI2C400) ? 0x03 : 0x0D;
                                ret = -EIO;
                                goto exit;
                        }

                }

                /* issue I2C STOP after write */
                buf[2] = I2C_CONTROL_REG;
                buf[3] = 0x4E;
                buf[4] = (HWI2C400) ? 0x03 : 0x0D;

        }

        /* read data from I2C bus */
        if ((msg->flags & I2C_M_RD) && (msg->len > 0)) {
                mxl_i2c("read buf len %d", msg->len);

                /* command to indicate data payload is
                   destined for I2C interface */
                buf[2] = I2C_CONTROL_REG;
                buf[3] = 0xDF;
                buf[4] = (HWI2C400) ? 0x03 : 0x0D;

                /* I2C xfer length */
                buf[5] = 0x14;
                buf[6] = (msg->len & 0xFF);
                buf[7] = 0;

                /* I2C slave device Address */
                buf[8] = I2C_SLAVE_ADDR_REG;
                buf[9] = msg->addr;
                buf[10] = 0x00;
                buf[11] = USB_END_I2C_CMD;
                ret = mxl111sf_i2c_send_data(state, 0, buf);

                /* check for I2C NACK status */
                if (mxl111sf_i2c_check_status(state) == 1) {
                        mxl_i2c("NACK reading slave address %02x",
                                msg->addr);

                        /* if NACK, stop I2C bus and exit */
                        buf[2] = I2C_CONTROL_REG;
                        buf[3] = 0xC7;
                        buf[4] = (HWI2C400) ? 0x03 : 0x0D;
                        ret = -EIO;
                        goto exit;
                }

                /* I2C interface can do I2C operations in block of 8 bytes of
                   I2C data. calculation to figure out number of blocks of
                   i2c data required to program */
                block_len = ((msg->len) / 8);
                left_over_len = ((msg->len) % 8);
                index = 0;

                mxl_i2c("block_len %d, left_over_len %d",
                        block_len, left_over_len);

                /* command to read data from I2C interface */
                buf[0] = USB_READ_I2C_CMD;
                buf[1] = 0x00;

                for (index = 0; index < block_len; index++) {
                        /* setup I2C read request packet on I2C interface */
                        for (i = 0; i < 8; i++) {
                                buf[2+(i*3)] = I2C_DATA_REG;
                                buf[3+(i*3)] = 0x00;
                                buf[4+(i*3)] = 0x00;
                        }

                        ret = mxl111sf_i2c_get_data(state, 0, buf, i2c_r_data);

                        /* check for I2C NACK status */
                        if (mxl111sf_i2c_check_status(state) == 1) {
                                mxl_i2c("NACK reading slave address %02x",
                                        msg->addr);

                                /* if NACK, stop I2C bus and exit */
                                buf[2] = I2C_CONTROL_REG;
                                buf[3] = 0xC7;
                                buf[4] = (HWI2C400) ? 0x03 : 0x0D;
                                ret = -EIO;
                                goto exit;
                        }

                        /* copy data from i2c data payload to read buffer */
                        for (i = 0; i < 8; i++) {
                                rd_status[i] = i2c_r_data[(i*3)+2];

                                if (rd_status[i] == 0x04) {
                                        if (i < 7) {
                                                mxl_i2c("i2c fifo empty! @ %d",
                                                        i);
                                                msg->buf[(index*8)+i] =
                                                        i2c_r_data[(i*3)+1];
                                                /* read again */
                                                ret_status =
                                                        mxl111sf_i2c_readagain(
                                                                state, 8-(i+1),
                                                                readbuff);
                                                if (ret_status == 1) {
                                                        for (k = 0;
                                                             k < 8-(i+1);
                                                             k++) {

                                        msg->buf[(index*8)+(k+i+1)] =
                                                readbuff[k];
                                        mxl_i2c("read data: %02x\t %02x",
                                                msg->buf[(index*8)+(k+i)],
                                                (index*8)+(k+i));
                                        mxl_i2c("read data: %02x\t %02x",
                                                msg->buf[(index*8)+(k+i+1)],
                                                readbuff[k]);

                                                        }
                                                        goto stop_copy;
                                                } else {
                                                        mxl_i2c("readagain ERROR!");
                                                }
                                        } else {
                                                msg->buf[(index*8)+i] =
                                                        i2c_r_data[(i*3)+1];
                                        }
                                } else {
                                        msg->buf[(index*8)+i] =
                                                i2c_r_data[(i*3)+1];
                                }
                        }
stop_copy:
                        ;

                }

                if (left_over_len) {
                        for (k = 0; k < 26; k++)
                                buf[k] = USB_END_I2C_CMD;

                        buf[0] = 0xDD;
                        buf[1] = 0x00;

                        for (i = 0; i < left_over_len; i++) {
                                buf[2+(i*3)] = I2C_DATA_REG;
                                buf[3+(i*3)] = 0x00;
                                buf[4+(i*3)] = 0x00;
                        }
                        ret = mxl111sf_i2c_get_data(state, 0, buf,
                                                    i2c_r_data);

                        /* check for I2C NACK status */
                        if (mxl111sf_i2c_check_status(state) == 1) {
                                mxl_i2c("NACK reading slave address %02x",
                                        msg->addr);

                                /* if NACK, stop I2C bus and exit */
                                buf[2] = I2C_CONTROL_REG;
                                buf[3] = 0xC7;
                                buf[4] = (HWI2C400) ? 0x03 : 0x0D;
                                ret = -EIO;
                                goto exit;
                        }

                        for (i = 0; i < left_over_len; i++) {
                                msg->buf[(block_len*8)+i] =
                                        i2c_r_data[(i*3)+1];
                                mxl_i2c("read data: %02x\t %02x",
                                        i2c_r_data[(i*3)+1],
                                        i2c_r_data[(i*3)+2]);
                        }
                }

                /* indicate I2C interface to issue NACK
                   after next I2C read op */
                buf[0] = USB_WRITE_I2C_CMD;
                buf[1] = 0x00;

                /* control register */
                buf[2] = I2C_CONTROL_REG;
                buf[3] = 0x17;
                buf[4] = (HWI2C400) ? 0x03 : 0x0D;

                buf[5] = USB_END_I2C_CMD;
                ret = mxl111sf_i2c_send_data(state, 0, buf);

                /* control register */
                buf[2] = I2C_CONTROL_REG;
                buf[3] = 0xC7;
                buf[4] = (HWI2C400) ? 0x03 : 0x0D;

        }
exit:
        /* STOP and disable I2C MUX */
        buf[0] = USB_WRITE_I2C_CMD;
        buf[1] = 0x00;

        /* de-initilize I2C BUS */
        buf[5] = USB_END_I2C_CMD;
        mxl111sf_i2c_send_data(state, 0, buf);

        /* Control Register */
        buf[2] = I2C_CONTROL_REG;
        buf[3] = 0xDF;
        buf[4] = 0x03;

        /* disable I2C interface */
        buf[5] = I2C_MUX_REG;
        buf[6] = 0x00;
        buf[7] = 0x00;

        /* de-initilize I2C BUS */
        buf[8] = USB_END_I2C_CMD;
        mxl111sf_i2c_send_data(state, 0, buf);

        /* disable I2C interface */
        buf[2] = I2C_MUX_REG;
        buf[3] = 0x81;
        buf[4] = 0x00;

        /* disable I2C interface */
        buf[5] = I2C_MUX_REG;
        buf[6] = 0x00;
        buf[7] = 0x00;

        /* disable I2C interface */
        buf[8] = I2C_MUX_REG;
        buf[9] = 0x00;
        buf[10] = 0x00;

        buf[11] = USB_END_I2C_CMD;
        mxl111sf_i2c_send_data(state, 0, buf);

        return ret;
}

/* ------------------------------------------------------------------------ */

int mxl111sf_i2c_xfer(struct i2c_adapter *adap,
                      struct i2c_msg msg[], int num)
{
        struct dvb_usb_device *d = i2c_get_adapdata(adap);
        struct mxl111sf_state *state = d->priv;
        int hwi2c = (state->chip_rev > MXL111SF_V6);
        int i, ret;

        if (mutex_lock_interruptible(&d->i2c_mutex) < 0)
                return -EAGAIN;

        for (i = 0; i < num; i++) {
                ret = (hwi2c) ?
                        mxl111sf_i2c_hw_xfer_msg(state, &msg[i]) :
                        mxl111sf_i2c_sw_xfer_msg(state, &msg[i]);
                if (mxl_fail(ret)) {
                        mxl_debug_adv("failed with error %d on i2c transaction %d of %d, %sing %d bytes to/from 0x%02x",
                                      ret, i+1, num,
                                      (msg[i].flags & I2C_M_RD) ?
                                      "read" : "writ",
                                      msg[i].len, msg[i].addr);

                        break;
                }
        }

        mutex_unlock(&d->i2c_mutex);

        return i == num ? num : -EREMOTEIO;
}