diff options
author | Paul Mundt <lethal@linux-sh.org> | 2009-08-24 09:49:17 -0400 |
---|---|---|
committer | Paul Mundt <lethal@linux-sh.org> | 2009-08-24 09:49:17 -0400 |
commit | 12cceb6251c2cd23e936b25eca66be99ba41b081 (patch) | |
tree | b7f62853e67b305519c375162760422fbfc81b8e /arch/sh/kernel | |
parent | f13327864f94c3a0e6acca923df537d20059639f (diff) | |
parent | 05ecd5a1f76c183cca381705b3adb7d77c9a0439 (diff) |
Merge branch 'sh/st-integration'
Diffstat (limited to 'arch/sh/kernel')
-rw-r--r-- | arch/sh/kernel/cpu/irq/ipr.c | 1 | ||||
-rw-r--r-- | arch/sh/kernel/cpu/sh3/entry.S | 2 | ||||
-rw-r--r-- | arch/sh/kernel/entry-common.S | 5 | ||||
-rw-r--r-- | arch/sh/kernel/io.c | 97 | ||||
-rw-r--r-- | arch/sh/kernel/io_generic.c | 50 | ||||
-rw-r--r-- | arch/sh/kernel/irq.c | 2 | ||||
-rw-r--r-- | arch/sh/kernel/kgdb.c | 2 | ||||
-rw-r--r-- | arch/sh/kernel/process_32.c | 20 | ||||
-rw-r--r-- | arch/sh/kernel/setup.c | 6 | ||||
-rw-r--r-- | arch/sh/kernel/signal_32.c | 12 | ||||
-rw-r--r-- | arch/sh/kernel/sys_sh.c | 43 | ||||
-rw-r--r-- | arch/sh/kernel/syscalls_32.S | 2 | ||||
-rw-r--r-- | arch/sh/kernel/syscalls_64.S | 2 | ||||
-rw-r--r-- | arch/sh/kernel/traps_32.c | 188 |
14 files changed, 337 insertions, 95 deletions
diff --git a/arch/sh/kernel/cpu/irq/ipr.c b/arch/sh/kernel/cpu/irq/ipr.c index 808d99a48ef..c1508a90fc6 100644 --- a/arch/sh/kernel/cpu/irq/ipr.c +++ b/arch/sh/kernel/cpu/irq/ipr.c | |||
@@ -35,6 +35,7 @@ static void disable_ipr_irq(unsigned int irq) | |||
35 | unsigned long addr = get_ipr_desc(irq)->ipr_offsets[p->ipr_idx]; | 35 | unsigned long addr = get_ipr_desc(irq)->ipr_offsets[p->ipr_idx]; |
36 | /* Set the priority in IPR to 0 */ | 36 | /* Set the priority in IPR to 0 */ |
37 | __raw_writew(__raw_readw(addr) & (0xffff ^ (0xf << p->shift)), addr); | 37 | __raw_writew(__raw_readw(addr) & (0xffff ^ (0xf << p->shift)), addr); |
38 | (void)__raw_readw(addr); /* Read back to flush write posting */ | ||
38 | } | 39 | } |
39 | 40 | ||
40 | static void enable_ipr_irq(unsigned int irq) | 41 | static void enable_ipr_irq(unsigned int irq) |
diff --git a/arch/sh/kernel/cpu/sh3/entry.S b/arch/sh/kernel/cpu/sh3/entry.S index 8c19e21847d..9421ec715fd 100644 --- a/arch/sh/kernel/cpu/sh3/entry.S +++ b/arch/sh/kernel/cpu/sh3/entry.S | |||
@@ -257,7 +257,7 @@ restore_all: | |||
257 | ! | 257 | ! |
258 | ! Calculate new SR value | 258 | ! Calculate new SR value |
259 | mov k3, k2 ! original SR value | 259 | mov k3, k2 ! original SR value |
260 | mov #0xf0, k1 | 260 | mov #0xfffffff0, k1 |
261 | extu.b k1, k1 | 261 | extu.b k1, k1 |
262 | not k1, k1 | 262 | not k1, k1 |
263 | and k1, k2 ! Mask original SR value | 263 | and k1, k2 ! Mask original SR value |
diff --git a/arch/sh/kernel/entry-common.S b/arch/sh/kernel/entry-common.S index 700477601c6..68d9223b145 100644 --- a/arch/sh/kernel/entry-common.S +++ b/arch/sh/kernel/entry-common.S | |||
@@ -98,8 +98,9 @@ need_resched: | |||
98 | 98 | ||
99 | mov #OFF_SR, r0 | 99 | mov #OFF_SR, r0 |
100 | mov.l @(r0,r15), r0 ! get status register | 100 | mov.l @(r0,r15), r0 ! get status register |
101 | and #0xf0, r0 ! interrupts off (exception path)? | 101 | shlr r0 |
102 | cmp/eq #0xf0, r0 | 102 | and #(0xf0>>1), r0 ! interrupts off (exception path)? |
103 | cmp/eq #(0xf0>>1), r0 | ||
103 | bt noresched | 104 | bt noresched |
104 | mov.l 3f, r0 | 105 | mov.l 3f, r0 |
105 | jsr @r0 ! call preempt_schedule_irq | 106 | jsr @r0 ! call preempt_schedule_irq |
diff --git a/arch/sh/kernel/io.c b/arch/sh/kernel/io.c index 4f85fffaa55..4770c241c67 100644 --- a/arch/sh/kernel/io.c +++ b/arch/sh/kernel/io.c | |||
@@ -1,12 +1,9 @@ | |||
1 | /* | 1 | /* |
2 | * linux/arch/sh/kernel/io.c | 2 | * arch/sh/kernel/io.c - Machine independent I/O functions. |
3 | * | 3 | * |
4 | * Copyright (C) 2000 Stuart Menefy | 4 | * Copyright (C) 2000 - 2009 Stuart Menefy |
5 | * Copyright (C) 2005 Paul Mundt | 5 | * Copyright (C) 2005 Paul Mundt |
6 | * | 6 | * |
7 | * Provide real functions which expand to whatever the header file defined. | ||
8 | * Also definitions of machine independent IO functions. | ||
9 | * | ||
10 | * This file is subject to the terms and conditions of the GNU General Public | 7 | * This file is subject to the terms and conditions of the GNU General Public |
11 | * License. See the file "COPYING" in the main directory of this archive | 8 | * License. See the file "COPYING" in the main directory of this archive |
12 | * for more details. | 9 | * for more details. |
@@ -18,33 +15,87 @@ | |||
18 | 15 | ||
19 | /* | 16 | /* |
20 | * Copy data from IO memory space to "real" memory space. | 17 | * Copy data from IO memory space to "real" memory space. |
21 | * This needs to be optimized. | ||
22 | */ | 18 | */ |
23 | void memcpy_fromio(void *to, const volatile void __iomem *from, unsigned long count) | 19 | void memcpy_fromio(void *to, const volatile void __iomem *from, unsigned long count) |
24 | { | 20 | { |
25 | unsigned char *p = to; | 21 | /* |
26 | while (count) { | 22 | * Would it be worthwhile doing byte and long transfers first |
27 | count--; | 23 | * to try and get aligned? |
28 | *p = readb(from); | 24 | */ |
29 | p++; | 25 | #ifdef CONFIG_CPU_SH4 |
30 | from++; | 26 | if ((count >= 0x20) && |
31 | } | 27 | (((u32)to & 0x1f) == 0) && (((u32)from & 0x3) == 0)) { |
28 | int tmp2, tmp3, tmp4, tmp5, tmp6; | ||
29 | |||
30 | __asm__ __volatile__( | ||
31 | "1: \n\t" | ||
32 | "mov.l @%7+, r0 \n\t" | ||
33 | "mov.l @%7+, %2 \n\t" | ||
34 | "movca.l r0, @%0 \n\t" | ||
35 | "mov.l @%7+, %3 \n\t" | ||
36 | "mov.l @%7+, %4 \n\t" | ||
37 | "mov.l @%7+, %5 \n\t" | ||
38 | "mov.l @%7+, %6 \n\t" | ||
39 | "mov.l @%7+, r7 \n\t" | ||
40 | "mov.l @%7+, r0 \n\t" | ||
41 | "mov.l %2, @(0x04,%0) \n\t" | ||
42 | "mov #0x20, %2 \n\t" | ||
43 | "mov.l %3, @(0x08,%0) \n\t" | ||
44 | "sub %2, %1 \n\t" | ||
45 | "mov.l %4, @(0x0c,%0) \n\t" | ||
46 | "cmp/hi %1, %2 ! T if 32 > count \n\t" | ||
47 | "mov.l %5, @(0x10,%0) \n\t" | ||
48 | "mov.l %6, @(0x14,%0) \n\t" | ||
49 | "mov.l r7, @(0x18,%0) \n\t" | ||
50 | "mov.l r0, @(0x1c,%0) \n\t" | ||
51 | "bf.s 1b \n\t" | ||
52 | " add #0x20, %0 \n\t" | ||
53 | : "=&r" (to), "=&r" (count), | ||
54 | "=&r" (tmp2), "=&r" (tmp3), "=&r" (tmp4), | ||
55 | "=&r" (tmp5), "=&r" (tmp6), "=&r" (from) | ||
56 | : "7"(from), "0" (to), "1" (count) | ||
57 | : "r0", "r7", "t", "memory"); | ||
58 | } | ||
59 | #endif | ||
60 | |||
61 | if ((((u32)to | (u32)from) & 0x3) == 0) { | ||
62 | for (; count > 3; count -= 4) { | ||
63 | *(u32 *)to = *(volatile u32 *)from; | ||
64 | to += 4; | ||
65 | from += 4; | ||
66 | } | ||
67 | } | ||
68 | |||
69 | for (; count > 0; count--) { | ||
70 | *(u8 *)to = *(volatile u8 *)from; | ||
71 | to++; | ||
72 | from++; | ||
73 | } | ||
74 | |||
75 | mb(); | ||
32 | } | 76 | } |
33 | EXPORT_SYMBOL(memcpy_fromio); | 77 | EXPORT_SYMBOL(memcpy_fromio); |
34 | 78 | ||
35 | /* | 79 | /* |
36 | * Copy data from "real" memory space to IO memory space. | 80 | * Copy data from "real" memory space to IO memory space. |
37 | * This needs to be optimized. | ||
38 | */ | 81 | */ |
39 | void memcpy_toio(volatile void __iomem *to, const void *from, unsigned long count) | 82 | void memcpy_toio(volatile void __iomem *to, const void *from, unsigned long count) |
40 | { | 83 | { |
41 | const unsigned char *p = from; | 84 | if ((((u32)to | (u32)from) & 0x3) == 0) { |
42 | while (count) { | 85 | for ( ; count > 3; count -= 4) { |
43 | count--; | 86 | *(volatile u32 *)to = *(u32 *)from; |
44 | writeb(*p, to); | 87 | to += 4; |
45 | p++; | 88 | from += 4; |
46 | to++; | 89 | } |
47 | } | 90 | } |
91 | |||
92 | for (; count > 0; count--) { | ||
93 | *(volatile u8 *)to = *(u8 *)from; | ||
94 | to++; | ||
95 | from++; | ||
96 | } | ||
97 | |||
98 | mb(); | ||
48 | } | 99 | } |
49 | EXPORT_SYMBOL(memcpy_toio); | 100 | EXPORT_SYMBOL(memcpy_toio); |
50 | 101 | ||
@@ -62,6 +113,8 @@ void memset_io(volatile void __iomem *dst, int c, unsigned long count) | |||
62 | } | 113 | } |
63 | EXPORT_SYMBOL(memset_io); | 114 | EXPORT_SYMBOL(memset_io); |
64 | 115 | ||
116 | #ifndef CONFIG_GENERIC_IOMAP | ||
117 | |||
65 | void __iomem *ioport_map(unsigned long port, unsigned int nr) | 118 | void __iomem *ioport_map(unsigned long port, unsigned int nr) |
66 | { | 119 | { |
67 | void __iomem *ret; | 120 | void __iomem *ret; |
@@ -79,3 +132,5 @@ void ioport_unmap(void __iomem *addr) | |||
79 | sh_mv.mv_ioport_unmap(addr); | 132 | sh_mv.mv_ioport_unmap(addr); |
80 | } | 133 | } |
81 | EXPORT_SYMBOL(ioport_unmap); | 134 | EXPORT_SYMBOL(ioport_unmap); |
135 | |||
136 | #endif /* CONFIG_GENERIC_IOMAP */ | ||
diff --git a/arch/sh/kernel/io_generic.c b/arch/sh/kernel/io_generic.c index 5a7f554d9ca..4ff50723928 100644 --- a/arch/sh/kernel/io_generic.c +++ b/arch/sh/kernel/io_generic.c | |||
@@ -73,35 +73,19 @@ u32 generic_inl_p(unsigned long port) | |||
73 | 73 | ||
74 | void generic_insb(unsigned long port, void *dst, unsigned long count) | 74 | void generic_insb(unsigned long port, void *dst, unsigned long count) |
75 | { | 75 | { |
76 | volatile u8 *port_addr; | 76 | __raw_readsb(__ioport_map(port, 1), dst, count); |
77 | u8 *buf = dst; | 77 | dummy_read(); |
78 | |||
79 | port_addr = (volatile u8 __force *)__ioport_map(port, 1); | ||
80 | while (count--) | ||
81 | *buf++ = *port_addr; | ||
82 | } | 78 | } |
83 | 79 | ||
84 | void generic_insw(unsigned long port, void *dst, unsigned long count) | 80 | void generic_insw(unsigned long port, void *dst, unsigned long count) |
85 | { | 81 | { |
86 | volatile u16 *port_addr; | 82 | __raw_readsw(__ioport_map(port, 2), dst, count); |
87 | u16 *buf = dst; | ||
88 | |||
89 | port_addr = (volatile u16 __force *)__ioport_map(port, 2); | ||
90 | while (count--) | ||
91 | *buf++ = *port_addr; | ||
92 | |||
93 | dummy_read(); | 83 | dummy_read(); |
94 | } | 84 | } |
95 | 85 | ||
96 | void generic_insl(unsigned long port, void *dst, unsigned long count) | 86 | void generic_insl(unsigned long port, void *dst, unsigned long count) |
97 | { | 87 | { |
98 | volatile u32 *port_addr; | 88 | __raw_readsl(__ioport_map(port, 4), dst, count); |
99 | u32 *buf = dst; | ||
100 | |||
101 | port_addr = (volatile u32 __force *)__ioport_map(port, 4); | ||
102 | while (count--) | ||
103 | *buf++ = *port_addr; | ||
104 | |||
105 | dummy_read(); | 89 | dummy_read(); |
106 | } | 90 | } |
107 | 91 | ||
@@ -145,37 +129,19 @@ void generic_outl_p(u32 b, unsigned long port) | |||
145 | */ | 129 | */ |
146 | void generic_outsb(unsigned long port, const void *src, unsigned long count) | 130 | void generic_outsb(unsigned long port, const void *src, unsigned long count) |
147 | { | 131 | { |
148 | volatile u8 *port_addr; | 132 | __raw_writesb(__ioport_map(port, 1), src, count); |
149 | const u8 *buf = src; | 133 | dummy_read(); |
150 | |||
151 | port_addr = (volatile u8 __force *)__ioport_map(port, 1); | ||
152 | |||
153 | while (count--) | ||
154 | *port_addr = *buf++; | ||
155 | } | 134 | } |
156 | 135 | ||
157 | void generic_outsw(unsigned long port, const void *src, unsigned long count) | 136 | void generic_outsw(unsigned long port, const void *src, unsigned long count) |
158 | { | 137 | { |
159 | volatile u16 *port_addr; | 138 | __raw_writesw(__ioport_map(port, 2), src, count); |
160 | const u16 *buf = src; | ||
161 | |||
162 | port_addr = (volatile u16 __force *)__ioport_map(port, 2); | ||
163 | |||
164 | while (count--) | ||
165 | *port_addr = *buf++; | ||
166 | |||
167 | dummy_read(); | 139 | dummy_read(); |
168 | } | 140 | } |
169 | 141 | ||
170 | void generic_outsl(unsigned long port, const void *src, unsigned long count) | 142 | void generic_outsl(unsigned long port, const void *src, unsigned long count) |
171 | { | 143 | { |
172 | volatile u32 *port_addr; | 144 | __raw_writesl(__ioport_map(port, 4), src, count); |
173 | const u32 *buf = src; | ||
174 | |||
175 | port_addr = (volatile u32 __force *)__ioport_map(port, 4); | ||
176 | while (count--) | ||
177 | *port_addr = *buf++; | ||
178 | |||
179 | dummy_read(); | 145 | dummy_read(); |
180 | } | 146 | } |
181 | 147 | ||
diff --git a/arch/sh/kernel/irq.c b/arch/sh/kernel/irq.c index 278c68c6048..d1053392e28 100644 --- a/arch/sh/kernel/irq.c +++ b/arch/sh/kernel/irq.c | |||
@@ -114,7 +114,7 @@ asmlinkage int do_IRQ(unsigned int irq, struct pt_regs *regs) | |||
114 | #endif | 114 | #endif |
115 | 115 | ||
116 | irq_enter(); | 116 | irq_enter(); |
117 | irq = irq_demux(intc_evt2irq(irq)); | 117 | irq = irq_demux(evt2irq(irq)); |
118 | 118 | ||
119 | #ifdef CONFIG_IRQSTACKS | 119 | #ifdef CONFIG_IRQSTACKS |
120 | curctx = (union irq_ctx *)current_thread_info(); | 120 | curctx = (union irq_ctx *)current_thread_info(); |
diff --git a/arch/sh/kernel/kgdb.c b/arch/sh/kernel/kgdb.c index 305aad742ae..d29de7864f3 100644 --- a/arch/sh/kernel/kgdb.c +++ b/arch/sh/kernel/kgdb.c | |||
@@ -195,8 +195,6 @@ void gdb_regs_to_pt_regs(unsigned long *gdb_regs, struct pt_regs *regs) | |||
195 | regs->gbr = gdb_regs[GDB_GBR]; | 195 | regs->gbr = gdb_regs[GDB_GBR]; |
196 | regs->mach = gdb_regs[GDB_MACH]; | 196 | regs->mach = gdb_regs[GDB_MACH]; |
197 | regs->macl = gdb_regs[GDB_MACL]; | 197 | regs->macl = gdb_regs[GDB_MACL]; |
198 | |||
199 | __asm__ __volatile__ ("ldc %0, vbr" : : "r" (gdb_regs[GDB_VBR])); | ||
200 | } | 198 | } |
201 | 199 | ||
202 | void sleeping_thread_to_gdb_regs(unsigned long *gdb_regs, struct task_struct *p) | 200 | void sleeping_thread_to_gdb_regs(unsigned long *gdb_regs, struct task_struct *p) |
diff --git a/arch/sh/kernel/process_32.c b/arch/sh/kernel/process_32.c index 9fee977f176..0673c4746be 100644 --- a/arch/sh/kernel/process_32.c +++ b/arch/sh/kernel/process_32.c | |||
@@ -32,15 +32,35 @@ | |||
32 | #include <asm/ubc.h> | 32 | #include <asm/ubc.h> |
33 | #include <asm/fpu.h> | 33 | #include <asm/fpu.h> |
34 | #include <asm/syscalls.h> | 34 | #include <asm/syscalls.h> |
35 | #include <asm/watchdog.h> | ||
35 | 36 | ||
36 | int ubc_usercnt = 0; | 37 | int ubc_usercnt = 0; |
37 | 38 | ||
39 | #ifdef CONFIG_32BIT | ||
40 | static void watchdog_trigger_immediate(void) | ||
41 | { | ||
42 | sh_wdt_write_cnt(0xFF); | ||
43 | sh_wdt_write_csr(0xC2); | ||
44 | } | ||
45 | |||
46 | void machine_restart(char * __unused) | ||
47 | { | ||
48 | local_irq_disable(); | ||
49 | |||
50 | /* Use watchdog timer to trigger reset */ | ||
51 | watchdog_trigger_immediate(); | ||
52 | |||
53 | while (1) | ||
54 | cpu_sleep(); | ||
55 | } | ||
56 | #else | ||
38 | void machine_restart(char * __unused) | 57 | void machine_restart(char * __unused) |
39 | { | 58 | { |
40 | /* SR.BL=1 and invoke address error to let CPU reset (manual reset) */ | 59 | /* SR.BL=1 and invoke address error to let CPU reset (manual reset) */ |
41 | asm volatile("ldc %0, sr\n\t" | 60 | asm volatile("ldc %0, sr\n\t" |
42 | "mov.l @%1, %0" : : "r" (0x10000000), "r" (0x80000001)); | 61 | "mov.l @%1, %0" : : "r" (0x10000000), "r" (0x80000001)); |
43 | } | 62 | } |
63 | #endif | ||
44 | 64 | ||
45 | void machine_halt(void) | 65 | void machine_halt(void) |
46 | { | 66 | { |
diff --git a/arch/sh/kernel/setup.c b/arch/sh/kernel/setup.c index 212e6bddaeb..d13bbafb4e1 100644 --- a/arch/sh/kernel/setup.c +++ b/arch/sh/kernel/setup.c | |||
@@ -404,10 +404,14 @@ void __init setup_arch(char **cmdline_p) | |||
404 | if (!memory_end) | 404 | if (!memory_end) |
405 | memory_end = memory_start + __MEMORY_SIZE; | 405 | memory_end = memory_start + __MEMORY_SIZE; |
406 | 406 | ||
407 | #ifdef CONFIG_CMDLINE_BOOL | 407 | #ifdef CONFIG_CMDLINE_OVERWRITE |
408 | strlcpy(command_line, CONFIG_CMDLINE, sizeof(command_line)); | 408 | strlcpy(command_line, CONFIG_CMDLINE, sizeof(command_line)); |
409 | #else | 409 | #else |
410 | strlcpy(command_line, COMMAND_LINE, sizeof(command_line)); | 410 | strlcpy(command_line, COMMAND_LINE, sizeof(command_line)); |
411 | #ifdef CONFIG_CMDLINE_EXTEND | ||
412 | strlcat(command_line, " ", sizeof(command_line)); | ||
413 | strlcat(command_line, CONFIG_CMDLINE, sizeof(command_line)); | ||
414 | #endif | ||
411 | #endif | 415 | #endif |
412 | 416 | ||
413 | /* Save unparsed command line copy for /proc/cmdline */ | 417 | /* Save unparsed command line copy for /proc/cmdline */ |
diff --git a/arch/sh/kernel/signal_32.c b/arch/sh/kernel/signal_32.c index b5afbec1db5..6010750c90b 100644 --- a/arch/sh/kernel/signal_32.c +++ b/arch/sh/kernel/signal_32.c | |||
@@ -41,6 +41,16 @@ struct fdpic_func_descriptor { | |||
41 | }; | 41 | }; |
42 | 42 | ||
43 | /* | 43 | /* |
44 | * The following define adds a 64 byte gap between the signal | ||
45 | * stack frame and previous contents of the stack. This allows | ||
46 | * frame unwinding in a function epilogue but only if a frame | ||
47 | * pointer is used in the function. This is necessary because | ||
48 | * current gcc compilers (<4.3) do not generate unwind info on | ||
49 | * SH for function epilogues. | ||
50 | */ | ||
51 | #define UNWINDGUARD 64 | ||
52 | |||
53 | /* | ||
44 | * Atomically swap in the new signal mask, and wait for a signal. | 54 | * Atomically swap in the new signal mask, and wait for a signal. |
45 | */ | 55 | */ |
46 | asmlinkage int | 56 | asmlinkage int |
@@ -327,7 +337,7 @@ get_sigframe(struct k_sigaction *ka, unsigned long sp, size_t frame_size) | |||
327 | sp = current->sas_ss_sp + current->sas_ss_size; | 337 | sp = current->sas_ss_sp + current->sas_ss_size; |
328 | } | 338 | } |
329 | 339 | ||
330 | return (void __user *)((sp - frame_size) & -8ul); | 340 | return (void __user *)((sp - (frame_size+UNWINDGUARD)) & -8ul); |
331 | } | 341 | } |
332 | 342 | ||
333 | /* These symbols are defined with the addresses in the vsyscall page. | 343 | /* These symbols are defined with the addresses in the vsyscall page. |
diff --git a/arch/sh/kernel/sys_sh.c b/arch/sh/kernel/sys_sh.c index 90d00e47264..8aa5d1ceaf1 100644 --- a/arch/sh/kernel/sys_sh.c +++ b/arch/sh/kernel/sys_sh.c | |||
@@ -25,6 +25,8 @@ | |||
25 | #include <asm/syscalls.h> | 25 | #include <asm/syscalls.h> |
26 | #include <asm/uaccess.h> | 26 | #include <asm/uaccess.h> |
27 | #include <asm/unistd.h> | 27 | #include <asm/unistd.h> |
28 | #include <asm/cacheflush.h> | ||
29 | #include <asm/cachectl.h> | ||
28 | 30 | ||
29 | static inline long | 31 | static inline long |
30 | do_mmap2(unsigned long addr, unsigned long len, unsigned long prot, | 32 | do_mmap2(unsigned long addr, unsigned long len, unsigned long prot, |
@@ -179,6 +181,47 @@ asmlinkage int sys_ipc(uint call, int first, int second, | |||
179 | return -EINVAL; | 181 | return -EINVAL; |
180 | } | 182 | } |
181 | 183 | ||
184 | /* sys_cacheflush -- flush (part of) the processor cache. */ | ||
185 | asmlinkage int sys_cacheflush(unsigned long addr, unsigned long len, int op) | ||
186 | { | ||
187 | struct vm_area_struct *vma; | ||
188 | |||
189 | if ((op <= 0) || (op > (CACHEFLUSH_D_PURGE|CACHEFLUSH_I))) | ||
190 | return -EINVAL; | ||
191 | |||
192 | /* | ||
193 | * Verify that the specified address region actually belongs | ||
194 | * to this process. | ||
195 | */ | ||
196 | if (addr + len < addr) | ||
197 | return -EFAULT; | ||
198 | |||
199 | down_read(¤t->mm->mmap_sem); | ||
200 | vma = find_vma (current->mm, addr); | ||
201 | if (vma == NULL || addr < vma->vm_start || addr + len > vma->vm_end) { | ||
202 | up_read(¤t->mm->mmap_sem); | ||
203 | return -EFAULT; | ||
204 | } | ||
205 | |||
206 | switch (op & CACHEFLUSH_D_PURGE) { | ||
207 | case CACHEFLUSH_D_INVAL: | ||
208 | __flush_invalidate_region((void *)addr, len); | ||
209 | break; | ||
210 | case CACHEFLUSH_D_WB: | ||
211 | __flush_wback_region((void *)addr, len); | ||
212 | break; | ||
213 | case CACHEFLUSH_D_PURGE: | ||
214 | __flush_purge_region((void *)addr, len); | ||
215 | break; | ||
216 | } | ||
217 | |||
218 | if (op & CACHEFLUSH_I) | ||
219 | flush_cache_all(); | ||
220 | |||
221 | up_read(¤t->mm->mmap_sem); | ||
222 | return 0; | ||
223 | } | ||
224 | |||
182 | asmlinkage int sys_uname(struct old_utsname __user *name) | 225 | asmlinkage int sys_uname(struct old_utsname __user *name) |
183 | { | 226 | { |
184 | int err; | 227 | int err; |
diff --git a/arch/sh/kernel/syscalls_32.S b/arch/sh/kernel/syscalls_32.S index f9e21fa2f59..16ba225ede8 100644 --- a/arch/sh/kernel/syscalls_32.S +++ b/arch/sh/kernel/syscalls_32.S | |||
@@ -139,7 +139,7 @@ ENTRY(sys_call_table) | |||
139 | .long sys_clone /* 120 */ | 139 | .long sys_clone /* 120 */ |
140 | .long sys_setdomainname | 140 | .long sys_setdomainname |
141 | .long sys_newuname | 141 | .long sys_newuname |
142 | .long sys_ni_syscall /* sys_modify_ldt */ | 142 | .long sys_cacheflush /* x86: sys_modify_ldt */ |
143 | .long sys_adjtimex | 143 | .long sys_adjtimex |
144 | .long sys_mprotect /* 125 */ | 144 | .long sys_mprotect /* 125 */ |
145 | .long sys_sigprocmask | 145 | .long sys_sigprocmask |
diff --git a/arch/sh/kernel/syscalls_64.S b/arch/sh/kernel/syscalls_64.S index bf420b616ae..af6fb7410c2 100644 --- a/arch/sh/kernel/syscalls_64.S +++ b/arch/sh/kernel/syscalls_64.S | |||
@@ -143,7 +143,7 @@ sys_call_table: | |||
143 | .long sys_clone /* 120 */ | 143 | .long sys_clone /* 120 */ |
144 | .long sys_setdomainname | 144 | .long sys_setdomainname |
145 | .long sys_newuname | 145 | .long sys_newuname |
146 | .long sys_ni_syscall /* sys_modify_ldt */ | 146 | .long sys_cacheflush /* x86: sys_modify_ldt */ |
147 | .long sys_adjtimex | 147 | .long sys_adjtimex |
148 | .long sys_mprotect /* 125 */ | 148 | .long sys_mprotect /* 125 */ |
149 | .long sys_sigprocmask | 149 | .long sys_sigprocmask |
diff --git a/arch/sh/kernel/traps_32.c b/arch/sh/kernel/traps_32.c index 05a04b6df84..c581dc31d92 100644 --- a/arch/sh/kernel/traps_32.c +++ b/arch/sh/kernel/traps_32.c | |||
@@ -24,6 +24,7 @@ | |||
24 | #include <linux/kdebug.h> | 24 | #include <linux/kdebug.h> |
25 | #include <linux/kexec.h> | 25 | #include <linux/kexec.h> |
26 | #include <linux/limits.h> | 26 | #include <linux/limits.h> |
27 | #include <linux/proc_fs.h> | ||
27 | #include <asm/system.h> | 28 | #include <asm/system.h> |
28 | #include <asm/uaccess.h> | 29 | #include <asm/uaccess.h> |
29 | #include <asm/fpu.h> | 30 | #include <asm/fpu.h> |
@@ -44,6 +45,87 @@ | |||
44 | #define TRAP_ILLEGAL_SLOT_INST 13 | 45 | #define TRAP_ILLEGAL_SLOT_INST 13 |
45 | #endif | 46 | #endif |
46 | 47 | ||
48 | static unsigned long se_user; | ||
49 | static unsigned long se_sys; | ||
50 | static unsigned long se_skipped; | ||
51 | static unsigned long se_half; | ||
52 | static unsigned long se_word; | ||
53 | static unsigned long se_dword; | ||
54 | static unsigned long se_multi; | ||
55 | /* bitfield: 1: warn 2: fixup 4: signal -> combinations 2|4 && 1|2|4 are not | ||
56 | valid! */ | ||
57 | static int se_usermode = 3; | ||
58 | /* 0: no warning 1: print a warning message */ | ||
59 | static int se_kernmode_warn = 1; | ||
60 | |||
61 | #ifdef CONFIG_PROC_FS | ||
62 | static const char *se_usermode_action[] = { | ||
63 | "ignored", | ||
64 | "warn", | ||
65 | "fixup", | ||
66 | "fixup+warn", | ||
67 | "signal", | ||
68 | "signal+warn" | ||
69 | }; | ||
70 | |||
71 | static int | ||
72 | proc_alignment_read(char *page, char **start, off_t off, int count, int *eof, | ||
73 | void *data) | ||
74 | { | ||
75 | char *p = page; | ||
76 | int len; | ||
77 | |||
78 | p += sprintf(p, "User:\t\t%lu\n", se_user); | ||
79 | p += sprintf(p, "System:\t\t%lu\n", se_sys); | ||
80 | p += sprintf(p, "Skipped:\t%lu\n", se_skipped); | ||
81 | p += sprintf(p, "Half:\t\t%lu\n", se_half); | ||
82 | p += sprintf(p, "Word:\t\t%lu\n", se_word); | ||
83 | p += sprintf(p, "DWord:\t\t%lu\n", se_dword); | ||
84 | p += sprintf(p, "Multi:\t\t%lu\n", se_multi); | ||
85 | p += sprintf(p, "User faults:\t%i (%s)\n", se_usermode, | ||
86 | se_usermode_action[se_usermode]); | ||
87 | p += sprintf(p, "Kernel faults:\t%i (fixup%s)\n", se_kernmode_warn, | ||
88 | se_kernmode_warn ? "+warn" : ""); | ||
89 | |||
90 | len = (p - page) - off; | ||
91 | if (len < 0) | ||
92 | len = 0; | ||
93 | |||
94 | *eof = (len <= count) ? 1 : 0; | ||
95 | *start = page + off; | ||
96 | |||
97 | return len; | ||
98 | } | ||
99 | |||
100 | static int proc_alignment_write(struct file *file, const char __user *buffer, | ||
101 | unsigned long count, void *data) | ||
102 | { | ||
103 | char mode; | ||
104 | |||
105 | if (count > 0) { | ||
106 | if (get_user(mode, buffer)) | ||
107 | return -EFAULT; | ||
108 | if (mode >= '0' && mode <= '5') | ||
109 | se_usermode = mode - '0'; | ||
110 | } | ||
111 | return count; | ||
112 | } | ||
113 | |||
114 | static int proc_alignment_kern_write(struct file *file, const char __user *buffer, | ||
115 | unsigned long count, void *data) | ||
116 | { | ||
117 | char mode; | ||
118 | |||
119 | if (count > 0) { | ||
120 | if (get_user(mode, buffer)) | ||
121 | return -EFAULT; | ||
122 | if (mode >= '0' && mode <= '1') | ||
123 | se_kernmode_warn = mode - '0'; | ||
124 | } | ||
125 | return count; | ||
126 | } | ||
127 | #endif | ||
128 | |||
47 | static void dump_mem(const char *str, unsigned long bottom, unsigned long top) | 129 | static void dump_mem(const char *str, unsigned long bottom, unsigned long top) |
48 | { | 130 | { |
49 | unsigned long p; | 131 | unsigned long p; |
@@ -194,6 +276,13 @@ static int handle_unaligned_ins(insn_size_t instruction, struct pt_regs *regs, | |||
194 | 276 | ||
195 | count = 1<<(instruction&3); | 277 | count = 1<<(instruction&3); |
196 | 278 | ||
279 | switch (count) { | ||
280 | case 1: se_half += 1; break; | ||
281 | case 2: se_word += 1; break; | ||
282 | case 4: se_dword += 1; break; | ||
283 | case 8: se_multi += 1; break; /* ??? */ | ||
284 | } | ||
285 | |||
197 | ret = -EFAULT; | 286 | ret = -EFAULT; |
198 | switch (instruction>>12) { | 287 | switch (instruction>>12) { |
199 | case 0: /* mov.[bwl] to/from memory via r0+rn */ | 288 | case 0: /* mov.[bwl] to/from memory via r0+rn */ |
@@ -359,13 +448,6 @@ static inline int handle_delayslot(struct pt_regs *regs, | |||
359 | #define SH_PC_8BIT_OFFSET(instr) ((((signed char)(instr))*2) + 4) | 448 | #define SH_PC_8BIT_OFFSET(instr) ((((signed char)(instr))*2) + 4) |
360 | #define SH_PC_12BIT_OFFSET(instr) ((((signed short)(instr<<4))>>3) + 4) | 449 | #define SH_PC_12BIT_OFFSET(instr) ((((signed short)(instr<<4))>>3) + 4) |
361 | 450 | ||
362 | /* | ||
363 | * XXX: SH-2A needs this too, but it needs an overhaul thanks to mixed 32-bit | ||
364 | * opcodes.. | ||
365 | */ | ||
366 | |||
367 | static int handle_unaligned_notify_count = 10; | ||
368 | |||
369 | int handle_unaligned_access(insn_size_t instruction, struct pt_regs *regs, | 451 | int handle_unaligned_access(insn_size_t instruction, struct pt_regs *regs, |
370 | struct mem_access *ma) | 452 | struct mem_access *ma) |
371 | { | 453 | { |
@@ -375,15 +457,13 @@ int handle_unaligned_access(insn_size_t instruction, struct pt_regs *regs, | |||
375 | index = (instruction>>8)&15; /* 0x0F00 */ | 457 | index = (instruction>>8)&15; /* 0x0F00 */ |
376 | rm = regs->regs[index]; | 458 | rm = regs->regs[index]; |
377 | 459 | ||
378 | /* shout about the first ten userspace fixups */ | 460 | /* shout about fixups */ |
379 | if (user_mode(regs) && handle_unaligned_notify_count>0) { | 461 | if (printk_ratelimit()) |
380 | handle_unaligned_notify_count--; | 462 | printk(KERN_NOTICE "Fixing up unaligned %s access " |
381 | |||
382 | printk(KERN_NOTICE "Fixing up unaligned userspace access " | ||
383 | "in \"%s\" pid=%d pc=0x%p ins=0x%04hx\n", | 463 | "in \"%s\" pid=%d pc=0x%p ins=0x%04hx\n", |
464 | user_mode(regs) ? "userspace" : "kernel", | ||
384 | current->comm, task_pid_nr(current), | 465 | current->comm, task_pid_nr(current), |
385 | (void *)regs->pc, instruction); | 466 | (void *)regs->pc, instruction); |
386 | } | ||
387 | 467 | ||
388 | ret = -EFAULT; | 468 | ret = -EFAULT; |
389 | switch (instruction&0xF000) { | 469 | switch (instruction&0xF000) { |
@@ -539,6 +619,36 @@ asmlinkage void do_address_error(struct pt_regs *regs, | |||
539 | 619 | ||
540 | local_irq_enable(); | 620 | local_irq_enable(); |
541 | 621 | ||
622 | se_user += 1; | ||
623 | |||
624 | #ifndef CONFIG_CPU_SH2A | ||
625 | set_fs(USER_DS); | ||
626 | if (copy_from_user(&instruction, (u16 *)(regs->pc & ~1), 2)) { | ||
627 | set_fs(oldfs); | ||
628 | goto uspace_segv; | ||
629 | } | ||
630 | set_fs(oldfs); | ||
631 | |||
632 | /* shout about userspace fixups */ | ||
633 | if (se_usermode & 1) | ||
634 | printk(KERN_NOTICE "Unaligned userspace access " | ||
635 | "in \"%s\" pid=%d pc=0x%p ins=0x%04hx\n", | ||
636 | current->comm, current->pid, (void *)regs->pc, | ||
637 | instruction); | ||
638 | #endif | ||
639 | |||
640 | if (se_usermode & 2) | ||
641 | goto fixup; | ||
642 | |||
643 | if (se_usermode & 4) | ||
644 | goto uspace_segv; | ||
645 | else { | ||
646 | /* ignore */ | ||
647 | regs->pc += instruction_size(instruction); | ||
648 | return; | ||
649 | } | ||
650 | |||
651 | fixup: | ||
542 | /* bad PC is not something we can fix */ | 652 | /* bad PC is not something we can fix */ |
543 | if (regs->pc & 1) { | 653 | if (regs->pc & 1) { |
544 | si_code = BUS_ADRALN; | 654 | si_code = BUS_ADRALN; |
@@ -546,15 +656,6 @@ asmlinkage void do_address_error(struct pt_regs *regs, | |||
546 | } | 656 | } |
547 | 657 | ||
548 | set_fs(USER_DS); | 658 | set_fs(USER_DS); |
549 | if (copy_from_user(&instruction, (void __user *)(regs->pc), | ||
550 | sizeof(instruction))) { | ||
551 | /* Argh. Fault on the instruction itself. | ||
552 | This should never happen non-SMP | ||
553 | */ | ||
554 | set_fs(oldfs); | ||
555 | goto uspace_segv; | ||
556 | } | ||
557 | |||
558 | tmp = handle_unaligned_access(instruction, regs, | 659 | tmp = handle_unaligned_access(instruction, regs, |
559 | &user_mem_access); | 660 | &user_mem_access); |
560 | set_fs(oldfs); | 661 | set_fs(oldfs); |
@@ -572,6 +673,14 @@ uspace_segv: | |||
572 | info.si_addr = (void __user *)address; | 673 | info.si_addr = (void __user *)address; |
573 | force_sig_info(SIGBUS, &info, current); | 674 | force_sig_info(SIGBUS, &info, current); |
574 | } else { | 675 | } else { |
676 | se_sys += 1; | ||
677 | |||
678 | if (se_kernmode_warn) | ||
679 | printk(KERN_NOTICE "Unaligned kernel access " | ||
680 | "on behalf of \"%s\" pid=%d pc=0x%p ins=0x%04hx\n", | ||
681 | current->comm, current->pid, (void *)regs->pc, | ||
682 | instruction); | ||
683 | |||
575 | if (regs->pc & 1) | 684 | if (regs->pc & 1) |
576 | die("unaligned program counter", regs, error_code); | 685 | die("unaligned program counter", regs, error_code); |
577 | 686 | ||
@@ -881,3 +990,38 @@ void dump_stack(void) | |||
881 | show_stack(NULL, NULL); | 990 | show_stack(NULL, NULL); |
882 | } | 991 | } |
883 | EXPORT_SYMBOL(dump_stack); | 992 | EXPORT_SYMBOL(dump_stack); |
993 | |||
994 | #ifdef CONFIG_PROC_FS | ||
995 | /* | ||
996 | * This needs to be done after sysctl_init, otherwise sys/ will be | ||
997 | * overwritten. Actually, this shouldn't be in sys/ at all since | ||
998 | * it isn't a sysctl, and it doesn't contain sysctl information. | ||
999 | * We now locate it in /proc/cpu/alignment instead. | ||
1000 | */ | ||
1001 | static int __init alignment_init(void) | ||
1002 | { | ||
1003 | struct proc_dir_entry *dir, *res; | ||
1004 | |||
1005 | dir = proc_mkdir("cpu", NULL); | ||
1006 | if (!dir) | ||
1007 | return -ENOMEM; | ||
1008 | |||
1009 | res = create_proc_entry("alignment", S_IWUSR | S_IRUGO, dir); | ||
1010 | if (!res) | ||
1011 | return -ENOMEM; | ||
1012 | |||
1013 | res->read_proc = proc_alignment_read; | ||
1014 | res->write_proc = proc_alignment_write; | ||
1015 | |||
1016 | res = create_proc_entry("kernel_alignment", S_IWUSR | S_IRUGO, dir); | ||
1017 | if (!res) | ||
1018 | return -ENOMEM; | ||
1019 | |||
1020 | res->read_proc = proc_alignment_read; | ||
1021 | res->write_proc = proc_alignment_kern_write; | ||
1022 | |||
1023 | return 0; | ||
1024 | } | ||
1025 | |||
1026 | fs_initcall(alignment_init); | ||
1027 | #endif | ||