diff options
Diffstat (limited to 'arch/arm')
-rw-r--r-- | arch/arm/Kconfig | 34 | ||||
-rw-r--r-- | arch/arm/include/asm/hwcap.h | 1 | ||||
-rw-r--r-- | arch/arm/include/asm/irq.h | 2 | ||||
-rw-r--r-- | arch/arm/include/asm/mach/arch.h | 1 | ||||
-rw-r--r-- | arch/arm/include/asm/mach/irq.h | 1 | ||||
-rw-r--r-- | arch/arm/include/asm/mach/map.h | 2 | ||||
-rw-r--r-- | arch/arm/include/asm/memory.h | 9 | ||||
-rw-r--r-- | arch/arm/include/asm/ptrace.h | 36 | ||||
-rw-r--r-- | arch/arm/include/asm/tls.h | 46 | ||||
-rw-r--r-- | arch/arm/include/asm/vfpmacros.h | 18 | ||||
-rw-r--r-- | arch/arm/kernel/entry-armv.S | 23 | ||||
-rw-r--r-- | arch/arm/kernel/irq.c | 41 | ||||
-rw-r--r-- | arch/arm/kernel/ptrace.c | 96 | ||||
-rw-r--r-- | arch/arm/kernel/setup.c | 18 | ||||
-rw-r--r-- | arch/arm/kernel/tcm.c | 118 | ||||
-rw-r--r-- | arch/arm/kernel/traps.c | 41 | ||||
-rw-r--r-- | arch/arm/mach-u300/include/mach/memory.h | 8 | ||||
-rw-r--r-- | arch/arm/mm/Kconfig | 11 | ||||
-rw-r--r-- | arch/arm/mm/init.c | 13 | ||||
-rw-r--r-- | arch/arm/mm/ioremap.c | 6 | ||||
-rw-r--r-- | arch/arm/mm/mmu.c | 13 | ||||
-rw-r--r-- | arch/arm/mm/proc-v6.S | 5 | ||||
-rw-r--r-- | arch/arm/mm/proc-v7.S | 2 | ||||
-rw-r--r-- | arch/arm/vfp/vfpmodule.c | 10 |
24 files changed, 439 insertions, 116 deletions
diff --git a/arch/arm/Kconfig b/arch/arm/Kconfig index 98922f7d2d1..8ef1e23a24a 100644 --- a/arch/arm/Kconfig +++ b/arch/arm/Kconfig | |||
@@ -24,6 +24,7 @@ config ARM | |||
24 | select HAVE_KERNEL_LZMA | 24 | select HAVE_KERNEL_LZMA |
25 | select HAVE_PERF_EVENTS | 25 | select HAVE_PERF_EVENTS |
26 | select PERF_USE_VMALLOC | 26 | select PERF_USE_VMALLOC |
27 | select HAVE_REGS_AND_STACK_ACCESS_API | ||
27 | help | 28 | help |
28 | The ARM series is a line of low-power-consumption RISC chip designs | 29 | The ARM series is a line of low-power-consumption RISC chip designs |
29 | licensed by ARM Ltd and targeted at embedded applications and | 30 | licensed by ARM Ltd and targeted at embedded applications and |
@@ -1031,11 +1032,6 @@ endmenu | |||
1031 | 1032 | ||
1032 | source "arch/arm/common/Kconfig" | 1033 | source "arch/arm/common/Kconfig" |
1033 | 1034 | ||
1034 | config FORCE_MAX_ZONEORDER | ||
1035 | int | ||
1036 | depends on SA1111 | ||
1037 | default "9" | ||
1038 | |||
1039 | menu "Bus support" | 1035 | menu "Bus support" |
1040 | 1036 | ||
1041 | config ARM_AMBA | 1037 | config ARM_AMBA |
@@ -1290,8 +1286,36 @@ config HW_PERF_EVENTS | |||
1290 | Enable hardware performance counter support for perf events. If | 1286 | Enable hardware performance counter support for perf events. If |
1291 | disabled, perf events will use software events only. | 1287 | disabled, perf events will use software events only. |
1292 | 1288 | ||
1289 | config SPARSE_IRQ | ||
1290 | bool "Support sparse irq numbering" | ||
1291 | depends on EXPERIMENTAL | ||
1292 | help | ||
1293 | This enables support for sparse irqs. This is useful in general | ||
1294 | as most CPUs have a fairly sparse array of IRQ vectors, which | ||
1295 | the irq_desc then maps directly on to. Systems with a high | ||
1296 | number of off-chip IRQs will want to treat this as | ||
1297 | experimental until they have been independently verified. | ||
1298 | |||
1299 | If you don't know what to do here, say N. | ||
1300 | |||
1293 | source "mm/Kconfig" | 1301 | source "mm/Kconfig" |
1294 | 1302 | ||
1303 | config FORCE_MAX_ZONEORDER | ||
1304 | int "Maximum zone order" if ARCH_SHMOBILE | ||
1305 | range 11 64 if ARCH_SHMOBILE | ||
1306 | default "9" if SA1111 | ||
1307 | default "11" | ||
1308 | help | ||
1309 | The kernel memory allocator divides physically contiguous memory | ||
1310 | blocks into "zones", where each zone is a power of two number of | ||
1311 | pages. This option selects the largest power of two that the kernel | ||
1312 | keeps in the memory allocator. If you need to allocate very large | ||
1313 | blocks of physically contiguous memory, then you may need to | ||
1314 | increase this value. | ||
1315 | |||
1316 | This config option is actually maximum order plus one. For example, | ||
1317 | a value of 11 means that the largest free memory block is 2^10 pages. | ||
1318 | |||
1295 | config LEDS | 1319 | config LEDS |
1296 | bool "Timer and CPU usage LEDs" | 1320 | bool "Timer and CPU usage LEDs" |
1297 | depends on ARCH_CDB89712 || ARCH_EBSA110 || \ | 1321 | depends on ARCH_CDB89712 || ARCH_EBSA110 || \ |
diff --git a/arch/arm/include/asm/hwcap.h b/arch/arm/include/asm/hwcap.h index f7bd52b1c36..c1062c31710 100644 --- a/arch/arm/include/asm/hwcap.h +++ b/arch/arm/include/asm/hwcap.h | |||
@@ -19,6 +19,7 @@ | |||
19 | #define HWCAP_NEON 4096 | 19 | #define HWCAP_NEON 4096 |
20 | #define HWCAP_VFPv3 8192 | 20 | #define HWCAP_VFPv3 8192 |
21 | #define HWCAP_VFPv3D16 16384 | 21 | #define HWCAP_VFPv3D16 16384 |
22 | #define HWCAP_TLS 32768 | ||
22 | 23 | ||
23 | #if defined(__KERNEL__) && !defined(__ASSEMBLY__) | 24 | #if defined(__KERNEL__) && !defined(__ASSEMBLY__) |
24 | /* | 25 | /* |
diff --git a/arch/arm/include/asm/irq.h b/arch/arm/include/asm/irq.h index 237282f7c76..2721a5814cb 100644 --- a/arch/arm/include/asm/irq.h +++ b/arch/arm/include/asm/irq.h | |||
@@ -7,6 +7,8 @@ | |||
7 | #define irq_canonicalize(i) (i) | 7 | #define irq_canonicalize(i) (i) |
8 | #endif | 8 | #endif |
9 | 9 | ||
10 | #define NR_IRQS_LEGACY 16 | ||
11 | |||
10 | /* | 12 | /* |
11 | * Use this value to indicate lack of interrupt | 13 | * Use this value to indicate lack of interrupt |
12 | * capability | 14 | * capability |
diff --git a/arch/arm/include/asm/mach/arch.h b/arch/arm/include/asm/mach/arch.h index c59842dc7cb..5ee6f85236a 100644 --- a/arch/arm/include/asm/mach/arch.h +++ b/arch/arm/include/asm/mach/arch.h | |||
@@ -20,6 +20,7 @@ struct machine_desc { | |||
20 | * by assembler code in head.S, head-common.S | 20 | * by assembler code in head.S, head-common.S |
21 | */ | 21 | */ |
22 | unsigned int nr; /* architecture number */ | 22 | unsigned int nr; /* architecture number */ |
23 | unsigned int nr_irqs; /* number of IRQs */ | ||
23 | unsigned int phys_io; /* start of physical io */ | 24 | unsigned int phys_io; /* start of physical io */ |
24 | unsigned int io_pg_offst; /* byte offset for io | 25 | unsigned int io_pg_offst; /* byte offset for io |
25 | * page tabe entry */ | 26 | * page tabe entry */ |
diff --git a/arch/arm/include/asm/mach/irq.h b/arch/arm/include/asm/mach/irq.h index 8920b2d6e3b..ce3eee9fe26 100644 --- a/arch/arm/include/asm/mach/irq.h +++ b/arch/arm/include/asm/mach/irq.h | |||
@@ -17,6 +17,7 @@ struct seq_file; | |||
17 | /* | 17 | /* |
18 | * This is internal. Do not use it. | 18 | * This is internal. Do not use it. |
19 | */ | 19 | */ |
20 | extern unsigned int arch_nr_irqs; | ||
20 | extern void (*init_arch_irq)(void); | 21 | extern void (*init_arch_irq)(void); |
21 | extern void init_FIQ(void); | 22 | extern void init_FIQ(void); |
22 | extern int show_fiq_list(struct seq_file *, void *); | 23 | extern int show_fiq_list(struct seq_file *, void *); |
diff --git a/arch/arm/include/asm/mach/map.h b/arch/arm/include/asm/mach/map.h index 742c2aaeb02..d2fedb5aeb1 100644 --- a/arch/arm/include/asm/mach/map.h +++ b/arch/arm/include/asm/mach/map.h | |||
@@ -27,6 +27,8 @@ struct map_desc { | |||
27 | #define MT_MEMORY 9 | 27 | #define MT_MEMORY 9 |
28 | #define MT_ROM 10 | 28 | #define MT_ROM 10 |
29 | #define MT_MEMORY_NONCACHED 11 | 29 | #define MT_MEMORY_NONCACHED 11 |
30 | #define MT_MEMORY_DTCM 12 | ||
31 | #define MT_MEMORY_ITCM 13 | ||
30 | 32 | ||
31 | #ifdef CONFIG_MMU | 33 | #ifdef CONFIG_MMU |
32 | extern void iotable_init(struct map_desc *, int); | 34 | extern void iotable_init(struct map_desc *, int); |
diff --git a/arch/arm/include/asm/memory.h b/arch/arm/include/asm/memory.h index 4312ee5e3d0..ab08d977ad4 100644 --- a/arch/arm/include/asm/memory.h +++ b/arch/arm/include/asm/memory.h | |||
@@ -124,6 +124,15 @@ | |||
124 | #endif /* !CONFIG_MMU */ | 124 | #endif /* !CONFIG_MMU */ |
125 | 125 | ||
126 | /* | 126 | /* |
127 | * We fix the TCM memories max 32 KiB ITCM resp DTCM at these | ||
128 | * locations | ||
129 | */ | ||
130 | #ifdef CONFIG_HAVE_TCM | ||
131 | #define ITCM_OFFSET UL(0xfffe0000) | ||
132 | #define DTCM_OFFSET UL(0xfffe8000) | ||
133 | #endif | ||
134 | |||
135 | /* | ||
127 | * Physical vs virtual RAM address space conversion. These are | 136 | * Physical vs virtual RAM address space conversion. These are |
128 | * private definitions which should NOT be used outside memory.h | 137 | * private definitions which should NOT be used outside memory.h |
129 | * files. Use virt_to_phys/phys_to_virt/__pa/__va instead. | 138 | * files. Use virt_to_phys/phys_to_virt/__pa/__va instead. |
diff --git a/arch/arm/include/asm/ptrace.h b/arch/arm/include/asm/ptrace.h index 9dcb11e5902..c974be8913a 100644 --- a/arch/arm/include/asm/ptrace.h +++ b/arch/arm/include/asm/ptrace.h | |||
@@ -184,6 +184,42 @@ extern unsigned long profile_pc(struct pt_regs *regs); | |||
184 | #define predicate(x) ((x) & 0xf0000000) | 184 | #define predicate(x) ((x) & 0xf0000000) |
185 | #define PREDICATE_ALWAYS 0xe0000000 | 185 | #define PREDICATE_ALWAYS 0xe0000000 |
186 | 186 | ||
187 | /* | ||
188 | * kprobe-based event tracer support | ||
189 | */ | ||
190 | #include <linux/stddef.h> | ||
191 | #include <linux/types.h> | ||
192 | #define MAX_REG_OFFSET (offsetof(struct pt_regs, ARM_ORIG_r0)) | ||
193 | |||
194 | extern int regs_query_register_offset(const char *name); | ||
195 | extern const char *regs_query_register_name(unsigned int offset); | ||
196 | extern bool regs_within_kernel_stack(struct pt_regs *regs, unsigned long addr); | ||
197 | extern unsigned long regs_get_kernel_stack_nth(struct pt_regs *regs, | ||
198 | unsigned int n); | ||
199 | |||
200 | /** | ||
201 | * regs_get_register() - get register value from its offset | ||
202 | * @regs: pt_regs from which register value is gotten | ||
203 | * @offset: offset number of the register. | ||
204 | * | ||
205 | * regs_get_register returns the value of a register whose offset from @regs. | ||
206 | * The @offset is the offset of the register in struct pt_regs. | ||
207 | * If @offset is bigger than MAX_REG_OFFSET, this returns 0. | ||
208 | */ | ||
209 | static inline unsigned long regs_get_register(struct pt_regs *regs, | ||
210 | unsigned int offset) | ||
211 | { | ||
212 | if (unlikely(offset > MAX_REG_OFFSET)) | ||
213 | return 0; | ||
214 | return *(unsigned long *)((unsigned long)regs + offset); | ||
215 | } | ||
216 | |||
217 | /* Valid only for Kernel mode traps. */ | ||
218 | static inline unsigned long kernel_stack_pointer(struct pt_regs *regs) | ||
219 | { | ||
220 | return regs->ARM_sp; | ||
221 | } | ||
222 | |||
187 | #endif /* __KERNEL__ */ | 223 | #endif /* __KERNEL__ */ |
188 | 224 | ||
189 | #endif /* __ASSEMBLY__ */ | 225 | #endif /* __ASSEMBLY__ */ |
diff --git a/arch/arm/include/asm/tls.h b/arch/arm/include/asm/tls.h new file mode 100644 index 00000000000..e71d6ff8d10 --- /dev/null +++ b/arch/arm/include/asm/tls.h | |||
@@ -0,0 +1,46 @@ | |||
1 | #ifndef __ASMARM_TLS_H | ||
2 | #define __ASMARM_TLS_H | ||
3 | |||
4 | #ifdef __ASSEMBLY__ | ||
5 | .macro set_tls_none, tp, tmp1, tmp2 | ||
6 | .endm | ||
7 | |||
8 | .macro set_tls_v6k, tp, tmp1, tmp2 | ||
9 | mcr p15, 0, \tp, c13, c0, 3 @ set TLS register | ||
10 | .endm | ||
11 | |||
12 | .macro set_tls_v6, tp, tmp1, tmp2 | ||
13 | ldr \tmp1, =elf_hwcap | ||
14 | ldr \tmp1, [\tmp1, #0] | ||
15 | mov \tmp2, #0xffff0fff | ||
16 | tst \tmp1, #HWCAP_TLS @ hardware TLS available? | ||
17 | mcrne p15, 0, \tp, c13, c0, 3 @ yes, set TLS register | ||
18 | streq \tp, [\tmp2, #-15] @ set TLS value at 0xffff0ff0 | ||
19 | .endm | ||
20 | |||
21 | .macro set_tls_software, tp, tmp1, tmp2 | ||
22 | mov \tmp1, #0xffff0fff | ||
23 | str \tp, [\tmp1, #-15] @ set TLS value at 0xffff0ff0 | ||
24 | .endm | ||
25 | #endif | ||
26 | |||
27 | #ifdef CONFIG_TLS_REG_EMUL | ||
28 | #define tls_emu 1 | ||
29 | #define has_tls_reg 1 | ||
30 | #define set_tls set_tls_none | ||
31 | #elif __LINUX_ARM_ARCH__ >= 7 || \ | ||
32 | (__LINUX_ARM_ARCH__ == 6 && defined(CONFIG_CPU_32v6K)) | ||
33 | #define tls_emu 0 | ||
34 | #define has_tls_reg 1 | ||
35 | #define set_tls set_tls_v6k | ||
36 | #elif __LINUX_ARM_ARCH__ == 6 | ||
37 | #define tls_emu 0 | ||
38 | #define has_tls_reg (elf_hwcap & HWCAP_TLS) | ||
39 | #define set_tls set_tls_v6 | ||
40 | #else | ||
41 | #define tls_emu 0 | ||
42 | #define has_tls_reg 0 | ||
43 | #define set_tls set_tls_software | ||
44 | #endif | ||
45 | |||
46 | #endif /* __ASMARM_TLS_H */ | ||
diff --git a/arch/arm/include/asm/vfpmacros.h b/arch/arm/include/asm/vfpmacros.h index 422f3cc204a..3d5fc41ae8d 100644 --- a/arch/arm/include/asm/vfpmacros.h +++ b/arch/arm/include/asm/vfpmacros.h | |||
@@ -3,6 +3,8 @@ | |||
3 | * | 3 | * |
4 | * Assembler-only file containing VFP macros and register definitions. | 4 | * Assembler-only file containing VFP macros and register definitions. |
5 | */ | 5 | */ |
6 | #include <asm/hwcap.h> | ||
7 | |||
6 | #include "vfp.h" | 8 | #include "vfp.h" |
7 | 9 | ||
8 | @ Macros to allow building with old toolkits (with no VFP support) | 10 | @ Macros to allow building with old toolkits (with no VFP support) |
@@ -22,12 +24,20 @@ | |||
22 | LDC p11, cr0, [\base],#32*4 @ FLDMIAD \base!, {d0-d15} | 24 | LDC p11, cr0, [\base],#32*4 @ FLDMIAD \base!, {d0-d15} |
23 | #endif | 25 | #endif |
24 | #ifdef CONFIG_VFPv3 | 26 | #ifdef CONFIG_VFPv3 |
27 | #if __LINUX_ARM_ARCH__ <= 6 | ||
28 | ldr \tmp, =elf_hwcap @ may not have MVFR regs | ||
29 | ldr \tmp, [\tmp, #0] | ||
30 | tst \tmp, #HWCAP_VFPv3D16 | ||
31 | ldceq p11, cr0, [\base],#32*4 @ FLDMIAD \base!, {d16-d31} | ||
32 | addne \base, \base, #32*4 @ step over unused register space | ||
33 | #else | ||
25 | VFPFMRX \tmp, MVFR0 @ Media and VFP Feature Register 0 | 34 | VFPFMRX \tmp, MVFR0 @ Media and VFP Feature Register 0 |
26 | and \tmp, \tmp, #MVFR0_A_SIMD_MASK @ A_SIMD field | 35 | and \tmp, \tmp, #MVFR0_A_SIMD_MASK @ A_SIMD field |
27 | cmp \tmp, #2 @ 32 x 64bit registers? | 36 | cmp \tmp, #2 @ 32 x 64bit registers? |
28 | ldceql p11, cr0, [\base],#32*4 @ FLDMIAD \base!, {d16-d31} | 37 | ldceql p11, cr0, [\base],#32*4 @ FLDMIAD \base!, {d16-d31} |
29 | addne \base, \base, #32*4 @ step over unused register space | 38 | addne \base, \base, #32*4 @ step over unused register space |
30 | #endif | 39 | #endif |
40 | #endif | ||
31 | .endm | 41 | .endm |
32 | 42 | ||
33 | @ write all the working registers out of the VFP | 43 | @ write all the working registers out of the VFP |
@@ -38,10 +48,18 @@ | |||
38 | STC p11, cr0, [\base],#32*4 @ FSTMIAD \base!, {d0-d15} | 48 | STC p11, cr0, [\base],#32*4 @ FSTMIAD \base!, {d0-d15} |
39 | #endif | 49 | #endif |
40 | #ifdef CONFIG_VFPv3 | 50 | #ifdef CONFIG_VFPv3 |
51 | #if __LINUX_ARM_ARCH__ <= 6 | ||
52 | ldr \tmp, =elf_hwcap @ may not have MVFR regs | ||
53 | ldr \tmp, [\tmp, #0] | ||
54 | tst \tmp, #HWCAP_VFPv3D16 | ||
55 | stceq p11, cr0, [\base],#32*4 @ FSTMIAD \base!, {d16-d31} | ||
56 | addne \base, \base, #32*4 @ step over unused register space | ||
57 | #else | ||
41 | VFPFMRX \tmp, MVFR0 @ Media and VFP Feature Register 0 | 58 | VFPFMRX \tmp, MVFR0 @ Media and VFP Feature Register 0 |
42 | and \tmp, \tmp, #MVFR0_A_SIMD_MASK @ A_SIMD field | 59 | and \tmp, \tmp, #MVFR0_A_SIMD_MASK @ A_SIMD field |
43 | cmp \tmp, #2 @ 32 x 64bit registers? | 60 | cmp \tmp, #2 @ 32 x 64bit registers? |
44 | stceql p11, cr0, [\base],#32*4 @ FSTMIAD \base!, {d16-d31} | 61 | stceql p11, cr0, [\base],#32*4 @ FSTMIAD \base!, {d16-d31} |
45 | addne \base, \base, #32*4 @ step over unused register space | 62 | addne \base, \base, #32*4 @ step over unused register space |
46 | #endif | 63 | #endif |
64 | #endif | ||
47 | .endm | 65 | .endm |
diff --git a/arch/arm/kernel/entry-armv.S b/arch/arm/kernel/entry-armv.S index 3fd7861de4d..e864e482118 100644 --- a/arch/arm/kernel/entry-armv.S +++ b/arch/arm/kernel/entry-armv.S | |||
@@ -22,6 +22,7 @@ | |||
22 | #include <asm/thread_notify.h> | 22 | #include <asm/thread_notify.h> |
23 | #include <asm/unwind.h> | 23 | #include <asm/unwind.h> |
24 | #include <asm/unistd.h> | 24 | #include <asm/unistd.h> |
25 | #include <asm/tls.h> | ||
25 | 26 | ||
26 | #include "entry-header.S" | 27 | #include "entry-header.S" |
27 | 28 | ||
@@ -735,12 +736,7 @@ ENTRY(__switch_to) | |||
735 | #ifdef CONFIG_MMU | 736 | #ifdef CONFIG_MMU |
736 | ldr r6, [r2, #TI_CPU_DOMAIN] | 737 | ldr r6, [r2, #TI_CPU_DOMAIN] |
737 | #endif | 738 | #endif |
738 | #if defined(CONFIG_HAS_TLS_REG) | 739 | set_tls r3, r4, r5 |
739 | mcr p15, 0, r3, c13, c0, 3 @ set TLS register | ||
740 | #elif !defined(CONFIG_TLS_REG_EMUL) | ||
741 | mov r4, #0xffff0fff | ||
742 | str r3, [r4, #-15] @ TLS val at 0xffff0ff0 | ||
743 | #endif | ||
744 | #ifdef CONFIG_MMU | 740 | #ifdef CONFIG_MMU |
745 | mcr p15, 0, r6, c3, c0, 0 @ Set domain register | 741 | mcr p15, 0, r6, c3, c0, 0 @ Set domain register |
746 | #endif | 742 | #endif |
@@ -1005,17 +1001,12 @@ kuser_cmpxchg_fixup: | |||
1005 | */ | 1001 | */ |
1006 | 1002 | ||
1007 | __kuser_get_tls: @ 0xffff0fe0 | 1003 | __kuser_get_tls: @ 0xffff0fe0 |
1008 | 1004 | ldr r0, [pc, #(16 - 8)] @ read TLS, set in kuser_get_tls_init | |
1009 | #if !defined(CONFIG_HAS_TLS_REG) && !defined(CONFIG_TLS_REG_EMUL) | ||
1010 | ldr r0, [pc, #(16 - 8)] @ TLS stored at 0xffff0ff0 | ||
1011 | #else | ||
1012 | mrc p15, 0, r0, c13, c0, 3 @ read TLS register | ||
1013 | #endif | ||
1014 | usr_ret lr | 1005 | usr_ret lr |
1015 | 1006 | mrc p15, 0, r0, c13, c0, 3 @ 0xffff0fe8 hardware TLS code | |
1016 | .rep 5 | 1007 | .rep 4 |
1017 | .word 0 @ pad up to __kuser_helper_version | 1008 | .word 0 @ 0xffff0ff0 software TLS value, then |
1018 | .endr | 1009 | .endr @ pad up to __kuser_helper_version |
1019 | 1010 | ||
1020 | /* | 1011 | /* |
1021 | * Reference declaration: | 1012 | * Reference declaration: |
diff --git a/arch/arm/kernel/irq.c b/arch/arm/kernel/irq.c index 3b3d2c80509..c0d5c3b3a76 100644 --- a/arch/arm/kernel/irq.c +++ b/arch/arm/kernel/irq.c | |||
@@ -47,12 +47,14 @@ | |||
47 | #define irq_finish(irq) do { } while (0) | 47 | #define irq_finish(irq) do { } while (0) |
48 | #endif | 48 | #endif |
49 | 49 | ||
50 | unsigned int arch_nr_irqs; | ||
50 | void (*init_arch_irq)(void) __initdata = NULL; | 51 | void (*init_arch_irq)(void) __initdata = NULL; |
51 | unsigned long irq_err_count; | 52 | unsigned long irq_err_count; |
52 | 53 | ||
53 | int show_interrupts(struct seq_file *p, void *v) | 54 | int show_interrupts(struct seq_file *p, void *v) |
54 | { | 55 | { |
55 | int i = *(loff_t *) v, cpu; | 56 | int i = *(loff_t *) v, cpu; |
57 | struct irq_desc *desc; | ||
56 | struct irqaction * action; | 58 | struct irqaction * action; |
57 | unsigned long flags; | 59 | unsigned long flags; |
58 | 60 | ||
@@ -67,24 +69,25 @@ int show_interrupts(struct seq_file *p, void *v) | |||
67 | seq_putc(p, '\n'); | 69 | seq_putc(p, '\n'); |
68 | } | 70 | } |
69 | 71 | ||
70 | if (i < NR_IRQS) { | 72 | if (i < nr_irqs) { |
71 | raw_spin_lock_irqsave(&irq_desc[i].lock, flags); | 73 | desc = irq_to_desc(i); |
72 | action = irq_desc[i].action; | 74 | raw_spin_lock_irqsave(&desc->lock, flags); |
75 | action = desc->action; | ||
73 | if (!action) | 76 | if (!action) |
74 | goto unlock; | 77 | goto unlock; |
75 | 78 | ||
76 | seq_printf(p, "%3d: ", i); | 79 | seq_printf(p, "%3d: ", i); |
77 | for_each_present_cpu(cpu) | 80 | for_each_present_cpu(cpu) |
78 | seq_printf(p, "%10u ", kstat_irqs_cpu(i, cpu)); | 81 | seq_printf(p, "%10u ", kstat_irqs_cpu(i, cpu)); |
79 | seq_printf(p, " %10s", irq_desc[i].chip->name ? : "-"); | 82 | seq_printf(p, " %10s", desc->chip->name ? : "-"); |
80 | seq_printf(p, " %s", action->name); | 83 | seq_printf(p, " %s", action->name); |
81 | for (action = action->next; action; action = action->next) | 84 | for (action = action->next; action; action = action->next) |
82 | seq_printf(p, ", %s", action->name); | 85 | seq_printf(p, ", %s", action->name); |
83 | 86 | ||
84 | seq_putc(p, '\n'); | 87 | seq_putc(p, '\n'); |
85 | unlock: | 88 | unlock: |
86 | raw_spin_unlock_irqrestore(&irq_desc[i].lock, flags); | 89 | raw_spin_unlock_irqrestore(&desc->lock, flags); |
87 | } else if (i == NR_IRQS) { | 90 | } else if (i == nr_irqs) { |
88 | #ifdef CONFIG_FIQ | 91 | #ifdef CONFIG_FIQ |
89 | show_fiq_list(p, v); | 92 | show_fiq_list(p, v); |
90 | #endif | 93 | #endif |
@@ -112,7 +115,7 @@ asmlinkage void __exception asm_do_IRQ(unsigned int irq, struct pt_regs *regs) | |||
112 | * Some hardware gives randomly wrong interrupts. Rather | 115 | * Some hardware gives randomly wrong interrupts. Rather |
113 | * than crashing, do something sensible. | 116 | * than crashing, do something sensible. |
114 | */ | 117 | */ |
115 | if (unlikely(irq >= NR_IRQS)) { | 118 | if (unlikely(irq >= nr_irqs)) { |
116 | if (printk_ratelimit()) | 119 | if (printk_ratelimit()) |
117 | printk(KERN_WARNING "Bad IRQ%u\n", irq); | 120 | printk(KERN_WARNING "Bad IRQ%u\n", irq); |
118 | ack_bad_irq(irq); | 121 | ack_bad_irq(irq); |
@@ -132,12 +135,12 @@ void set_irq_flags(unsigned int irq, unsigned int iflags) | |||
132 | struct irq_desc *desc; | 135 | struct irq_desc *desc; |
133 | unsigned long flags; | 136 | unsigned long flags; |
134 | 137 | ||
135 | if (irq >= NR_IRQS) { | 138 | if (irq >= nr_irqs) { |
136 | printk(KERN_ERR "Trying to set irq flags for IRQ%d\n", irq); | 139 | printk(KERN_ERR "Trying to set irq flags for IRQ%d\n", irq); |
137 | return; | 140 | return; |
138 | } | 141 | } |
139 | 142 | ||
140 | desc = irq_desc + irq; | 143 | desc = irq_to_desc(irq); |
141 | raw_spin_lock_irqsave(&desc->lock, flags); | 144 | raw_spin_lock_irqsave(&desc->lock, flags); |
142 | desc->status |= IRQ_NOREQUEST | IRQ_NOPROBE | IRQ_NOAUTOEN; | 145 | desc->status |= IRQ_NOREQUEST | IRQ_NOPROBE | IRQ_NOAUTOEN; |
143 | if (iflags & IRQF_VALID) | 146 | if (iflags & IRQF_VALID) |
@@ -151,14 +154,25 @@ void set_irq_flags(unsigned int irq, unsigned int iflags) | |||
151 | 154 | ||
152 | void __init init_IRQ(void) | 155 | void __init init_IRQ(void) |
153 | { | 156 | { |
157 | struct irq_desc *desc; | ||
154 | int irq; | 158 | int irq; |
155 | 159 | ||
156 | for (irq = 0; irq < NR_IRQS; irq++) | 160 | for (irq = 0; irq < nr_irqs; irq++) { |
157 | irq_desc[irq].status |= IRQ_NOREQUEST | IRQ_NOPROBE; | 161 | desc = irq_to_desc_alloc_node(irq, 0); |
162 | desc->status |= IRQ_NOREQUEST | IRQ_NOPROBE; | ||
163 | } | ||
158 | 164 | ||
159 | init_arch_irq(); | 165 | init_arch_irq(); |
160 | } | 166 | } |
161 | 167 | ||
168 | #ifdef CONFIG_SPARSE_IRQ | ||
169 | int __init arch_probe_nr_irqs(void) | ||
170 | { | ||
171 | nr_irqs = arch_nr_irqs ? arch_nr_irqs : NR_IRQS; | ||
172 | return 0; | ||
173 | } | ||
174 | #endif | ||
175 | |||
162 | #ifdef CONFIG_HOTPLUG_CPU | 176 | #ifdef CONFIG_HOTPLUG_CPU |
163 | 177 | ||
164 | static void route_irq(struct irq_desc *desc, unsigned int irq, unsigned int cpu) | 178 | static void route_irq(struct irq_desc *desc, unsigned int irq, unsigned int cpu) |
@@ -178,10 +192,9 @@ static void route_irq(struct irq_desc *desc, unsigned int irq, unsigned int cpu) | |||
178 | void migrate_irqs(void) | 192 | void migrate_irqs(void) |
179 | { | 193 | { |
180 | unsigned int i, cpu = smp_processor_id(); | 194 | unsigned int i, cpu = smp_processor_id(); |
195 | struct irq_desc *desc; | ||
181 | 196 | ||
182 | for (i = 0; i < NR_IRQS; i++) { | 197 | for_each_irq_desc(i, desc) { |
183 | struct irq_desc *desc = irq_desc + i; | ||
184 | |||
185 | if (desc->node == cpu) { | 198 | if (desc->node == cpu) { |
186 | unsigned int newcpu = cpumask_any_and(desc->affinity, | 199 | unsigned int newcpu = cpumask_any_and(desc->affinity, |
187 | cpu_online_mask); | 200 | cpu_online_mask); |
diff --git a/arch/arm/kernel/ptrace.c b/arch/arm/kernel/ptrace.c index 3f562a7c0a9..f99d489822d 100644 --- a/arch/arm/kernel/ptrace.c +++ b/arch/arm/kernel/ptrace.c | |||
@@ -52,6 +52,102 @@ | |||
52 | #define BREAKINST_THUMB 0xde01 | 52 | #define BREAKINST_THUMB 0xde01 |
53 | #endif | 53 | #endif |
54 | 54 | ||
55 | struct pt_regs_offset { | ||
56 | const char *name; | ||
57 | int offset; | ||
58 | }; | ||
59 | |||
60 | #define REG_OFFSET_NAME(r) \ | ||
61 | {.name = #r, .offset = offsetof(struct pt_regs, ARM_##r)} | ||
62 | #define REG_OFFSET_END {.name = NULL, .offset = 0} | ||
63 | |||
64 | static const struct pt_regs_offset regoffset_table[] = { | ||
65 | REG_OFFSET_NAME(r0), | ||
66 | REG_OFFSET_NAME(r1), | ||
67 | REG_OFFSET_NAME(r2), | ||
68 | REG_OFFSET_NAME(r3), | ||
69 | REG_OFFSET_NAME(r4), | ||
70 | REG_OFFSET_NAME(r5), | ||
71 | REG_OFFSET_NAME(r6), | ||
72 | REG_OFFSET_NAME(r7), | ||
73 | REG_OFFSET_NAME(r8), | ||
74 | REG_OFFSET_NAME(r9), | ||
75 | REG_OFFSET_NAME(r10), | ||
76 | REG_OFFSET_NAME(fp), | ||
77 | REG_OFFSET_NAME(ip), | ||
78 | REG_OFFSET_NAME(sp), | ||
79 | REG_OFFSET_NAME(lr), | ||
80 | REG_OFFSET_NAME(pc), | ||
81 | REG_OFFSET_NAME(cpsr), | ||
82 | REG_OFFSET_NAME(ORIG_r0), | ||
83 | REG_OFFSET_END, | ||
84 | }; | ||
85 | |||
86 | /** | ||
87 | * regs_query_register_offset() - query register offset from its name | ||
88 | * @name: the name of a register | ||
89 | * | ||
90 | * regs_query_register_offset() returns the offset of a register in struct | ||
91 | * pt_regs from its name. If the name is invalid, this returns -EINVAL; | ||
92 | */ | ||
93 | int regs_query_register_offset(const char *name) | ||
94 | { | ||
95 | const struct pt_regs_offset *roff; | ||
96 | for (roff = regoffset_table; roff->name != NULL; roff++) | ||
97 | if (!strcmp(roff->name, name)) | ||
98 | return roff->offset; | ||
99 | return -EINVAL; | ||
100 | } | ||
101 | |||
102 | /** | ||
103 | * regs_query_register_name() - query register name from its offset | ||
104 | * @offset: the offset of a register in struct pt_regs. | ||
105 | * | ||
106 | * regs_query_register_name() returns the name of a register from its | ||
107 | * offset in struct pt_regs. If the @offset is invalid, this returns NULL; | ||
108 | */ | ||
109 | const char *regs_query_register_name(unsigned int offset) | ||
110 | { | ||
111 | const struct pt_regs_offset *roff; | ||
112 | for (roff = regoffset_table; roff->name != NULL; roff++) | ||
113 | if (roff->offset == offset) | ||
114 | return roff->name; | ||
115 | return NULL; | ||
116 | } | ||
117 | |||
118 | /** | ||
119 | * regs_within_kernel_stack() - check the address in the stack | ||
120 | * @regs: pt_regs which contains kernel stack pointer. | ||
121 | * @addr: address which is checked. | ||
122 | * | ||
123 | * regs_within_kernel_stack() checks @addr is within the kernel stack page(s). | ||
124 | * If @addr is within the kernel stack, it returns true. If not, returns false. | ||
125 | */ | ||
126 | bool regs_within_kernel_stack(struct pt_regs *regs, unsigned long addr) | ||
127 | { | ||
128 | return ((addr & ~(THREAD_SIZE - 1)) == | ||
129 | (kernel_stack_pointer(regs) & ~(THREAD_SIZE - 1))); | ||
130 | } | ||
131 | |||
132 | /** | ||
133 | * regs_get_kernel_stack_nth() - get Nth entry of the stack | ||
134 | * @regs: pt_regs which contains kernel stack pointer. | ||
135 | * @n: stack entry number. | ||
136 | * | ||
137 | * regs_get_kernel_stack_nth() returns @n th entry of the kernel stack which | ||
138 | * is specified by @regs. If the @n th entry is NOT in the kernel stack, | ||
139 | * this returns 0. | ||
140 | */ | ||
141 | unsigned long regs_get_kernel_stack_nth(struct pt_regs *regs, unsigned int n) | ||
142 | { | ||
143 | unsigned long *addr = (unsigned long *)kernel_stack_pointer(regs); | ||
144 | addr += n; | ||
145 | if (regs_within_kernel_stack(regs, (unsigned long)addr)) | ||
146 | return *addr; | ||
147 | else | ||
148 | return 0; | ||
149 | } | ||
150 | |||
55 | /* | 151 | /* |
56 | * this routine will get a word off of the processes privileged stack. | 152 | * this routine will get a word off of the processes privileged stack. |
57 | * the offset is how far from the base addr as stored in the THREAD. | 153 | * the offset is how far from the base addr as stored in the THREAD. |
diff --git a/arch/arm/kernel/setup.c b/arch/arm/kernel/setup.c index 122d999bdc7..648c3c1e16c 100644 --- a/arch/arm/kernel/setup.c +++ b/arch/arm/kernel/setup.c | |||
@@ -269,6 +269,21 @@ static void __init cacheid_init(void) | |||
269 | extern struct proc_info_list *lookup_processor_type(unsigned int); | 269 | extern struct proc_info_list *lookup_processor_type(unsigned int); |
270 | extern struct machine_desc *lookup_machine_type(unsigned int); | 270 | extern struct machine_desc *lookup_machine_type(unsigned int); |
271 | 271 | ||
272 | static void __init feat_v6_fixup(void) | ||
273 | { | ||
274 | int id = read_cpuid_id(); | ||
275 | |||
276 | if ((id & 0xff0f0000) != 0x41070000) | ||
277 | return; | ||
278 | |||
279 | /* | ||
280 | * HWCAP_TLS is available only on 1136 r1p0 and later, | ||
281 | * see also kuser_get_tls_init. | ||
282 | */ | ||
283 | if ((((id >> 4) & 0xfff) == 0xb36) && (((id >> 20) & 3) == 0)) | ||
284 | elf_hwcap &= ~HWCAP_TLS; | ||
285 | } | ||
286 | |||
272 | static void __init setup_processor(void) | 287 | static void __init setup_processor(void) |
273 | { | 288 | { |
274 | struct proc_info_list *list; | 289 | struct proc_info_list *list; |
@@ -311,6 +326,8 @@ static void __init setup_processor(void) | |||
311 | elf_hwcap &= ~HWCAP_THUMB; | 326 | elf_hwcap &= ~HWCAP_THUMB; |
312 | #endif | 327 | #endif |
313 | 328 | ||
329 | feat_v6_fixup(); | ||
330 | |||
314 | cacheid_init(); | 331 | cacheid_init(); |
315 | cpu_proc_init(); | 332 | cpu_proc_init(); |
316 | } | 333 | } |
@@ -729,6 +746,7 @@ void __init setup_arch(char **cmdline_p) | |||
729 | /* | 746 | /* |
730 | * Set up various architecture-specific pointers | 747 | * Set up various architecture-specific pointers |
731 | */ | 748 | */ |
749 | arch_nr_irqs = mdesc->nr_irqs; | ||
732 | init_arch_irq = mdesc->init_irq; | 750 | init_arch_irq = mdesc->init_irq; |
733 | system_timer = mdesc->timer; | 751 | system_timer = mdesc->timer; |
734 | init_machine = mdesc->init_machine; | 752 | init_machine = mdesc->init_machine; |
diff --git a/arch/arm/kernel/tcm.c b/arch/arm/kernel/tcm.c index e50303868f1..26685c2f7a4 100644 --- a/arch/arm/kernel/tcm.c +++ b/arch/arm/kernel/tcm.c | |||
@@ -13,38 +13,35 @@ | |||
13 | #include <linux/ioport.h> | 13 | #include <linux/ioport.h> |
14 | #include <linux/genalloc.h> | 14 | #include <linux/genalloc.h> |
15 | #include <linux/string.h> /* memcpy */ | 15 | #include <linux/string.h> /* memcpy */ |
16 | #include <asm/page.h> /* PAGE_SHIFT */ | ||
17 | #include <asm/cputype.h> | 16 | #include <asm/cputype.h> |
18 | #include <asm/mach/map.h> | 17 | #include <asm/mach/map.h> |
19 | #include <mach/memory.h> | 18 | #include <mach/memory.h> |
20 | #include "tcm.h" | 19 | #include "tcm.h" |
21 | 20 | ||
22 | /* Scream and warn about misuse */ | ||
23 | #if !defined(ITCM_OFFSET) || !defined(ITCM_END) || \ | ||
24 | !defined(DTCM_OFFSET) || !defined(DTCM_END) | ||
25 | #error "TCM support selected but offsets not defined!" | ||
26 | #endif | ||
27 | |||
28 | static struct gen_pool *tcm_pool; | 21 | static struct gen_pool *tcm_pool; |
29 | 22 | ||
30 | /* TCM section definitions from the linker */ | 23 | /* TCM section definitions from the linker */ |
31 | extern char __itcm_start, __sitcm_text, __eitcm_text; | 24 | extern char __itcm_start, __sitcm_text, __eitcm_text; |
32 | extern char __dtcm_start, __sdtcm_data, __edtcm_data; | 25 | extern char __dtcm_start, __sdtcm_data, __edtcm_data; |
33 | 26 | ||
27 | /* These will be increased as we run */ | ||
28 | u32 dtcm_end = DTCM_OFFSET; | ||
29 | u32 itcm_end = ITCM_OFFSET; | ||
30 | |||
34 | /* | 31 | /* |
35 | * TCM memory resources | 32 | * TCM memory resources |
36 | */ | 33 | */ |
37 | static struct resource dtcm_res = { | 34 | static struct resource dtcm_res = { |
38 | .name = "DTCM RAM", | 35 | .name = "DTCM RAM", |
39 | .start = DTCM_OFFSET, | 36 | .start = DTCM_OFFSET, |
40 | .end = DTCM_END, | 37 | .end = DTCM_OFFSET, |
41 | .flags = IORESOURCE_MEM | 38 | .flags = IORESOURCE_MEM |
42 | }; | 39 | }; |
43 | 40 | ||
44 | static struct resource itcm_res = { | 41 | static struct resource itcm_res = { |
45 | .name = "ITCM RAM", | 42 | .name = "ITCM RAM", |
46 | .start = ITCM_OFFSET, | 43 | .start = ITCM_OFFSET, |
47 | .end = ITCM_END, | 44 | .end = ITCM_OFFSET, |
48 | .flags = IORESOURCE_MEM | 45 | .flags = IORESOURCE_MEM |
49 | }; | 46 | }; |
50 | 47 | ||
@@ -52,8 +49,8 @@ static struct map_desc dtcm_iomap[] __initdata = { | |||
52 | { | 49 | { |
53 | .virtual = DTCM_OFFSET, | 50 | .virtual = DTCM_OFFSET, |
54 | .pfn = __phys_to_pfn(DTCM_OFFSET), | 51 | .pfn = __phys_to_pfn(DTCM_OFFSET), |
55 | .length = (DTCM_END - DTCM_OFFSET + 1), | 52 | .length = 0, |
56 | .type = MT_UNCACHED | 53 | .type = MT_MEMORY_DTCM |
57 | } | 54 | } |
58 | }; | 55 | }; |
59 | 56 | ||
@@ -61,8 +58,8 @@ static struct map_desc itcm_iomap[] __initdata = { | |||
61 | { | 58 | { |
62 | .virtual = ITCM_OFFSET, | 59 | .virtual = ITCM_OFFSET, |
63 | .pfn = __phys_to_pfn(ITCM_OFFSET), | 60 | .pfn = __phys_to_pfn(ITCM_OFFSET), |
64 | .length = (ITCM_END - ITCM_OFFSET + 1), | 61 | .length = 0, |
65 | .type = MT_UNCACHED | 62 | .type = MT_MEMORY_ITCM |
66 | } | 63 | } |
67 | }; | 64 | }; |
68 | 65 | ||
@@ -93,14 +90,24 @@ void tcm_free(void *addr, size_t len) | |||
93 | } | 90 | } |
94 | EXPORT_SYMBOL(tcm_free); | 91 | EXPORT_SYMBOL(tcm_free); |
95 | 92 | ||
96 | 93 | static int __init setup_tcm_bank(u8 type, u8 bank, u8 banks, | |
97 | static void __init setup_tcm_bank(u8 type, u32 offset, u32 expected_size) | 94 | u32 *offset) |
98 | { | 95 | { |
99 | const int tcm_sizes[16] = { 0, -1, -1, 4, 8, 16, 32, 64, 128, | 96 | const int tcm_sizes[16] = { 0, -1, -1, 4, 8, 16, 32, 64, 128, |
100 | 256, 512, 1024, -1, -1, -1, -1 }; | 97 | 256, 512, 1024, -1, -1, -1, -1 }; |
101 | u32 tcm_region; | 98 | u32 tcm_region; |
102 | int tcm_size; | 99 | int tcm_size; |
103 | 100 | ||
101 | /* | ||
102 | * If there are more than one TCM bank of this type, | ||
103 | * select the TCM bank to operate on in the TCM selection | ||
104 | * register. | ||
105 | */ | ||
106 | if (banks > 1) | ||
107 | asm("mcr p15, 0, %0, c9, c2, 0" | ||
108 | : /* No output operands */ | ||
109 | : "r" (bank)); | ||
110 | |||
104 | /* Read the special TCM region register c9, 0 */ | 111 | /* Read the special TCM region register c9, 0 */ |
105 | if (!type) | 112 | if (!type) |
106 | asm("mrc p15, 0, %0, c9, c1, 0" | 113 | asm("mrc p15, 0, %0, c9, c1, 0" |
@@ -111,26 +118,24 @@ static void __init setup_tcm_bank(u8 type, u32 offset, u32 expected_size) | |||
111 | 118 | ||
112 | tcm_size = tcm_sizes[(tcm_region >> 2) & 0x0f]; | 119 | tcm_size = tcm_sizes[(tcm_region >> 2) & 0x0f]; |
113 | if (tcm_size < 0) { | 120 | if (tcm_size < 0) { |
114 | pr_err("CPU: %sTCM of unknown size!\n", | 121 | pr_err("CPU: %sTCM%d of unknown size\n", |
115 | type ? "I" : "D"); | 122 | type ? "I" : "D", bank); |
123 | return -EINVAL; | ||
124 | } else if (tcm_size > 32) { | ||
125 | pr_err("CPU: %sTCM%d larger than 32k found\n", | ||
126 | type ? "I" : "D", bank); | ||
127 | return -EINVAL; | ||
116 | } else { | 128 | } else { |
117 | pr_info("CPU: found %sTCM %dk @ %08x, %senabled\n", | 129 | pr_info("CPU: found %sTCM%d %dk @ %08x, %senabled\n", |
118 | type ? "I" : "D", | 130 | type ? "I" : "D", |
131 | bank, | ||
119 | tcm_size, | 132 | tcm_size, |
120 | (tcm_region & 0xfffff000U), | 133 | (tcm_region & 0xfffff000U), |
121 | (tcm_region & 1) ? "" : "not "); | 134 | (tcm_region & 1) ? "" : "not "); |
122 | } | 135 | } |
123 | 136 | ||
124 | if (tcm_size != expected_size) { | ||
125 | pr_crit("CPU: %sTCM was detected %dk but expected %dk!\n", | ||
126 | type ? "I" : "D", | ||
127 | tcm_size, | ||
128 | expected_size); | ||
129 | /* Adjust to the expected size? what can we do... */ | ||
130 | } | ||
131 | |||
132 | /* Force move the TCM bank to where we want it, enable */ | 137 | /* Force move the TCM bank to where we want it, enable */ |
133 | tcm_region = offset | (tcm_region & 0x00000ffeU) | 1; | 138 | tcm_region = *offset | (tcm_region & 0x00000ffeU) | 1; |
134 | 139 | ||
135 | if (!type) | 140 | if (!type) |
136 | asm("mcr p15, 0, %0, c9, c1, 0" | 141 | asm("mcr p15, 0, %0, c9, c1, 0" |
@@ -141,10 +146,15 @@ static void __init setup_tcm_bank(u8 type, u32 offset, u32 expected_size) | |||
141 | : /* No output operands */ | 146 | : /* No output operands */ |
142 | : "r" (tcm_region)); | 147 | : "r" (tcm_region)); |
143 | 148 | ||
144 | pr_debug("CPU: moved %sTCM %dk to %08x, enabled\n", | 149 | /* Increase offset */ |
145 | type ? "I" : "D", | 150 | *offset += (tcm_size << 10); |
146 | tcm_size, | 151 | |
147 | (tcm_region & 0xfffff000U)); | 152 | pr_info("CPU: moved %sTCM%d %dk to %08x, enabled\n", |
153 | type ? "I" : "D", | ||
154 | bank, | ||
155 | tcm_size, | ||
156 | (tcm_region & 0xfffff000U)); | ||
157 | return 0; | ||
148 | } | 158 | } |
149 | 159 | ||
150 | /* | 160 | /* |
@@ -153,34 +163,52 @@ static void __init setup_tcm_bank(u8 type, u32 offset, u32 expected_size) | |||
153 | void __init tcm_init(void) | 163 | void __init tcm_init(void) |
154 | { | 164 | { |
155 | u32 tcm_status = read_cpuid_tcmstatus(); | 165 | u32 tcm_status = read_cpuid_tcmstatus(); |
166 | u8 dtcm_banks = (tcm_status >> 16) & 0x03; | ||
167 | u8 itcm_banks = (tcm_status & 0x03); | ||
156 | char *start; | 168 | char *start; |
157 | char *end; | 169 | char *end; |
158 | char *ram; | 170 | char *ram; |
171 | int ret; | ||
172 | int i; | ||
159 | 173 | ||
160 | /* Setup DTCM if present */ | 174 | /* Setup DTCM if present */ |
161 | if (tcm_status & (1 << 16)) { | 175 | if (dtcm_banks > 0) { |
162 | setup_tcm_bank(0, DTCM_OFFSET, | 176 | for (i = 0; i < dtcm_banks; i++) { |
163 | (DTCM_END - DTCM_OFFSET + 1) >> 10); | 177 | ret = setup_tcm_bank(0, i, dtcm_banks, &dtcm_end); |
178 | if (ret) | ||
179 | return; | ||
180 | } | ||
181 | dtcm_res.end = dtcm_end - 1; | ||
164 | request_resource(&iomem_resource, &dtcm_res); | 182 | request_resource(&iomem_resource, &dtcm_res); |
183 | dtcm_iomap[0].length = dtcm_end - DTCM_OFFSET; | ||
165 | iotable_init(dtcm_iomap, 1); | 184 | iotable_init(dtcm_iomap, 1); |
166 | /* Copy data from RAM to DTCM */ | 185 | /* Copy data from RAM to DTCM */ |
167 | start = &__sdtcm_data; | 186 | start = &__sdtcm_data; |
168 | end = &__edtcm_data; | 187 | end = &__edtcm_data; |
169 | ram = &__dtcm_start; | 188 | ram = &__dtcm_start; |
189 | /* This means you compiled more code than fits into DTCM */ | ||
190 | BUG_ON((end - start) > (dtcm_end - DTCM_OFFSET)); | ||
170 | memcpy(start, ram, (end-start)); | 191 | memcpy(start, ram, (end-start)); |
171 | pr_debug("CPU DTCM: copied data from %p - %p\n", start, end); | 192 | pr_debug("CPU DTCM: copied data from %p - %p\n", start, end); |
172 | } | 193 | } |
173 | 194 | ||
174 | /* Setup ITCM if present */ | 195 | /* Setup ITCM if present */ |
175 | if (tcm_status & 1) { | 196 | if (itcm_banks > 0) { |
176 | setup_tcm_bank(1, ITCM_OFFSET, | 197 | for (i = 0; i < itcm_banks; i++) { |
177 | (ITCM_END - ITCM_OFFSET + 1) >> 10); | 198 | ret = setup_tcm_bank(1, i, itcm_banks, &itcm_end); |
199 | if (ret) | ||
200 | return; | ||
201 | } | ||
202 | itcm_res.end = itcm_end - 1; | ||
178 | request_resource(&iomem_resource, &itcm_res); | 203 | request_resource(&iomem_resource, &itcm_res); |
204 | itcm_iomap[0].length = itcm_end - ITCM_OFFSET; | ||
179 | iotable_init(itcm_iomap, 1); | 205 | iotable_init(itcm_iomap, 1); |
180 | /* Copy code from RAM to ITCM */ | 206 | /* Copy code from RAM to ITCM */ |
181 | start = &__sitcm_text; | 207 | start = &__sitcm_text; |
182 | end = &__eitcm_text; | 208 | end = &__eitcm_text; |
183 | ram = &__itcm_start; | 209 | ram = &__itcm_start; |
210 | /* This means you compiled more code than fits into ITCM */ | ||
211 | BUG_ON((end - start) > (itcm_end - ITCM_OFFSET)); | ||
184 | memcpy(start, ram, (end-start)); | 212 | memcpy(start, ram, (end-start)); |
185 | pr_debug("CPU ITCM: copied code from %p - %p\n", start, end); | 213 | pr_debug("CPU ITCM: copied code from %p - %p\n", start, end); |
186 | } | 214 | } |
@@ -208,10 +236,10 @@ static int __init setup_tcm_pool(void) | |||
208 | pr_debug("Setting up TCM memory pool\n"); | 236 | pr_debug("Setting up TCM memory pool\n"); |
209 | 237 | ||
210 | /* Add the rest of DTCM to the TCM pool */ | 238 | /* Add the rest of DTCM to the TCM pool */ |
211 | if (tcm_status & (1 << 16)) { | 239 | if (tcm_status & (0x03 << 16)) { |
212 | if (dtcm_pool_start < DTCM_END) { | 240 | if (dtcm_pool_start < dtcm_end) { |
213 | ret = gen_pool_add(tcm_pool, dtcm_pool_start, | 241 | ret = gen_pool_add(tcm_pool, dtcm_pool_start, |
214 | DTCM_END - dtcm_pool_start + 1, -1); | 242 | dtcm_end - dtcm_pool_start, -1); |
215 | if (ret) { | 243 | if (ret) { |
216 | pr_err("CPU DTCM: could not add DTCM " \ | 244 | pr_err("CPU DTCM: could not add DTCM " \ |
217 | "remainder to pool!\n"); | 245 | "remainder to pool!\n"); |
@@ -219,16 +247,16 @@ static int __init setup_tcm_pool(void) | |||
219 | } | 247 | } |
220 | pr_debug("CPU DTCM: Added %08x bytes @ %08x to " \ | 248 | pr_debug("CPU DTCM: Added %08x bytes @ %08x to " \ |
221 | "the TCM memory pool\n", | 249 | "the TCM memory pool\n", |
222 | DTCM_END - dtcm_pool_start + 1, | 250 | dtcm_end - dtcm_pool_start, |
223 | dtcm_pool_start); | 251 | dtcm_pool_start); |
224 | } | 252 | } |
225 | } | 253 | } |
226 | 254 | ||
227 | /* Add the rest of ITCM to the TCM pool */ | 255 | /* Add the rest of ITCM to the TCM pool */ |
228 | if (tcm_status & 1) { | 256 | if (tcm_status & 0x03) { |
229 | if (itcm_pool_start < ITCM_END) { | 257 | if (itcm_pool_start < itcm_end) { |
230 | ret = gen_pool_add(tcm_pool, itcm_pool_start, | 258 | ret = gen_pool_add(tcm_pool, itcm_pool_start, |
231 | ITCM_END - itcm_pool_start + 1, -1); | 259 | itcm_end - itcm_pool_start, -1); |
232 | if (ret) { | 260 | if (ret) { |
233 | pr_err("CPU ITCM: could not add ITCM " \ | 261 | pr_err("CPU ITCM: could not add ITCM " \ |
234 | "remainder to pool!\n"); | 262 | "remainder to pool!\n"); |
@@ -236,7 +264,7 @@ static int __init setup_tcm_pool(void) | |||
236 | } | 264 | } |
237 | pr_debug("CPU ITCM: Added %08x bytes @ %08x to " \ | 265 | pr_debug("CPU ITCM: Added %08x bytes @ %08x to " \ |
238 | "the TCM memory pool\n", | 266 | "the TCM memory pool\n", |
239 | ITCM_END - itcm_pool_start + 1, | 267 | itcm_end - itcm_pool_start, |
240 | itcm_pool_start); | 268 | itcm_pool_start); |
241 | } | 269 | } |
242 | } | 270 | } |
diff --git a/arch/arm/kernel/traps.c b/arch/arm/kernel/traps.c index 1621e5327b2..cda78d59aa3 100644 --- a/arch/arm/kernel/traps.c +++ b/arch/arm/kernel/traps.c | |||
@@ -30,6 +30,7 @@ | |||
30 | #include <asm/unistd.h> | 30 | #include <asm/unistd.h> |
31 | #include <asm/traps.h> | 31 | #include <asm/traps.h> |
32 | #include <asm/unwind.h> | 32 | #include <asm/unwind.h> |
33 | #include <asm/tls.h> | ||
33 | 34 | ||
34 | #include "ptrace.h" | 35 | #include "ptrace.h" |
35 | #include "signal.h" | 36 | #include "signal.h" |
@@ -518,17 +519,20 @@ asmlinkage int arm_syscall(int no, struct pt_regs *regs) | |||
518 | 519 | ||
519 | case NR(set_tls): | 520 | case NR(set_tls): |
520 | thread->tp_value = regs->ARM_r0; | 521 | thread->tp_value = regs->ARM_r0; |
521 | #if defined(CONFIG_HAS_TLS_REG) | 522 | if (tls_emu) |
522 | asm ("mcr p15, 0, %0, c13, c0, 3" : : "r" (regs->ARM_r0) ); | 523 | return 0; |
523 | #elif !defined(CONFIG_TLS_REG_EMUL) | 524 | if (has_tls_reg) { |
524 | /* | 525 | asm ("mcr p15, 0, %0, c13, c0, 3" |
525 | * User space must never try to access this directly. | 526 | : : "r" (regs->ARM_r0)); |
526 | * Expect your app to break eventually if you do so. | 527 | } else { |
527 | * The user helper at 0xffff0fe0 must be used instead. | 528 | /* |
528 | * (see entry-armv.S for details) | 529 | * User space must never try to access this directly. |
529 | */ | 530 | * Expect your app to break eventually if you do so. |
530 | *((unsigned int *)0xffff0ff0) = regs->ARM_r0; | 531 | * The user helper at 0xffff0fe0 must be used instead. |
531 | #endif | 532 | * (see entry-armv.S for details) |
533 | */ | ||
534 | *((unsigned int *)0xffff0ff0) = regs->ARM_r0; | ||
535 | } | ||
532 | return 0; | 536 | return 0; |
533 | 537 | ||
534 | #ifdef CONFIG_NEEDS_SYSCALL_FOR_CMPXCHG | 538 | #ifdef CONFIG_NEEDS_SYSCALL_FOR_CMPXCHG |
@@ -743,6 +747,16 @@ void __init trap_init(void) | |||
743 | return; | 747 | return; |
744 | } | 748 | } |
745 | 749 | ||
750 | static void __init kuser_get_tls_init(unsigned long vectors) | ||
751 | { | ||
752 | /* | ||
753 | * vectors + 0xfe0 = __kuser_get_tls | ||
754 | * vectors + 0xfe8 = hardware TLS instruction at 0xffff0fe8 | ||
755 | */ | ||
756 | if (tls_emu || has_tls_reg) | ||
757 | memcpy((void *)vectors + 0xfe0, (void *)vectors + 0xfe8, 4); | ||
758 | } | ||
759 | |||
746 | void __init early_trap_init(void) | 760 | void __init early_trap_init(void) |
747 | { | 761 | { |
748 | unsigned long vectors = CONFIG_VECTORS_BASE; | 762 | unsigned long vectors = CONFIG_VECTORS_BASE; |
@@ -761,6 +775,11 @@ void __init early_trap_init(void) | |||
761 | memcpy((void *)vectors + 0x1000 - kuser_sz, __kuser_helper_start, kuser_sz); | 775 | memcpy((void *)vectors + 0x1000 - kuser_sz, __kuser_helper_start, kuser_sz); |
762 | 776 | ||
763 | /* | 777 | /* |
778 | * Do processor specific fixups for the kuser helpers | ||
779 | */ | ||
780 | kuser_get_tls_init(vectors); | ||
781 | |||
782 | /* | ||
764 | * Copy signal return handlers into the vector page, and | 783 | * Copy signal return handlers into the vector page, and |
765 | * set sigreturn to be a pointer to these. | 784 | * set sigreturn to be a pointer to these. |
766 | */ | 785 | */ |
diff --git a/arch/arm/mach-u300/include/mach/memory.h b/arch/arm/mach-u300/include/mach/memory.h index ab000df7fc0..bf134bcc129 100644 --- a/arch/arm/mach-u300/include/mach/memory.h +++ b/arch/arm/mach-u300/include/mach/memory.h | |||
@@ -35,14 +35,6 @@ | |||
35 | #endif | 35 | #endif |
36 | 36 | ||
37 | /* | 37 | /* |
38 | * TCM memory whereabouts | ||
39 | */ | ||
40 | #define ITCM_OFFSET 0xffff2000 | ||
41 | #define ITCM_END 0xffff3fff | ||
42 | #define DTCM_OFFSET 0xffff4000 | ||
43 | #define DTCM_END 0xffff5fff | ||
44 | |||
45 | /* | ||
46 | * We enable a real big DMA buffer if need be. | 38 | * We enable a real big DMA buffer if need be. |
47 | */ | 39 | */ |
48 | #define CONSISTENT_DMA_SIZE SZ_4M | 40 | #define CONSISTENT_DMA_SIZE SZ_4M |
diff --git a/arch/arm/mm/Kconfig b/arch/arm/mm/Kconfig index 101105e5261..87ec141fcaa 100644 --- a/arch/arm/mm/Kconfig +++ b/arch/arm/mm/Kconfig | |||
@@ -717,17 +717,6 @@ config TLS_REG_EMUL | |||
717 | a few prototypes like that in existence) and therefore access to | 717 | a few prototypes like that in existence) and therefore access to |
718 | that required register must be emulated. | 718 | that required register must be emulated. |
719 | 719 | ||
720 | config HAS_TLS_REG | ||
721 | bool | ||
722 | depends on !TLS_REG_EMUL | ||
723 | default y if SMP || CPU_32v7 | ||
724 | help | ||
725 | This selects support for the CP15 thread register. | ||
726 | It is defined to be available on some ARMv6 processors (including | ||
727 | all SMP capable ARMv6's) or later processors. User space may | ||
728 | assume directly accessing that register and always obtain the | ||
729 | expected value only on ARMv7 and above. | ||
730 | |||
731 | config NEEDS_SYSCALL_FOR_CMPXCHG | 720 | config NEEDS_SYSCALL_FOR_CMPXCHG |
732 | bool | 721 | bool |
733 | help | 722 | help |
diff --git a/arch/arm/mm/init.c b/arch/arm/mm/init.c index f6a99946532..e00404e6f45 100644 --- a/arch/arm/mm/init.c +++ b/arch/arm/mm/init.c | |||
@@ -529,6 +529,11 @@ void __init mem_init(void) | |||
529 | { | 529 | { |
530 | unsigned long reserved_pages, free_pages; | 530 | unsigned long reserved_pages, free_pages; |
531 | int i, node; | 531 | int i, node; |
532 | #ifdef CONFIG_HAVE_TCM | ||
533 | /* These pointers are filled in on TCM detection */ | ||
534 | extern u32 dtcm_end; | ||
535 | extern u32 itcm_end; | ||
536 | #endif | ||
532 | 537 | ||
533 | #ifndef CONFIG_DISCONTIGMEM | 538 | #ifndef CONFIG_DISCONTIGMEM |
534 | max_mapnr = pfn_to_page(max_pfn + PHYS_PFN_OFFSET) - mem_map; | 539 | max_mapnr = pfn_to_page(max_pfn + PHYS_PFN_OFFSET) - mem_map; |
@@ -611,6 +616,10 @@ void __init mem_init(void) | |||
611 | 616 | ||
612 | printk(KERN_NOTICE "Virtual kernel memory layout:\n" | 617 | printk(KERN_NOTICE "Virtual kernel memory layout:\n" |
613 | " vector : 0x%08lx - 0x%08lx (%4ld kB)\n" | 618 | " vector : 0x%08lx - 0x%08lx (%4ld kB)\n" |
619 | #ifdef CONFIG_HAVE_TCM | ||
620 | " DTCM : 0x%08lx - 0x%08lx (%4ld kB)\n" | ||
621 | " ITCM : 0x%08lx - 0x%08lx (%4ld kB)\n" | ||
622 | #endif | ||
614 | " fixmap : 0x%08lx - 0x%08lx (%4ld kB)\n" | 623 | " fixmap : 0x%08lx - 0x%08lx (%4ld kB)\n" |
615 | #ifdef CONFIG_MMU | 624 | #ifdef CONFIG_MMU |
616 | " DMA : 0x%08lx - 0x%08lx (%4ld MB)\n" | 625 | " DMA : 0x%08lx - 0x%08lx (%4ld MB)\n" |
@@ -627,6 +636,10 @@ void __init mem_init(void) | |||
627 | 636 | ||
628 | MLK(UL(CONFIG_VECTORS_BASE), UL(CONFIG_VECTORS_BASE) + | 637 | MLK(UL(CONFIG_VECTORS_BASE), UL(CONFIG_VECTORS_BASE) + |
629 | (PAGE_SIZE)), | 638 | (PAGE_SIZE)), |
639 | #ifdef CONFIG_HAVE_TCM | ||
640 | MLK(DTCM_OFFSET, (unsigned long) dtcm_end), | ||
641 | MLK(ITCM_OFFSET, (unsigned long) itcm_end), | ||
642 | #endif | ||
630 | MLK(FIXADDR_START, FIXADDR_TOP), | 643 | MLK(FIXADDR_START, FIXADDR_TOP), |
631 | #ifdef CONFIG_MMU | 644 | #ifdef CONFIG_MMU |
632 | MLM(CONSISTENT_BASE, CONSISTENT_END), | 645 | MLM(CONSISTENT_BASE, CONSISTENT_END), |
diff --git a/arch/arm/mm/ioremap.c b/arch/arm/mm/ioremap.c index 28c8b950ef0..03f11935ed0 100644 --- a/arch/arm/mm/ioremap.c +++ b/arch/arm/mm/ioremap.c | |||
@@ -268,6 +268,12 @@ void __iomem * __arm_ioremap_pfn_caller(unsigned long pfn, | |||
268 | if (pfn >= 0x100000 && (__pfn_to_phys(pfn) & ~SUPERSECTION_MASK)) | 268 | if (pfn >= 0x100000 && (__pfn_to_phys(pfn) & ~SUPERSECTION_MASK)) |
269 | return NULL; | 269 | return NULL; |
270 | 270 | ||
271 | /* | ||
272 | * Don't allow RAM to be mapped - this causes problems with ARMv6+ | ||
273 | */ | ||
274 | if (WARN_ON(pfn_valid(pfn))) | ||
275 | return NULL; | ||
276 | |||
271 | type = get_mem_type(mtype); | 277 | type = get_mem_type(mtype); |
272 | if (!type) | 278 | if (!type) |
273 | return NULL; | 279 | return NULL; |
diff --git a/arch/arm/mm/mmu.c b/arch/arm/mm/mmu.c index 28589417118..e53480148c0 100644 --- a/arch/arm/mm/mmu.c +++ b/arch/arm/mm/mmu.c | |||
@@ -258,6 +258,19 @@ static struct mem_type mem_types[] = { | |||
258 | .prot_sect = PMD_TYPE_SECT | PMD_SECT_AP_WRITE, | 258 | .prot_sect = PMD_TYPE_SECT | PMD_SECT_AP_WRITE, |
259 | .domain = DOMAIN_KERNEL, | 259 | .domain = DOMAIN_KERNEL, |
260 | }, | 260 | }, |
261 | [MT_MEMORY_DTCM] = { | ||
262 | .prot_pte = L_PTE_PRESENT | L_PTE_YOUNG | | ||
263 | L_PTE_DIRTY | L_PTE_WRITE, | ||
264 | .prot_l1 = PMD_TYPE_TABLE, | ||
265 | .prot_sect = PMD_TYPE_SECT | PMD_SECT_XN, | ||
266 | .domain = DOMAIN_KERNEL, | ||
267 | }, | ||
268 | [MT_MEMORY_ITCM] = { | ||
269 | .prot_pte = L_PTE_PRESENT | L_PTE_YOUNG | L_PTE_DIRTY | | ||
270 | L_PTE_USER | L_PTE_EXEC, | ||
271 | .prot_l1 = PMD_TYPE_TABLE, | ||
272 | .domain = DOMAIN_IO, | ||
273 | }, | ||
261 | }; | 274 | }; |
262 | 275 | ||
263 | const struct mem_type *get_mem_type(unsigned int type) | 276 | const struct mem_type *get_mem_type(unsigned int type) |
diff --git a/arch/arm/mm/proc-v6.S b/arch/arm/mm/proc-v6.S index 7a5337ed7d6..2f5a3c23a0f 100644 --- a/arch/arm/mm/proc-v6.S +++ b/arch/arm/mm/proc-v6.S | |||
@@ -239,7 +239,8 @@ __v6_proc_info: | |||
239 | b __v6_setup | 239 | b __v6_setup |
240 | .long cpu_arch_name | 240 | .long cpu_arch_name |
241 | .long cpu_elf_name | 241 | .long cpu_elf_name |
242 | .long HWCAP_SWP|HWCAP_HALF|HWCAP_THUMB|HWCAP_FAST_MULT|HWCAP_EDSP|HWCAP_JAVA | 242 | /* See also feat_v6_fixup() for HWCAP_TLS */ |
243 | .long HWCAP_SWP|HWCAP_HALF|HWCAP_THUMB|HWCAP_FAST_MULT|HWCAP_EDSP|HWCAP_JAVA|HWCAP_TLS | ||
243 | .long cpu_v6_name | 244 | .long cpu_v6_name |
244 | .long v6_processor_functions | 245 | .long v6_processor_functions |
245 | .long v6wbi_tlb_fns | 246 | .long v6wbi_tlb_fns |
@@ -262,7 +263,7 @@ __pj4_v6_proc_info: | |||
262 | b __v6_setup | 263 | b __v6_setup |
263 | .long cpu_arch_name | 264 | .long cpu_arch_name |
264 | .long cpu_elf_name | 265 | .long cpu_elf_name |
265 | .long HWCAP_SWP|HWCAP_HALF|HWCAP_THUMB|HWCAP_FAST_MULT|HWCAP_EDSP | 266 | .long HWCAP_SWP|HWCAP_HALF|HWCAP_THUMB|HWCAP_FAST_MULT|HWCAP_EDSP|HWCAP_TLS |
266 | .long cpu_pj4_name | 267 | .long cpu_pj4_name |
267 | .long v6_processor_functions | 268 | .long v6_processor_functions |
268 | .long v6wbi_tlb_fns | 269 | .long v6wbi_tlb_fns |
diff --git a/arch/arm/mm/proc-v7.S b/arch/arm/mm/proc-v7.S index 7aaf88a3b7a..8071bcd4c99 100644 --- a/arch/arm/mm/proc-v7.S +++ b/arch/arm/mm/proc-v7.S | |||
@@ -344,7 +344,7 @@ __v7_proc_info: | |||
344 | b __v7_setup | 344 | b __v7_setup |
345 | .long cpu_arch_name | 345 | .long cpu_arch_name |
346 | .long cpu_elf_name | 346 | .long cpu_elf_name |
347 | .long HWCAP_SWP|HWCAP_HALF|HWCAP_THUMB|HWCAP_FAST_MULT|HWCAP_EDSP | 347 | .long HWCAP_SWP|HWCAP_HALF|HWCAP_THUMB|HWCAP_FAST_MULT|HWCAP_EDSP|HWCAP_TLS |
348 | .long cpu_v7_name | 348 | .long cpu_v7_name |
349 | .long v7_processor_functions | 349 | .long v7_processor_functions |
350 | .long v7wbi_tlb_fns | 350 | .long v7wbi_tlb_fns |
diff --git a/arch/arm/vfp/vfpmodule.c b/arch/arm/vfp/vfpmodule.c index 315a540c7ce..8063a322c79 100644 --- a/arch/arm/vfp/vfpmodule.c +++ b/arch/arm/vfp/vfpmodule.c | |||
@@ -15,6 +15,7 @@ | |||
15 | #include <linux/sched.h> | 15 | #include <linux/sched.h> |
16 | #include <linux/init.h> | 16 | #include <linux/init.h> |
17 | 17 | ||
18 | #include <asm/cputype.h> | ||
18 | #include <asm/thread_notify.h> | 19 | #include <asm/thread_notify.h> |
19 | #include <asm/vfp.h> | 20 | #include <asm/vfp.h> |
20 | 21 | ||
@@ -549,10 +550,13 @@ static int __init vfp_init(void) | |||
549 | /* | 550 | /* |
550 | * Check for the presence of the Advanced SIMD | 551 | * Check for the presence of the Advanced SIMD |
551 | * load/store instructions, integer and single | 552 | * load/store instructions, integer and single |
552 | * precision floating point operations. | 553 | * precision floating point operations. Only check |
554 | * for NEON if the hardware has the MVFR registers. | ||
553 | */ | 555 | */ |
554 | if ((fmrx(MVFR1) & 0x000fff00) == 0x00011100) | 556 | if ((read_cpuid_id() & 0x000f0000) == 0x000f0000) { |
555 | elf_hwcap |= HWCAP_NEON; | 557 | if ((fmrx(MVFR1) & 0x000fff00) == 0x00011100) |
558 | elf_hwcap |= HWCAP_NEON; | ||
559 | } | ||
556 | #endif | 560 | #endif |
557 | } | 561 | } |
558 | return 0; | 562 | return 0; |