From: Boris Brezillon <boris.brezillon@bootlin.com>
To: Yogesh Narayan Gaur <yogeshnarayan.gaur@nxp.com>
Cc: Mark Brown <broonie@kernel.org>,
"linux-spi@vger.kernel.org" <linux-spi@vger.kernel.org>,
David Woodhouse <dwmw2@infradead.org>,
Brian Norris <computersforpeace@gmail.com>,
Marek Vasut <marek.vasut@gmail.com>,
Richard Weinberger <richard@nod.at>,
"linux-mtd@lists.infradead.org" <linux-mtd@lists.infradead.org>,
Vignesh R <vigneshr@ti.com>,
Cyrille Pitchen <cyrille.pitchen@microchip.com>,
Tudor Ambarus <tudor.ambarus@microchip.com>,
Frieder Schrempf <frieder.schrempf@exceet.de>,
Miquel Raynal <miquel.raynal@bootlin.com>
Subject: Re: [PATCH v2 5/7] mtd: devices: m25p80: Use the spi-mem dirmap API
Date: Wed, 31 Oct 2018 12:03:55 +0100 [thread overview]
Message-ID: <20181031120355.4c2da37e@bbrezillon> (raw)
In-Reply-To: <20181031114951.55f662cc@bbrezillon>
On Wed, 31 Oct 2018 11:49:51 +0100
Boris Brezillon <boris.brezillon@bootlin.com> wrote:
> On Wed, 31 Oct 2018 10:28:04 +0000
> Yogesh Narayan Gaur <yogeshnarayan.gaur@nxp.com> wrote:
>
>
> > > +static int m25p_create_read_dirmap(struct m25p *flash) {
> > > + struct spi_nor *nor = &flash->spi_nor;
> > > + struct spi_mem_dirmap_info info = {
> > > + .op_tmpl = SPI_MEM_OP(SPI_MEM_OP_CMD(nor-
> > > >read_opcode, 1),
> > > + SPI_MEM_OP_ADDR(nor->addr_width, 0, 1),
> > > + SPI_MEM_OP_DUMMY(nor->read_dummy,
> > > 1),
> > > + SPI_MEM_OP_DATA_IN(0, NULL, 1)),
> > > + .offset = 0,
> > > + .length = flash->spi_nor.mtd.size,
> >
> > I want to understand the sequence of calling up of this routine from drivers/mtd/spi-nor/spi-nor.c file.
> > Because till point of JedecID read and SFDP command read mtd.size is not being populated and having value as 0. This would make length as 0.
>
> When the dirmap is created mtd->size has been initialized already. But
> you're right, nor->read() might be called before the read dirmap has
> been created (SFDP read), which will lead to a NULL pointer exception.
The following version should fix the problem:
--->8---
diff --git a/drivers/mtd/devices/m25p80.c b/drivers/mtd/devices/m25p80.c
index c4a1d04b8c80..847188e8a99e 100644
--- a/drivers/mtd/devices/m25p80.c
+++ b/drivers/mtd/devices/m25p80.c
@@ -31,8 +31,70 @@
struct m25p {
struct spi_mem *spimem;
struct spi_nor spi_nor;
+ struct {
+ struct spi_mem_dirmap_desc *rdesc;
+ struct spi_mem_dirmap_desc *wdesc;
+ } dirmap;
};
+static int m25p_create_write_dirmap(struct m25p *flash)
+{
+ struct spi_nor *nor = &flash->spi_nor;
+ struct spi_mem_dirmap_info info = {
+ .op_tmpl = SPI_MEM_OP(SPI_MEM_OP_CMD(nor->program_opcode, 1),
+ SPI_MEM_OP_ADDR(nor->addr_width, 0, 1),
+ SPI_MEM_OP_NO_DUMMY,
+ SPI_MEM_OP_DATA_OUT(0, NULL, 1)),
+ .offset = 0,
+ .length = flash->spi_nor.mtd.size,
+ };
+ struct spi_mem_op *op = &info.op_tmpl;
+
+ /* get transfer protocols. */
+ op->cmd.buswidth = spi_nor_get_protocol_inst_nbits(nor->write_proto);
+ op->addr.buswidth = spi_nor_get_protocol_addr_nbits(nor->write_proto);
+ op->dummy.buswidth = op->addr.buswidth;
+ op->data.buswidth = spi_nor_get_protocol_data_nbits(nor->write_proto);
+
+ if (nor->program_opcode == SPINOR_OP_AAI_WP && nor->sst_write_second)
+ op->addr.nbytes = 0;
+
+ flash->dirmap.wdesc = spi_mem_dirmap_create(flash->spimem, &info);
+ if (IS_ERR(flash->dirmap.wdesc))
+ return PTR_ERR(flash->dirmap.wdesc);
+
+ return 0;
+}
+
+static int m25p_create_read_dirmap(struct m25p *flash)
+{
+ struct spi_nor *nor = &flash->spi_nor;
+ struct spi_mem_dirmap_info info = {
+ .op_tmpl = SPI_MEM_OP(SPI_MEM_OP_CMD(nor->read_opcode, 1),
+ SPI_MEM_OP_ADDR(nor->addr_width, 0, 1),
+ SPI_MEM_OP_DUMMY(nor->read_dummy, 1),
+ SPI_MEM_OP_DATA_IN(0, NULL, 1)),
+ .offset = 0,
+ .length = flash->spi_nor.mtd.size,
+ };
+ struct spi_mem_op *op = &info.op_tmpl;
+
+ /* get transfer protocols. */
+ op->cmd.buswidth = spi_nor_get_protocol_inst_nbits(nor->read_proto);
+ op->addr.buswidth = spi_nor_get_protocol_addr_nbits(nor->read_proto);
+ op->dummy.buswidth = op->addr.buswidth;
+ op->data.buswidth = spi_nor_get_protocol_data_nbits(nor->read_proto);
+
+ /* convert the dummy cycles to the number of bytes */
+ op->dummy.nbytes = (nor->read_dummy * op->dummy.buswidth) / 8;
+
+ flash->dirmap.rdesc = spi_mem_dirmap_create(flash->spimem, &info);
+ if (IS_ERR(flash->dirmap.rdesc))
+ return PTR_ERR(flash->dirmap.rdesc);
+
+ return 0;
+}
+
static int m25p80_read_reg(struct spi_nor *nor, u8 code, u8 *val, int len)
{
struct m25p *flash = nor->priv;
@@ -92,6 +154,9 @@ static ssize_t m25p80_write(struct spi_nor *nor, loff_t to, size_t len,
SPI_MEM_OP_DATA_OUT(len, buf, 1));
int ret;
+ if (flash->dirmap.wdesc)
+ return spi_mem_dirmap_write(flash->dirmap.wdesc, to, len, buf);
+
/* get transfer protocols. */
op.cmd.buswidth = spi_nor_get_protocol_inst_nbits(nor->write_proto);
op.addr.buswidth = spi_nor_get_protocol_addr_nbits(nor->write_proto);
@@ -128,6 +193,9 @@ static ssize_t m25p80_read(struct spi_nor *nor, loff_t from, size_t len,
size_t remaining = len;
int ret;
+ if (flash->dirmap.rdesc)
+ return spi_mem_dirmap_read(flash->dirmap.rdesc, from, len, buf);
+
/* get transfer protocols. */
op.cmd.buswidth = spi_nor_get_protocol_inst_nbits(nor->read_proto);
op.addr.buswidth = spi_nor_get_protocol_addr_nbits(nor->read_proto);
@@ -231,19 +299,47 @@ static int m25p_probe(struct spi_mem *spimem)
if (ret)
return ret;
- return mtd_device_register(&nor->mtd, data ? data->parts : NULL,
- data ? data->nr_parts : 0);
+ ret = m25p_create_write_dirmap(flash);
+ if (ret)
+ return ret;
+
+ ret = m25p_create_read_dirmap(flash);
+ if (ret)
+ goto err_destroy_write_dirmap;
+
+ ret = mtd_device_register(&nor->mtd, data ? data->parts : NULL,
+ data ? data->nr_parts : 0);
+ if (ret)
+ goto err_destroy_read_dirmap;
+
+ return 0;
+
+err_destroy_read_dirmap:
+ spi_mem_dirmap_destroy(flash->dirmap.rdesc);
+
+err_destroy_write_dirmap:
+ spi_mem_dirmap_destroy(flash->dirmap.wdesc);
+
+ return ret;
}
static int m25p_remove(struct spi_mem *spimem)
{
struct m25p *flash = spi_mem_get_drvdata(spimem);
+ int ret;
spi_nor_restore(&flash->spi_nor);
/* Clean up MTD stuff. */
- return mtd_device_unregister(&flash->spi_nor.mtd);
+ ret = mtd_device_unregister(&flash->spi_nor.mtd);
+ if (ret)
+ return ret;
+
+ spi_mem_dirmap_destroy(flash->dirmap.rdesc);
+ spi_mem_dirmap_destroy(flash->dirmap.wdesc);
+
+ return 0;
}
static void m25p_shutdown(struct spi_mem *spimem)
WARNING: multiple messages have this Message-ID (diff)
From: Boris Brezillon <boris.brezillon@bootlin.com>
To: Yogesh Narayan Gaur <yogeshnarayan.gaur@nxp.com>
Cc: Vignesh R <vigneshr@ti.com>,
Tudor Ambarus <tudor.ambarus@microchip.com>,
Richard Weinberger <richard@nod.at>,
Cyrille Pitchen <cyrille.pitchen@microchip.com>,
"linux-spi@vger.kernel.org" <linux-spi@vger.kernel.org>,
Marek Vasut <marek.vasut@gmail.com>,
Frieder Schrempf <frieder.schrempf@exceet.de>,
Mark Brown <broonie@kernel.org>,
"linux-mtd@lists.infradead.org" <linux-mtd@lists.infradead.org>,
Miquel Raynal <miquel.raynal@bootlin.com>,
Brian Norris <computersforpeace@gmail.com>,
David Woodhouse <dwmw2@infradead.org>
Subject: Re: [PATCH v2 5/7] mtd: devices: m25p80: Use the spi-mem dirmap API
Date: Wed, 31 Oct 2018 12:03:55 +0100 [thread overview]
Message-ID: <20181031120355.4c2da37e@bbrezillon> (raw)
In-Reply-To: <20181031114951.55f662cc@bbrezillon>
On Wed, 31 Oct 2018 11:49:51 +0100
Boris Brezillon <boris.brezillon@bootlin.com> wrote:
> On Wed, 31 Oct 2018 10:28:04 +0000
> Yogesh Narayan Gaur <yogeshnarayan.gaur@nxp.com> wrote:
>
>
> > > +static int m25p_create_read_dirmap(struct m25p *flash) {
> > > + struct spi_nor *nor = &flash->spi_nor;
> > > + struct spi_mem_dirmap_info info = {
> > > + .op_tmpl = SPI_MEM_OP(SPI_MEM_OP_CMD(nor-
> > > >read_opcode, 1),
> > > + SPI_MEM_OP_ADDR(nor->addr_width, 0, 1),
> > > + SPI_MEM_OP_DUMMY(nor->read_dummy,
> > > 1),
> > > + SPI_MEM_OP_DATA_IN(0, NULL, 1)),
> > > + .offset = 0,
> > > + .length = flash->spi_nor.mtd.size,
> >
> > I want to understand the sequence of calling up of this routine from drivers/mtd/spi-nor/spi-nor.c file.
> > Because till point of JedecID read and SFDP command read mtd.size is not being populated and having value as 0. This would make length as 0.
>
> When the dirmap is created mtd->size has been initialized already. But
> you're right, nor->read() might be called before the read dirmap has
> been created (SFDP read), which will lead to a NULL pointer exception.
The following version should fix the problem:
--->8---
diff --git a/drivers/mtd/devices/m25p80.c b/drivers/mtd/devices/m25p80.c
index c4a1d04b8c80..847188e8a99e 100644
--- a/drivers/mtd/devices/m25p80.c
+++ b/drivers/mtd/devices/m25p80.c
@@ -31,8 +31,70 @@
struct m25p {
struct spi_mem *spimem;
struct spi_nor spi_nor;
+ struct {
+ struct spi_mem_dirmap_desc *rdesc;
+ struct spi_mem_dirmap_desc *wdesc;
+ } dirmap;
};
+static int m25p_create_write_dirmap(struct m25p *flash)
+{
+ struct spi_nor *nor = &flash->spi_nor;
+ struct spi_mem_dirmap_info info = {
+ .op_tmpl = SPI_MEM_OP(SPI_MEM_OP_CMD(nor->program_opcode, 1),
+ SPI_MEM_OP_ADDR(nor->addr_width, 0, 1),
+ SPI_MEM_OP_NO_DUMMY,
+ SPI_MEM_OP_DATA_OUT(0, NULL, 1)),
+ .offset = 0,
+ .length = flash->spi_nor.mtd.size,
+ };
+ struct spi_mem_op *op = &info.op_tmpl;
+
+ /* get transfer protocols. */
+ op->cmd.buswidth = spi_nor_get_protocol_inst_nbits(nor->write_proto);
+ op->addr.buswidth = spi_nor_get_protocol_addr_nbits(nor->write_proto);
+ op->dummy.buswidth = op->addr.buswidth;
+ op->data.buswidth = spi_nor_get_protocol_data_nbits(nor->write_proto);
+
+ if (nor->program_opcode == SPINOR_OP_AAI_WP && nor->sst_write_second)
+ op->addr.nbytes = 0;
+
+ flash->dirmap.wdesc = spi_mem_dirmap_create(flash->spimem, &info);
+ if (IS_ERR(flash->dirmap.wdesc))
+ return PTR_ERR(flash->dirmap.wdesc);
+
+ return 0;
+}
+
+static int m25p_create_read_dirmap(struct m25p *flash)
+{
+ struct spi_nor *nor = &flash->spi_nor;
+ struct spi_mem_dirmap_info info = {
+ .op_tmpl = SPI_MEM_OP(SPI_MEM_OP_CMD(nor->read_opcode, 1),
+ SPI_MEM_OP_ADDR(nor->addr_width, 0, 1),
+ SPI_MEM_OP_DUMMY(nor->read_dummy, 1),
+ SPI_MEM_OP_DATA_IN(0, NULL, 1)),
+ .offset = 0,
+ .length = flash->spi_nor.mtd.size,
+ };
+ struct spi_mem_op *op = &info.op_tmpl;
+
+ /* get transfer protocols. */
+ op->cmd.buswidth = spi_nor_get_protocol_inst_nbits(nor->read_proto);
+ op->addr.buswidth = spi_nor_get_protocol_addr_nbits(nor->read_proto);
+ op->dummy.buswidth = op->addr.buswidth;
+ op->data.buswidth = spi_nor_get_protocol_data_nbits(nor->read_proto);
+
+ /* convert the dummy cycles to the number of bytes */
+ op->dummy.nbytes = (nor->read_dummy * op->dummy.buswidth) / 8;
+
+ flash->dirmap.rdesc = spi_mem_dirmap_create(flash->spimem, &info);
+ if (IS_ERR(flash->dirmap.rdesc))
+ return PTR_ERR(flash->dirmap.rdesc);
+
+ return 0;
+}
+
static int m25p80_read_reg(struct spi_nor *nor, u8 code, u8 *val, int len)
{
struct m25p *flash = nor->priv;
@@ -92,6 +154,9 @@ static ssize_t m25p80_write(struct spi_nor *nor, loff_t to, size_t len,
SPI_MEM_OP_DATA_OUT(len, buf, 1));
int ret;
+ if (flash->dirmap.wdesc)
+ return spi_mem_dirmap_write(flash->dirmap.wdesc, to, len, buf);
+
/* get transfer protocols. */
op.cmd.buswidth = spi_nor_get_protocol_inst_nbits(nor->write_proto);
op.addr.buswidth = spi_nor_get_protocol_addr_nbits(nor->write_proto);
@@ -128,6 +193,9 @@ static ssize_t m25p80_read(struct spi_nor *nor, loff_t from, size_t len,
size_t remaining = len;
int ret;
+ if (flash->dirmap.rdesc)
+ return spi_mem_dirmap_read(flash->dirmap.rdesc, from, len, buf);
+
/* get transfer protocols. */
op.cmd.buswidth = spi_nor_get_protocol_inst_nbits(nor->read_proto);
op.addr.buswidth = spi_nor_get_protocol_addr_nbits(nor->read_proto);
@@ -231,19 +299,47 @@ static int m25p_probe(struct spi_mem *spimem)
if (ret)
return ret;
- return mtd_device_register(&nor->mtd, data ? data->parts : NULL,
- data ? data->nr_parts : 0);
+ ret = m25p_create_write_dirmap(flash);
+ if (ret)
+ return ret;
+
+ ret = m25p_create_read_dirmap(flash);
+ if (ret)
+ goto err_destroy_write_dirmap;
+
+ ret = mtd_device_register(&nor->mtd, data ? data->parts : NULL,
+ data ? data->nr_parts : 0);
+ if (ret)
+ goto err_destroy_read_dirmap;
+
+ return 0;
+
+err_destroy_read_dirmap:
+ spi_mem_dirmap_destroy(flash->dirmap.rdesc);
+
+err_destroy_write_dirmap:
+ spi_mem_dirmap_destroy(flash->dirmap.wdesc);
+
+ return ret;
}
static int m25p_remove(struct spi_mem *spimem)
{
struct m25p *flash = spi_mem_get_drvdata(spimem);
+ int ret;
spi_nor_restore(&flash->spi_nor);
/* Clean up MTD stuff. */
- return mtd_device_unregister(&flash->spi_nor.mtd);
+ ret = mtd_device_unregister(&flash->spi_nor.mtd);
+ if (ret)
+ return ret;
+
+ spi_mem_dirmap_destroy(flash->dirmap.rdesc);
+ spi_mem_dirmap_destroy(flash->dirmap.wdesc);
+
+ return 0;
}
static void m25p_shutdown(struct spi_mem *spimem)
______________________________________________________
Linux MTD discussion mailing list
http://lists.infradead.org/mailman/listinfo/linux-mtd/
next prev parent reply other threads:[~2018-10-31 11:08 UTC|newest]
Thread overview: 24+ messages / expand[flat|nested] mbox.gz Atom feed top
2018-10-30 13:36 [PATCH v2 0/7] spi: spi-mem: Add a direct mapping API Boris Brezillon
2018-10-30 13:36 ` Boris Brezillon
2018-10-30 13:36 ` [PATCH v2 1/7] spi: spi-mem: Add missing word in the SPI_MEM_DATA_OUT description Boris Brezillon
2018-10-30 13:36 ` Boris Brezillon
2018-10-30 13:36 ` [PATCH v2 2/7] spi: spi-mem: Add SPI_MEM_NO_DATA to the spi_mem_data_dir enum Boris Brezillon
2018-10-30 13:36 ` Boris Brezillon
2018-10-30 13:36 ` [PATCH v2 3/7] spi: spi-mem: Split spi_mem_exec_op() code Boris Brezillon
2018-10-30 13:36 ` Boris Brezillon
2018-10-30 13:36 ` [PATCH v2 4/7] spi: spi-mem: Add a new API to support direct mapping Boris Brezillon
2018-10-30 13:36 ` Boris Brezillon
2018-10-30 13:36 ` [PATCH v2 5/7] mtd: devices: m25p80: Use the spi-mem dirmap API Boris Brezillon
2018-10-30 13:36 ` Boris Brezillon
2018-10-31 10:28 ` Yogesh Narayan Gaur
2018-10-31 10:28 ` Yogesh Narayan Gaur
2018-10-31 10:49 ` Boris Brezillon
2018-10-31 10:49 ` Boris Brezillon
2018-10-31 11:03 ` Boris Brezillon [this message]
2018-10-31 11:03 ` Boris Brezillon
2018-10-30 13:36 ` [PATCH v2 6/7] mtd: spinand: " Boris Brezillon
2018-10-30 13:36 ` Boris Brezillon
2018-10-30 13:36 ` [PATCH v2 7/7] spi: mxic: Add support for direct mapping Boris Brezillon
2018-10-30 13:36 ` Boris Brezillon
2018-11-06 13:21 ` [PATCH v2 0/7] spi: spi-mem: Add a direct mapping API Miquel Raynal
2018-11-06 13:21 ` Miquel Raynal
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20181031120355.4c2da37e@bbrezillon \
--to=boris.brezillon@bootlin.com \
--cc=broonie@kernel.org \
--cc=computersforpeace@gmail.com \
--cc=cyrille.pitchen@microchip.com \
--cc=dwmw2@infradead.org \
--cc=frieder.schrempf@exceet.de \
--cc=linux-mtd@lists.infradead.org \
--cc=linux-spi@vger.kernel.org \
--cc=marek.vasut@gmail.com \
--cc=miquel.raynal@bootlin.com \
--cc=richard@nod.at \
--cc=tudor.ambarus@microchip.com \
--cc=vigneshr@ti.com \
--cc=yogeshnarayan.gaur@nxp.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.