Commit 3e1ccb2a authored by Pontus Andersson's avatar Pontus Andersson Committed by Thadeu Lima de Souza Cascardo

i2c: ismt: Separate I2C block read from SMBus block read

BugLink: http://bugs.launchpad.net/bugs/1729107

commit c6ebcedb upstream.

Commit b6c159a9 ("i2c: ismt: Don't duplicate the receive length for
block reads") broke I2C block reads. It aimed to fix normal SMBus block
read, but changed the correct behavior of I2C block read in the process.

According to Documentation/i2c/smbus-protocol, one vital difference
between normal SMBus block read and I2C block read is that there is no
byte count prefixed in the data sent on the wire:

 SMBus Block Read:  i2c_smbus_read_block_data()
 S Addr Wr [A] Comm [A]
            S Addr Rd [A] [Count] A [Data] A [Data] A ... A [Data] NA P

 I2C Block Read:  i2c_smbus_read_i2c_block_data()
 S Addr Wr [A] Comm [A]
            S Addr Rd [A] [Data] A [Data] A ... A [Data] NA P

Therefore the two transaction types need to be processed differently in
the driver by copying of the dma_buffer as done previously for the
I2C_SMBUS_I2C_BLOCK_DATA case.

Fixes: b6c159a9 ("i2c: ismt: Don't duplicate the receive length for block reads")
Signed-off-by: default avatarPontus Andersson <epontan@gmail.com>
Tested-by: default avatarStephen Douthit <stephend@adiengineering.com>
Signed-off-by: default avatarWolfram Sang <wsa@the-dreams.de>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 12574a4d
...@@ -338,12 +338,15 @@ static int ismt_process_desc(const struct ismt_desc *desc, ...@@ -338,12 +338,15 @@ static int ismt_process_desc(const struct ismt_desc *desc,
data->word = dma_buffer[0] | (dma_buffer[1] << 8); data->word = dma_buffer[0] | (dma_buffer[1] << 8);
break; break;
case I2C_SMBUS_BLOCK_DATA: case I2C_SMBUS_BLOCK_DATA:
case I2C_SMBUS_I2C_BLOCK_DATA:
if (desc->rxbytes != dma_buffer[0] + 1) if (desc->rxbytes != dma_buffer[0] + 1)
return -EMSGSIZE; return -EMSGSIZE;
memcpy(data->block, dma_buffer, desc->rxbytes); memcpy(data->block, dma_buffer, desc->rxbytes);
break; break;
case I2C_SMBUS_I2C_BLOCK_DATA:
memcpy(&data->block[1], dma_buffer, desc->rxbytes);
data->block[0] = desc->rxbytes;
break;
} }
return 0; return 0;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment