Commit c9723750 authored by Martin Fuzzey's avatar Martin Fuzzey Committed by Greg Kroah-Hartman

w1: mxc_w1: Fix timeout resolution problem leading to bus error

On my platform (i.MX53) bus access sometimes fails with
	w1_search: max_slave_count 64 reached, will continue next search.

The reason is the use of jiffies to implement a 200us timeout in
mxc_w1_ds2_touch_bit().
On some platforms the jiffies timer resolution is insufficient for this.

Fix by replacing jiffies by ktime_get().

For consistency apply the same change to the other use of jiffies in
mxc_w1_ds2_reset_bus().

Fixes: f80b2581 ("w1: mxc_w1: Optimize mxc_w1_ds2_touch_bit()")
Cc: stable <stable@vger.kernel.org>
Signed-off-by: default avatarMartin Fuzzey <martin.fuzzey@flowbird.group>
Link: https://lore.kernel.org/r/1601455030-6607-1-git-send-email-martin.fuzzey@flowbird.groupSigned-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent e84d3896
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
#include <linux/clk.h> #include <linux/clk.h>
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/io.h> #include <linux/io.h>
#include <linux/jiffies.h> #include <linux/ktime.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/mod_devicetable.h> #include <linux/mod_devicetable.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
...@@ -40,12 +40,12 @@ struct mxc_w1_device { ...@@ -40,12 +40,12 @@ struct mxc_w1_device {
static u8 mxc_w1_ds2_reset_bus(void *data) static u8 mxc_w1_ds2_reset_bus(void *data)
{ {
struct mxc_w1_device *dev = data; struct mxc_w1_device *dev = data;
unsigned long timeout; ktime_t timeout;
writeb(MXC_W1_CONTROL_RPP, dev->regs + MXC_W1_CONTROL); writeb(MXC_W1_CONTROL_RPP, dev->regs + MXC_W1_CONTROL);
/* Wait for reset sequence 511+512us, use 1500us for sure */ /* Wait for reset sequence 511+512us, use 1500us for sure */
timeout = jiffies + usecs_to_jiffies(1500); timeout = ktime_add_us(ktime_get(), 1500);
udelay(511 + 512); udelay(511 + 512);
...@@ -55,7 +55,7 @@ static u8 mxc_w1_ds2_reset_bus(void *data) ...@@ -55,7 +55,7 @@ static u8 mxc_w1_ds2_reset_bus(void *data)
/* PST bit is valid after the RPP bit is self-cleared */ /* PST bit is valid after the RPP bit is self-cleared */
if (!(ctrl & MXC_W1_CONTROL_RPP)) if (!(ctrl & MXC_W1_CONTROL_RPP))
return !(ctrl & MXC_W1_CONTROL_PST); return !(ctrl & MXC_W1_CONTROL_PST);
} while (time_is_after_jiffies(timeout)); } while (ktime_before(ktime_get(), timeout));
return 1; return 1;
} }
...@@ -68,12 +68,12 @@ static u8 mxc_w1_ds2_reset_bus(void *data) ...@@ -68,12 +68,12 @@ static u8 mxc_w1_ds2_reset_bus(void *data)
static u8 mxc_w1_ds2_touch_bit(void *data, u8 bit) static u8 mxc_w1_ds2_touch_bit(void *data, u8 bit)
{ {
struct mxc_w1_device *dev = data; struct mxc_w1_device *dev = data;
unsigned long timeout; ktime_t timeout;
writeb(MXC_W1_CONTROL_WR(bit), dev->regs + MXC_W1_CONTROL); writeb(MXC_W1_CONTROL_WR(bit), dev->regs + MXC_W1_CONTROL);
/* Wait for read/write bit (60us, Max 120us), use 200us for sure */ /* Wait for read/write bit (60us, Max 120us), use 200us for sure */
timeout = jiffies + usecs_to_jiffies(200); timeout = ktime_add_us(ktime_get(), 200);
udelay(60); udelay(60);
...@@ -83,7 +83,7 @@ static u8 mxc_w1_ds2_touch_bit(void *data, u8 bit) ...@@ -83,7 +83,7 @@ static u8 mxc_w1_ds2_touch_bit(void *data, u8 bit)
/* RDST bit is valid after the WR1/RD bit is self-cleared */ /* RDST bit is valid after the WR1/RD bit is self-cleared */
if (!(ctrl & MXC_W1_CONTROL_WR(bit))) if (!(ctrl & MXC_W1_CONTROL_WR(bit)))
return !!(ctrl & MXC_W1_CONTROL_RDST); return !!(ctrl & MXC_W1_CONTROL_RDST);
} while (time_is_after_jiffies(timeout)); } while (ktime_before(ktime_get(), timeout));
return 0; return 0;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment