/* * 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);