diff options
| author | Aaron Sierra <asierra@xes-inc.com> | 2011-11-14 19:44:34 -0500 |
|---|---|---|
| committer | David Woodhouse <David.Woodhouse@intel.com> | 2012-01-09 13:06:13 -0500 |
| commit | 8e987465a137d4824710e02550f06aa891c9b865 (patch) | |
| tree | 26edb55e4a9fe8f2f9832ba045286d5b18d3cc40 | |
| parent | 342ff28f5a2e5aa3236617bd2bddf6c749677ef2 (diff) | |
mtd: cfi: Allow per-mapping CFI device endianness
This patch allows each CFI device map to use its own endianness. The
globally defined CFI endianness (CONFIG_MTD_CFI_NOSWAP,
CONFIG_MTD_CFI_BE_BYTE_SWAP or CONFIG_MTD_CFI_LE_BYTE_SWAP) becomes the
default value which can be overridden by a driver for a particular device.
Signed-off-by: Aaron Sierra <asierra@xes-inc.com>
Signed-off-by: Artem Bityutskiy <Artem.Bityutskiy@linux.intel.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
| -rw-r--r-- | drivers/mtd/chips/cfi_cmdset_0020.c | 5 | ||||
| -rw-r--r-- | include/linux/mtd/cfi.h | 16 | ||||
| -rw-r--r-- | include/linux/mtd/cfi_endian.h | 76 | ||||
| -rw-r--r-- | include/linux/mtd/map.h | 1 |
4 files changed, 41 insertions, 57 deletions
diff --git a/drivers/mtd/chips/cfi_cmdset_0020.c b/drivers/mtd/chips/cfi_cmdset_0020.c index 179814a95f3a..666c52f8bf8d 100644 --- a/drivers/mtd/chips/cfi_cmdset_0020.c +++ b/drivers/mtd/chips/cfi_cmdset_0020.c | |||
| @@ -139,8 +139,9 @@ struct mtd_info *cfi_cmdset_0020(struct map_info *map, int primary) | |||
| 139 | } | 139 | } |
| 140 | 140 | ||
| 141 | /* Do some byteswapping if necessary */ | 141 | /* Do some byteswapping if necessary */ |
| 142 | extp->FeatureSupport = cfi32_to_cpu(extp->FeatureSupport); | 142 | extp->FeatureSupport = cfi32_to_cpu(map, extp->FeatureSupport); |
| 143 | extp->BlkStatusRegMask = cfi32_to_cpu(extp->BlkStatusRegMask); | 143 | extp->BlkStatusRegMask = cfi32_to_cpu(map, |
| 144 | extp->BlkStatusRegMask); | ||
| 144 | 145 | ||
| 145 | #ifdef DEBUG_CFI_FEATURES | 146 | #ifdef DEBUG_CFI_FEATURES |
| 146 | /* Tell the user about it in lots of lovely detail */ | 147 | /* Tell the user about it in lots of lovely detail */ |
diff --git a/include/linux/mtd/cfi.h b/include/linux/mtd/cfi.h index d24925492972..d5d2ec6494bb 100644 --- a/include/linux/mtd/cfi.h +++ b/include/linux/mtd/cfi.h | |||
| @@ -354,10 +354,10 @@ static inline map_word cfi_build_cmd(u_long cmd, struct map_info *map, struct cf | |||
| 354 | onecmd = cmd; | 354 | onecmd = cmd; |
| 355 | break; | 355 | break; |
| 356 | case 2: | 356 | case 2: |
| 357 | onecmd = cpu_to_cfi16(cmd); | 357 | onecmd = cpu_to_cfi16(map, cmd); |
| 358 | break; | 358 | break; |
| 359 | case 4: | 359 | case 4: |
| 360 | onecmd = cpu_to_cfi32(cmd); | 360 | onecmd = cpu_to_cfi32(map, cmd); |
| 361 | break; | 361 | break; |
| 362 | } | 362 | } |
| 363 | 363 | ||
| @@ -437,10 +437,10 @@ static inline unsigned long cfi_merge_status(map_word val, struct map_info *map, | |||
| 437 | case 1: | 437 | case 1: |
| 438 | break; | 438 | break; |
| 439 | case 2: | 439 | case 2: |
| 440 | res = cfi16_to_cpu(res); | 440 | res = cfi16_to_cpu(map, res); |
| 441 | break; | 441 | break; |
| 442 | case 4: | 442 | case 4: |
| 443 | res = cfi32_to_cpu(res); | 443 | res = cfi32_to_cpu(map, res); |
| 444 | break; | 444 | break; |
| 445 | default: BUG(); | 445 | default: BUG(); |
| 446 | } | 446 | } |
| @@ -480,12 +480,12 @@ static inline uint8_t cfi_read_query(struct map_info *map, uint32_t addr) | |||
| 480 | if (map_bankwidth_is_1(map)) { | 480 | if (map_bankwidth_is_1(map)) { |
| 481 | return val.x[0]; | 481 | return val.x[0]; |
| 482 | } else if (map_bankwidth_is_2(map)) { | 482 | } else if (map_bankwidth_is_2(map)) { |
| 483 | return cfi16_to_cpu(val.x[0]); | 483 | return cfi16_to_cpu(map, val.x[0]); |
| 484 | } else { | 484 | } else { |
| 485 | /* No point in a 64-bit byteswap since that would just be | 485 | /* No point in a 64-bit byteswap since that would just be |
| 486 | swapping the responses from different chips, and we are | 486 | swapping the responses from different chips, and we are |
| 487 | only interested in one chip (a representative sample) */ | 487 | only interested in one chip (a representative sample) */ |
| 488 | return cfi32_to_cpu(val.x[0]); | 488 | return cfi32_to_cpu(map, val.x[0]); |
| 489 | } | 489 | } |
| 490 | } | 490 | } |
| 491 | 491 | ||
| @@ -496,12 +496,12 @@ static inline uint16_t cfi_read_query16(struct map_info *map, uint32_t addr) | |||
| 496 | if (map_bankwidth_is_1(map)) { | 496 | if (map_bankwidth_is_1(map)) { |
| 497 | return val.x[0] & 0xff; | 497 | return val.x[0] & 0xff; |
| 498 | } else if (map_bankwidth_is_2(map)) { | 498 | } else if (map_bankwidth_is_2(map)) { |
| 499 | return cfi16_to_cpu(val.x[0]); | 499 | return cfi16_to_cpu(map, val.x[0]); |
| 500 | } else { | 500 | } else { |
| 501 | /* No point in a 64-bit byteswap since that would just be | 501 | /* No point in a 64-bit byteswap since that would just be |
| 502 | swapping the responses from different chips, and we are | 502 | swapping the responses from different chips, and we are |
| 503 | only interested in one chip (a representative sample) */ | 503 | only interested in one chip (a representative sample) */ |
| 504 | return cfi32_to_cpu(val.x[0]); | 504 | return cfi32_to_cpu(map, val.x[0]); |
| 505 | } | 505 | } |
| 506 | } | 506 | } |
| 507 | 507 | ||
diff --git a/include/linux/mtd/cfi_endian.h b/include/linux/mtd/cfi_endian.h index 51cc3f5917a8..b97a625071f8 100644 --- a/include/linux/mtd/cfi_endian.h +++ b/include/linux/mtd/cfi_endian.h | |||
| @@ -19,53 +19,35 @@ | |||
| 19 | 19 | ||
| 20 | #include <asm/byteorder.h> | 20 | #include <asm/byteorder.h> |
| 21 | 21 | ||
| 22 | #ifndef CONFIG_MTD_CFI_ADV_OPTIONS | 22 | #define CFI_HOST_ENDIAN 1 |
| 23 | 23 | #define CFI_LITTLE_ENDIAN 2 | |
| 24 | #define CFI_HOST_ENDIAN | 24 | #define CFI_BIG_ENDIAN 3 |
| 25 | 25 | ||
| 26 | #else | 26 | #if !defined(CONFIG_MTD_CFI_ADV_OPTIONS) || defined(CONFIG_MTD_CFI_NOSWAP) |
| 27 | 27 | #define CFI_DEFAULT_ENDIAN CFI_HOST_ENDIAN | |
| 28 | #ifdef CONFIG_MTD_CFI_NOSWAP | 28 | #elif defined(CONFIG_MTD_CFI_LE_BYTE_SWAP) |
| 29 | #define CFI_HOST_ENDIAN | 29 | #define CFI_DEFAULT_ENDIAN CFI_LITTLE_ENDIAN |
| 30 | #endif | 30 | #elif defined(CONFIG_MTD_CFI_BE_BYTE_SWAP) |
| 31 | 31 | #define CFI_DEFAULT_ENDIAN CFI_BIG_ENDIAN | |
| 32 | #ifdef CONFIG_MTD_CFI_LE_BYTE_SWAP | ||
| 33 | #define CFI_LITTLE_ENDIAN | ||
| 34 | #endif | ||
| 35 | |||
| 36 | #ifdef CONFIG_MTD_CFI_BE_BYTE_SWAP | ||
| 37 | #define CFI_BIG_ENDIAN | ||
| 38 | #endif | ||
| 39 | |||
| 40 | #endif | ||
| 41 | |||
| 42 | #if defined(CFI_LITTLE_ENDIAN) | ||
| 43 | #define cpu_to_cfi8(x) (x) | ||
| 44 | #define cfi8_to_cpu(x) (x) | ||
| 45 | #define cpu_to_cfi16(x) cpu_to_le16(x) | ||
| 46 | #define cpu_to_cfi32(x) cpu_to_le32(x) | ||
| 47 | #define cpu_to_cfi64(x) cpu_to_le64(x) | ||
| 48 | #define cfi16_to_cpu(x) le16_to_cpu(x) | ||
| 49 | #define cfi32_to_cpu(x) le32_to_cpu(x) | ||
| 50 | #define cfi64_to_cpu(x) le64_to_cpu(x) | ||
| 51 | #elif defined (CFI_BIG_ENDIAN) | ||
| 52 | #define cpu_to_cfi8(x) (x) | ||
| 53 | #define cfi8_to_cpu(x) (x) | ||
| 54 | #define cpu_to_cfi16(x) cpu_to_be16(x) | ||
| 55 | #define cpu_to_cfi32(x) cpu_to_be32(x) | ||
| 56 | #define cpu_to_cfi64(x) cpu_to_be64(x) | ||
| 57 | #define cfi16_to_cpu(x) be16_to_cpu(x) | ||
| 58 | #define cfi32_to_cpu(x) be32_to_cpu(x) | ||
| 59 | #define cfi64_to_cpu(x) be64_to_cpu(x) | ||
| 60 | #elif defined (CFI_HOST_ENDIAN) | ||
| 61 | #define cpu_to_cfi8(x) (x) | ||
| 62 | #define cfi8_to_cpu(x) (x) | ||
| 63 | #define cpu_to_cfi16(x) (x) | ||
| 64 | #define cpu_to_cfi32(x) (x) | ||
| 65 | #define cpu_to_cfi64(x) (x) | ||
| 66 | #define cfi16_to_cpu(x) (x) | ||
| 67 | #define cfi32_to_cpu(x) (x) | ||
| 68 | #define cfi64_to_cpu(x) (x) | ||
| 69 | #else | 32 | #else |
| 70 | #error No CFI endianness defined | 33 | #error No CFI endianness defined |
| 71 | #endif | 34 | #endif |
| 35 | |||
| 36 | #define cfi_default(s) ((s)?:CFI_DEFAULT_ENDIAN) | ||
| 37 | #define cfi_be(s) (cfi_default(s) == CFI_BIG_ENDIAN) | ||
| 38 | #define cfi_le(s) (cfi_default(s) == CFI_LITTLE_ENDIAN) | ||
| 39 | #define cfi_host(s) (cfi_default(s) == CFI_HOST_ENDIAN) | ||
| 40 | |||
| 41 | #define cpu_to_cfi8(map, x) (x) | ||
| 42 | #define cfi8_to_cpu(map, x) (x) | ||
| 43 | #define cpu_to_cfi16(map, x) _cpu_to_cfi(16, (map)->swap, (x)) | ||
| 44 | #define cpu_to_cfi32(map, x) _cpu_to_cfi(32, (map)->swap, (x)) | ||
| 45 | #define cpu_to_cfi64(map, x) _cpu_to_cfi(64, (map)->swap, (x)) | ||
| 46 | #define cfi16_to_cpu(map, x) _cfi_to_cpu(16, (map)->swap, (x)) | ||
| 47 | #define cfi32_to_cpu(map, x) _cfi_to_cpu(32, (map)->swap, (x)) | ||
| 48 | #define cfi64_to_cpu(map, x) _cfi_to_cpu(64, (map)->swap, (x)) | ||
| 49 | |||
| 50 | #define _cpu_to_cfi(w, s, x) (cfi_host(s)?(x):_swap_to_cfi(w, s, x)) | ||
| 51 | #define _cfi_to_cpu(w, s, x) (cfi_host(s)?(x):_swap_to_cpu(w, s, x)) | ||
| 52 | #define _swap_to_cfi(w, s, x) (cfi_be(s)?cpu_to_be##w(x):cpu_to_le##w(x)) | ||
| 53 | #define _swap_to_cpu(w, s, x) (cfi_be(s)?be##w##_to_cpu(x):le##w##_to_cpu(x)) | ||
diff --git a/include/linux/mtd/map.h b/include/linux/mtd/map.h index a9e6ba46865e..1132410f14c6 100644 --- a/include/linux/mtd/map.h +++ b/include/linux/mtd/map.h | |||
| @@ -214,6 +214,7 @@ struct map_info { | |||
| 214 | void __iomem *virt; | 214 | void __iomem *virt; |
| 215 | void *cached; | 215 | void *cached; |
| 216 | 216 | ||
| 217 | int swap; /* this mapping's byte-swapping requirement */ | ||
| 217 | int bankwidth; /* in octets. This isn't necessarily the width | 218 | int bankwidth; /* in octets. This isn't necessarily the width |
| 218 | of actual bus cycles -- it's the repeat interval | 219 | of actual bus cycles -- it's the repeat interval |
| 219 | in bytes, before you are talking to the first chip again. | 220 | in bytes, before you are talking to the first chip again. |
