diff options
author | Andrew Morton <akpm@linux-foundation.org> | 2011-07-18 09:28:20 -0400 |
---|---|---|
committer | Jonas Bonn <jonas@southpole.se> | 2011-07-22 12:45:33 -0400 |
commit | a87e553fabe8ceadc6f90889066559234cf194c7 (patch) | |
tree | 680a59e97378218fd13e0d1400706d84681c921c /include | |
parent | 30ab2b034fa87472d700f584e277e3aeb7a84d2c (diff) |
asm-generic: delay.h fix udelay and ndelay for 8 bit args
With a non-constant 8-bit argument, a call to udelay() generates a warning:
drivers/gpu/drm/radeon/atom.c: In function 'atom_op_delay':
drivers/gpu/drm/radeon/atom.c:654: warning: comparison is always false due to limited range of data type
The code looks like it works OK with an 8-bit arg, and the calling code is
doing nothing wrong, so udelay() needs fixing.
Fixing it was rather tricky. Simply typecasting `n' in the comparison with
20000 didn't change anything. Hence the divide-by-20000 trick.
Using a do{}while loop didn't work because udelay() is used in ?: statements,
hence the ({...}) construct.
While I was there I replaced the brain-bending ?:?:?: mess with nice if/else
code.
Probably other architectures are generating the same warning and can use a
similar change.
[Taken from the x86 tree and moved to asm-generic by Jonas Bonn]
Cc: Ingo Molnar <mingo@elte.hu>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: <linux-arch@vger.kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Jonas Bonn <jonas@southpole.se>
Diffstat (limited to 'include')
-rw-r--r-- | include/asm-generic/delay.h | 33 |
1 files changed, 27 insertions, 6 deletions
diff --git a/include/asm-generic/delay.h b/include/asm-generic/delay.h index 6511b99c5f17..0f79054ce7cd 100644 --- a/include/asm-generic/delay.h +++ b/include/asm-generic/delay.h | |||
@@ -10,14 +10,35 @@ extern void __ndelay(unsigned long nsecs); | |||
10 | extern void __const_udelay(unsigned long xloops); | 10 | extern void __const_udelay(unsigned long xloops); |
11 | extern void __delay(unsigned long loops); | 11 | extern void __delay(unsigned long loops); |
12 | 12 | ||
13 | /* | ||
14 | * The weird n/20000 thing suppresses a "comparison is always false due to | ||
15 | * limited range of data type" warning with non-const 8-bit arguments. | ||
16 | */ | ||
17 | |||
13 | /* 0x10c7 is 2**32 / 1000000 (rounded up) */ | 18 | /* 0x10c7 is 2**32 / 1000000 (rounded up) */ |
14 | #define udelay(n) (__builtin_constant_p(n) ? \ | 19 | #define udelay(n) \ |
15 | ((n) > 20000 ? __bad_udelay() : __const_udelay((n) * 0x10c7ul)) : \ | 20 | ({ \ |
16 | __udelay(n)) | 21 | if (__builtin_constant_p(n)) { \ |
22 | if ((n) / 20000 >= 1) \ | ||
23 | __bad_udelay(); \ | ||
24 | else \ | ||
25 | __const_udelay((n) * 0x10c7ul); \ | ||
26 | } else { \ | ||
27 | __udelay(n); \ | ||
28 | } \ | ||
29 | }) | ||
17 | 30 | ||
18 | /* 0x5 is 2**32 / 1000000000 (rounded up) */ | 31 | /* 0x5 is 2**32 / 1000000000 (rounded up) */ |
19 | #define ndelay(n) (__builtin_constant_p(n) ? \ | 32 | #define ndelay(n) \ |
20 | ((n) > 20000 ? __bad_ndelay() : __const_udelay((n) * 5ul)) : \ | 33 | ({ \ |
21 | __ndelay(n)) | 34 | if (__builtin_constant_p(n)) { \ |
35 | if ((n) / 20000 >= 1) \ | ||
36 | __bad_ndelay(); \ | ||
37 | else \ | ||
38 | __const_udelay((n) * 5ul); \ | ||
39 | } else { \ | ||
40 | __ndelay(n); \ | ||
41 | } \ | ||
42 | }) | ||
22 | 43 | ||
23 | #endif /* __ASM_GENERIC_DELAY_H */ | 44 | #endif /* __ASM_GENERIC_DELAY_H */ |