This patch generalises the calculation of block protect bits based on the number
of sectors and implements the _is_locked function.

Existing calculation of block protect bits only works for devices with 64
sectors or more. This new logic is applicable to the STmicro devices:
m25p10, p20, p40, p80, p16, pe16, p32, p64, p128.
Note devices with >64 sectors only allow the protected region to be specified
to a resolution of 1/64th of the total size (such as m25p64).

New return codes for ioctl(MEMISLOCKED) have been added to uapi/mtd/mtd-abi.h
because the _is_locked function can query a region which is partially unlocked.

Added flag to m25p_ids table to indicate if flash protection is supported.

Added n_sectors and sector_size to m25p flash structure so it can be used in
block protect bit calculation.

From: Austin Boyle <boyle.aus...@gmail.com>
Signed-off-by: Austin Boyle <boyle.aus...@gmail.com>
---
diff --git a/drivers/mtd/devices/m25p80.c b/drivers/mtd/devices/m25p80.c
index ad19139..f632e41 100644
--- a/drivers/mtd/devices/m25p80.c
+++ b/drivers/mtd/devices/m25p80.c
@@ -77,8 +77,13 @@
 #define        SR_BP0                  4       /* Block protect 0 */
 #define        SR_BP1                  8       /* Block protect 1 */
 #define        SR_BP2                  0x10    /* Block protect 2 */
+#define        SR_BP_BIT_OFFSET        2       /* Offset to Block protect 0 */
+#define        SR_BP_BIT_MASK          (SR_BP2 | SR_BP1 | SR_BP0)
 #define        SR_SRWD                 0x80    /* SR write protect */
 
+/* Highest resolution of sector locking */
+#define M25P_MAX_LOCKABLE_SECTORS      64
+
 #define SR_QUAD_EN_MX           0x40    /* Macronix Quad I/O */
 
 /* Configuration Register bits. */
@@ -104,6 +109,8 @@ struct m25p {
        struct mtd_info         mtd;
        u16                     page_size;
        u16                     addr_width;
+       u16                     n_sectors;
+       u32                     sector_size;
        u8                      erase_opcode;
        u8                      read_opcode;
        u8                      program_opcode;
@@ -736,11 +743,25 @@ time_out:
        return ret;
 }
 
+static inline uint16_t min_lockable_sectors(uint16_t n_sectors)
+{
+       return max(1, n_sectors/M25P_MAX_LOCKABLE_SECTORS);
+}
+
+static inline uint32_t get_protected_area(struct m25p *flash,
+                                                       uint8_t lock_bits)
+{
+       return (1<<(lock_bits-1)) * min_lockable_sectors(flash->n_sectors) *
+                                                       flash->sector_size;
+}
+
 static int m25p80_lock(struct mtd_info *mtd, loff_t ofs, uint64_t len)
 {
        struct m25p *flash = mtd_to_m25p(mtd);
        uint32_t offset = ofs;
        uint8_t status_old, status_new;
+       uint8_t lock_bits;
+       uint32_t protected_area_start;
        int res = 0;
 
        mutex_lock(&flash->lock);
@@ -752,24 +773,18 @@ static int m25p80_lock(struct mtd_info *mtd, loff_t ofs, 
uint64_t len)
 
        status_old = read_sr(flash);
 
-       if (offset < flash->mtd.size-(flash->mtd.size/2))
-               status_new = status_old | SR_BP2 | SR_BP1 | SR_BP0;
-       else if (offset < flash->mtd.size-(flash->mtd.size/4))
-               status_new = (status_old & ~SR_BP0) | SR_BP2 | SR_BP1;
-       else if (offset < flash->mtd.size-(flash->mtd.size/8))
-               status_new = (status_old & ~SR_BP1) | SR_BP2 | SR_BP0;
-       else if (offset < flash->mtd.size-(flash->mtd.size/16))
-               status_new = (status_old & ~(SR_BP0|SR_BP1)) | SR_BP2;
-       else if (offset < flash->mtd.size-(flash->mtd.size/32))
-               status_new = (status_old & ~SR_BP2) | SR_BP1 | SR_BP0;
-       else if (offset < flash->mtd.size-(flash->mtd.size/64))
-               status_new = (status_old & ~(SR_BP2|SR_BP0)) | SR_BP1;
-       else
-               status_new = (status_old & ~(SR_BP2|SR_BP1)) | SR_BP0;
+       for (lock_bits = 1; lock_bits < 7; lock_bits++) {
+               protected_area_start = flash->mtd.size -
+                                       get_protected_area(flash, lock_bits);
+               if (offset >= protected_area_start)
+                       break;
+       }
+
+       status_new = (status_old & ~SR_BP_BIT_MASK) |
+                       ((lock_bits << SR_BP_BIT_OFFSET) & SR_BP_BIT_MASK);
 
        /* Only modify protection if it will not unlock other areas */
-       if ((status_new&(SR_BP2|SR_BP1|SR_BP0)) >
-                                       (status_old&(SR_BP2|SR_BP1|SR_BP0))) {
+       if ((status_new & SR_BP_BIT_MASK) > (status_old & SR_BP_BIT_MASK)) {
                write_enable(flash);
                if (write_sr(flash, status_new) < 0) {
                        res = 1;
@@ -786,6 +801,8 @@ static int m25p80_unlock(struct mtd_info *mtd, loff_t ofs, 
uint64_t len)
        struct m25p *flash = mtd_to_m25p(mtd);
        uint32_t offset = ofs;
        uint8_t status_old, status_new;
+       uint8_t lock_bits;
+       uint32_t protected_area_start;
        int res = 0;
 
        mutex_lock(&flash->lock);
@@ -797,24 +814,19 @@ static int m25p80_unlock(struct mtd_info *mtd, loff_t 
ofs, uint64_t len)
 
        status_old = read_sr(flash);
 
-       if (offset+len > flash->mtd.size-(flash->mtd.size/64))
-               status_new = status_old & ~(SR_BP2|SR_BP1|SR_BP0);
-       else if (offset+len > flash->mtd.size-(flash->mtd.size/32))
-               status_new = (status_old & ~(SR_BP2|SR_BP1)) | SR_BP0;
-       else if (offset+len > flash->mtd.size-(flash->mtd.size/16))
-               status_new = (status_old & ~(SR_BP2|SR_BP0)) | SR_BP1;
-       else if (offset+len > flash->mtd.size-(flash->mtd.size/8))
-               status_new = (status_old & ~SR_BP2) | SR_BP1 | SR_BP0;
-       else if (offset+len > flash->mtd.size-(flash->mtd.size/4))
-               status_new = (status_old & ~(SR_BP0|SR_BP1)) | SR_BP2;
-       else if (offset+len > flash->mtd.size-(flash->mtd.size/2))
-               status_new = (status_old & ~SR_BP1) | SR_BP2 | SR_BP0;
-       else
-               status_new = (status_old & ~SR_BP0) | SR_BP2 | SR_BP1;
+       for (lock_bits = 1; lock_bits < 7; lock_bits++) {
+               protected_area_start = flash->mtd.size -
+                                       get_protected_area(flash, lock_bits);
+               if (offset+len >= protected_area_start)
+                       break;
+       }
+       lock_bits--;
+
+       status_new = (status_old & ~SR_BP_BIT_MASK) |
+                       ((lock_bits << SR_BP_BIT_OFFSET) & SR_BP_BIT_MASK);
 
        /* Only modify protection if it will not lock other areas */
-       if ((status_new&(SR_BP2|SR_BP1|SR_BP0)) <
-                                       (status_old&(SR_BP2|SR_BP1|SR_BP0))) {
+       if ((status_new & SR_BP_BIT_MASK) < (status_old & SR_BP_BIT_MASK)) {
                write_enable(flash);
                if (write_sr(flash, status_new) < 0) {
                        res = 1;
@@ -826,6 +838,39 @@ err:       mutex_unlock(&flash->lock);
        return res;
 }
 
+static int m25p80_is_locked(struct mtd_info *mtd, loff_t ofs, uint64_t len)
+{
+       struct m25p *flash = mtd_to_m25p(mtd);
+       uint32_t offset = ofs;
+       uint8_t status;
+       uint8_t lock_bits;
+       uint32_t protected_area_start;
+       int res;
+
+       mutex_lock(&flash->lock);
+       /* Wait until finished previous command */
+       if (wait_till_ready(flash)) {
+               mutex_unlock(&flash->lock);
+               return -EBUSY;
+       }
+       status = read_sr(flash);
+       mutex_unlock(&flash->lock);
+
+       lock_bits = ((status & SR_BP_BIT_MASK) >> SR_BP_BIT_OFFSET);
+
+       protected_area_start = flash->mtd.size -
+                                       get_protected_area(flash, lock_bits);
+
+       if (offset > protected_area_start)
+               res = MTD_IS_LOCKED;
+       else if (offset+len < protected_area_start)
+               res = MTD_IS_UNLOCKED;
+       else
+               res = MTD_IS_PARTIALLY_LOCKED;
+
+       return res;
+}
+
 /****************************************************************************/
 
 /*
@@ -856,6 +901,7 @@ struct flash_info {
 #define        M25P_NO_FR      0x08            /* Can't do fastread */
 #define        SECT_4K_PMC     0x10            /* OPCODE_BE_4K_PMC works 
uniformly */
 #define        M25P80_QUAD_READ        0x20    /* Flash supports Quad Read */
+#define        M25P_FLASH_LOCK 0x40            /* Flash protection support */
 };
 
 #define INFO(_jedec_id, _ext_id, _sector_size, _n_sectors, _flags)     \
@@ -981,25 +1027,25 @@ static const struct spi_device_id m25p_ids[] = {
 
        /* ST Microelectronics -- newer production may have feature updates */
        { "m25p05",  INFO(0x202010,  0,  32 * 1024,   2, 0) },
-       { "m25p10",  INFO(0x202011,  0,  32 * 1024,   4, 0) },
-       { "m25p20",  INFO(0x202012,  0,  64 * 1024,   4, 0) },
-       { "m25p40",  INFO(0x202013,  0,  64 * 1024,   8, 0) },
-       { "m25p80",  INFO(0x202014,  0,  64 * 1024,  16, 0) },
-       { "m25p16",  INFO(0x202015,  0,  64 * 1024,  32, 0) },
-       { "m25p32",  INFO(0x202016,  0,  64 * 1024,  64, 0) },
-       { "m25p64",  INFO(0x202017,  0,  64 * 1024, 128, 0) },
-       { "m25p128", INFO(0x202018,  0, 256 * 1024,  64, 0) },
+       { "m25p10",  INFO(0x202011,  0,  32 * 1024,   4, M25P_FLASH_LOCK) },
+       { "m25p20",  INFO(0x202012,  0,  64 * 1024,   4, M25P_FLASH_LOCK) },
+       { "m25p40",  INFO(0x202013,  0,  64 * 1024,   8, M25P_FLASH_LOCK) },
+       { "m25p80",  INFO(0x202014,  0,  64 * 1024,  16, M25P_FLASH_LOCK) },
+       { "m25p16",  INFO(0x202015,  0,  64 * 1024,  32, M25P_FLASH_LOCK) },
+       { "m25p32",  INFO(0x202016,  0,  64 * 1024,  64, M25P_FLASH_LOCK) },
+       { "m25p64",  INFO(0x202017,  0,  64 * 1024, 128, M25P_FLASH_LOCK) },
+       { "m25p128", INFO(0x202018,  0, 256 * 1024,  64, M25P_FLASH_LOCK) },
        { "n25q032", INFO(0x20ba16,  0,  64 * 1024,  64, 0) },
 
        { "m25p05-nonjedec",  INFO(0, 0,  32 * 1024,   2, 0) },
-       { "m25p10-nonjedec",  INFO(0, 0,  32 * 1024,   4, 0) },
-       { "m25p20-nonjedec",  INFO(0, 0,  64 * 1024,   4, 0) },
-       { "m25p40-nonjedec",  INFO(0, 0,  64 * 1024,   8, 0) },
-       { "m25p80-nonjedec",  INFO(0, 0,  64 * 1024,  16, 0) },
-       { "m25p16-nonjedec",  INFO(0, 0,  64 * 1024,  32, 0) },
-       { "m25p32-nonjedec",  INFO(0, 0,  64 * 1024,  64, 0) },
-       { "m25p64-nonjedec",  INFO(0, 0,  64 * 1024, 128, 0) },
-       { "m25p128-nonjedec", INFO(0, 0, 256 * 1024,  64, 0) },
+       { "m25p10-nonjedec",  INFO(0, 0,  32 * 1024,   4, M25P_FLASH_LOCK) },
+       { "m25p20-nonjedec",  INFO(0, 0,  64 * 1024,   4, M25P_FLASH_LOCK) },
+       { "m25p40-nonjedec",  INFO(0, 0,  64 * 1024,   8, M25P_FLASH_LOCK) },
+       { "m25p80-nonjedec",  INFO(0, 0,  64 * 1024,  16, M25P_FLASH_LOCK) },
+       { "m25p16-nonjedec",  INFO(0, 0,  64 * 1024,  32, M25P_FLASH_LOCK) },
+       { "m25p32-nonjedec",  INFO(0, 0,  64 * 1024,  64, M25P_FLASH_LOCK) },
+       { "m25p64-nonjedec",  INFO(0, 0,  64 * 1024, 128, M25P_FLASH_LOCK) },
+       { "m25p128-nonjedec", INFO(0, 0, 256 * 1024,  64, M25P_FLASH_LOCK) },
 
        { "m45pe10", INFO(0x204011,  0, 64 * 1024,    2, 0) },
        { "m45pe80", INFO(0x204014,  0, 64 * 1024,   16, 0) },
@@ -1007,7 +1053,7 @@ static const struct spi_device_id m25p_ids[] = {
 
        { "m25pe20", INFO(0x208012,  0, 64 * 1024,  4,       0) },
        { "m25pe80", INFO(0x208014,  0, 64 * 1024, 16,       0) },
-       { "m25pe16", INFO(0x208015,  0, 64 * 1024, 32, SECT_4K) },
+       { "m25pe16", INFO(0x208015,  0, 64 * 1024, 32, SECT_4K | 
M25P_FLASH_LOCK) },
 
        { "m25px16",    INFO(0x207115,  0, 64 * 1024, 32, SECT_4K) },
        { "m25px32",    INFO(0x207116,  0, 64 * 1024, 64, SECT_4K) },
@@ -1176,13 +1222,17 @@ static int m25p_probe(struct spi_device *spi)
        flash->mtd.writesize = 1;
        flash->mtd.flags = MTD_CAP_NORFLASH;
        flash->mtd.size = info->sector_size * info->n_sectors;
+       flash->n_sectors = info->n_sectors;
+       flash->sector_size = info->sector_size;
        flash->mtd._erase = m25p80_erase;
        flash->mtd._read = m25p80_read;
 
        /* flash protection support for STmicro chips */
-       if (JEDEC_MFR(info->jedec_id) == CFI_MFR_ST) {
+       if (JEDEC_MFR(info->jedec_id) == CFI_MFR_ST &&
+                                       (info->flags & M25P_FLASH_LOCK)) {
                flash->mtd._lock = m25p80_lock;
                flash->mtd._unlock = m25p80_unlock;
+               flash->mtd._is_locked = m25p80_is_locked;
        }
 
        /* sst flash chips use AAI word program */
diff --git a/include/uapi/mtd/mtd-abi.h b/include/uapi/mtd/mtd-abi.h
index e272ea0..f8f5e9d 100644
--- a/include/uapi/mtd/mtd-abi.h
+++ b/include/uapi/mtd/mtd-abi.h
@@ -251,7 +251,7 @@ struct mtd_ecc_stats {
        __u32 bbtblocks;
 };
 
-/*
+/**
  * MTD file modes - for read/write access to MTD
  *
  * @MTD_FILE_MODE_NORMAL:      OTP disabled, ECC enabled
@@ -275,6 +275,19 @@ enum mtd_file_modes {
        MTD_FILE_MODE_RAW,
 };
 
+/**
+ * MTD locking states - return codes for ioctl(MEMISLOCKED)
+ *
+ * @MTD_IS_UNLOCKED:           Specified region is completely unlocked
+ * @MTD_IS_LOCKED:             Specified region is completely locked
+ * @MTD_IS_PARTIALLY_LOCKED:   Specified region is partially locked
+ */
+enum mtd_locking_states {
+       MTD_IS_UNLOCKED,
+       MTD_IS_LOCKED,
+       MTD_IS_PARTIALLY_LOCKED,
+};
+
 static inline int mtd_type_is_nand_user(const struct mtd_info_user *mtd)
 {
        return mtd->type == MTD_NANDFLASH || mtd->type == MTD_MLCNANDFLASH;
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to