| /* |
| * Copyright 2012 Jean-Christophe PLAGNIOL-VILLARD <plagnioj@jcrosoft.com> |
| * |
| * OF helpers for mtd. |
| * |
| * This file is released under the GPLv2 |
| * |
| */ |
| #include <linux/kernel.h> |
| #include <linux/of_mtd.h> |
| #include <linux/mtd/nand.h> |
| #include <linux/export.h> |
| |
| /** |
| * It maps 'enum nand_ecc_modes_t' found in include/linux/mtd/nand.h |
| * into the device tree binding of 'nand-ecc', so that MTD |
| * device driver can get nand ecc from device tree. |
| */ |
| static const char *nand_ecc_modes[] = { |
| [NAND_ECC_NONE] = "none", |
| [NAND_ECC_SOFT] = "soft", |
| [NAND_ECC_HW] = "hw", |
| [NAND_ECC_HW_SYNDROME] = "hw_syndrome", |
| [NAND_ECC_HW_OOB_FIRST] = "hw_oob_first", |
| [NAND_ECC_SOFT_BCH] = "soft_bch", |
| }; |
| |
| /** |
| * of_get_nand_ecc_mode - Get nand ecc mode for given device_node |
| * @np: Pointer to the given device_node |
| * |
| * The function gets ecc mode string from property 'nand-ecc-mode', |
| * and return its index in nand_ecc_modes table, or errno in error case. |
| */ |
| int of_get_nand_ecc_mode(struct device_node *np) |
| { |
| const char *pm; |
| int err, i; |
| |
| err = of_property_read_string(np, "nand-ecc-mode", &pm); |
| if (err < 0) |
| return err; |
| |
| for (i = 0; i < ARRAY_SIZE(nand_ecc_modes); i++) |
| if (!strcasecmp(pm, nand_ecc_modes[i])) |
| return i; |
| |
| return -ENODEV; |
| } |
| EXPORT_SYMBOL_GPL(of_get_nand_ecc_mode); |
| |
| /** |
| * of_get_nand_ecc_step_size - Get ECC step size associated to |
| * the required ECC strength (see below). |
| * @np: Pointer to the given device_node |
| * |
| * return the ECC step size, or errno in error case. |
| */ |
| int of_get_nand_ecc_step_size(struct device_node *np) |
| { |
| int ret; |
| u32 val; |
| |
| ret = of_property_read_u32(np, "nand-ecc-step-size", &val); |
| return ret ? ret : val; |
| } |
| EXPORT_SYMBOL_GPL(of_get_nand_ecc_step_size); |
| |
| /** |
| * of_get_nand_ecc_strength - Get required ECC strength over the |
| * correspnding step size as defined by 'nand-ecc-size' |
| * @np: Pointer to the given device_node |
| * |
| * return the ECC strength, or errno in error case. |
| */ |
| int of_get_nand_ecc_strength(struct device_node *np) |
| { |
| int ret; |
| u32 val; |
| |
| ret = of_property_read_u32(np, "nand-ecc-strength", &val); |
| return ret ? ret : val; |
| } |
| EXPORT_SYMBOL_GPL(of_get_nand_ecc_strength); |
| |
| /** |
| * of_get_nand_bus_width - Get nand bus witdh for given device_node |
| * @np: Pointer to the given device_node |
| * |
| * return bus width option, or errno in error case. |
| */ |
| int of_get_nand_bus_width(struct device_node *np) |
| { |
| u32 val; |
| |
| if (of_property_read_u32(np, "nand-bus-width", &val)) |
| return 8; |
| |
| switch(val) { |
| case 8: |
| case 16: |
| return val; |
| default: |
| return -EIO; |
| } |
| } |
| EXPORT_SYMBOL_GPL(of_get_nand_bus_width); |
| |
| /** |
| * of_get_nand_on_flash_bbt - Get nand on flash bbt for given device_node |
| * @np: Pointer to the given device_node |
| * |
| * return true if present false other wise |
| */ |
| bool of_get_nand_on_flash_bbt(struct device_node *np) |
| { |
| return of_property_read_bool(np, "nand-on-flash-bbt"); |
| } |
| EXPORT_SYMBOL_GPL(of_get_nand_on_flash_bbt); |