2017-01-27 13:32:10 +00:00
|
|
|
From: Hauke Mehrtens <hauke@hauke-m.de>
|
2016-12-25 19:11:34 +00:00
|
|
|
Subject: mtd: part: add generic parsing of linux,part-probe
|
2017-01-27 13:32:10 +00:00
|
|
|
|
|
|
|
This moves the linux,part-probe device tree parsing code from
|
|
|
|
physmap_of.c to mtdpart.c. Now all drivers can use this feature by just
|
|
|
|
providing a reference to their device tree node in struct
|
|
|
|
mtd_part_parser_data.
|
|
|
|
|
|
|
|
Signed-off-by: Hauke Mehrtens <hauke@hauke-m.de>
|
|
|
|
---
|
|
|
|
Documentation/devicetree/bindings/mtd/nand.txt | 16 +++++++++
|
|
|
|
drivers/mtd/maps/physmap_of.c | 46 +-------------------------
|
|
|
|
drivers/mtd/mtdpart.c | 45 +++++++++++++++++++++++++
|
|
|
|
3 files changed, 62 insertions(+), 45 deletions(-)
|
|
|
|
|
|
|
|
--- a/Documentation/devicetree/bindings/mtd/nand.txt
|
|
|
|
+++ b/Documentation/devicetree/bindings/mtd/nand.txt
|
|
|
|
@@ -44,6 +44,22 @@ Optional NAND chip properties:
|
|
|
|
used by the upper layers, and you want to make your NAND
|
|
|
|
as reliable as possible.
|
|
|
|
|
|
|
|
+- linux,part-probe: list of name as strings of the partition parser
|
|
|
|
+ which should be used to parse the partition table.
|
|
|
|
+ They will be tried in the specified ordering and
|
|
|
|
+ the next one will be used if the previous one
|
|
|
|
+ failed.
|
|
|
|
+
|
|
|
|
+ Example: linux,part-probe = "cmdlinepart", "ofpart";
|
|
|
|
+
|
|
|
|
+ This is also the default value, which will be used
|
|
|
|
+ if this attribute is not specified. It could be
|
|
|
|
+ that the flash driver in use overwrote the default
|
|
|
|
+ value and uses some other default.
|
|
|
|
+
|
|
|
|
+ Possible values are: bcm47xxpart, afs, ar7part,
|
|
|
|
+ ofoldpart, ofpart, bcm63xxpart, RedBoot, cmdlinepart
|
|
|
|
+
|
|
|
|
The ECC strength and ECC step size properties define the correction capability
|
|
|
|
of a controller. Together, they say a controller can correct "{strength} bit
|
|
|
|
errors per {size} bytes".
|
|
|
|
--- a/drivers/mtd/maps/physmap_of.c
|
|
|
|
+++ b/drivers/mtd/maps/physmap_of.c
|
2019-11-29 09:52:01 +00:00
|
|
|
@@ -104,47 +104,9 @@ static struct mtd_info *obsolete_probe(s
|
2017-01-27 13:32:10 +00:00
|
|
|
static const char * const part_probe_types_def[] = {
|
|
|
|
"cmdlinepart", "RedBoot", "ofpart", "ofoldpart", NULL };
|
|
|
|
|
|
|
|
-static const char * const *of_get_probes(struct device_node *dp)
|
|
|
|
-{
|
|
|
|
- const char *cp;
|
|
|
|
- int cplen;
|
|
|
|
- unsigned int l;
|
|
|
|
- unsigned int count;
|
|
|
|
- const char **res;
|
|
|
|
-
|
|
|
|
- cp = of_get_property(dp, "linux,part-probe", &cplen);
|
|
|
|
- if (cp == NULL)
|
|
|
|
- return part_probe_types_def;
|
|
|
|
-
|
|
|
|
- count = 0;
|
|
|
|
- for (l = 0; l != cplen; l++)
|
|
|
|
- if (cp[l] == 0)
|
|
|
|
- count++;
|
|
|
|
-
|
|
|
|
- res = kzalloc((count + 1)*sizeof(*res), GFP_KERNEL);
|
|
|
|
- if (!res)
|
|
|
|
- return NULL;
|
|
|
|
- count = 0;
|
|
|
|
- while (cplen > 0) {
|
|
|
|
- res[count] = cp;
|
|
|
|
- l = strlen(cp) + 1;
|
|
|
|
- cp += l;
|
|
|
|
- cplen -= l;
|
|
|
|
- count++;
|
|
|
|
- }
|
|
|
|
- return res;
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-static void of_free_probes(const char * const *probes)
|
|
|
|
-{
|
|
|
|
- if (probes != part_probe_types_def)
|
|
|
|
- kfree(probes);
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
static const struct of_device_id of_flash_match[];
|
|
|
|
static int of_flash_probe(struct platform_device *dev)
|
|
|
|
{
|
|
|
|
- const char * const *part_probe_types;
|
|
|
|
const struct of_device_id *match;
|
|
|
|
struct device_node *dp = dev->dev.of_node;
|
|
|
|
struct resource res;
|
2019-11-29 09:52:01 +00:00
|
|
|
@@ -300,14 +262,8 @@ static int of_flash_probe(struct platfor
|
2017-01-27 13:32:10 +00:00
|
|
|
|
|
|
|
info->cmtd->dev.parent = &dev->dev;
|
|
|
|
mtd_set_of_node(info->cmtd, dp);
|
|
|
|
- part_probe_types = of_get_probes(dp);
|
|
|
|
- if (!part_probe_types) {
|
|
|
|
- err = -ENOMEM;
|
|
|
|
- goto err_out;
|
|
|
|
- }
|
|
|
|
- mtd_device_parse_register(info->cmtd, part_probe_types, NULL,
|
|
|
|
+ mtd_device_parse_register(info->cmtd, part_probe_types_def, NULL,
|
|
|
|
NULL, 0);
|
|
|
|
- of_free_probes(part_probe_types);
|
|
|
|
|
|
|
|
kfree(mtd_list);
|
|
|
|
|
|
|
|
--- a/drivers/mtd/mtdpart.c
|
|
|
|
+++ b/drivers/mtd/mtdpart.c
|
|
|
|
@@ -29,6 +29,7 @@
|
|
|
|
#include <linux/kmod.h>
|
|
|
|
#include <linux/mtd/mtd.h>
|
|
|
|
#include <linux/mtd/partitions.h>
|
|
|
|
+#include <linux/of.h>
|
|
|
|
#include <linux/err.h>
|
2018-01-11 10:59:39 +00:00
|
|
|
#include <linux/of.h>
|
2017-01-27 13:32:10 +00:00
|
|
|
|
2019-12-06 13:06:33 +00:00
|
|
|
@@ -852,6 +853,42 @@ void deregister_mtd_parser(struct mtd_pa
|
2017-01-27 13:32:10 +00:00
|
|
|
EXPORT_SYMBOL_GPL(deregister_mtd_parser);
|
|
|
|
|
|
|
|
/*
|
|
|
|
+ * Parses the linux,part-probe device tree property.
|
|
|
|
+ * When a non null value is returned it has to be freed with kfree() by
|
|
|
|
+ * the caller.
|
|
|
|
+ */
|
|
|
|
+static const char * const *of_get_probes(struct device_node *dp)
|
|
|
|
+{
|
|
|
|
+ const char *cp;
|
|
|
|
+ int cplen;
|
|
|
|
+ unsigned int l;
|
|
|
|
+ unsigned int count;
|
|
|
|
+ const char **res;
|
|
|
|
+
|
|
|
|
+ cp = of_get_property(dp, "linux,part-probe", &cplen);
|
|
|
|
+ if (cp == NULL)
|
|
|
|
+ return NULL;
|
|
|
|
+
|
|
|
|
+ count = 0;
|
|
|
|
+ for (l = 0; l != cplen; l++)
|
|
|
|
+ if (cp[l] == 0)
|
|
|
|
+ count++;
|
|
|
|
+
|
|
|
|
+ res = kzalloc((count + 1) * sizeof(*res), GFP_KERNEL);
|
|
|
|
+ if (!res)
|
|
|
|
+ return NULL;
|
|
|
|
+ count = 0;
|
|
|
|
+ while (cplen > 0) {
|
|
|
|
+ res[count] = cp;
|
|
|
|
+ l = strlen(cp) + 1;
|
|
|
|
+ cp += l;
|
|
|
|
+ cplen -= l;
|
|
|
|
+ count++;
|
|
|
|
+ }
|
|
|
|
+ return res;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/*
|
|
|
|
* Do not forget to update 'parse_mtd_partitions()' kerneldoc comment if you
|
|
|
|
* are changing this array!
|
|
|
|
*/
|
2019-12-06 13:06:33 +00:00
|
|
|
@@ -1000,6 +1037,13 @@ int parse_mtd_partitions(struct mtd_info
|
2018-05-12 21:12:32 +00:00
|
|
|
struct mtd_partitions pparts = { };
|
2018-03-14 14:07:03 +00:00
|
|
|
struct mtd_part_parser *parser;
|
2017-01-27 13:32:10 +00:00
|
|
|
int ret, err = 0;
|
|
|
|
+ const char *const *types_of = NULL;
|
|
|
|
+
|
|
|
|
+ if (mtd_get_of_node(master)) {
|
|
|
|
+ types_of = of_get_probes(mtd_get_of_node(master));
|
|
|
|
+ if (types_of != NULL)
|
|
|
|
+ types = types_of;
|
|
|
|
+ }
|
|
|
|
|
2018-03-14 14:07:03 +00:00
|
|
|
if (!types)
|
2018-07-27 19:54:08 +00:00
|
|
|
types = mtd_is_partition(master) ? default_subpartition_types :
|
2019-12-06 13:06:33 +00:00
|
|
|
@@ -1041,6 +1085,7 @@ int parse_mtd_partitions(struct mtd_info
|
2017-01-27 13:32:10 +00:00
|
|
|
if (ret < 0 && !err)
|
|
|
|
err = ret;
|
|
|
|
}
|
|
|
|
+ kfree(types_of);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|