[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <20150403195751.GD2016@katana>
Date: Fri, 3 Apr 2015 21:57:51 +0200
From: Wolfram Sang <wsa@...-dreams.de>
To: Andrey Danin <danindrey@...l.ru>
Cc: devicetree@...r.kernel.org, linux-i2c@...r.kernel.org,
linux-arm-kernel@...ts.infradead.org, linux-tegra@...r.kernel.org,
linux-kernel@...r.kernel.org, ac100@...ts.launchpad.net,
Laxman Dewangan <ldewangan@...dia.com>,
Rob Herring <robh+dt@...nel.org>,
Pawel Moll <pawel.moll@....com>,
Mark Rutland <mark.rutland@....com>,
Ian Campbell <ijc+devicetree@...lion.org.uk>,
Kumar Gala <galak@...eaurora.org>,
Russell King <linux@....linux.org.uk>,
Stephen Warren <swarren@...dotorg.org>,
Thierry Reding <thierry.reding@...il.com>,
Alexandre Courbot <gnurou@...il.com>,
Greg Kroah-Hartman <gregkh@...uxfoundation.org>,
Julian Andres Klode <jak@...-linux.org>,
Marc Dietrich <marvin24@....de>
Subject: Re: [PATCH v2 2/4] staging/nvec: reimplement on top of tegra i2c
driver
> +/**
> + * nvec_slave_cb - I2C slave callback
> + *
> + * This callback fills our RX buffers and empties our TX
> + * buffers. This uses a finite state machine.
> + */
> +static int nvec_slave_cb(struct i2c_client *client,
> + enum i2c_slave_event event, u8 *val)
> +{
> + struct nvec_chip *nvec = i2c_get_clientdata(client);
> +
> + switch (event) {
> + case I2C_SLAVE_WRITE_REQUESTED:
> + /* Alloc new msg only if prev transaction finished */
> + if (nvec->state == ST_NONE)
> + nvec->rx = nvec_msg_alloc(nvec, NVEC_MSG_RX);
> +
> + /* Should not happen in a normal world */
> + if (unlikely(nvec->rx == NULL)) {
> + nvec->state = ST_NONE;
> + return -1;
> + }
> + nvec->rx->pos = 0;
> +
> + if (client->addr != ((*val) >> 1)) {
Uh, I2C_SLAVE_WRITE_REQUESTED should not use val.
> + dev_err(&client->dev,
> + "received address 0x%02x, expected 0x%02x\n",
> + ((*val) >> 1), client->addr);
> + return -1;
> + }
> + nvec->state = ST_TRANS_START;
> + break;
> +
...
> + case I2C_SLAVE_READ_PROCESSED:
> + if (nvec->state != ST_RX &&
> + nvec->state != ST_TX) {
> + dev_err(&client->dev,
> + "unexpected read: state %d\n",
> + nvec->state);
> + return -1;
> + }
> +
> + if (!nvec->tx || nvec->tx->pos >= nvec->tx->size) {
> + dev_err(nvec->dev,
> + "tx buffer underflow on %p (%u > %u)\n",
> + nvec->tx,
> + (uint) (nvec->tx ? nvec->tx->pos : 0),
> + (uint) (nvec->tx ? nvec->tx->size : 0));
> + nvec->state = ST_NONE;
> + break;
> + }
> +
> + nvec->state = ST_TX;
> + *val = nvec->tx->data[nvec->tx->pos++];
Are you sure you want to increase the pointer here? Remember that this
byte is requested but might not be sent out if the remote master stops
the transfer after the previous byte using NACK instead of ACK.
> + break;
> +
Download attachment "signature.asc" of type "application/pgp-signature" (820 bytes)
Powered by blists - more mailing lists