diff options
Diffstat (limited to 'include')
-rw-r--r-- | include/asm-mips/io.h | 31 | ||||
-rw-r--r-- | include/asm-mips/mach-au1x00/ioremap.h | 30 | ||||
-rw-r--r-- | include/asm-mips/mach-generic/ioremap.h | 23 |
3 files changed, 82 insertions, 2 deletions
diff --git a/include/asm-mips/io.h b/include/asm-mips/io.h index 3b4d97d80643..42f80782acd2 100644 --- a/include/asm-mips/io.h +++ b/include/asm-mips/io.h | |||
@@ -27,6 +27,7 @@ | |||
27 | #include <asm/processor.h> | 27 | #include <asm/processor.h> |
28 | #include <asm/string.h> | 28 | #include <asm/string.h> |
29 | 29 | ||
30 | #include <ioremap.h> | ||
30 | #include <mangle-port.h> | 31 | #include <mangle-port.h> |
31 | 32 | ||
32 | /* | 33 | /* |
@@ -209,6 +210,8 @@ extern void __iounmap(volatile void __iomem *addr); | |||
209 | static inline void __iomem * __ioremap_mode(phys_t offset, unsigned long size, | 210 | static inline void __iomem * __ioremap_mode(phys_t offset, unsigned long size, |
210 | unsigned long flags) | 211 | unsigned long flags) |
211 | { | 212 | { |
213 | #define __IS_LOW512(addr) (!((phys_t)(addr) & (phys_t) ~0x1fffffffULL)) | ||
214 | |||
212 | if (cpu_has_64bit_addresses) { | 215 | if (cpu_has_64bit_addresses) { |
213 | u64 base = UNCAC_BASE; | 216 | u64 base = UNCAC_BASE; |
214 | 217 | ||
@@ -219,9 +222,29 @@ static inline void __iomem * __ioremap_mode(phys_t offset, unsigned long size, | |||
219 | if (flags == _CACHE_UNCACHED) | 222 | if (flags == _CACHE_UNCACHED) |
220 | base = (u64) IO_BASE; | 223 | base = (u64) IO_BASE; |
221 | return (void __iomem *) (unsigned long) (base + offset); | 224 | return (void __iomem *) (unsigned long) (base + offset); |
225 | } else if (__builtin_constant_p(offset) && | ||
226 | __builtin_constant_p(size) && __builtin_constant_p(flags)) { | ||
227 | phys_t phys_addr, last_addr; | ||
228 | |||
229 | phys_addr = fixup_bigphys_addr(offset, size); | ||
230 | |||
231 | /* Don't allow wraparound or zero size. */ | ||
232 | last_addr = phys_addr + size - 1; | ||
233 | if (!size || last_addr < phys_addr) | ||
234 | return NULL; | ||
235 | |||
236 | /* | ||
237 | * Map uncached objects in the low 512MB of address | ||
238 | * space using KSEG1. | ||
239 | */ | ||
240 | if (__IS_LOW512(phys_addr) && __IS_LOW512(last_addr) && | ||
241 | flags == _CACHE_UNCACHED) | ||
242 | return (void __iomem *)CKSEG1ADDR(phys_addr); | ||
222 | } | 243 | } |
223 | 244 | ||
224 | return __ioremap(offset, size, flags); | 245 | return __ioremap(offset, size, flags); |
246 | |||
247 | #undef __IS_LOW512 | ||
225 | } | 248 | } |
226 | 249 | ||
227 | /* | 250 | /* |
@@ -273,12 +296,16 @@ static inline void __iomem * __ioremap_mode(phys_t offset, unsigned long size, | |||
273 | 296 | ||
274 | static inline void iounmap(volatile void __iomem *addr) | 297 | static inline void iounmap(volatile void __iomem *addr) |
275 | { | 298 | { |
276 | if (cpu_has_64bit_addresses) | 299 | #define __IS_KSEG1(addr) (((unsigned long)(addr) & ~0x1fffffffUL) == CKSEG1) |
300 | |||
301 | if (cpu_has_64bit_addresses || | ||
302 | (__builtin_constant_p(addr) && __IS_KSEG1(addr))) | ||
277 | return; | 303 | return; |
278 | 304 | ||
279 | __iounmap(addr); | 305 | __iounmap(addr); |
280 | } | ||
281 | 306 | ||
307 | #undef __IS_KSEG1 | ||
308 | } | ||
282 | 309 | ||
283 | #define __BUILD_MEMORY_SINGLE(pfx, bwlq, type, irq) \ | 310 | #define __BUILD_MEMORY_SINGLE(pfx, bwlq, type, irq) \ |
284 | \ | 311 | \ |
diff --git a/include/asm-mips/mach-au1x00/ioremap.h b/include/asm-mips/mach-au1x00/ioremap.h new file mode 100644 index 000000000000..f1c8c1087115 --- /dev/null +++ b/include/asm-mips/mach-au1x00/ioremap.h | |||
@@ -0,0 +1,30 @@ | |||
1 | /* | ||
2 | * include/asm-mips/mach-au1x00/ioremap.h | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or | ||
5 | * modify it under the terms of the GNU General Public License | ||
6 | * as published by the Free Software Foundation; either version | ||
7 | * 2 of the License, or (at your option) any later version. | ||
8 | */ | ||
9 | #ifndef __ASM_MACH_AU1X00_IOREMAP_H | ||
10 | #define __ASM_MACH_AU1X00_IOREMAP_H | ||
11 | |||
12 | #include <linux/config.h> | ||
13 | #include <linux/types.h> | ||
14 | |||
15 | #ifndef CONFIG_64BIT_PHYS_ADDR | ||
16 | static inline phys_t __fixup_bigphys_addr(phys_t phys_addr, phys_t size) | ||
17 | { | ||
18 | return phys_addr; | ||
19 | } | ||
20 | #endif | ||
21 | |||
22 | /* | ||
23 | * Allow physical addresses to be fixed up to help 36-bit peripherals. | ||
24 | */ | ||
25 | static inline phys_t fixup_bigphys_addr(phys_t phys_addr, phys_t size) | ||
26 | { | ||
27 | return __fixup_bigphys_addr(phys_addr, size); | ||
28 | } | ||
29 | |||
30 | #endif /* __ASM_MACH_AU1X00_IOREMAP_H */ | ||
diff --git a/include/asm-mips/mach-generic/ioremap.h b/include/asm-mips/mach-generic/ioremap.h new file mode 100644 index 000000000000..9b64ff6e485d --- /dev/null +++ b/include/asm-mips/mach-generic/ioremap.h | |||
@@ -0,0 +1,23 @@ | |||
1 | /* | ||
2 | * include/asm-mips/mach-generic/ioremap.h | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or | ||
5 | * modify it under the terms of the GNU General Public License | ||
6 | * as published by the Free Software Foundation; either version | ||
7 | * 2 of the License, or (at your option) any later version. | ||
8 | */ | ||
9 | #ifndef __ASM_MACH_GENERIC_IOREMAP_H | ||
10 | #define __ASM_MACH_GENERIC_IOREMAP_H | ||
11 | |||
12 | #include <linux/types.h> | ||
13 | |||
14 | /* | ||
15 | * Allow physical addresses to be fixed up to help peripherals located | ||
16 | * outside the low 32-bit range -- generic pass-through version. | ||
17 | */ | ||
18 | static inline phys_t fixup_bigphys_addr(phys_t phys_addr, phys_t size) | ||
19 | { | ||
20 | return phys_addr; | ||
21 | } | ||
22 | |||
23 | #endif /* __ASM_MACH_GENERIC_IOREMAP_H */ | ||