Commit 5c135e15 authored by Abhishek Sahu's avatar Abhishek Sahu Committed by Wolfram Sang

i2c: qup: Fixed the DMA segments length

1. The current QCOM I2C driver code is failing for transfer length
greater than 255. This is happening due to improper segments length
as the I2C DMA segments can be maximum of 256 bytes.

2. The transfer length tlen was being initialized with 0 for 256
bytes, which is being passed for DMA mappings resulting in improper
DMA mapping length.

This patch fixes the above said problems by initializing the block
count with the values calculated in qup_i2c_set_blk_data and calculating
the remaining length for last DMA segment. Also, the block data length
need to be decremented after each transfer. Additionally, this patch
corrects the tlen assignment for DMA mapping.
Signed-off-by: default avatarAbhishek Sahu <absahu@codeaurora.org>
Signed-off-by: default avatarWolfram Sang <wsa@the-dreams.de>
parent 2b84a4dd
...@@ -659,23 +659,24 @@ static int qup_i2c_bam_do_xfer(struct qup_i2c_dev *qup, struct i2c_msg *msg, ...@@ -659,23 +659,24 @@ static int qup_i2c_bam_do_xfer(struct qup_i2c_dev *qup, struct i2c_msg *msg,
u8 *tags; u8 *tags;
while (idx < num) { while (idx < num) {
blocks = (msg->len + limit) / limit;
rem = msg->len % limit;
tx_len = 0, len = 0, i = 0; tx_len = 0, len = 0, i = 0;
qup->is_last = (idx == (num - 1)); qup->is_last = (idx == (num - 1));
qup_i2c_set_blk_data(qup, msg); qup_i2c_set_blk_data(qup, msg);
blocks = qup->blk.count;
rem = msg->len - (blocks - 1) * limit;
if (msg->flags & I2C_M_RD) { if (msg->flags & I2C_M_RD) {
rx_nents += (blocks * 2) + 1; rx_nents += (blocks * 2) + 1;
tx_nents += 1; tx_nents += 1;
while (qup->blk.pos < blocks) { while (qup->blk.pos < blocks) {
/* length set to '0' implies 256 bytes */ tlen = (i == (blocks - 1)) ? rem : limit;
tlen = (i == (blocks - 1)) ? rem : 0;
tags = &qup->start_tag.start[off + len]; tags = &qup->start_tag.start[off + len];
len += qup_i2c_set_tags(tags, qup, msg, 1); len += qup_i2c_set_tags(tags, qup, msg, 1);
qup->blk.data_len -= tlen;
/* scratch buf to read the start and len tags */ /* scratch buf to read the start and len tags */
ret = qup_sg_set_buf(&qup->brx.sg[rx_buf++], ret = qup_sg_set_buf(&qup->brx.sg[rx_buf++],
...@@ -712,9 +713,10 @@ static int qup_i2c_bam_do_xfer(struct qup_i2c_dev *qup, struct i2c_msg *msg, ...@@ -712,9 +713,10 @@ static int qup_i2c_bam_do_xfer(struct qup_i2c_dev *qup, struct i2c_msg *msg,
tx_nents += (blocks * 2); tx_nents += (blocks * 2);
while (qup->blk.pos < blocks) { while (qup->blk.pos < blocks) {
tlen = (i == (blocks - 1)) ? rem : 0; tlen = (i == (blocks - 1)) ? rem : limit;
tags = &qup->start_tag.start[off + tx_len]; tags = &qup->start_tag.start[off + tx_len];
len = qup_i2c_set_tags(tags, qup, msg, 1); len = qup_i2c_set_tags(tags, qup, msg, 1);
qup->blk.data_len -= tlen;
ret = qup_sg_set_buf(&qup->btx.sg[tx_buf++], ret = qup_sg_set_buf(&qup->btx.sg[tx_buf++],
tags, len, tags, len,
......
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