Re: [PATCH] i2c:octeon:Add block-mode r/w

From: Aryan Srivastava
Date: Mon Sep 11 2023 - 21:11:56 EST


Hi, Andi,

On Tue, 2023-09-05 at 12:22 +0200, Andi Shyti wrote:
> Hi Aryan,
>
> In the title, please leave a space after the ':'
>
> i2c: octeon: Add block-mode r/w
>
> Please check with "git log drivers..." to see what's the rule in
> a particular community.
>
> I guess Wolfram can fix this, though, before pushing.
>
> [...]
>
Done

> > +/* high-level-controller composite block write+read, msg0=addr,
> > msg1=data */
>
> I think this comment is fine and great to have it, but it's
> missing a bit of clarity, can you please expand the concept?
>
Done, let me know if you want me to add more here.

> > +static int octeon_i2c_hlc_block_comp_read(struct octeon_i2c *i2c,
> > struct i2c_msg *msgs)
> > +{
> > + int i, j, len, ret = 0;
> > + u64 cmd = 0, rd = 0;
>
> can please you move rd, j inside the for loop? The basic common
> sense is to have all variable declared in the innermost section
> in order to avoid confusion.
>
> It's a nitpick though, not a strong comment and, afaik, not a
> real rule.
>
> Same comment for the function below.
>
Done, I agree they should be defined within loop. I was just trying to
match the other hlc r/w.

> > +
> > + octeon_i2c_hlc_enable(i2c);
> > + octeon_i2c_block_enable(i2c);
> > +
> > + /* Write (size - 1) into block control register */
> > + len = msgs[1].len - 1;
> > + octeon_i2c_writeq_flush((u64)(len), i2c->twsi_base +
> > TWSI_BLOCK_CTL(i2c));
> > +
> > + /* Prepare core command */
> > + cmd = SW_TWSI_V | SW_TWSI_R | SW_TWSI_SOVR;
> > + cmd |= (u64)(msgs[0].addr & 0x7full) << SW_TWSI_ADDR_SHIFT;
> > +
> > + if (msgs[0].flags & I2C_M_TEN)
> > + cmd |= SW_TWSI_OP_10_IA;
> > + else
> > + cmd |= SW_TWSI_OP_7_IA;
> > +No, but this doesnt really matter as the internal j loop will take
> > care of the remaining bytes.
> >
> > e.g. if the len is 9, then we will do
> > 0-7 in the first, then i = 8, which is < len, and then the internal
> > loop will do 8-17.
> > + if (msgs[0].len == 2) {
> > + u64 ext = 0;
> > +
> > + cmd |= SW_TWSI_EIA;
> > + ext = (u64)msgs[0].buf[0] << SW_TWSI_IA_SHIFT;
> > + cmd |= (u64)msgs[0].buf[1] << SW_TWSI_IA_SHIFT;
> > + octeon_i2c_writeq_flush(ext, i2c->twsi_base +
> > SW_TWSI_EXT(i2c));
> > + } else {
> > + cmd |= (u64)msgs[0].buf[0] << SW_TWSI_IA_SHIFT;
> > + }
>
> This first part is basically a copy/paste with the write()
> function... can we put them together in a common function?
>
> Can we put as much as we can in a single function?
>
Done. Could not make a common for the write+writes, as they way we
insert data into buffers for writing are significantly different,
and don't occur after the core command send (i.e. the core command is
sent with/concurrently with the write data). Unlike the read, which
sends almost identical core commands and reads the buffer differently
afterwards. Though if would like it I could mangle together a function
for these as well.

> > + /* Send command to core (send data to FIFO) */
> > + octeon_i2c_hlc_int_clear(i2c);
> > + octeon_i2c_writeq_flush(cmd, i2c->twsi_base + SW_TWSI(i2c));
> > +
> > + /* Wait for transaction to complete */
> > + ret = octeon_i2c_hlc_wait(i2c);
> > + if (ret)
> > + return ret;
> > +
> > + cmd = __raw_readq(i2c->twsi_base + SW_TWSI(i2c));
> > + if ((cmd & SW_TWSI_R) == 0)
> > + return octeon_i2c_check_status(i2c, false);
> > +
> > + /* read data in FIFO */
> > + octeon_i2c_writeq_flush(TWSI_BLOCK_STS_RESET_PTR, i2c-
> > >twsi_base + TWSI_BLOCK_STS(i2c));
> > + for (i = 0; i < len; i += 8) {
> > + rd = __raw_readq(i2c->twsi_base +
> > TWSI_BLOCK_FIFO(i2c));
> > + for (j = 7; j >= 0; j--)
>
> is len always a multiple of 8?
>
> > + msgs[1].buf[i + (7 - j)] = (rd >> (8 * j)) &
> > 0xff;
> > + }
> > +
> > + octeon_i2c_block_disable(i2c);
> > + return ret;
> > +}
>
> [...]

No, but this doesnt really matter as the internal j loop will take care
of the remaining bytes.

e.g. if the len is 9, then we will do
0-7 in the first, then i = 8, which is < len, and then the internal
loop will do 8-17.
>
> > - msgs[1].len > 0 && msgs[1].len <= 8 &&
> > + msgs[1].len > 0 &&
> > msgs[0].addr == msgs[1].addr) {
> > - if (msgs[1].flags & I2C_M_RD)
> > - ret = octeon_i2c_hlc_comp_read(i2c,
> > msgs);
> > - else
> > - ret = octeon_i2c_hlc_comp_write(i2c,
> > msgs);
> > - goto out;
> > + if (msgs[1].len <= 8) {
> > + if (msgs[1].flags & I2C_M_RD)
> > + ret =
> > octeon_i2c_hlc_comp_read(i2c, msgs);
> > + else
> > + ret =
> > octeon_i2c_hlc_comp_write(i2c, msgs);
> > + goto out;
> > + } else if (msgs[1].len <= 1024 &&
> > TWSI_BLOCK_CTL(i2c)) {
> > + if (msgs[1].flags & I2C_M_RD)
> > + ret =
> > octeon_i2c_hlc_block_comp_read(i2c, msgs);
> > + else
> > + ret =
> > octeon_i2c_hlc_block_comp_write(i2c, msgs);
> > + goto out;
> > + }
>
> the rest looks good...
>
> Thanks,
> Andi

Thanks,
Aryan