The patch titled delay: add generic udelay(), mdelay() and ssleep() has been added to the -mm tree. Its filename is delay-add-generic-udelay-mdelay-and-ssleep.patch See http://www.zip.com.au/~akpm/linux/patches/stuff/added-to-mm.txt to find out what to do about this ------------------------------------------------------ Subject: delay: add generic udelay(), mdelay() and ssleep() From: Denis Vlasenko <vda.linux@xxxxxxxxxxxxxx> * make it so than asm/delay.h does not define udelay(), only __udelay(), to be used in generic udelay() * add generic udelay() which calls __udelay() repeatedly as needed. Protect against overflow in udelay() argument. * similarly for mdelay() and ssleep() * __const_udelay for all arches is removed or renamed to __const_delay (it did not do microsecond delays anyway) if still used by arch ndelay() function/macro * remove EXPORT_SYMBOL(__udelay). It is not used in modules anymore * remove MAX_UDELAY_MS We specifically do not touch ndelay() in these patches. Signed-off-by: Denis Vlasenko <vda.linux@xxxxxxxxxxxxxx> Signed-off-by: Andrew Morton <akpm@xxxxxxxx> --- arch/arm/kernel/armksyms.c | 4 -- arch/arm/lib/delay.S | 2 - arch/i386/lib/delay.c | 10 +++---- arch/m32r/kernel/m32r_ksyms.c | 2 - arch/m32r/lib/delay.c | 6 ++-- arch/m68knommu/lib/Makefile | 2 - arch/m68knommu/lib/delay.c | 21 -------------- arch/powerpc/kernel/time.c | 4 +- arch/sh/kernel/sh_ksyms.c | 3 -- arch/sh/lib/delay.c | 8 ++--- arch/sh64/lib/udelay.c | 11 +++---- arch/sparc64/kernel/sparc64_ksyms.c | 2 - arch/sparc64/lib/delay.c | 6 ++-- arch/um/sys-i386/delay.c | 13 --------- arch/um/sys-x86_64/delay.c | 13 --------- arch/x86_64/kernel/functionlist | 2 - arch/x86_64/lib/delay.c | 9 +++--- include/asm-arm/delay.h | 30 --------------------- include/asm-arm26/delay.h | 4 -- include/asm-cris/delay.h | 5 --- include/asm-frv/delay.h | 3 +- include/asm-h8300/delay.h | 2 - include/asm-i386/delay.h | 9 +----- include/asm-ia64/delay.h | 2 - include/asm-m32r/delay.h | 9 +----- include/asm-m68k/delay.h | 10 +------ include/asm-m68knommu/delay.h | 10 ------- include/asm-mips/delay.h | 13 +-------- include/asm-parisc/delay.h | 2 - include/asm-powerpc/delay.h | 2 - include/asm-ppc/delay.h | 14 +++------ include/asm-s390/delay.h | 2 - include/asm-sh/delay.h | 10 +------ include/asm-sh64/delay.h | 6 ++-- include/asm-sparc/delay.h | 1 include/asm-sparc64/delay.h | 8 +---- include/asm-v850/delay.h | 2 - include/asm-x86_64/delay.h | 9 +----- include/asm-xtensa/delay.h | 2 - include/linux/delay.h | 30 ++------------------- kernel/timer.c | 37 ++++++++++++++++++++++++++ 41 files changed, 103 insertions(+), 237 deletions(-) diff -puN arch/arm/kernel/armksyms.c~delay-add-generic-udelay-mdelay-and-ssleep arch/arm/kernel/armksyms.c --- a/arch/arm/kernel/armksyms.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/arm/kernel/armksyms.c @@ -70,10 +70,6 @@ EXPORT_SYMBOL_ALIAS(fp_send_sig,send_sig EXPORT_SYMBOL(__backtrace); - /* platform dependent support */ -EXPORT_SYMBOL(__udelay); -EXPORT_SYMBOL(__const_udelay); - /* networking */ EXPORT_SYMBOL(csum_partial); EXPORT_SYMBOL(csum_partial_copy_nocheck); diff -puN arch/arm/lib/delay.S~delay-add-generic-udelay-mdelay-and-ssleep arch/arm/lib/delay.S --- a/arch/arm/lib/delay.S~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/arm/lib/delay.S @@ -24,7 +24,7 @@ ENTRY(__udelay) ldr r2, .LC1 mul r0, r2, r0 -ENTRY(__const_udelay) @ 0 <= r0 <= 0x7fffff06 + @ 0 <= r0 <= 0x7fffff06 ldr r2, .LC0 ldr r2, [r2] @ max = 0x01ffffff mov r0, r0, lsr #14 @ max = 0x0001ffff diff -puN arch/i386/lib/delay.c~delay-add-generic-udelay-mdelay-and-ssleep arch/i386/lib/delay.c --- a/arch/i386/lib/delay.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/i386/lib/delay.c @@ -74,7 +74,8 @@ void __delay(unsigned long loops) delay_fn(loops); } -inline void __const_udelay(unsigned long xloops) +/* cannot be static: ndelay macro needs it */ +inline void __const_delay(unsigned long xloops) { int d0; @@ -89,15 +90,14 @@ inline void __const_udelay(unsigned long void __udelay(unsigned long usecs) { - __const_udelay(usecs * 0x000010c7); /* 2**32 / 1000000 (rounded up) */ + __const_delay(usecs * 0x000010c7); /* 2**32 / 1000000 (rounded up) */ } void __ndelay(unsigned long nsecs) { - __const_udelay(nsecs * 0x00005); /* 2**32 / 1000000000 (rounded up) */ + __const_delay(nsecs * 0x00005); /* 2**32 / 1000000000 (rounded up) */ } EXPORT_SYMBOL(__delay); -EXPORT_SYMBOL(__const_udelay); -EXPORT_SYMBOL(__udelay); +EXPORT_SYMBOL(__const_delay); EXPORT_SYMBOL(__ndelay); diff -puN arch/m32r/kernel/m32r_ksyms.c~delay-add-generic-udelay-mdelay-and-ssleep arch/m32r/kernel/m32r_ksyms.c --- a/arch/m32r/kernel/m32r_ksyms.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/m32r/kernel/m32r_ksyms.c @@ -30,9 +30,7 @@ EXPORT_SYMBOL(__down_trylock); /* Networking helper routines. */ /* Delay loops */ -EXPORT_SYMBOL(__udelay); EXPORT_SYMBOL(__delay); -EXPORT_SYMBOL(__const_udelay); EXPORT_SYMBOL(strncpy_from_user); EXPORT_SYMBOL(__strncpy_from_user); diff -puN arch/m32r/lib/delay.c~delay-add-generic-udelay-mdelay-and-ssleep arch/m32r/lib/delay.c --- a/arch/m32r/lib/delay.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/m32r/lib/delay.c @@ -57,7 +57,7 @@ void __delay(unsigned long loops) #endif } -void __const_udelay(unsigned long xloops) +void __const_delay(unsigned long xloops) { #if defined(CONFIG_ISA_M32R2) && defined(CONFIG_ISA_DSP_LEVEL2) /* @@ -116,10 +116,10 @@ void __const_udelay(unsigned long xloops void __udelay(unsigned long usecs) { - __const_udelay(usecs * 0x000010c7); /* 2**32 / 1000000 (rounded up) */ + __const_delay(usecs * 0x000010c7); /* 2**32 / 1000000 (rounded up) */ } void __ndelay(unsigned long nsecs) { - __const_udelay(nsecs * 0x00005); /* 2**32 / 1000000000 (rounded up) */ + __const_delay(nsecs * 0x00005); /* 2**32 / 1000000000 (rounded up) */ } diff -puN arch/m68knommu/lib/Makefile~delay-add-generic-udelay-mdelay-and-ssleep arch/m68knommu/lib/Makefile --- a/arch/m68knommu/lib/Makefile~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/m68knommu/lib/Makefile @@ -4,4 +4,4 @@ lib-y := ashldi3.o ashrdi3.o lshrdi3.o \ muldi3.o mulsi3.o divsi3.o udivsi3.o modsi3.o umodsi3.o \ - checksum.o semaphore.o memcpy.o memset.o delay.o + checksum.o semaphore.o memcpy.o memset.o diff -puN arch/m68knommu/lib/delay.c~delay-add-generic-udelay-mdelay-and-ssleep /dev/null --- a/arch/m68knommu/lib/delay.c +++ /dev/null @@ -1,21 +0,0 @@ -/* - * arch/m68knommu/lib/delay.c - * - * (C) Copyright 2004, Greg Ungerer <gerg@xxxxxxxxxxxx> - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License version 2 as - * published by the Free Software Foundation. - */ - -#include <linux/module.h> -#include <asm/param.h> -#include <asm/delay.h> - -EXPORT_SYMBOL(udelay); - -void udelay(unsigned long usecs) -{ - _udelay(usecs); -} - diff -puN arch/powerpc/kernel/time.c~delay-add-generic-udelay-mdelay-and-ssleep arch/powerpc/kernel/time.c --- a/arch/powerpc/kernel/time.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/powerpc/kernel/time.c @@ -368,11 +368,11 @@ void __delay(unsigned long loops) } EXPORT_SYMBOL(__delay); -void udelay(unsigned long usecs) +void __udelay(unsigned long usecs) { __delay(tb_ticks_per_usec * usecs); } -EXPORT_SYMBOL(udelay); +EXPORT_SYMBOL(__udelay); static __inline__ void timer_check_rtc(void) { diff -puN arch/sh/kernel/sh_ksyms.c~delay-add-generic-udelay-mdelay-and-ssleep arch/sh/kernel/sh_ksyms.c --- a/arch/sh/kernel/sh_ksyms.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/sh/kernel/sh_ksyms.c @@ -71,9 +71,8 @@ EXPORT_SYMBOL(__up); EXPORT_SYMBOL(__down); EXPORT_SYMBOL(__down_interruptible); -EXPORT_SYMBOL(__udelay); EXPORT_SYMBOL(__ndelay); -EXPORT_SYMBOL(__const_udelay); +EXPORT_SYMBOL(__const_delay); EXPORT_SYMBOL(__div64_32); diff -puN arch/sh/lib/delay.c~delay-add-generic-udelay-mdelay-and-ssleep arch/sh/lib/delay.c --- a/arch/sh/lib/delay.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/sh/lib/delay.c @@ -19,7 +19,8 @@ void __delay(unsigned long loops) : "t"); } -inline void __const_udelay(unsigned long xloops) +/* cannot be static: ndelay macro needs it */ +inline void __const_delay(unsigned long xloops) { __asm__("dmulu.l %0, %2\n\t" "sts mach, %0" @@ -31,11 +32,10 @@ inline void __const_udelay(unsigned long void __udelay(unsigned long usecs) { - __const_udelay(usecs * 0x000010c6); /* 2**32 / 1000000 */ + __const_delay(usecs * 0x000010c6); /* 2**32 / 1000000 */ } void __ndelay(unsigned long nsecs) { - __const_udelay(nsecs * 0x00000005); + __const_delay(nsecs * 0x00000005); } - diff -puN arch/sh64/lib/udelay.c~delay-add-generic-udelay-mdelay-and-ssleep arch/sh64/lib/udelay.c --- a/arch/sh64/lib/udelay.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/sh64/lib/udelay.c @@ -35,25 +35,24 @@ void __delay(int loops) :"0"(loops)); } -void __udelay(unsigned long long usecs, unsigned long lpj) +static inline void __arch_udelay(unsigned long long usecs, unsigned long lpj) { usecs *= (((unsigned long long) HZ << 32) / 1000000) * lpj; __delay((long long) usecs >> 32); } -void __ndelay(unsigned long long nsecs, unsigned long lpj) +static inline void __arch_ndelay(unsigned long long nsecs, unsigned long lpj) { nsecs *= (((unsigned long long) HZ << 32) / 1000000000) * lpj; __delay((long long) nsecs >> 32); } -void udelay(unsigned long usecs) +void __udelay(unsigned long usecs) { - __udelay(usecs, loops_per_jiffy); + __arch_udelay(usecs, loops_per_jiffy); } void ndelay(unsigned long nsecs) { - __ndelay(nsecs, loops_per_jiffy); + __arch_ndelay(nsecs, loops_per_jiffy); } - diff -puN arch/sparc64/kernel/sparc64_ksyms.c~delay-add-generic-udelay-mdelay-and-ssleep arch/sparc64/kernel/sparc64_ksyms.c --- a/arch/sparc64/kernel/sparc64_ksyms.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/sparc64/kernel/sparc64_ksyms.c @@ -334,7 +334,7 @@ EXPORT_SYMBOL(strncmp); /* Delay routines. */ EXPORT_SYMBOL(__udelay); EXPORT_SYMBOL(__ndelay); -EXPORT_SYMBOL(__const_udelay); +EXPORT_SYMBOL(__const_delay); EXPORT_SYMBOL(__delay); void VISenter(void); diff -puN arch/sparc64/lib/delay.c~delay-add-generic-udelay-mdelay-and-ssleep arch/sparc64/lib/delay.c --- a/arch/sparc64/lib/delay.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/sparc64/lib/delay.c @@ -24,7 +24,7 @@ void __delay(unsigned long loops) * but that runs into problems for higher values of HZ and * slow cpus. */ -void __const_udelay(unsigned long n) +void __const_delay(unsigned long n) { n *= 4; @@ -36,11 +36,11 @@ void __const_udelay(unsigned long n) void __udelay(unsigned long n) { - __const_udelay(n * 0x10c7UL); + __const_delay(n * 0x10c7UL); } void __ndelay(unsigned long n) { - __const_udelay(n * 0x5UL); + __const_delay(n * 0x5UL); } diff -puN arch/um/sys-i386/delay.c~delay-add-generic-udelay-mdelay-and-ssleep arch/um/sys-i386/delay.c --- a/arch/um/sys-i386/delay.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/um/sys-i386/delay.c @@ -25,16 +25,3 @@ void __udelay(unsigned long usecs) for(i=0;i<n;i++) cpu_relax(); } - -EXPORT_SYMBOL(__udelay); - -void __const_udelay(unsigned long usecs) -{ - int i, n; - - n = (loops_per_jiffy * HZ * usecs) / MILLION; - for(i=0;i<n;i++) - cpu_relax(); -} - -EXPORT_SYMBOL(__const_udelay); diff -puN arch/um/sys-x86_64/delay.c~delay-add-generic-udelay-mdelay-and-ssleep arch/um/sys-x86_64/delay.c --- a/arch/um/sys-x86_64/delay.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/um/sys-x86_64/delay.c @@ -26,16 +26,3 @@ void __udelay(unsigned long usecs) for(i=0;i<n;i++) cpu_relax(); } - -EXPORT_SYMBOL(__udelay); - -void __const_udelay(unsigned long usecs) -{ - unsigned long i, n; - - n = (loops_per_jiffy * HZ * usecs) / MILLION; - for(i=0;i<n;i++) - cpu_relax(); -} - -EXPORT_SYMBOL(__const_udelay); diff -puN arch/x86_64/kernel/functionlist~delay-add-generic-udelay-mdelay-and-ssleep arch/x86_64/kernel/functionlist --- a/arch/x86_64/kernel/functionlist~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/x86_64/kernel/functionlist @@ -494,7 +494,7 @@ *(.text.do_setitimer) *(.text.dev_queue_xmit_nit) *(.text.copy_from_read_buf) -*(.text.__const_udelay) +*(.text.__const_delay) *(.text.console_conditional_schedule) *(.text.wake_up_new_task) *(.text.wait_for_completion_interruptible) diff -puN arch/x86_64/lib/delay.c~delay-add-generic-udelay-mdelay-and-ssleep arch/x86_64/lib/delay.c --- a/arch/x86_64/lib/delay.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/arch/x86_64/lib/delay.c @@ -39,20 +39,21 @@ void __delay(unsigned long loops) } EXPORT_SYMBOL(__delay); -inline void __const_udelay(unsigned long xloops) +/* cannot be static: ndelay macro needs it */ +inline void __const_delay(unsigned long xloops) { __delay((xloops * HZ * cpu_data[raw_smp_processor_id()].loops_per_jiffy) >> 32); } -EXPORT_SYMBOL(__const_udelay); +EXPORT_SYMBOL(__const_delay); void __udelay(unsigned long usecs) { - __const_udelay(usecs * 0x000010c6); /* 2**32 / 1000000 */ + __const_delay(usecs * 0x000010c6); /* 2**32 / 1000000 */ } EXPORT_SYMBOL(__udelay); void __ndelay(unsigned long nsecs) { - __const_udelay(nsecs * 0x00005); /* 2**32 / 1000000000 (rounded up) */ + __const_delay(nsecs * 0x00005); /* 2**32 / 1000000000 (rounded up) */ } EXPORT_SYMBOL(__ndelay); diff -puN include/asm-arm/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-arm/delay.h --- a/include/asm-arm/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-arm/delay.h @@ -9,36 +9,6 @@ #include <asm/param.h> /* HZ */ extern void __delay(int loops); - -/* - * This function intentionally does not exist; if you see references to - * it, it means that you're calling udelay() with an out of range value. - * - * With currently imposed limits, this means that we support a max delay - * of 2000us. Further limits: HZ<=1000 and bogomips<=3355 - */ -extern void __bad_udelay(void); - -/* - * division by multiplication: you don't have to worry about - * loss of precision. - * - * Use only for very small delays ( < 1 msec). Should probably use a - * lookup table, really, as the multiplications take much too long with - * short delays. This is a "reasonable" implementation, though (and the - * first constant multiplications gets optimized away if the delay is - * a constant) - */ extern void __udelay(unsigned long usecs); -extern void __const_udelay(unsigned long); - -#define MAX_UDELAY_MS 2 - -#define udelay(n) \ - (__builtin_constant_p(n) ? \ - ((n) > (MAX_UDELAY_MS * 1000) ? __bad_udelay() : \ - __const_udelay((n) * ((2199023U*HZ)>>11))) : \ - __udelay(n)) #endif /* defined(_ARM_DELAY_H) */ - diff -puN include/asm-arm26/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-arm26/delay.h --- a/include/asm-arm26/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-arm26/delay.h @@ -21,14 +21,10 @@ extern void __delay(int loops); * * FIXME - lets improve it then... */ -extern void udelay(unsigned long usecs); - static inline unsigned long muldiv(unsigned long a, unsigned long b, unsigned long c) { return a * b / c; } - - #endif /* defined(_ARM_DELAY_H) */ diff -puN include/asm-cris/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-cris/delay.h --- a/include/asm-cris/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-cris/delay.h @@ -13,12 +13,9 @@ extern unsigned long loops_per_usec; /* arch/cris/mm/init.c */ -static inline void udelay(unsigned long usecs) +static inline void __udelay(unsigned long usecs) { __delay(usecs * loops_per_usec); } #endif /* defined(_CRIS_DELAY_H) */ - - - diff -puN include/asm-frv/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-frv/delay.h --- a/include/asm-frv/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-frv/delay.h @@ -40,11 +40,12 @@ static inline void __delay(unsigned long extern unsigned long loops_per_jiffy; -static inline void udelay(unsigned long usecs) +static inline void __udelay(unsigned long usecs) { __delay(usecs * __delay_loops_MHz); } +/* FIXME: looks like bug to me: */ #define ndelay(n) udelay((n) * 5) #endif /* _ASM_DELAY_H */ diff -puN include/asm-h8300/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-h8300/delay.h --- a/include/asm-h8300/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-h8300/delay.h @@ -27,7 +27,7 @@ extern __inline__ void __delay(unsigned extern unsigned long loops_per_jiffy; -extern __inline__ void udelay(unsigned long usecs) +extern __inline__ void __udelay(unsigned long usecs) { usecs *= 4295; /* 2**32 / 1000000 */ usecs /= (loops_per_jiffy*HZ); diff -puN include/asm-i386/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-i386/delay.h --- a/include/asm-i386/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-i386/delay.h @@ -7,20 +7,15 @@ * Delay routines calling functions in arch/i386/lib/delay.c */ -extern void __bad_udelay(void); extern void __bad_ndelay(void); extern void __udelay(unsigned long usecs); extern void __ndelay(unsigned long nsecs); -extern void __const_udelay(unsigned long usecs); +extern void __const_delay(unsigned long usecs); extern void __delay(unsigned long loops); -#define udelay(n) (__builtin_constant_p(n) ? \ - ((n) > 20000 ? __bad_udelay() : __const_udelay((n) * 0x10c7ul)) : \ - __udelay(n)) - #define ndelay(n) (__builtin_constant_p(n) ? \ - ((n) > 20000 ? __bad_ndelay() : __const_udelay((n) * 5ul)) : \ + ((n) > 20000 ? __bad_ndelay() : __const_delay((n) * 5ul)) : \ __ndelay(n)) void use_tsc_delay(void); diff -puN include/asm-ia64/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-ia64/delay.h --- a/include/asm-ia64/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-ia64/delay.h @@ -83,6 +83,4 @@ __delay (unsigned long loops) ia64_delay_loop (loops - 1); } -extern void udelay (unsigned long usecs); - #endif /* _ASM_IA64_DELAY_H */ diff -puN include/asm-m32r/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-m32r/delay.h --- a/include/asm-m32r/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-m32r/delay.h @@ -9,20 +9,15 @@ * Delay routines calling functions in arch/m32r/lib/delay.c */ -extern void __bad_udelay(void); extern void __bad_ndelay(void); extern void __udelay(unsigned long usecs); extern void __ndelay(unsigned long nsecs); -extern void __const_udelay(unsigned long usecs); +extern void __const_delay(unsigned long usecs); extern void __delay(unsigned long loops); -#define udelay(n) (__builtin_constant_p(n) ? \ - ((n) > 20000 ? __bad_udelay() : __const_udelay((n) * 0x10c7ul)) : \ - __udelay(n)) - #define ndelay(n) (__builtin_constant_p(n) ? \ - ((n) > 20000 ? __bad_ndelay() : __const_udelay((n) * 5ul)) : \ + ((n) > 20000 ? __bad_ndelay() : __const_delay((n) * 5ul)) : \ __ndelay(n)) #endif /* _ASM_M32R_DELAY_H */ diff -puN include/asm-m68k/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-m68k/delay.h --- a/include/asm-m68k/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-m68k/delay.h @@ -15,8 +15,6 @@ static inline void __delay(unsigned long : "=d" (loops) : "0" (loops)); } -extern void __bad_udelay(void); - /* * Use only for very small delays ( < 1 msec). Should probably use a * lookup table, really, as the multiplications take much too long with @@ -24,7 +22,7 @@ extern void __bad_udelay(void); * first constant multiplications gets optimized away if the delay is * a constant) */ -static inline void __const_udelay(unsigned long xloops) +static inline void __const_delay(unsigned long xloops) { unsigned long tmp; @@ -36,13 +34,9 @@ static inline void __const_udelay(unsign static inline void __udelay(unsigned long usecs) { - __const_udelay(usecs * 4295); /* 2**32 / 1000000 */ + __const_delay(usecs * 4295); /* 2**32 / 1000000 */ } -#define udelay(n) (__builtin_constant_p(n) ? \ - ((n) > 20000 ? __bad_udelay() : __const_udelay((n) * 4295)) : \ - __udelay(n)) - static inline unsigned long muldiv(unsigned long a, unsigned long b, unsigned long c) { diff -puN include/asm-m68knommu/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-m68knommu/delay.h --- a/include/asm-m68knommu/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-m68knommu/delay.h @@ -48,7 +48,7 @@ static inline void __delay(unsigned long extern unsigned long loops_per_jiffy; -static inline void _udelay(unsigned long usecs) +static inline void __udelay(unsigned long usecs) { #if defined(CONFIG_M68328) || defined(CONFIG_M68EZ328) || \ defined(CONFIG_M68VZ328) || defined(CONFIG_M68360) || \ @@ -65,12 +65,4 @@ static inline void _udelay(unsigned long #endif } -/* - * Moved the udelay() function into library code, no longer inlined. - * I had to change the algorithm because we are overflowing now on - * the faster ColdFire parts. The code is a little biger, so it makes - * sense to library it. - */ -extern void udelay(unsigned long usecs); - #endif /* defined(_M68KNOMMU_DELAY_H) */ diff -puN include/asm-mips/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-mips/delay.h --- a/include/asm-mips/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-mips/delay.h @@ -48,7 +48,7 @@ static inline void __delay(unsigned long * a constant) */ -static inline void __udelay(unsigned long usecs, unsigned long lpj) +static inline void __arch_udelay(unsigned long usecs, unsigned long lpj) { unsigned long lo; @@ -81,15 +81,6 @@ static inline void __udelay(unsigned lon #define __udelay_val cpu_data[smp_processor_id()].udelay_val -#define udelay(usecs) __udelay((usecs),__udelay_val) - -/* make sure "usecs *= ..." in udelay do not overflow. */ -#if HZ >= 1000 -#define MAX_UDELAY_MS 1 -#elif HZ <= 200 -#define MAX_UDELAY_MS 5 -#else -#define MAX_UDELAY_MS (1000 / HZ) -#endif +#define __udelay(usecs) __arch_udelay((usecs),__udelay_val) #endif /* _ASM_DELAY_H */ diff -puN include/asm-parisc/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-parisc/delay.h --- a/include/asm-parisc/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-parisc/delay.h @@ -37,6 +37,4 @@ static __inline__ void __udelay(unsigned __cr16_delay(usecs * ((unsigned long)boot_cpu_data.cpu_hz / 1000000UL)); } -#define udelay(n) __udelay(n) - #endif /* defined(_PARISC_DELAY_H) */ diff -puN include/asm-powerpc/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-powerpc/delay.h --- a/include/asm-powerpc/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-powerpc/delay.h @@ -15,7 +15,7 @@ */ extern void __delay(unsigned long loops); -extern void udelay(unsigned long usecs); +extern void __udelay(unsigned long usecs); /* * On shared processor machines the generic implementation of mdelay can diff -puN include/asm-ppc/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-ppc/delay.h --- a/include/asm-ppc/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-ppc/delay.h @@ -30,10 +30,9 @@ extern void __delay(unsigned int loops); * (which corresponds to ~3800 bogomips at HZ = 100). * -- paulus */ -#define __MAX_UDELAY (226050910UL/HZ) /* maximum udelay argument */ #define __MAX_NDELAY (4294967295UL/HZ) /* maximum ndelay argument */ -extern __inline__ void __udelay(unsigned int x) +extern __inline__ void __arch_udelay(unsigned int x) { unsigned int loops; @@ -42,7 +41,7 @@ extern __inline__ void __udelay(unsigned __delay(loops); } -extern __inline__ void __ndelay(unsigned int x) +extern __inline__ void __arch_ndelay(unsigned int x) { unsigned int loops; @@ -51,16 +50,13 @@ extern __inline__ void __ndelay(unsigned __delay(loops); } -extern void __bad_udelay(void); /* deliberately undefined */ extern void __bad_ndelay(void); /* deliberately undefined */ -#define udelay(n) (__builtin_constant_p(n)? \ - ((n) > __MAX_UDELAY? __bad_udelay(): __udelay((n) * (19 * HZ))) : \ - __udelay((n) * (19 * HZ))) +#define __udelay(n) __arch_udelay((n) * (19 * HZ)) #define ndelay(n) (__builtin_constant_p(n)? \ - ((n) > __MAX_NDELAY? __bad_ndelay(): __ndelay((n) * HZ)) : \ - __ndelay((n) * HZ)) + ((n) > __MAX_NDELAY? __bad_ndelay(): __arch_ndelay((n) * HZ)) : \ + __arch_ndelay((n) * HZ)) #endif /* defined(_PPC_DELAY_H) */ #endif /* __KERNEL__ */ diff -puN include/asm-s390/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-s390/delay.h --- a/include/asm-s390/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-s390/delay.h @@ -17,6 +17,4 @@ extern void __udelay(unsigned long usecs); extern void __delay(unsigned long loops); -#define udelay(n) __udelay(n) - #endif /* defined(_S390_DELAY_H) */ diff -puN include/asm-sh/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-sh/delay.h --- a/include/asm-sh/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-sh/delay.h @@ -7,21 +7,15 @@ * Delay routines calling functions in arch/sh/lib/delay.c */ -extern void __bad_udelay(void); extern void __bad_ndelay(void); extern void __udelay(unsigned long usecs); extern void __ndelay(unsigned long nsecs); -extern void __const_udelay(unsigned long usecs); +extern void __const_delay(unsigned long usecs); extern void __delay(unsigned long loops); -#define udelay(n) (__builtin_constant_p(n) ? \ - ((n) > 20000 ? __bad_udelay() : __const_udelay((n) * 0x10c6ul)) : \ - __udelay(n)) - - #define ndelay(n) (__builtin_constant_p(n) ? \ - ((n) > 20000 ? __bad_ndelay() : __const_udelay((n) * 5ul)) : \ + ((n) > 20000 ? __bad_ndelay() : __const_delay((n) * 5ul)) : \ __ndelay(n)) #endif /* __ASM_SH_DELAY_H */ diff -puN include/asm-sh64/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-sh64/delay.h --- a/include/asm-sh64/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-sh64/delay.h @@ -2,10 +2,10 @@ #define __ASM_SH64_DELAY_H extern void __delay(int loops); -extern void __udelay(unsigned long long usecs, unsigned long lpj); -extern void __ndelay(unsigned long long nsecs, unsigned long lpj); -extern void udelay(unsigned long usecs); +extern void __udelay(unsigned long usecs); extern void ndelay(unsigned long nsecs); +/* for linux/delay.h: */ +#define ndelay ndelay #endif /* __ASM_SH64_DELAY_H */ diff -puN include/asm-sparc/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-sparc/delay.h --- a/include/asm-sparc/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-sparc/delay.h @@ -28,7 +28,6 @@ extern void __ndelay(unsigned long nsecs #else /* SMP */ #define __udelay_val loops_per_jiffy #endif /* SMP */ -#define udelay(__usecs) __udelay(__usecs, __udelay_val) #define ndelay(__nsecs) __ndelay(__nsecs, __udelay_val) #endif /* defined(__SPARC_DELAY_H) */ diff -puN include/asm-sparc64/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-sparc64/delay.h --- a/include/asm-sparc64/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-sparc64/delay.h @@ -21,15 +21,11 @@ extern void __bad_ndelay(void); extern void __udelay(unsigned long usecs); extern void __ndelay(unsigned long nsecs); -extern void __const_udelay(unsigned long usecs); +extern void __const_delay(unsigned long usecs); extern void __delay(unsigned long loops); -#define udelay(n) (__builtin_constant_p(n) ? \ - ((n) > 20000 ? __bad_udelay() : __const_udelay((n) * 0x10c7ul)) : \ - __udelay(n)) - #define ndelay(n) (__builtin_constant_p(n) ? \ - ((n) > 20000 ? __bad_ndelay() : __const_udelay((n) * 5ul)) : \ + ((n) > 20000 ? __bad_ndelay() : __const_delay((n) * 5ul)) : \ __ndelay(n)) #endif /* !__ASSEMBLY__ */ diff -puN include/asm-v850/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-v850/delay.h --- a/include/asm-v850/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-v850/delay.h @@ -33,7 +33,7 @@ static inline void __delay(unsigned long extern unsigned long loops_per_jiffy; -static inline void udelay(unsigned long usecs) +static inline void __udelay(unsigned long usecs) { register unsigned long full_loops, part_loops; diff -puN include/asm-x86_64/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-x86_64/delay.h --- a/include/asm-x86_64/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-x86_64/delay.h @@ -7,20 +7,15 @@ * Delay routines calling functions in arch/x86_64/lib/delay.c */ -extern void __bad_udelay(void); extern void __bad_ndelay(void); extern void __udelay(unsigned long usecs); extern void __ndelay(unsigned long usecs); -extern void __const_udelay(unsigned long usecs); +extern void __const_delay(unsigned long usecs); extern void __delay(unsigned long loops); -#define udelay(n) (__builtin_constant_p(n) ? \ - ((n) > 20000 ? __bad_udelay() : __const_udelay((n) * 0x10c6ul)) : \ - __udelay(n)) - #define ndelay(n) (__builtin_constant_p(n) ? \ - ((n) > 20000 ? __bad_ndelay() : __const_udelay((n) * 5ul)) : \ + ((n) > 20000 ? __bad_ndelay() : __const_delay((n) * 5ul)) : \ __ndelay(n)) diff -puN include/asm-xtensa/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/asm-xtensa/delay.h --- a/include/asm-xtensa/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/asm-xtensa/delay.h @@ -35,7 +35,7 @@ static __inline__ u32 xtensa_get_ccount( * local_cpu_data->... where local_cpu_data points to the current * cpu. */ -static __inline__ void udelay (unsigned long usecs) +static __inline__ void __udelay (unsigned long usecs) { unsigned long start = xtensa_get_ccount(); unsigned long cycles = usecs * (loops_per_jiffy / (1000000UL / HZ)); diff -puN include/linux/delay.h~delay-add-generic-udelay-mdelay-and-ssleep include/linux/delay.h --- a/include/linux/delay.h~delay-add-generic-udelay-mdelay-and-ssleep +++ a/include/linux/delay.h @@ -11,37 +11,15 @@ extern unsigned long loops_per_jiffy; #include <asm/delay.h> -/* - * Using udelay() for intervals greater than a few milliseconds can - * risk overflow for high loops_per_jiffy (high bogomips) machines. The - * mdelay() provides a wrapper to prevent this. For delays greater - * than MAX_UDELAY_MS milliseconds, the wrapper is used. Architecture - * specific values can be defined in asm-???/delay.h as an override. - * The 2nd mdelay() definition ensures GCC will optimize away the - * while loop for the common cases where n <= MAX_UDELAY_MS -- Paul G. - */ - -#ifndef MAX_UDELAY_MS -#define MAX_UDELAY_MS 5 -#endif - -#ifndef mdelay -#define mdelay(n) (\ - (__builtin_constant_p(n) && (n)<=MAX_UDELAY_MS) ? udelay((n)*1000) : \ - ({unsigned long __ms=(n); while (__ms--) udelay(1000);})) -#endif - +void calibrate_delay(void); +void udelay(unsigned int usecs); +void mdelay(unsigned int msecs); #ifndef ndelay #define ndelay(x) udelay(((x)+999)/1000) #endif -void calibrate_delay(void); void msleep(unsigned int msecs); +void ssleep(unsigned int secs); unsigned long msleep_interruptible(unsigned int msecs); -static inline void ssleep(unsigned int seconds) -{ - msleep(seconds * 1000); -} - #endif /* defined(_LINUX_DELAY_H) */ diff -puN kernel/timer.c~delay-add-generic-udelay-mdelay-and-ssleep kernel/timer.c --- a/kernel/timer.c~delay-add-generic-udelay-mdelay-and-ssleep +++ a/kernel/timer.c @@ -1883,6 +1883,36 @@ unregister_time_interpolator(struct time } #endif /* CONFIG_TIME_INTERPOLATION */ +/* + * Not inlined because we do not optimize delays for speed. ;) + */ +void udelay(unsigned int usecs) +{ + unsigned int k; + if (unlikely(usecs > 200*1000)) { + printk("BUG: delay too large: udelay(%u)\n", usecs); + dump_stack(); + usecs = 200*1000; + } + k = usecs / 1024; + usecs %= 1024; + while (k) { + __udelay(1024); + k--; + } + __udelay(usecs); +} + +EXPORT_SYMBOL(udelay); + +void mdelay(unsigned int msecs) +{ + while (msecs--) + __udelay(1000); +} + +EXPORT_SYMBOL(mdelay); + /** * msleep - sleep safely even with waitqueue interruptions * @msecs: Time in milliseconds to sleep for @@ -1897,6 +1927,13 @@ void msleep(unsigned int msecs) EXPORT_SYMBOL(msleep); +void ssleep(unsigned int secs) +{ + msleep(secs * 1000); +} + +EXPORT_SYMBOL(ssleep); + /** * msleep_interruptible - sleep waiting for signals * @msecs: Time in milliseconds to sleep for _ Patches currently in -mm which might be from vda.linux@xxxxxxxxxxxxxx are git-drm.patch aic7xxx-fix-byte-i-o-order-in-ahd_inw.patch delay-s-include-asm-delayh-include-linux-delayh.patch delay-remove-references-to-max_udelay_ms-fix-comment.patch delay-add-generic-udelay-mdelay-and-ssleep.patch - To unsubscribe from this list: send the line "unsubscribe mm-commits" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html