aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/base/regmap/regmap.c
diff options
context:
space:
mode:
authorJavier Martinez Canillas <javier.martinez@collabora.co.uk>2014-08-18 09:09:02 -0400
committerMark Brown <broonie@linaro.org>2014-08-18 12:39:27 -0400
commitba1b53feb8cacbd84bcf0e48925e30ad29e141a6 (patch)
tree794e0535f93a355853484f60b84973d4625616a9 /drivers/base/regmap/regmap.c
parent275876e208e28abf4b96ec89030e482b1331ee75 (diff)
regmap: Fix DT endianess parsing logic
Commit d647c199510c ("regmap: add DT endianness binding support.") added support to parse the device endianness from the device tree but unfortunately the added logic doesn't have the same semantics than the old code. This leads to a NULL dereference pointer error when these properties are not provided by the Device Tree: Unable to handle kernel NULL pointer dereference at virtual address 00000044 pgd = c0004000 [00000044] *pgd=00000000 Internal error: Oops: 5 [#1] PREEMPT SMP ARM Modules linked in: CPU: 5 PID: 1 Comm: swapper/0 Not tainted 3.17.0-rc1-next-20140818ccu #671 task: ea412800 ti: ea484000 task.ti: ea484000 PC is at regmap_update_bits+0xc/0x5c The problem is that platforms that rely on the default value now gets different values due two related issues in the current code: a) It only parses the endianness from DT for the regmap registers and not for the regmap values but it checks unconditionally in both cases if the resulting endiannes is REGMAP_ENDIAN_NATIVE. b) REGMAP_ENDIAN_NATIVE is not even a valid DT property according to the regmap DT binding documentation so it shouldn't be set. Signed-off-by: Javier Martinez Canillas <javier.martinez@collabora.co.uk> Signed-off-by: Mark Brown <broonie@linaro.org>
Diffstat (limited to 'drivers/base/regmap/regmap.c')
-rw-r--r--drivers/base/regmap/regmap.c16
1 files changed, 4 insertions, 12 deletions
diff --git a/drivers/base/regmap/regmap.c b/drivers/base/regmap/regmap.c
index e4e567e82b84..055a9c3a3b12 100644
--- a/drivers/base/regmap/regmap.c
+++ b/drivers/base/regmap/regmap.c
@@ -499,7 +499,6 @@ static int of_regmap_get_endian(struct device *dev,
499 * From the DT node the endianness value maybe: 499 * From the DT node the endianness value maybe:
500 * REGMAP_ENDIAN_BIG, 500 * REGMAP_ENDIAN_BIG,
501 * REGMAP_ENDIAN_LITTLE, 501 * REGMAP_ENDIAN_LITTLE,
502 * REGMAP_ENDIAN_NATIVE,
503 */ 502 */
504 switch (type) { 503 switch (type) {
505 case REGMAP_ENDIAN_VAL: 504 case REGMAP_ENDIAN_VAL:
@@ -507,8 +506,10 @@ static int of_regmap_get_endian(struct device *dev,
507 *endian = REGMAP_ENDIAN_BIG; 506 *endian = REGMAP_ENDIAN_BIG;
508 else if (of_property_read_bool(np, "little-endian")) 507 else if (of_property_read_bool(np, "little-endian"))
509 *endian = REGMAP_ENDIAN_LITTLE; 508 *endian = REGMAP_ENDIAN_LITTLE;
510 else 509
511 *endian = REGMAP_ENDIAN_NATIVE; 510 if (*endian != REGMAP_ENDIAN_DEFAULT)
511 return 0;
512
512 break; 513 break;
513 case REGMAP_ENDIAN_REG: 514 case REGMAP_ENDIAN_REG:
514 break; 515 break;
@@ -517,15 +518,6 @@ static int of_regmap_get_endian(struct device *dev,
517 } 518 }
518 519
519 /* 520 /*
520 * If the endianness parsed from DT node is REGMAP_ENDIAN_NATIVE, that
521 * maybe means the DT does not care the endianness or it should use
522 * the regmap bus's default endianness, then we should try to check
523 * whether the regmap bus has specified the default endianness.
524 */
525 if (*endian != REGMAP_ENDIAN_NATIVE)
526 return 0;
527
528 /*
529 * Finally, try to parse the endianness from regmap bus config 521 * Finally, try to parse the endianness from regmap bus config
530 * if in device's DT node the endianness property is absent. 522 * if in device's DT node the endianness property is absent.
531 */ 523 */