[PATCH v1] regmap: Fix regmap_bulk_read in BE mode

From: Arun Chandran
Date: Mon Jun 15 2015 - 06:30:14 EST


In big endian mode regmap_bulk_read gives incorrect data
for byte reads.

This is because memcpy of a single byte from an address
after full word read gives different results when
endianness differs. ie. we get little-end in LE and big-end in BE.

Signed-off-by: Arun Chandran <achandran@xxxxxxxxxx>
---
---
I tested this on xilinx zynq platform in BE mode. On zynq
drivers/clk/clk-si570.c uses regmap and it fails in BE mode.

Please see the error below.

[ 2.634380] Reg val0 =0
[ 2.636788] Reg val1 =0
[ 2.639168] Reg val2 =0
[ 2.641598] Reg val3 =0
[ 2.644028] Reg val4 =0
[ 2.646513] Reg val5 =0
[ 2.648890] Division by zero in kernel.
[ 2.652715] CPU: 0 PID: 1 Comm: swapper/0 Not tainted 4.1.0-rc6+ #116
[ 2.659181] Hardware name: Xilinx Zynq Platform
[ 2.663676] [<800158a0>] (unwind_backtrace) from [<8001286c>] (show_stack+0x10/0x14)
[ 2.671429] [<8001286c>] (show_stack) from [<804f8e00>] (dump_stack+0x88/0xc8)
[ 2.678624] [<804f8e00>] (dump_stack) from [<8027a408>] (Ldiv0_64+0x8/0x18)
[ 2.685575] [<8027a408>] (Ldiv0_64) from [<80288830>] (div64_u64+0x1c/0xac)
[ 2.692477] [<80288830>] (div64_u64) from [<803caf48>] (si570_probe+0x140/0x2dc)
[ 2.699896] [<803caf48>] (si570_probe) from [<802d9da0>] (really_probe+0x1b0/0x28c)
[ 2.707529] [<802d9da0>] (really_probe) from [<802d9f68>] (__driver_attach+0x98/0x9c)
[ 2.715301] [<802d9f68>] (__driver_attach) from [<802d8308>] (bus_for_each_dev+0x68/0x9c)
[ 2.723498] [<802d8308>] (bus_for_each_dev) from [<802d954c>] (bus_add_driver+0x148/0x1f0)
[ 2.731746] [<802d954c>] (bus_add_driver) from [<802da35c>] (driver_register+0x78/0xf8)
[ 2.739731] [<802da35c>] (driver_register) from [<8036b154>] (i2c_register_driver+0x30/0x7c)
[ 2.748146] [<8036b154>] (i2c_register_driver) from [<800097d0>] (do_one_initcall+0x8c/0x1e0)
[ 2.756652] [<800097d0>] (do_one_initcall) from [<806bde50>] (kernel_init_freeable+0x1cc/0x270)
[ 2.765298] [<806bde50>] (kernel_init_freeable) from [<804f5ec0>] (kernel_init+0xc/0xe8)
[ 2.773393] [<804f5ec0>] (kernel_init) from [<8000f628>] (ret_from_fork+0x14/0x2c)


drivers/clk/clk-si570.c is modified to print the register values
after using regmap_bulk_read() as shown below

static int si570_get_divs(struct clk_si570 *data, u64 *rfreq,
unsigned int *n1, unsigned int *hs_div)
{
- int err;
+ int err, i;
u8 reg[6];
u64 tmp;

err = regmap_bulk_read(data->regmap, SI570_REG_HS_N1 + data->div_offset,
reg, ARRAY_SIZE(reg));
+ for (i = 0; i < 6; i++)
+ printk("Reg val%d =%x\n", i, reg[i]);
+

---
drivers/base/regmap/regmap.c | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/drivers/base/regmap/regmap.c b/drivers/base/regmap/regmap.c
index 6273ff0..9f7f78e 100644
--- a/drivers/base/regmap/regmap.c
+++ b/drivers/base/regmap/regmap.c
@@ -2318,7 +2318,7 @@ int regmap_bulk_read(struct regmap *map, unsigned int reg, void *val,
&ival);
if (ret != 0)
return ret;
- memcpy(val + (i * val_bytes), &ival, val_bytes);
+ map->format.format_val(val + (i * val_bytes), ival, 0);
}
}

--
1.9.1

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/