diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2013-03-02 10:58:56 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2013-03-02 10:58:56 -0500 |
commit | e23b62256a361611cbd45cd1456638f1a5106b5c (patch) | |
tree | 472968c961432a1d9d0c3634ca20433f1d9cd29b | |
parent | aebb2afd5420c860b7fbc3882a323ef1247fbf16 (diff) | |
parent | 8ccfe6675fa974bd06d64f74d0fdee6a5267d2aa (diff) |
Merge tag 'arc-v3.9-rc1-late' of git://git.kernel.org/pub/scm/linux/kernel/git/vgupta/arc
Pull new ARC architecture from Vineet Gupta:
"Initial ARC Linux port with some fixes on top for 3.9-rc1:
I would like to introduce the Linux port to ARC Processors (from
Synopsys) for 3.9-rc1. The patch-set has been discussed on the public
lists since Nov and has received a fair bit of review, specially from
Arnd, tglx, Al and other subsystem maintainers for DeviceTree, kgdb...
The arch bits are in arch/arc, some asm-generic changes (acked by
Arnd), a minor change to PARISC (acked by Helge).
The series is a touch bigger for a new port for 2 main reasons:
1. It enables a basic kernel in first sub-series and adds
ptrace/kgdb/.. later
2. Some of the fallout of review (DeviceTree support, multi-platform-
image support) were added on top of orig series, primarily to
record the revision history.
This updated pull request additionally contains
- fixes due to our GNU tools catching up with the new syscall/ptrace
ABI
- some (minor) cross-arch Kconfig updates."
* tag 'arc-v3.9-rc1-late' of git://git.kernel.org/pub/scm/linux/kernel/git/vgupta/arc: (82 commits)
ARC: split elf.h into uapi and export it for userspace
ARC: Fixup the current ABI version
ARC: gdbserver using regset interface possibly broken
ARC: Kconfig cleanup tracking cross-arch Kconfig pruning in merge window
ARC: make a copy of flat DT
ARC: [plat-arcfpga] DT arc-uart bindings change: "baud" => "current-speed"
ARC: Ensure CONFIG_VIRT_TO_BUS is not enabled
ARC: Fix pt_orig_r8 access
ARC: [3.9] Fallout of hlist iterator update
ARC: 64bit RTSC timestamp hardware issue
ARC: Don't fiddle with non-existent caches
ARC: Add self to MAINTAINERS
ARC: Provide a default serial.h for uart drivers needing BASE_BAUD
ARC: [plat-arcfpga] defconfig for fully loaded ARC Linux
ARC: [Review] Multi-platform image #8: platform registers SMP callbacks
ARC: [Review] Multi-platform image #7: SMP common code to use callbacks
ARC: [Review] Multi-platform image #6: cpu-to-dma-addr optional
ARC: [Review] Multi-platform image #5: NR_IRQS defined by ARC core
ARC: [Review] Multi-platform image #4: Isolate platform headers
ARC: [Review] Multi-platform image #3: switch to board callback
...
147 files changed, 19552 insertions, 1 deletions
diff --git a/Documentation/devicetree/bindings/arc/interrupts.txt b/Documentation/devicetree/bindings/arc/interrupts.txt new file mode 100644 index 000000000000..9a5d562435ea --- /dev/null +++ b/Documentation/devicetree/bindings/arc/interrupts.txt | |||
@@ -0,0 +1,24 @@ | |||
1 | * ARC700 incore Interrupt Controller | ||
2 | |||
3 | The core interrupt controller provides 32 prioritised interrupts (2 levels) | ||
4 | to ARC700 core. | ||
5 | |||
6 | Properties: | ||
7 | |||
8 | - compatible: "snps,arc700-intc" | ||
9 | - interrupt-controller: This is an interrupt controller. | ||
10 | - #interrupt-cells: Must be <1>. | ||
11 | |||
12 | Single Cell "interrupts" property of a device specifies the IRQ number | ||
13 | between 0 to 31 | ||
14 | |||
15 | intc accessed via the special ARC AUX register interface, hence "reg" property | ||
16 | is not specified. | ||
17 | |||
18 | Example: | ||
19 | |||
20 | intc: interrupt-controller { | ||
21 | compatible = "snps,arc700-intc"; | ||
22 | interrupt-controller; | ||
23 | #interrupt-cells = <1>; | ||
24 | }; | ||
diff --git a/MAINTAINERS b/MAINTAINERS index 6db1c6bdf015..aea0adf414dc 100644 --- a/MAINTAINERS +++ b/MAINTAINERS | |||
@@ -7682,6 +7682,12 @@ F: lib/swiotlb.c | |||
7682 | F: arch/*/kernel/pci-swiotlb.c | 7682 | F: arch/*/kernel/pci-swiotlb.c |
7683 | F: include/linux/swiotlb.h | 7683 | F: include/linux/swiotlb.h |
7684 | 7684 | ||
7685 | SYNOPSYS ARC ARCHITECTURE | ||
7686 | M: Vineet Gupta <vgupta@synopsys.com> | ||
7687 | L: linux-snps-arc@vger.kernel.org | ||
7688 | S: Supported | ||
7689 | F: arch/arc/ | ||
7690 | |||
7685 | SYSV FILESYSTEM | 7691 | SYSV FILESYSTEM |
7686 | M: Christoph Hellwig <hch@infradead.org> | 7692 | M: Christoph Hellwig <hch@infradead.org> |
7687 | S: Maintained | 7693 | S: Maintained |
diff --git a/arch/arc/Kbuild b/arch/arc/Kbuild new file mode 100644 index 000000000000..082d329d3245 --- /dev/null +++ b/arch/arc/Kbuild | |||
@@ -0,0 +1,2 @@ | |||
1 | obj-y += kernel/ | ||
2 | obj-y += mm/ | ||
diff --git a/arch/arc/Kconfig b/arch/arc/Kconfig new file mode 100644 index 000000000000..e6f4eca09ee3 --- /dev/null +++ b/arch/arc/Kconfig | |||
@@ -0,0 +1,453 @@ | |||
1 | # | ||
2 | # Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | # | ||
4 | # This program is free software; you can redistribute it and/or modify | ||
5 | # it under the terms of the GNU General Public License version 2 as | ||
6 | # published by the Free Software Foundation. | ||
7 | # | ||
8 | |||
9 | config ARC | ||
10 | def_bool y | ||
11 | select CLONE_BACKWARDS | ||
12 | # ARC Busybox based initramfs absolutely relies on DEVTMPFS for /dev | ||
13 | select DEVTMPFS if !INITRAMFS_SOURCE="" | ||
14 | select GENERIC_ATOMIC64 | ||
15 | select GENERIC_CLOCKEVENTS | ||
16 | select GENERIC_FIND_FIRST_BIT | ||
17 | # for now, we don't need GENERIC_IRQ_PROBE, CONFIG_GENERIC_IRQ_CHIP | ||
18 | select GENERIC_IRQ_SHOW | ||
19 | select GENERIC_KERNEL_EXECVE | ||
20 | select GENERIC_KERNEL_THREAD | ||
21 | select GENERIC_PENDING_IRQ if SMP | ||
22 | select GENERIC_SMP_IDLE_THREAD | ||
23 | select HAVE_ARCH_KGDB | ||
24 | select HAVE_ARCH_TRACEHOOK | ||
25 | select HAVE_GENERIC_HARDIRQS | ||
26 | select HAVE_IOREMAP_PROT | ||
27 | select HAVE_KPROBES | ||
28 | select HAVE_KRETPROBES | ||
29 | select HAVE_MEMBLOCK | ||
30 | select HAVE_MOD_ARCH_SPECIFIC if ARC_DW2_UNWIND | ||
31 | select HAVE_OPROFILE | ||
32 | select HAVE_PERF_EVENTS | ||
33 | select IRQ_DOMAIN | ||
34 | select MODULES_USE_ELF_RELA | ||
35 | select NO_BOOTMEM | ||
36 | select OF | ||
37 | select OF_EARLY_FLATTREE | ||
38 | select PERF_USE_VMALLOC | ||
39 | |||
40 | config SCHED_OMIT_FRAME_POINTER | ||
41 | def_bool y | ||
42 | |||
43 | config GENERIC_CSUM | ||
44 | def_bool y | ||
45 | |||
46 | config RWSEM_GENERIC_SPINLOCK | ||
47 | def_bool y | ||
48 | |||
49 | config ARCH_FLATMEM_ENABLE | ||
50 | def_bool y | ||
51 | |||
52 | config MMU | ||
53 | def_bool y | ||
54 | |||
55 | config NO_IOPORT | ||
56 | def_bool y | ||
57 | |||
58 | config GENERIC_CALIBRATE_DELAY | ||
59 | def_bool y | ||
60 | |||
61 | config GENERIC_HWEIGHT | ||
62 | def_bool y | ||
63 | |||
64 | config BINFMT_ELF | ||
65 | def_bool y | ||
66 | |||
67 | config STACKTRACE_SUPPORT | ||
68 | def_bool y | ||
69 | select STACKTRACE | ||
70 | |||
71 | config HAVE_LATENCYTOP_SUPPORT | ||
72 | def_bool y | ||
73 | |||
74 | config NO_DMA | ||
75 | def_bool n | ||
76 | |||
77 | source "init/Kconfig" | ||
78 | source "kernel/Kconfig.freezer" | ||
79 | |||
80 | menu "ARC Architecture Configuration" | ||
81 | |||
82 | menu "ARC Platform/SoC/Board" | ||
83 | |||
84 | source "arch/arc/plat-arcfpga/Kconfig" | ||
85 | #New platform adds here | ||
86 | |||
87 | endmenu | ||
88 | |||
89 | menu "ARC CPU Configuration" | ||
90 | |||
91 | choice | ||
92 | prompt "ARC Core" | ||
93 | default ARC_CPU_770 | ||
94 | |||
95 | config ARC_CPU_750D | ||
96 | bool "ARC750D" | ||
97 | help | ||
98 | Support for ARC750 core | ||
99 | |||
100 | config ARC_CPU_770 | ||
101 | bool "ARC770" | ||
102 | select ARC_CPU_REL_4_10 | ||
103 | help | ||
104 | Support for ARC770 core introduced with Rel 4.10 (Summer 2011) | ||
105 | This core has a bunch of cool new features: | ||
106 | -MMU-v3: Variable Page Sz (4k, 8k, 16k), bigger J-TLB (128x4) | ||
107 | Shared Address Spaces (for sharing TLB entires in MMU) | ||
108 | -Caches: New Prog Model, Region Flush | ||
109 | -Insns: endian swap, load-locked/store-conditional, time-stamp-ctr | ||
110 | |||
111 | endchoice | ||
112 | |||
113 | config CPU_BIG_ENDIAN | ||
114 | bool "Enable Big Endian Mode" | ||
115 | default n | ||
116 | help | ||
117 | Build kernel for Big Endian Mode of ARC CPU | ||
118 | |||
119 | # If a platform can't work with 0x8000_0000 based dma_addr_t | ||
120 | config ARC_PLAT_NEEDS_CPU_TO_DMA | ||
121 | bool | ||
122 | |||
123 | config SMP | ||
124 | bool "Symmetric Multi-Processing (Incomplete)" | ||
125 | default n | ||
126 | select USE_GENERIC_SMP_HELPERS | ||
127 | help | ||
128 | This enables support for systems with more than one CPU. If you have | ||
129 | a system with only one CPU, like most personal computers, say N. If | ||
130 | you have a system with more than one CPU, say Y. | ||
131 | |||
132 | if SMP | ||
133 | |||
134 | config ARC_HAS_COH_CACHES | ||
135 | def_bool n | ||
136 | |||
137 | config ARC_HAS_COH_LLSC | ||
138 | def_bool n | ||
139 | |||
140 | config ARC_HAS_COH_RTSC | ||
141 | def_bool n | ||
142 | |||
143 | config ARC_HAS_REENTRANT_IRQ_LV2 | ||
144 | def_bool n | ||
145 | |||
146 | endif | ||
147 | |||
148 | config NR_CPUS | ||
149 | int "Maximum number of CPUs (2-32)" | ||
150 | range 2 32 | ||
151 | depends on SMP | ||
152 | default "2" | ||
153 | |||
154 | menuconfig ARC_CACHE | ||
155 | bool "Enable Cache Support" | ||
156 | default y | ||
157 | # if SMP, cache enabled ONLY if ARC implementation has cache coherency | ||
158 | depends on !SMP || ARC_HAS_COH_CACHES | ||
159 | |||
160 | if ARC_CACHE | ||
161 | |||
162 | config ARC_CACHE_LINE_SHIFT | ||
163 | int "Cache Line Length (as power of 2)" | ||
164 | range 5 7 | ||
165 | default "6" | ||
166 | help | ||
167 | Starting with ARC700 4.9, Cache line length is configurable, | ||
168 | This option specifies "N", with Line-len = 2 power N | ||
169 | So line lengths of 32, 64, 128 are specified by 5,6,7, respectively | ||
170 | Linux only supports same line lengths for I and D caches. | ||
171 | |||
172 | config ARC_HAS_ICACHE | ||
173 | bool "Use Instruction Cache" | ||
174 | default y | ||
175 | |||
176 | config ARC_HAS_DCACHE | ||
177 | bool "Use Data Cache" | ||
178 | default y | ||
179 | |||
180 | config ARC_CACHE_PAGES | ||
181 | bool "Per Page Cache Control" | ||
182 | default y | ||
183 | depends on ARC_HAS_ICACHE || ARC_HAS_DCACHE | ||
184 | help | ||
185 | This can be used to over-ride the global I/D Cache Enable on a | ||
186 | per-page basis (but only for pages accessed via MMU such as | ||
187 | Kernel Virtual address or User Virtual Address) | ||
188 | TLB entries have a per-page Cache Enable Bit. | ||
189 | Note that Global I/D ENABLE + Per Page DISABLE works but corollary | ||
190 | Global DISABLE + Per Page ENABLE won't work | ||
191 | |||
192 | endif #ARC_CACHE | ||
193 | |||
194 | config ARC_HAS_ICCM | ||
195 | bool "Use ICCM" | ||
196 | help | ||
197 | Single Cycle RAMS to store Fast Path Code | ||
198 | default n | ||
199 | |||
200 | config ARC_ICCM_SZ | ||
201 | int "ICCM Size in KB" | ||
202 | default "64" | ||
203 | depends on ARC_HAS_ICCM | ||
204 | |||
205 | config ARC_HAS_DCCM | ||
206 | bool "Use DCCM" | ||
207 | help | ||
208 | Single Cycle RAMS to store Fast Path Data | ||
209 | default n | ||
210 | |||
211 | config ARC_DCCM_SZ | ||
212 | int "DCCM Size in KB" | ||
213 | default "64" | ||
214 | depends on ARC_HAS_DCCM | ||
215 | |||
216 | config ARC_DCCM_BASE | ||
217 | hex "DCCM map address" | ||
218 | default "0xA0000000" | ||
219 | depends on ARC_HAS_DCCM | ||
220 | |||
221 | config ARC_HAS_HW_MPY | ||
222 | bool "Use Hardware Multiplier (Normal or Faster XMAC)" | ||
223 | default y | ||
224 | help | ||
225 | Influences how gcc generates code for MPY operations. | ||
226 | If enabled, MPYxx insns are generated, provided by Standard/XMAC | ||
227 | Multipler. Otherwise software multipy lib is used | ||
228 | |||
229 | choice | ||
230 | prompt "ARC700 MMU Version" | ||
231 | default ARC_MMU_V3 if ARC_CPU_770 | ||
232 | default ARC_MMU_V2 if ARC_CPU_750D | ||
233 | |||
234 | config ARC_MMU_V1 | ||
235 | bool "MMU v1" | ||
236 | help | ||
237 | Orig ARC700 MMU | ||
238 | |||
239 | config ARC_MMU_V2 | ||
240 | bool "MMU v2" | ||
241 | help | ||
242 | Fixed the deficiency of v1 - possible thrashing in memcpy sceanrio | ||
243 | when 2 D-TLB and 1 I-TLB entries index into same 2way set. | ||
244 | |||
245 | config ARC_MMU_V3 | ||
246 | bool "MMU v3" | ||
247 | depends on ARC_CPU_770 | ||
248 | help | ||
249 | Introduced with ARC700 4.10: New Features | ||
250 | Variable Page size (1k-16k), var JTLB size 128 x (2 or 4) | ||
251 | Shared Address Spaces (SASID) | ||
252 | |||
253 | endchoice | ||
254 | |||
255 | |||
256 | choice | ||
257 | prompt "MMU Page Size" | ||
258 | default ARC_PAGE_SIZE_8K | ||
259 | |||
260 | config ARC_PAGE_SIZE_8K | ||
261 | bool "8KB" | ||
262 | help | ||
263 | Choose between 8k vs 16k | ||
264 | |||
265 | config ARC_PAGE_SIZE_16K | ||
266 | bool "16KB" | ||
267 | depends on ARC_MMU_V3 | ||
268 | |||
269 | config ARC_PAGE_SIZE_4K | ||
270 | bool "4KB" | ||
271 | depends on ARC_MMU_V3 | ||
272 | |||
273 | endchoice | ||
274 | |||
275 | config ARC_COMPACT_IRQ_LEVELS | ||
276 | bool "ARCompact IRQ Priorities: High(2)/Low(1)" | ||
277 | default n | ||
278 | # Timer HAS to be high priority, for any other high priority config | ||
279 | select ARC_IRQ3_LV2 | ||
280 | # if SMP, LV2 enabled ONLY if ARC implementation has LV2 re-entrancy | ||
281 | depends on !SMP || ARC_HAS_REENTRANT_IRQ_LV2 | ||
282 | |||
283 | if ARC_COMPACT_IRQ_LEVELS | ||
284 | |||
285 | config ARC_IRQ3_LV2 | ||
286 | bool | ||
287 | |||
288 | config ARC_IRQ5_LV2 | ||
289 | bool | ||
290 | |||
291 | config ARC_IRQ6_LV2 | ||
292 | bool | ||
293 | |||
294 | endif | ||
295 | |||
296 | config ARC_FPU_SAVE_RESTORE | ||
297 | bool "Enable FPU state persistence across context switch" | ||
298 | default n | ||
299 | help | ||
300 | Double Precision Floating Point unit had dedictaed regs which | ||
301 | need to be saved/restored across context-switch. | ||
302 | Note that ARC FPU is overly simplistic, unlike say x86, which has | ||
303 | hardware pieces to allow software to conditionally save/restore, | ||
304 | based on actual usage of FPU by a task. Thus our implemn does | ||
305 | this for all tasks in system. | ||
306 | |||
307 | menuconfig ARC_CPU_REL_4_10 | ||
308 | bool "Enable support for Rel 4.10 features" | ||
309 | default n | ||
310 | help | ||
311 | -ARC770 (and dependent features) enabled | ||
312 | -ARC750 also shares some of the new features with 770 | ||
313 | |||
314 | config ARC_HAS_LLSC | ||
315 | bool "Insn: LLOCK/SCOND (efficient atomic ops)" | ||
316 | default y | ||
317 | depends on ARC_CPU_770 | ||
318 | # if SMP, enable LLSC ONLY if ARC implementation has coherent atomics | ||
319 | depends on !SMP || ARC_HAS_COH_LLSC | ||
320 | |||
321 | config ARC_HAS_SWAPE | ||
322 | bool "Insn: SWAPE (endian-swap)" | ||
323 | default y | ||
324 | depends on ARC_CPU_REL_4_10 | ||
325 | |||
326 | config ARC_HAS_RTSC | ||
327 | bool "Insn: RTSC (64-bit r/o cycle counter)" | ||
328 | default y | ||
329 | depends on ARC_CPU_REL_4_10 | ||
330 | # if SMP, enable RTSC only if counter is coherent across cores | ||
331 | depends on !SMP || ARC_HAS_COH_RTSC | ||
332 | |||
333 | endmenu # "ARC CPU Configuration" | ||
334 | |||
335 | config LINUX_LINK_BASE | ||
336 | hex "Linux Link Address" | ||
337 | default "0x80000000" | ||
338 | help | ||
339 | ARC700 divides the 32 bit phy address space into two equal halves | ||
340 | -Lower 2G (0 - 0x7FFF_FFFF ) is user virtual, translated by MMU | ||
341 | -Upper 2G (0x8000_0000 onwards) is untranslated, for kernel | ||
342 | Typically Linux kernel is linked at the start of untransalted addr, | ||
343 | hence the default value of 0x8zs. | ||
344 | However some customers have peripherals mapped at this addr, so | ||
345 | Linux needs to be scooted a bit. | ||
346 | If you don't know what the above means, leave this setting alone. | ||
347 | |||
348 | config ARC_CURR_IN_REG | ||
349 | bool "Dedicate Register r25 for current_task pointer" | ||
350 | default y | ||
351 | help | ||
352 | This reserved Register R25 to point to Current Task in | ||
353 | kernel mode. This saves memory access for each such access | ||
354 | |||
355 | |||
356 | config ARC_MISALIGN_ACCESS | ||
357 | bool "Emulate unaligned memory access (userspace only)" | ||
358 | default N | ||
359 | select SYSCTL_ARCH_UNALIGN_NO_WARN | ||
360 | select SYSCTL_ARCH_UNALIGN_ALLOW | ||
361 | help | ||
362 | This enables misaligned 16 & 32 bit memory access from user space. | ||
363 | Use ONLY-IF-ABS-NECESSARY as it will be very slow and also can hide | ||
364 | potential bugs in code | ||
365 | |||
366 | config ARC_STACK_NONEXEC | ||
367 | bool "Make stack non-executable" | ||
368 | default n | ||
369 | help | ||
370 | To disable the execute permissions of stack/heap of processes | ||
371 | which are enabled by default. | ||
372 | |||
373 | config HZ | ||
374 | int "Timer Frequency" | ||
375 | default 100 | ||
376 | |||
377 | config ARC_METAWARE_HLINK | ||
378 | bool "Support for Metaware debugger assisted Host access" | ||
379 | default n | ||
380 | help | ||
381 | This options allows a Linux userland apps to directly access | ||
382 | host file system (open/creat/read/write etc) with help from | ||
383 | Metaware Debugger. This can come in handy for Linux-host communication | ||
384 | when there is no real usable peripheral such as EMAC. | ||
385 | |||
386 | menuconfig ARC_DBG | ||
387 | bool "ARC debugging" | ||
388 | default y | ||
389 | |||
390 | config ARC_DW2_UNWIND | ||
391 | bool "Enable DWARF specific kernel stack unwind" | ||
392 | depends on ARC_DBG | ||
393 | default y | ||
394 | select KALLSYMS | ||
395 | help | ||
396 | Compiles the kernel with DWARF unwind information and can be used | ||
397 | to get stack backtraces. | ||
398 | |||
399 | If you say Y here the resulting kernel image will be slightly larger | ||
400 | but not slower, and it will give very useful debugging information. | ||
401 | If you don't debug the kernel, you can say N, but we may not be able | ||
402 | to solve problems without frame unwind information | ||
403 | |||
404 | config ARC_DBG_TLB_PARANOIA | ||
405 | bool "Paranoia Checks in Low Level TLB Handlers" | ||
406 | depends on ARC_DBG | ||
407 | default n | ||
408 | |||
409 | config ARC_DBG_TLB_MISS_COUNT | ||
410 | bool "Profile TLB Misses" | ||
411 | default n | ||
412 | select DEBUG_FS | ||
413 | depends on ARC_DBG | ||
414 | help | ||
415 | Counts number of I and D TLB Misses and exports them via Debugfs | ||
416 | The counters can be cleared via Debugfs as well | ||
417 | |||
418 | config CMDLINE | ||
419 | string "Kernel command line to built-in" | ||
420 | default "print-fatal-signals=1" | ||
421 | help | ||
422 | The default command line which will be appended to the optional | ||
423 | u-boot provided command line (see below) | ||
424 | |||
425 | config CMDLINE_UBOOT | ||
426 | bool "Support U-boot kernel command line passing" | ||
427 | default n | ||
428 | help | ||
429 | If you are using U-boot (www.denx.de) and wish to pass the kernel | ||
430 | command line from the U-boot environment to the Linux kernel then | ||
431 | switch this option on. | ||
432 | ARC U-boot will setup the cmdline in RAM/flash and set r2 to point | ||
433 | to it. kernel startup code will copy the string into cmdline buffer | ||
434 | and also append CONFIG_CMDLINE. | ||
435 | |||
436 | config ARC_BUILTIN_DTB_NAME | ||
437 | string "Built in DTB" | ||
438 | help | ||
439 | Set the name of the DTB to embed in the vmlinux binary | ||
440 | Leaving it blank selects the minimal "skeleton" dtb | ||
441 | |||
442 | source "kernel/Kconfig.preempt" | ||
443 | |||
444 | endmenu # "ARC Architecture Configuration" | ||
445 | |||
446 | source "mm/Kconfig" | ||
447 | source "net/Kconfig" | ||
448 | source "drivers/Kconfig" | ||
449 | source "fs/Kconfig" | ||
450 | source "arch/arc/Kconfig.debug" | ||
451 | source "security/Kconfig" | ||
452 | source "crypto/Kconfig" | ||
453 | source "lib/Kconfig" | ||
diff --git a/arch/arc/Kconfig.debug b/arch/arc/Kconfig.debug new file mode 100644 index 000000000000..962c6099659e --- /dev/null +++ b/arch/arc/Kconfig.debug | |||
@@ -0,0 +1,34 @@ | |||
1 | menu "Kernel hacking" | ||
2 | |||
3 | source "lib/Kconfig.debug" | ||
4 | |||
5 | config EARLY_PRINTK | ||
6 | bool "Early printk" if EMBEDDED | ||
7 | default y | ||
8 | help | ||
9 | Write kernel log output directly into the VGA buffer or to a serial | ||
10 | port. | ||
11 | |||
12 | This is useful for kernel debugging when your machine crashes very | ||
13 | early before the console code is initialized. For normal operation | ||
14 | it is not recommended because it looks ugly and doesn't cooperate | ||
15 | with klogd/syslogd or the X server. You should normally N here, | ||
16 | unless you want to debug such a crash. | ||
17 | |||
18 | config DEBUG_STACKOVERFLOW | ||
19 | bool "Check for stack overflows" | ||
20 | depends on DEBUG_KERNEL | ||
21 | help | ||
22 | This option will cause messages to be printed if free stack space | ||
23 | drops below a certain limit. | ||
24 | |||
25 | config 16KSTACKS | ||
26 | bool "Use 16Kb for kernel stacks instead of 8Kb" | ||
27 | help | ||
28 | If you say Y here the kernel will use a 16Kb stacksize for the | ||
29 | kernel stack attached to each process/thread. The default is 8K. | ||
30 | This increases the resident kernel footprint and will cause less | ||
31 | threads to run on the system and also increase the pressure | ||
32 | on the VM subsystem for higher order allocations. | ||
33 | |||
34 | endmenu | ||
diff --git a/arch/arc/Makefile b/arch/arc/Makefile new file mode 100644 index 000000000000..92379c7cbc1a --- /dev/null +++ b/arch/arc/Makefile | |||
@@ -0,0 +1,126 @@ | |||
1 | # | ||
2 | # Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | # | ||
4 | # This program is free software; you can redistribute it and/or modify | ||
5 | # it under the terms of the GNU General Public License version 2 as | ||
6 | # published by the Free Software Foundation. | ||
7 | # | ||
8 | |||
9 | UTS_MACHINE := arc | ||
10 | |||
11 | KBUILD_DEFCONFIG := fpga_defconfig | ||
12 | |||
13 | cflags-y += -mA7 -fno-common -pipe -fno-builtin -D__linux__ | ||
14 | |||
15 | LINUXINCLUDE += -include ${src}/arch/arc/include/asm/defines.h | ||
16 | |||
17 | ifdef CONFIG_ARC_CURR_IN_REG | ||
18 | # For a global register defintion, make sure it gets passed to every file | ||
19 | # We had a customer reported bug where some code built in kernel was NOT using | ||
20 | # any kernel headers, and missing the r25 global register | ||
21 | # Can't do unconditionally (like above) because of recursive include issues | ||
22 | # due to <linux/thread_info.h> | ||
23 | LINUXINCLUDE += -include ${src}/arch/arc/include/asm/current.h | ||
24 | endif | ||
25 | |||
26 | atleast_gcc44 := $(call cc-ifversion, -gt, 0402, y) | ||
27 | cflags-$(atleast_gcc44) += -fsection-anchors | ||
28 | |||
29 | cflags-$(CONFIG_ARC_HAS_LLSC) += -mlock | ||
30 | cflags-$(CONFIG_ARC_HAS_SWAPE) += -mswape | ||
31 | cflags-$(CONFIG_ARC_HAS_RTSC) += -mrtsc | ||
32 | cflags-$(CONFIG_ARC_DW2_UNWIND) += -fasynchronous-unwind-tables | ||
33 | |||
34 | ifndef CONFIG_CC_OPTIMIZE_FOR_SIZE | ||
35 | # Generic build system uses -O2, we want -O3 | ||
36 | cflags-y += -O3 | ||
37 | endif | ||
38 | |||
39 | # small data is default for elf32 tool-chain. If not usable, disable it | ||
40 | # This also allows repurposing GP as scratch reg to gcc reg allocator | ||
41 | disable_small_data := y | ||
42 | cflags-$(disable_small_data) += -mno-sdata -fcall-used-gp | ||
43 | |||
44 | cflags-$(CONFIG_CPU_BIG_ENDIAN) += -mbig-endian | ||
45 | ldflags-$(CONFIG_CPU_BIG_ENDIAN) += -EB | ||
46 | |||
47 | # STAR 9000518362: | ||
48 | # arc-linux-uclibc-ld (buildroot) or arceb-elf32-ld (EZChip) don't accept | ||
49 | # --build-id w/o "-marclinux". | ||
50 | # Default arc-elf32-ld is OK | ||
51 | ldflags-y += -marclinux | ||
52 | |||
53 | ARC_LIBGCC := -mA7 | ||
54 | cflags-$(CONFIG_ARC_HAS_HW_MPY) += -multcost=16 | ||
55 | |||
56 | ifndef CONFIG_ARC_HAS_HW_MPY | ||
57 | cflags-y += -mno-mpy | ||
58 | |||
59 | # newlib for ARC700 assumes MPY to be always present, which is generally true | ||
60 | # However, if someone really doesn't want MPY, we need to use the 600 ver | ||
61 | # which coupled with -mno-mpy will use mpy emulation | ||
62 | # With gcc 4.4.7, -mno-mpy is enough to make any other related adjustments, | ||
63 | # e.g. increased cost of MPY. With gcc 4.2.1 this had to be explicitly hinted | ||
64 | |||
65 | ARC_LIBGCC := -marc600 | ||
66 | ifneq ($(atleast_gcc44),y) | ||
67 | cflags-y += -multcost=30 | ||
68 | endif | ||
69 | endif | ||
70 | |||
71 | LIBGCC := $(shell $(CC) $(ARC_LIBGCC) $(cflags-y) --print-libgcc-file-name) | ||
72 | |||
73 | # Modules with short calls might break for calls into builtin-kernel | ||
74 | KBUILD_CFLAGS_MODULE += -mlong-calls | ||
75 | |||
76 | # Finally dump eveything into kernel build system | ||
77 | KBUILD_CFLAGS += $(cflags-y) | ||
78 | KBUILD_AFLAGS += $(KBUILD_CFLAGS) | ||
79 | LDFLAGS += $(ldflags-y) | ||
80 | |||
81 | head-y := arch/arc/kernel/head.o | ||
82 | |||
83 | # See arch/arc/Kbuild for content of core part of the kernel | ||
84 | core-y += arch/arc/ | ||
85 | |||
86 | # w/o this dtb won't embed into kernel binary | ||
87 | core-y += arch/arc/boot/dts/ | ||
88 | |||
89 | core-$(CONFIG_ARC_PLAT_FPGA_LEGACY) += arch/arc/plat-arcfpga/ | ||
90 | |||
91 | drivers-$(CONFIG_OPROFILE) += arch/arc/oprofile/ | ||
92 | |||
93 | libs-y += arch/arc/lib/ $(LIBGCC) | ||
94 | |||
95 | #default target for make without any arguements. | ||
96 | KBUILD_IMAGE := bootpImage | ||
97 | |||
98 | all: $(KBUILD_IMAGE) | ||
99 | boot := arch/arc/boot | ||
100 | |||
101 | bootpImage: vmlinux | ||
102 | |||
103 | uImage: vmlinux | ||
104 | $(Q)$(MAKE) $(build)=$(boot) $(boot)/$@ | ||
105 | |||
106 | %.dtb %.dtb.S %.dtb.o: scripts | ||
107 | $(Q)$(MAKE) $(build)=$(boot)/dts $(boot)/dts/$@ | ||
108 | |||
109 | dtbs: scripts | ||
110 | $(Q)$(MAKE) $(build)=$(boot)/dts dtbs | ||
111 | |||
112 | archclean: | ||
113 | $(Q)$(MAKE) $(clean)=$(boot) | ||
114 | |||
115 | # Hacks to enable final link due to absence of link-time branch relexation | ||
116 | # and gcc choosing optimal(shorter) branches at -O3 | ||
117 | # | ||
118 | # vineetg Feb 2010: -mlong-calls switched off for overall kernel build | ||
119 | # However lib/decompress_inflate.o (.init.text) calls | ||
120 | # zlib_inflate_workspacesize (.text) causing relocation errors. | ||
121 | # Thus forcing all exten calls in this file to be long calls | ||
122 | export CFLAGS_decompress_inflate.o = -mmedium-calls | ||
123 | export CFLAGS_initramfs.o = -mmedium-calls | ||
124 | ifdef CONFIG_SMP | ||
125 | export CFLAGS_core.o = -mmedium-calls | ||
126 | endif | ||
diff --git a/arch/arc/boot/Makefile b/arch/arc/boot/Makefile new file mode 100644 index 000000000000..7d514c24e095 --- /dev/null +++ b/arch/arc/boot/Makefile | |||
@@ -0,0 +1,26 @@ | |||
1 | targets := vmlinux.bin vmlinux.bin.gz uImage | ||
2 | |||
3 | # uImage build relies on mkimage being availble on your host for ARC target | ||
4 | # You will need to build u-boot for ARC, rename mkimage to arc-elf32-mkimage | ||
5 | # and make sure it's reacable from your PATH | ||
6 | MKIMAGE := $(srctree)/scripts/mkuboot.sh | ||
7 | |||
8 | OBJCOPYFLAGS= -O binary -R .note -R .note.gnu.build-id -R .comment -S | ||
9 | |||
10 | LINUX_START_TEXT = $$(readelf -h vmlinux | \ | ||
11 | grep "Entry point address" | grep -o 0x.*) | ||
12 | |||
13 | UIMAGE_LOADADDR = $(CONFIG_LINUX_LINK_BASE) | ||
14 | UIMAGE_ENTRYADDR = $(LINUX_START_TEXT) | ||
15 | UIMAGE_COMPRESSION = gzip | ||
16 | |||
17 | $(obj)/vmlinux.bin: vmlinux FORCE | ||
18 | $(call if_changed,objcopy) | ||
19 | |||
20 | $(obj)/vmlinux.bin.gz: $(obj)/vmlinux.bin FORCE | ||
21 | $(call if_changed,gzip) | ||
22 | |||
23 | $(obj)/uImage: $(obj)/vmlinux.bin.gz FORCE | ||
24 | $(call if_changed,uimage) | ||
25 | |||
26 | PHONY += FORCE | ||
diff --git a/arch/arc/boot/dts/Makefile b/arch/arc/boot/dts/Makefile new file mode 100644 index 000000000000..5776835d583f --- /dev/null +++ b/arch/arc/boot/dts/Makefile | |||
@@ -0,0 +1,13 @@ | |||
1 | # Built-in dtb | ||
2 | builtindtb-y := angel4 | ||
3 | |||
4 | ifneq ($(CONFIG_ARC_BUILTIN_DTB_NAME),"") | ||
5 | builtindtb-y := $(patsubst "%",%,$(CONFIG_ARC_BUILTIN_DTB_NAME)) | ||
6 | endif | ||
7 | |||
8 | obj-y += $(builtindtb-y).dtb.o | ||
9 | targets += $(builtindtb-y).dtb | ||
10 | |||
11 | dtbs: $(addprefix $(obj)/, $(builtindtb-y).dtb) | ||
12 | |||
13 | clean-files := *.dtb | ||
diff --git a/arch/arc/boot/dts/angel4.dts b/arch/arc/boot/dts/angel4.dts new file mode 100644 index 000000000000..bae4f936cb03 --- /dev/null +++ b/arch/arc/boot/dts/angel4.dts | |||
@@ -0,0 +1,55 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | /dts-v1/; | ||
9 | |||
10 | /include/ "skeleton.dtsi" | ||
11 | |||
12 | / { | ||
13 | compatible = "snps,arc-angel4"; | ||
14 | clock-frequency = <80000000>; /* 80 MHZ */ | ||
15 | #address-cells = <1>; | ||
16 | #size-cells = <1>; | ||
17 | interrupt-parent = <&intc>; | ||
18 | |||
19 | chosen { | ||
20 | bootargs = "console=ttyARC0,115200n8"; | ||
21 | }; | ||
22 | |||
23 | aliases { | ||
24 | serial0 = &arcuart0; | ||
25 | }; | ||
26 | |||
27 | memory { | ||
28 | device_type = "memory"; | ||
29 | reg = <0x00000000 0x10000000>; /* 256M */ | ||
30 | }; | ||
31 | |||
32 | fpga { | ||
33 | compatible = "simple-bus"; | ||
34 | #address-cells = <1>; | ||
35 | #size-cells = <1>; | ||
36 | |||
37 | /* child and parent address space 1:1 mapped */ | ||
38 | ranges; | ||
39 | |||
40 | intc: interrupt-controller { | ||
41 | compatible = "snps,arc700-intc"; | ||
42 | interrupt-controller; | ||
43 | #interrupt-cells = <1>; | ||
44 | }; | ||
45 | |||
46 | arcuart0: serial@c0fc1000 { | ||
47 | compatible = "snps,arc-uart"; | ||
48 | reg = <0xc0fc1000 0x100>; | ||
49 | interrupts = <5>; | ||
50 | clock-frequency = <80000000>; | ||
51 | current-speed = <115200>; | ||
52 | status = "okay"; | ||
53 | }; | ||
54 | }; | ||
55 | }; | ||
diff --git a/arch/arc/boot/dts/skeleton.dts b/arch/arc/boot/dts/skeleton.dts new file mode 100644 index 000000000000..25a84fb5b3dc --- /dev/null +++ b/arch/arc/boot/dts/skeleton.dts | |||
@@ -0,0 +1,10 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | /dts-v1/; | ||
9 | |||
10 | /include/ "skeleton.dtsi" | ||
diff --git a/arch/arc/boot/dts/skeleton.dtsi b/arch/arc/boot/dts/skeleton.dtsi new file mode 100644 index 000000000000..a870bdd5e404 --- /dev/null +++ b/arch/arc/boot/dts/skeleton.dtsi | |||
@@ -0,0 +1,37 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | /* | ||
10 | * Skeleton device tree; the bare minimum needed to boot; just include and | ||
11 | * add a compatible value. | ||
12 | */ | ||
13 | |||
14 | / { | ||
15 | compatible = "snps,arc"; | ||
16 | clock-frequency = <80000000>; /* 80 MHZ */ | ||
17 | #address-cells = <1>; | ||
18 | #size-cells = <1>; | ||
19 | chosen { }; | ||
20 | aliases { }; | ||
21 | |||
22 | cpus { | ||
23 | #address-cells = <1>; | ||
24 | #size-cells = <0>; | ||
25 | |||
26 | cpu@0 { | ||
27 | device_type = "cpu"; | ||
28 | compatible = "snps,arc770d"; | ||
29 | reg = <0>; | ||
30 | }; | ||
31 | }; | ||
32 | |||
33 | memory { | ||
34 | device_type = "memory"; | ||
35 | reg = <0x00000000 0x10000000>; /* 256M */ | ||
36 | }; | ||
37 | }; | ||
diff --git a/arch/arc/configs/fpga_defconfig b/arch/arc/configs/fpga_defconfig new file mode 100644 index 000000000000..b8698067ebbe --- /dev/null +++ b/arch/arc/configs/fpga_defconfig | |||
@@ -0,0 +1,61 @@ | |||
1 | CONFIG_CROSS_COMPILE="arc-elf32-" | ||
2 | # CONFIG_LOCALVERSION_AUTO is not set | ||
3 | CONFIG_DEFAULT_HOSTNAME="ARCLinux" | ||
4 | # CONFIG_SWAP is not set | ||
5 | CONFIG_HIGH_RES_TIMERS=y | ||
6 | CONFIG_IKCONFIG=y | ||
7 | CONFIG_IKCONFIG_PROC=y | ||
8 | CONFIG_NAMESPACES=y | ||
9 | # CONFIG_UTS_NS is not set | ||
10 | # CONFIG_PID_NS is not set | ||
11 | CONFIG_BLK_DEV_INITRD=y | ||
12 | CONFIG_INITRAMFS_SOURCE="../arc_initramfs" | ||
13 | CONFIG_KALLSYMS_ALL=y | ||
14 | CONFIG_EMBEDDED=y | ||
15 | # CONFIG_SLUB_DEBUG is not set | ||
16 | # CONFIG_COMPAT_BRK is not set | ||
17 | CONFIG_KPROBES=y | ||
18 | CONFIG_MODULES=y | ||
19 | # CONFIG_LBDAF is not set | ||
20 | # CONFIG_BLK_DEV_BSG is not set | ||
21 | # CONFIG_IOSCHED_DEADLINE is not set | ||
22 | # CONFIG_IOSCHED_CFQ is not set | ||
23 | CONFIG_ARC_PLAT_FPGA_LEGACY=y | ||
24 | CONFIG_ARC_BOARD_ML509=y | ||
25 | # CONFIG_ARC_HAS_RTSC is not set | ||
26 | CONFIG_ARC_BUILTIN_DTB_NAME="angel4" | ||
27 | # CONFIG_COMPACTION is not set | ||
28 | # CONFIG_CROSS_MEMORY_ATTACH is not set | ||
29 | CONFIG_NET=y | ||
30 | CONFIG_PACKET=y | ||
31 | CONFIG_UNIX=y | ||
32 | CONFIG_UNIX_DIAG=y | ||
33 | CONFIG_NET_KEY=y | ||
34 | CONFIG_INET=y | ||
35 | # CONFIG_IPV6 is not set | ||
36 | # CONFIG_STANDALONE is not set | ||
37 | # CONFIG_PREVENT_FIRMWARE_BUILD is not set | ||
38 | # CONFIG_FIRMWARE_IN_KERNEL is not set | ||
39 | # CONFIG_BLK_DEV is not set | ||
40 | # CONFIG_INPUT_MOUSEDEV_PSAUX is not set | ||
41 | # CONFIG_INPUT_KEYBOARD is not set | ||
42 | # CONFIG_INPUT_MOUSE is not set | ||
43 | # CONFIG_SERIO is not set | ||
44 | # CONFIG_LEGACY_PTYS is not set | ||
45 | # CONFIG_DEVKMEM is not set | ||
46 | CONFIG_SERIAL_ARC=y | ||
47 | CONFIG_SERIAL_ARC_CONSOLE=y | ||
48 | # CONFIG_HW_RANDOM is not set | ||
49 | # CONFIG_HWMON is not set | ||
50 | # CONFIG_VGA_CONSOLE is not set | ||
51 | # CONFIG_HID is not set | ||
52 | # CONFIG_USB_SUPPORT is not set | ||
53 | # CONFIG_IOMMU_SUPPORT is not set | ||
54 | CONFIG_EXT2_FS=y | ||
55 | CONFIG_EXT2_FS_XATTR=y | ||
56 | CONFIG_TMPFS=y | ||
57 | # CONFIG_MISC_FILESYSTEMS is not set | ||
58 | CONFIG_NFS_FS=y | ||
59 | # CONFIG_ENABLE_WARN_DEPRECATED is not set | ||
60 | # CONFIG_ENABLE_MUST_CHECK is not set | ||
61 | CONFIG_XZ_DEC=y | ||
diff --git a/arch/arc/include/asm/Kbuild b/arch/arc/include/asm/Kbuild new file mode 100644 index 000000000000..48af742f8b5a --- /dev/null +++ b/arch/arc/include/asm/Kbuild | |||
@@ -0,0 +1,49 @@ | |||
1 | generic-y += auxvec.h | ||
2 | generic-y += bugs.h | ||
3 | generic-y += bitsperlong.h | ||
4 | generic-y += clkdev.h | ||
5 | generic-y += cputime.h | ||
6 | generic-y += device.h | ||
7 | generic-y += div64.h | ||
8 | generic-y += emergency-restart.h | ||
9 | generic-y += errno.h | ||
10 | generic-y += fcntl.h | ||
11 | generic-y += fb.h | ||
12 | generic-y += ftrace.h | ||
13 | generic-y += hardirq.h | ||
14 | generic-y += hw_irq.h | ||
15 | generic-y += ioctl.h | ||
16 | generic-y += ioctls.h | ||
17 | generic-y += ipcbuf.h | ||
18 | generic-y += irq_regs.h | ||
19 | generic-y += kmap_types.h | ||
20 | generic-y += kvm_para.h | ||
21 | generic-y += local.h | ||
22 | generic-y += local64.h | ||
23 | generic-y += mman.h | ||
24 | generic-y += msgbuf.h | ||
25 | generic-y += param.h | ||
26 | generic-y += parport.h | ||
27 | generic-y += pci.h | ||
28 | generic-y += percpu.h | ||
29 | generic-y += poll.h | ||
30 | generic-y += posix_types.h | ||
31 | generic-y += resource.h | ||
32 | generic-y += scatterlist.h | ||
33 | generic-y += sembuf.h | ||
34 | generic-y += shmbuf.h | ||
35 | generic-y += shmparam.h | ||
36 | generic-y += siginfo.h | ||
37 | generic-y += socket.h | ||
38 | generic-y += sockios.h | ||
39 | generic-y += stat.h | ||
40 | generic-y += statfs.h | ||
41 | generic-y += termbits.h | ||
42 | generic-y += termios.h | ||
43 | generic-y += topology.h | ||
44 | generic-y += trace_clock.h | ||
45 | generic-y += types.h | ||
46 | generic-y += ucontext.h | ||
47 | generic-y += user.h | ||
48 | generic-y += vga.h | ||
49 | generic-y += xor.h | ||
diff --git a/arch/arc/include/asm/arcregs.h b/arch/arc/include/asm/arcregs.h new file mode 100644 index 000000000000..1b907c465666 --- /dev/null +++ b/arch/arc/include/asm/arcregs.h | |||
@@ -0,0 +1,433 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_ARCREGS_H | ||
10 | #define _ASM_ARC_ARCREGS_H | ||
11 | |||
12 | #ifdef __KERNEL__ | ||
13 | |||
14 | /* Build Configuration Registers */ | ||
15 | #define ARC_REG_DCCMBASE_BCR 0x61 /* DCCM Base Addr */ | ||
16 | #define ARC_REG_CRC_BCR 0x62 | ||
17 | #define ARC_REG_DVFB_BCR 0x64 | ||
18 | #define ARC_REG_EXTARITH_BCR 0x65 | ||
19 | #define ARC_REG_VECBASE_BCR 0x68 | ||
20 | #define ARC_REG_PERIBASE_BCR 0x69 | ||
21 | #define ARC_REG_FP_BCR 0x6B /* Single-Precision FPU */ | ||
22 | #define ARC_REG_DPFP_BCR 0x6C /* Dbl Precision FPU */ | ||
23 | #define ARC_REG_MMU_BCR 0x6f | ||
24 | #define ARC_REG_DCCM_BCR 0x74 /* DCCM Present + SZ */ | ||
25 | #define ARC_REG_TIMERS_BCR 0x75 | ||
26 | #define ARC_REG_ICCM_BCR 0x78 | ||
27 | #define ARC_REG_XY_MEM_BCR 0x79 | ||
28 | #define ARC_REG_MAC_BCR 0x7a | ||
29 | #define ARC_REG_MUL_BCR 0x7b | ||
30 | #define ARC_REG_SWAP_BCR 0x7c | ||
31 | #define ARC_REG_NORM_BCR 0x7d | ||
32 | #define ARC_REG_MIXMAX_BCR 0x7e | ||
33 | #define ARC_REG_BARREL_BCR 0x7f | ||
34 | #define ARC_REG_D_UNCACH_BCR 0x6A | ||
35 | |||
36 | /* status32 Bits Positions */ | ||
37 | #define STATUS_H_BIT 0 /* CPU Halted */ | ||
38 | #define STATUS_E1_BIT 1 /* Int 1 enable */ | ||
39 | #define STATUS_E2_BIT 2 /* Int 2 enable */ | ||
40 | #define STATUS_A1_BIT 3 /* Int 1 active */ | ||
41 | #define STATUS_A2_BIT 4 /* Int 2 active */ | ||
42 | #define STATUS_AE_BIT 5 /* Exception active */ | ||
43 | #define STATUS_DE_BIT 6 /* PC is in delay slot */ | ||
44 | #define STATUS_U_BIT 7 /* User/Kernel mode */ | ||
45 | #define STATUS_L_BIT 12 /* Loop inhibit */ | ||
46 | |||
47 | /* These masks correspond to the status word(STATUS_32) bits */ | ||
48 | #define STATUS_H_MASK (1<<STATUS_H_BIT) | ||
49 | #define STATUS_E1_MASK (1<<STATUS_E1_BIT) | ||
50 | #define STATUS_E2_MASK (1<<STATUS_E2_BIT) | ||
51 | #define STATUS_A1_MASK (1<<STATUS_A1_BIT) | ||
52 | #define STATUS_A2_MASK (1<<STATUS_A2_BIT) | ||
53 | #define STATUS_AE_MASK (1<<STATUS_AE_BIT) | ||
54 | #define STATUS_DE_MASK (1<<STATUS_DE_BIT) | ||
55 | #define STATUS_U_MASK (1<<STATUS_U_BIT) | ||
56 | #define STATUS_L_MASK (1<<STATUS_L_BIT) | ||
57 | |||
58 | /* | ||
59 | * ECR: Exception Cause Reg bits-n-pieces | ||
60 | * [23:16] = Exception Vector | ||
61 | * [15: 8] = Exception Cause Code | ||
62 | * [ 7: 0] = Exception Parameters (for certain types only) | ||
63 | */ | ||
64 | #define ECR_VEC_MASK 0xff0000 | ||
65 | #define ECR_CODE_MASK 0x00ff00 | ||
66 | #define ECR_PARAM_MASK 0x0000ff | ||
67 | |||
68 | /* Exception Cause Vector Values */ | ||
69 | #define ECR_V_INSN_ERR 0x02 | ||
70 | #define ECR_V_MACH_CHK 0x20 | ||
71 | #define ECR_V_ITLB_MISS 0x21 | ||
72 | #define ECR_V_DTLB_MISS 0x22 | ||
73 | #define ECR_V_PROTV 0x23 | ||
74 | |||
75 | /* Protection Violation Exception Cause Code Values */ | ||
76 | #define ECR_C_PROTV_INST_FETCH 0x00 | ||
77 | #define ECR_C_PROTV_LOAD 0x01 | ||
78 | #define ECR_C_PROTV_STORE 0x02 | ||
79 | #define ECR_C_PROTV_XCHG 0x03 | ||
80 | #define ECR_C_PROTV_MISALIG_DATA 0x04 | ||
81 | |||
82 | /* DTLB Miss Exception Cause Code Values */ | ||
83 | #define ECR_C_BIT_DTLB_LD_MISS 8 | ||
84 | #define ECR_C_BIT_DTLB_ST_MISS 9 | ||
85 | |||
86 | |||
87 | /* Auxiliary registers */ | ||
88 | #define AUX_IDENTITY 4 | ||
89 | #define AUX_INTR_VEC_BASE 0x25 | ||
90 | #define AUX_IRQ_LEV 0x200 /* IRQ Priority: L1 or L2 */ | ||
91 | #define AUX_IRQ_HINT 0x201 /* For generating Soft Interrupts */ | ||
92 | #define AUX_IRQ_LV12 0x43 /* interrupt level register */ | ||
93 | |||
94 | #define AUX_IENABLE 0x40c | ||
95 | #define AUX_ITRIGGER 0x40d | ||
96 | #define AUX_IPULSE 0x415 | ||
97 | |||
98 | /* Timer related Aux registers */ | ||
99 | #define ARC_REG_TIMER0_LIMIT 0x23 /* timer 0 limit */ | ||
100 | #define ARC_REG_TIMER0_CTRL 0x22 /* timer 0 control */ | ||
101 | #define ARC_REG_TIMER0_CNT 0x21 /* timer 0 count */ | ||
102 | #define ARC_REG_TIMER1_LIMIT 0x102 /* timer 1 limit */ | ||
103 | #define ARC_REG_TIMER1_CTRL 0x101 /* timer 1 control */ | ||
104 | #define ARC_REG_TIMER1_CNT 0x100 /* timer 1 count */ | ||
105 | |||
106 | #define TIMER_CTRL_IE (1 << 0) /* Interupt when Count reachs limit */ | ||
107 | #define TIMER_CTRL_NH (1 << 1) /* Count only when CPU NOT halted */ | ||
108 | |||
109 | /* MMU Management regs */ | ||
110 | #define ARC_REG_TLBPD0 0x405 | ||
111 | #define ARC_REG_TLBPD1 0x406 | ||
112 | #define ARC_REG_TLBINDEX 0x407 | ||
113 | #define ARC_REG_TLBCOMMAND 0x408 | ||
114 | #define ARC_REG_PID 0x409 | ||
115 | #define ARC_REG_SCRATCH_DATA0 0x418 | ||
116 | |||
117 | /* Bits in MMU PID register */ | ||
118 | #define MMU_ENABLE (1 << 31) /* Enable MMU for process */ | ||
119 | |||
120 | /* Error code if probe fails */ | ||
121 | #define TLB_LKUP_ERR 0x80000000 | ||
122 | |||
123 | /* TLB Commands */ | ||
124 | #define TLBWrite 0x1 | ||
125 | #define TLBRead 0x2 | ||
126 | #define TLBGetIndex 0x3 | ||
127 | #define TLBProbe 0x4 | ||
128 | |||
129 | #if (CONFIG_ARC_MMU_VER >= 2) | ||
130 | #define TLBWriteNI 0x5 /* write JTLB without inv uTLBs */ | ||
131 | #define TLBIVUTLB 0x6 /* explicitly inv uTLBs */ | ||
132 | #else | ||
133 | #undef TLBWriteNI /* These cmds don't exist on older MMU */ | ||
134 | #undef TLBIVUTLB | ||
135 | #endif | ||
136 | |||
137 | /* Instruction cache related Auxiliary registers */ | ||
138 | #define ARC_REG_IC_BCR 0x77 /* Build Config reg */ | ||
139 | #define ARC_REG_IC_IVIC 0x10 | ||
140 | #define ARC_REG_IC_CTRL 0x11 | ||
141 | #define ARC_REG_IC_IVIL 0x19 | ||
142 | #if (CONFIG_ARC_MMU_VER > 2) | ||
143 | #define ARC_REG_IC_PTAG 0x1E | ||
144 | #endif | ||
145 | |||
146 | /* Bit val in IC_CTRL */ | ||
147 | #define IC_CTRL_CACHE_DISABLE 0x1 | ||
148 | |||
149 | /* Data cache related Auxiliary registers */ | ||
150 | #define ARC_REG_DC_BCR 0x72 | ||
151 | #define ARC_REG_DC_IVDC 0x47 | ||
152 | #define ARC_REG_DC_CTRL 0x48 | ||
153 | #define ARC_REG_DC_IVDL 0x4A | ||
154 | #define ARC_REG_DC_FLSH 0x4B | ||
155 | #define ARC_REG_DC_FLDL 0x4C | ||
156 | #if (CONFIG_ARC_MMU_VER > 2) | ||
157 | #define ARC_REG_DC_PTAG 0x5C | ||
158 | #endif | ||
159 | |||
160 | /* Bit val in DC_CTRL */ | ||
161 | #define DC_CTRL_INV_MODE_FLUSH 0x40 | ||
162 | #define DC_CTRL_FLUSH_STATUS 0x100 | ||
163 | |||
164 | /* MMU Management regs */ | ||
165 | #define ARC_REG_PID 0x409 | ||
166 | #define ARC_REG_SCRATCH_DATA0 0x418 | ||
167 | |||
168 | /* Bits in MMU PID register */ | ||
169 | #define MMU_ENABLE (1 << 31) /* Enable MMU for process */ | ||
170 | |||
171 | /* | ||
172 | * Floating Pt Registers | ||
173 | * Status regs are read-only (build-time) so need not be saved/restored | ||
174 | */ | ||
175 | #define ARC_AUX_FP_STAT 0x300 | ||
176 | #define ARC_AUX_DPFP_1L 0x301 | ||
177 | #define ARC_AUX_DPFP_1H 0x302 | ||
178 | #define ARC_AUX_DPFP_2L 0x303 | ||
179 | #define ARC_AUX_DPFP_2H 0x304 | ||
180 | #define ARC_AUX_DPFP_STAT 0x305 | ||
181 | |||
182 | #ifndef __ASSEMBLY__ | ||
183 | |||
184 | /* | ||
185 | ****************************************************************** | ||
186 | * Inline ASM macros to read/write AUX Regs | ||
187 | * Essentially invocation of lr/sr insns from "C" | ||
188 | */ | ||
189 | |||
190 | #if 1 | ||
191 | |||
192 | #define read_aux_reg(reg) __builtin_arc_lr(reg) | ||
193 | |||
194 | /* gcc builtin sr needs reg param to be long immediate */ | ||
195 | #define write_aux_reg(reg_immed, val) \ | ||
196 | __builtin_arc_sr((unsigned int)val, reg_immed) | ||
197 | |||
198 | #else | ||
199 | |||
200 | #define read_aux_reg(reg) \ | ||
201 | ({ \ | ||
202 | unsigned int __ret; \ | ||
203 | __asm__ __volatile__( \ | ||
204 | " lr %0, [%1]" \ | ||
205 | : "=r"(__ret) \ | ||
206 | : "i"(reg)); \ | ||
207 | __ret; \ | ||
208 | }) | ||
209 | |||
210 | /* | ||
211 | * Aux Reg address is specified as long immediate by caller | ||
212 | * e.g. | ||
213 | * write_aux_reg(0x69, some_val); | ||
214 | * This generates tightest code. | ||
215 | */ | ||
216 | #define write_aux_reg(reg_imm, val) \ | ||
217 | ({ \ | ||
218 | __asm__ __volatile__( \ | ||
219 | " sr %0, [%1] \n" \ | ||
220 | : \ | ||
221 | : "ir"(val), "i"(reg_imm)); \ | ||
222 | }) | ||
223 | |||
224 | /* | ||
225 | * Aux Reg address is specified in a variable | ||
226 | * * e.g. | ||
227 | * reg_num = 0x69 | ||
228 | * write_aux_reg2(reg_num, some_val); | ||
229 | * This has to generate glue code to load the reg num from | ||
230 | * memory to a reg hence not recommended. | ||
231 | */ | ||
232 | #define write_aux_reg2(reg_in_var, val) \ | ||
233 | ({ \ | ||
234 | unsigned int tmp; \ | ||
235 | \ | ||
236 | __asm__ __volatile__( \ | ||
237 | " ld %0, [%2] \n\t" \ | ||
238 | " sr %1, [%0] \n\t" \ | ||
239 | : "=&r"(tmp) \ | ||
240 | : "r"(val), "memory"(®_in_var)); \ | ||
241 | }) | ||
242 | |||
243 | #endif | ||
244 | |||
245 | #define READ_BCR(reg, into) \ | ||
246 | { \ | ||
247 | unsigned int tmp; \ | ||
248 | tmp = read_aux_reg(reg); \ | ||
249 | if (sizeof(tmp) == sizeof(into)) { \ | ||
250 | into = *((typeof(into) *)&tmp); \ | ||
251 | } else { \ | ||
252 | extern void bogus_undefined(void); \ | ||
253 | bogus_undefined(); \ | ||
254 | } \ | ||
255 | } | ||
256 | |||
257 | #define WRITE_BCR(reg, into) \ | ||
258 | { \ | ||
259 | unsigned int tmp; \ | ||
260 | if (sizeof(tmp) == sizeof(into)) { \ | ||
261 | tmp = (*(unsigned int *)(into)); \ | ||
262 | write_aux_reg(reg, tmp); \ | ||
263 | } else { \ | ||
264 | extern void bogus_undefined(void); \ | ||
265 | bogus_undefined(); \ | ||
266 | } \ | ||
267 | } | ||
268 | |||
269 | /* Helpers */ | ||
270 | #define TO_KB(bytes) ((bytes) >> 10) | ||
271 | #define TO_MB(bytes) (TO_KB(bytes) >> 10) | ||
272 | #define PAGES_TO_KB(n_pages) ((n_pages) << (PAGE_SHIFT - 10)) | ||
273 | #define PAGES_TO_MB(n_pages) (PAGES_TO_KB(n_pages) >> 10) | ||
274 | |||
275 | #ifdef CONFIG_ARC_FPU_SAVE_RESTORE | ||
276 | /* These DPFP regs need to be saved/restored across ctx-sw */ | ||
277 | struct arc_fpu { | ||
278 | struct { | ||
279 | unsigned int l, h; | ||
280 | } aux_dpfp[2]; | ||
281 | }; | ||
282 | #endif | ||
283 | |||
284 | /* | ||
285 | *************************************************************** | ||
286 | * Build Configuration Registers, with encoded hardware config | ||
287 | */ | ||
288 | struct bcr_identity { | ||
289 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
290 | unsigned int chip_id:16, cpu_id:8, family:8; | ||
291 | #else | ||
292 | unsigned int family:8, cpu_id:8, chip_id:16; | ||
293 | #endif | ||
294 | }; | ||
295 | |||
296 | struct bcr_mmu_1_2 { | ||
297 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
298 | unsigned int ver:8, ways:4, sets:4, u_itlb:8, u_dtlb:8; | ||
299 | #else | ||
300 | unsigned int u_dtlb:8, u_itlb:8, sets:4, ways:4, ver:8; | ||
301 | #endif | ||
302 | }; | ||
303 | |||
304 | struct bcr_mmu_3 { | ||
305 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
306 | unsigned int ver:8, ways:4, sets:4, osm:1, reserv:3, pg_sz:4, | ||
307 | u_itlb:4, u_dtlb:4; | ||
308 | #else | ||
309 | unsigned int u_dtlb:4, u_itlb:4, pg_sz:4, reserv:3, osm:1, sets:4, | ||
310 | ways:4, ver:8; | ||
311 | #endif | ||
312 | }; | ||
313 | |||
314 | #define EXTN_SWAP_VALID 0x1 | ||
315 | #define EXTN_NORM_VALID 0x2 | ||
316 | #define EXTN_MINMAX_VALID 0x2 | ||
317 | #define EXTN_BARREL_VALID 0x2 | ||
318 | |||
319 | struct bcr_extn { | ||
320 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
321 | unsigned int pad:20, crc:1, ext_arith:2, mul:2, barrel:2, minmax:2, | ||
322 | norm:2, swap:1; | ||
323 | #else | ||
324 | unsigned int swap:1, norm:2, minmax:2, barrel:2, mul:2, ext_arith:2, | ||
325 | crc:1, pad:20; | ||
326 | #endif | ||
327 | }; | ||
328 | |||
329 | /* DSP Options Ref Manual */ | ||
330 | struct bcr_extn_mac_mul { | ||
331 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
332 | unsigned int pad:16, type:8, ver:8; | ||
333 | #else | ||
334 | unsigned int ver:8, type:8, pad:16; | ||
335 | #endif | ||
336 | }; | ||
337 | |||
338 | struct bcr_extn_xymem { | ||
339 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
340 | unsigned int ram_org:2, num_banks:4, bank_sz:4, ver:8; | ||
341 | #else | ||
342 | unsigned int ver:8, bank_sz:4, num_banks:4, ram_org:2; | ||
343 | #endif | ||
344 | }; | ||
345 | |||
346 | struct bcr_cache { | ||
347 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
348 | unsigned int pad:12, line_len:4, sz:4, config:4, ver:8; | ||
349 | #else | ||
350 | unsigned int ver:8, config:4, sz:4, line_len:4, pad:12; | ||
351 | #endif | ||
352 | }; | ||
353 | |||
354 | struct bcr_perip { | ||
355 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
356 | unsigned int start:8, pad2:8, sz:8, pad:8; | ||
357 | #else | ||
358 | unsigned int pad:8, sz:8, pad2:8, start:8; | ||
359 | #endif | ||
360 | }; | ||
361 | struct bcr_iccm { | ||
362 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
363 | unsigned int base:16, pad:5, sz:3, ver:8; | ||
364 | #else | ||
365 | unsigned int ver:8, sz:3, pad:5, base:16; | ||
366 | #endif | ||
367 | }; | ||
368 | |||
369 | /* DCCM Base Address Register: ARC_REG_DCCMBASE_BCR */ | ||
370 | struct bcr_dccm_base { | ||
371 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
372 | unsigned int addr:24, ver:8; | ||
373 | #else | ||
374 | unsigned int ver:8, addr:24; | ||
375 | #endif | ||
376 | }; | ||
377 | |||
378 | /* DCCM RAM Configuration Register: ARC_REG_DCCM_BCR */ | ||
379 | struct bcr_dccm { | ||
380 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
381 | unsigned int res:21, sz:3, ver:8; | ||
382 | #else | ||
383 | unsigned int ver:8, sz:3, res:21; | ||
384 | #endif | ||
385 | }; | ||
386 | |||
387 | /* Both SP and DP FPU BCRs have same format */ | ||
388 | struct bcr_fp { | ||
389 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
390 | unsigned int fast:1, ver:8; | ||
391 | #else | ||
392 | unsigned int ver:8, fast:1; | ||
393 | #endif | ||
394 | }; | ||
395 | |||
396 | /* | ||
397 | ******************************************************************* | ||
398 | * Generic structures to hold build configuration used at runtime | ||
399 | */ | ||
400 | |||
401 | struct cpuinfo_arc_mmu { | ||
402 | unsigned int ver, pg_sz, sets, ways, u_dtlb, u_itlb, num_tlb; | ||
403 | }; | ||
404 | |||
405 | struct cpuinfo_arc_cache { | ||
406 | unsigned int has_aliasing, sz, line_len, assoc, ver; | ||
407 | }; | ||
408 | |||
409 | struct cpuinfo_arc_ccm { | ||
410 | unsigned int base_addr, sz; | ||
411 | }; | ||
412 | |||
413 | struct cpuinfo_arc { | ||
414 | struct cpuinfo_arc_cache icache, dcache; | ||
415 | struct cpuinfo_arc_mmu mmu; | ||
416 | struct bcr_identity core; | ||
417 | unsigned int timers; | ||
418 | unsigned int vec_base; | ||
419 | unsigned int uncached_base; | ||
420 | struct cpuinfo_arc_ccm iccm, dccm; | ||
421 | struct bcr_extn extn; | ||
422 | struct bcr_extn_xymem extn_xymem; | ||
423 | struct bcr_extn_mac_mul extn_mac_mul; | ||
424 | struct bcr_fp fp, dpfp; | ||
425 | }; | ||
426 | |||
427 | extern struct cpuinfo_arc cpuinfo_arc700[]; | ||
428 | |||
429 | #endif /* __ASEMBLY__ */ | ||
430 | |||
431 | #endif /* __KERNEL__ */ | ||
432 | |||
433 | #endif /* _ASM_ARC_ARCREGS_H */ | ||
diff --git a/arch/arc/include/asm/asm-offsets.h b/arch/arc/include/asm/asm-offsets.h new file mode 100644 index 000000000000..dad18768fe43 --- /dev/null +++ b/arch/arc/include/asm/asm-offsets.h | |||
@@ -0,0 +1,9 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <generated/asm-offsets.h> | ||
diff --git a/arch/arc/include/asm/atomic.h b/arch/arc/include/asm/atomic.h new file mode 100644 index 000000000000..83f03ca6caf6 --- /dev/null +++ b/arch/arc/include/asm/atomic.h | |||
@@ -0,0 +1,232 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_ATOMIC_H | ||
10 | #define _ASM_ARC_ATOMIC_H | ||
11 | |||
12 | #ifdef __KERNEL__ | ||
13 | |||
14 | #ifndef __ASSEMBLY__ | ||
15 | |||
16 | #include <linux/types.h> | ||
17 | #include <linux/compiler.h> | ||
18 | #include <asm/cmpxchg.h> | ||
19 | #include <asm/barrier.h> | ||
20 | #include <asm/smp.h> | ||
21 | |||
22 | #define atomic_read(v) ((v)->counter) | ||
23 | |||
24 | #ifdef CONFIG_ARC_HAS_LLSC | ||
25 | |||
26 | #define atomic_set(v, i) (((v)->counter) = (i)) | ||
27 | |||
28 | static inline void atomic_add(int i, atomic_t *v) | ||
29 | { | ||
30 | unsigned int temp; | ||
31 | |||
32 | __asm__ __volatile__( | ||
33 | "1: llock %0, [%1] \n" | ||
34 | " add %0, %0, %2 \n" | ||
35 | " scond %0, [%1] \n" | ||
36 | " bnz 1b \n" | ||
37 | : "=&r"(temp) /* Early clobber, to prevent reg reuse */ | ||
38 | : "r"(&v->counter), "ir"(i) | ||
39 | : "cc"); | ||
40 | } | ||
41 | |||
42 | static inline void atomic_sub(int i, atomic_t *v) | ||
43 | { | ||
44 | unsigned int temp; | ||
45 | |||
46 | __asm__ __volatile__( | ||
47 | "1: llock %0, [%1] \n" | ||
48 | " sub %0, %0, %2 \n" | ||
49 | " scond %0, [%1] \n" | ||
50 | " bnz 1b \n" | ||
51 | : "=&r"(temp) | ||
52 | : "r"(&v->counter), "ir"(i) | ||
53 | : "cc"); | ||
54 | } | ||
55 | |||
56 | /* add and also return the new value */ | ||
57 | static inline int atomic_add_return(int i, atomic_t *v) | ||
58 | { | ||
59 | unsigned int temp; | ||
60 | |||
61 | __asm__ __volatile__( | ||
62 | "1: llock %0, [%1] \n" | ||
63 | " add %0, %0, %2 \n" | ||
64 | " scond %0, [%1] \n" | ||
65 | " bnz 1b \n" | ||
66 | : "=&r"(temp) | ||
67 | : "r"(&v->counter), "ir"(i) | ||
68 | : "cc"); | ||
69 | |||
70 | return temp; | ||
71 | } | ||
72 | |||
73 | static inline int atomic_sub_return(int i, atomic_t *v) | ||
74 | { | ||
75 | unsigned int temp; | ||
76 | |||
77 | __asm__ __volatile__( | ||
78 | "1: llock %0, [%1] \n" | ||
79 | " sub %0, %0, %2 \n" | ||
80 | " scond %0, [%1] \n" | ||
81 | " bnz 1b \n" | ||
82 | : "=&r"(temp) | ||
83 | : "r"(&v->counter), "ir"(i) | ||
84 | : "cc"); | ||
85 | |||
86 | return temp; | ||
87 | } | ||
88 | |||
89 | static inline void atomic_clear_mask(unsigned long mask, unsigned long *addr) | ||
90 | { | ||
91 | unsigned int temp; | ||
92 | |||
93 | __asm__ __volatile__( | ||
94 | "1: llock %0, [%1] \n" | ||
95 | " bic %0, %0, %2 \n" | ||
96 | " scond %0, [%1] \n" | ||
97 | " bnz 1b \n" | ||
98 | : "=&r"(temp) | ||
99 | : "r"(addr), "ir"(mask) | ||
100 | : "cc"); | ||
101 | } | ||
102 | |||
103 | #else /* !CONFIG_ARC_HAS_LLSC */ | ||
104 | |||
105 | #ifndef CONFIG_SMP | ||
106 | |||
107 | /* violating atomic_xxx API locking protocol in UP for optimization sake */ | ||
108 | #define atomic_set(v, i) (((v)->counter) = (i)) | ||
109 | |||
110 | #else | ||
111 | |||
112 | static inline void atomic_set(atomic_t *v, int i) | ||
113 | { | ||
114 | /* | ||
115 | * Independent of hardware support, all of the atomic_xxx() APIs need | ||
116 | * to follow the same locking rules to make sure that a "hardware" | ||
117 | * atomic insn (e.g. LD) doesn't clobber an "emulated" atomic insn | ||
118 | * sequence | ||
119 | * | ||
120 | * Thus atomic_set() despite being 1 insn (and seemingly atomic) | ||
121 | * requires the locking. | ||
122 | */ | ||
123 | unsigned long flags; | ||
124 | |||
125 | atomic_ops_lock(flags); | ||
126 | v->counter = i; | ||
127 | atomic_ops_unlock(flags); | ||
128 | } | ||
129 | #endif | ||
130 | |||
131 | /* | ||
132 | * Non hardware assisted Atomic-R-M-W | ||
133 | * Locking would change to irq-disabling only (UP) and spinlocks (SMP) | ||
134 | */ | ||
135 | |||
136 | static inline void atomic_add(int i, atomic_t *v) | ||
137 | { | ||
138 | unsigned long flags; | ||
139 | |||
140 | atomic_ops_lock(flags); | ||
141 | v->counter += i; | ||
142 | atomic_ops_unlock(flags); | ||
143 | } | ||
144 | |||
145 | static inline void atomic_sub(int i, atomic_t *v) | ||
146 | { | ||
147 | unsigned long flags; | ||
148 | |||
149 | atomic_ops_lock(flags); | ||
150 | v->counter -= i; | ||
151 | atomic_ops_unlock(flags); | ||
152 | } | ||
153 | |||
154 | static inline int atomic_add_return(int i, atomic_t *v) | ||
155 | { | ||
156 | unsigned long flags; | ||
157 | unsigned long temp; | ||
158 | |||
159 | atomic_ops_lock(flags); | ||
160 | temp = v->counter; | ||
161 | temp += i; | ||
162 | v->counter = temp; | ||
163 | atomic_ops_unlock(flags); | ||
164 | |||
165 | return temp; | ||
166 | } | ||
167 | |||
168 | static inline int atomic_sub_return(int i, atomic_t *v) | ||
169 | { | ||
170 | unsigned long flags; | ||
171 | unsigned long temp; | ||
172 | |||
173 | atomic_ops_lock(flags); | ||
174 | temp = v->counter; | ||
175 | temp -= i; | ||
176 | v->counter = temp; | ||
177 | atomic_ops_unlock(flags); | ||
178 | |||
179 | return temp; | ||
180 | } | ||
181 | |||
182 | static inline void atomic_clear_mask(unsigned long mask, unsigned long *addr) | ||
183 | { | ||
184 | unsigned long flags; | ||
185 | |||
186 | atomic_ops_lock(flags); | ||
187 | *addr &= ~mask; | ||
188 | atomic_ops_unlock(flags); | ||
189 | } | ||
190 | |||
191 | #endif /* !CONFIG_ARC_HAS_LLSC */ | ||
192 | |||
193 | /** | ||
194 | * __atomic_add_unless - add unless the number is a given value | ||
195 | * @v: pointer of type atomic_t | ||
196 | * @a: the amount to add to v... | ||
197 | * @u: ...unless v is equal to u. | ||
198 | * | ||
199 | * Atomically adds @a to @v, so long as it was not @u. | ||
200 | * Returns the old value of @v | ||
201 | */ | ||
202 | #define __atomic_add_unless(v, a, u) \ | ||
203 | ({ \ | ||
204 | int c, old; \ | ||
205 | c = atomic_read(v); \ | ||
206 | while (c != (u) && (old = atomic_cmpxchg((v), c, c + (a))) != c)\ | ||
207 | c = old; \ | ||
208 | c; \ | ||
209 | }) | ||
210 | |||
211 | #define atomic_inc_not_zero(v) atomic_add_unless((v), 1, 0) | ||
212 | |||
213 | #define atomic_inc(v) atomic_add(1, v) | ||
214 | #define atomic_dec(v) atomic_sub(1, v) | ||
215 | |||
216 | #define atomic_inc_and_test(v) (atomic_add_return(1, v) == 0) | ||
217 | #define atomic_dec_and_test(v) (atomic_sub_return(1, v) == 0) | ||
218 | #define atomic_inc_return(v) atomic_add_return(1, (v)) | ||
219 | #define atomic_dec_return(v) atomic_sub_return(1, (v)) | ||
220 | #define atomic_sub_and_test(i, v) (atomic_sub_return(i, v) == 0) | ||
221 | |||
222 | #define atomic_add_negative(i, v) (atomic_add_return(i, v) < 0) | ||
223 | |||
224 | #define ATOMIC_INIT(i) { (i) } | ||
225 | |||
226 | #include <asm-generic/atomic64.h> | ||
227 | |||
228 | #endif | ||
229 | |||
230 | #endif | ||
231 | |||
232 | #endif | ||
diff --git a/arch/arc/include/asm/barrier.h b/arch/arc/include/asm/barrier.h new file mode 100644 index 000000000000..f6cb7c4ffb35 --- /dev/null +++ b/arch/arc/include/asm/barrier.h | |||
@@ -0,0 +1,42 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_BARRIER_H | ||
10 | #define __ASM_BARRIER_H | ||
11 | |||
12 | #ifndef __ASSEMBLY__ | ||
13 | |||
14 | /* TODO-vineetg: Need to see what this does, don't we need sync anywhere */ | ||
15 | #define mb() __asm__ __volatile__ ("" : : : "memory") | ||
16 | #define rmb() mb() | ||
17 | #define wmb() mb() | ||
18 | #define set_mb(var, value) do { var = value; mb(); } while (0) | ||
19 | #define set_wmb(var, value) do { var = value; wmb(); } while (0) | ||
20 | #define read_barrier_depends() mb() | ||
21 | |||
22 | /* TODO-vineetg verify the correctness of macros here */ | ||
23 | #ifdef CONFIG_SMP | ||
24 | #define smp_mb() mb() | ||
25 | #define smp_rmb() rmb() | ||
26 | #define smp_wmb() wmb() | ||
27 | #else | ||
28 | #define smp_mb() barrier() | ||
29 | #define smp_rmb() barrier() | ||
30 | #define smp_wmb() barrier() | ||
31 | #endif | ||
32 | |||
33 | #define smp_mb__before_atomic_dec() barrier() | ||
34 | #define smp_mb__after_atomic_dec() barrier() | ||
35 | #define smp_mb__before_atomic_inc() barrier() | ||
36 | #define smp_mb__after_atomic_inc() barrier() | ||
37 | |||
38 | #define smp_read_barrier_depends() do { } while (0) | ||
39 | |||
40 | #endif | ||
41 | |||
42 | #endif | ||
diff --git a/arch/arc/include/asm/bitops.h b/arch/arc/include/asm/bitops.h new file mode 100644 index 000000000000..647a83a8e756 --- /dev/null +++ b/arch/arc/include/asm/bitops.h | |||
@@ -0,0 +1,516 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_BITOPS_H | ||
10 | #define _ASM_BITOPS_H | ||
11 | |||
12 | #ifndef _LINUX_BITOPS_H | ||
13 | #error only <linux/bitops.h> can be included directly | ||
14 | #endif | ||
15 | |||
16 | #ifdef __KERNEL__ | ||
17 | |||
18 | #ifndef __ASSEMBLY__ | ||
19 | |||
20 | #include <linux/types.h> | ||
21 | #include <linux/compiler.h> | ||
22 | |||
23 | /* | ||
24 | * Hardware assisted read-modify-write using ARC700 LLOCK/SCOND insns. | ||
25 | * The Kconfig glue ensures that in SMP, this is only set if the container | ||
26 | * SoC/platform has cross-core coherent LLOCK/SCOND | ||
27 | */ | ||
28 | #if defined(CONFIG_ARC_HAS_LLSC) | ||
29 | |||
30 | static inline void set_bit(unsigned long nr, volatile unsigned long *m) | ||
31 | { | ||
32 | unsigned int temp; | ||
33 | |||
34 | m += nr >> 5; | ||
35 | |||
36 | if (__builtin_constant_p(nr)) | ||
37 | nr &= 0x1f; | ||
38 | |||
39 | __asm__ __volatile__( | ||
40 | "1: llock %0, [%1] \n" | ||
41 | " bset %0, %0, %2 \n" | ||
42 | " scond %0, [%1] \n" | ||
43 | " bnz 1b \n" | ||
44 | : "=&r"(temp) | ||
45 | : "r"(m), "ir"(nr) | ||
46 | : "cc"); | ||
47 | } | ||
48 | |||
49 | static inline void clear_bit(unsigned long nr, volatile unsigned long *m) | ||
50 | { | ||
51 | unsigned int temp; | ||
52 | |||
53 | m += nr >> 5; | ||
54 | |||
55 | if (__builtin_constant_p(nr)) | ||
56 | nr &= 0x1f; | ||
57 | |||
58 | __asm__ __volatile__( | ||
59 | "1: llock %0, [%1] \n" | ||
60 | " bclr %0, %0, %2 \n" | ||
61 | " scond %0, [%1] \n" | ||
62 | " bnz 1b \n" | ||
63 | : "=&r"(temp) | ||
64 | : "r"(m), "ir"(nr) | ||
65 | : "cc"); | ||
66 | } | ||
67 | |||
68 | static inline void change_bit(unsigned long nr, volatile unsigned long *m) | ||
69 | { | ||
70 | unsigned int temp; | ||
71 | |||
72 | m += nr >> 5; | ||
73 | |||
74 | if (__builtin_constant_p(nr)) | ||
75 | nr &= 0x1f; | ||
76 | |||
77 | __asm__ __volatile__( | ||
78 | "1: llock %0, [%1] \n" | ||
79 | " bxor %0, %0, %2 \n" | ||
80 | " scond %0, [%1] \n" | ||
81 | " bnz 1b \n" | ||
82 | : "=&r"(temp) | ||
83 | : "r"(m), "ir"(nr) | ||
84 | : "cc"); | ||
85 | } | ||
86 | |||
87 | /* | ||
88 | * Semantically: | ||
89 | * Test the bit | ||
90 | * if clear | ||
91 | * set it and return 0 (old value) | ||
92 | * else | ||
93 | * return 1 (old value). | ||
94 | * | ||
95 | * Since ARC lacks a equivalent h/w primitive, the bit is set unconditionally | ||
96 | * and the old value of bit is returned | ||
97 | */ | ||
98 | static inline int test_and_set_bit(unsigned long nr, volatile unsigned long *m) | ||
99 | { | ||
100 | unsigned long old, temp; | ||
101 | |||
102 | m += nr >> 5; | ||
103 | |||
104 | if (__builtin_constant_p(nr)) | ||
105 | nr &= 0x1f; | ||
106 | |||
107 | __asm__ __volatile__( | ||
108 | "1: llock %0, [%2] \n" | ||
109 | " bset %1, %0, %3 \n" | ||
110 | " scond %1, [%2] \n" | ||
111 | " bnz 1b \n" | ||
112 | : "=&r"(old), "=&r"(temp) | ||
113 | : "r"(m), "ir"(nr) | ||
114 | : "cc"); | ||
115 | |||
116 | return (old & (1 << nr)) != 0; | ||
117 | } | ||
118 | |||
119 | static inline int | ||
120 | test_and_clear_bit(unsigned long nr, volatile unsigned long *m) | ||
121 | { | ||
122 | unsigned int old, temp; | ||
123 | |||
124 | m += nr >> 5; | ||
125 | |||
126 | if (__builtin_constant_p(nr)) | ||
127 | nr &= 0x1f; | ||
128 | |||
129 | __asm__ __volatile__( | ||
130 | "1: llock %0, [%2] \n" | ||
131 | " bclr %1, %0, %3 \n" | ||
132 | " scond %1, [%2] \n" | ||
133 | " bnz 1b \n" | ||
134 | : "=&r"(old), "=&r"(temp) | ||
135 | : "r"(m), "ir"(nr) | ||
136 | : "cc"); | ||
137 | |||
138 | return (old & (1 << nr)) != 0; | ||
139 | } | ||
140 | |||
141 | static inline int | ||
142 | test_and_change_bit(unsigned long nr, volatile unsigned long *m) | ||
143 | { | ||
144 | unsigned int old, temp; | ||
145 | |||
146 | m += nr >> 5; | ||
147 | |||
148 | if (__builtin_constant_p(nr)) | ||
149 | nr &= 0x1f; | ||
150 | |||
151 | __asm__ __volatile__( | ||
152 | "1: llock %0, [%2] \n" | ||
153 | " bxor %1, %0, %3 \n" | ||
154 | " scond %1, [%2] \n" | ||
155 | " bnz 1b \n" | ||
156 | : "=&r"(old), "=&r"(temp) | ||
157 | : "r"(m), "ir"(nr) | ||
158 | : "cc"); | ||
159 | |||
160 | return (old & (1 << nr)) != 0; | ||
161 | } | ||
162 | |||
163 | #else /* !CONFIG_ARC_HAS_LLSC */ | ||
164 | |||
165 | #include <asm/smp.h> | ||
166 | |||
167 | /* | ||
168 | * Non hardware assisted Atomic-R-M-W | ||
169 | * Locking would change to irq-disabling only (UP) and spinlocks (SMP) | ||
170 | * | ||
171 | * There's "significant" micro-optimization in writing our own variants of | ||
172 | * bitops (over generic variants) | ||
173 | * | ||
174 | * (1) The generic APIs have "signed" @nr while we have it "unsigned" | ||
175 | * This avoids extra code to be generated for pointer arithmatic, since | ||
176 | * is "not sure" that index is NOT -ve | ||
177 | * (2) Utilize the fact that ARCompact bit fidding insn (BSET/BCLR/ASL) etc | ||
178 | * only consider bottom 5 bits of @nr, so NO need to mask them off. | ||
179 | * (GCC Quirk: however for constant @nr we still need to do the masking | ||
180 | * at compile time) | ||
181 | */ | ||
182 | |||
183 | static inline void set_bit(unsigned long nr, volatile unsigned long *m) | ||
184 | { | ||
185 | unsigned long temp, flags; | ||
186 | m += nr >> 5; | ||
187 | |||
188 | if (__builtin_constant_p(nr)) | ||
189 | nr &= 0x1f; | ||
190 | |||
191 | bitops_lock(flags); | ||
192 | |||
193 | temp = *m; | ||
194 | *m = temp | (1UL << nr); | ||
195 | |||
196 | bitops_unlock(flags); | ||
197 | } | ||
198 | |||
199 | static inline void clear_bit(unsigned long nr, volatile unsigned long *m) | ||
200 | { | ||
201 | unsigned long temp, flags; | ||
202 | m += nr >> 5; | ||
203 | |||
204 | if (__builtin_constant_p(nr)) | ||
205 | nr &= 0x1f; | ||
206 | |||
207 | bitops_lock(flags); | ||
208 | |||
209 | temp = *m; | ||
210 | *m = temp & ~(1UL << nr); | ||
211 | |||
212 | bitops_unlock(flags); | ||
213 | } | ||
214 | |||
215 | static inline void change_bit(unsigned long nr, volatile unsigned long *m) | ||
216 | { | ||
217 | unsigned long temp, flags; | ||
218 | m += nr >> 5; | ||
219 | |||
220 | if (__builtin_constant_p(nr)) | ||
221 | nr &= 0x1f; | ||
222 | |||
223 | bitops_lock(flags); | ||
224 | |||
225 | temp = *m; | ||
226 | *m = temp ^ (1UL << nr); | ||
227 | |||
228 | bitops_unlock(flags); | ||
229 | } | ||
230 | |||
231 | static inline int test_and_set_bit(unsigned long nr, volatile unsigned long *m) | ||
232 | { | ||
233 | unsigned long old, flags; | ||
234 | m += nr >> 5; | ||
235 | |||
236 | if (__builtin_constant_p(nr)) | ||
237 | nr &= 0x1f; | ||
238 | |||
239 | bitops_lock(flags); | ||
240 | |||
241 | old = *m; | ||
242 | *m = old | (1 << nr); | ||
243 | |||
244 | bitops_unlock(flags); | ||
245 | |||
246 | return (old & (1 << nr)) != 0; | ||
247 | } | ||
248 | |||
249 | static inline int | ||
250 | test_and_clear_bit(unsigned long nr, volatile unsigned long *m) | ||
251 | { | ||
252 | unsigned long old, flags; | ||
253 | m += nr >> 5; | ||
254 | |||
255 | if (__builtin_constant_p(nr)) | ||
256 | nr &= 0x1f; | ||
257 | |||
258 | bitops_lock(flags); | ||
259 | |||
260 | old = *m; | ||
261 | *m = old & ~(1 << nr); | ||
262 | |||
263 | bitops_unlock(flags); | ||
264 | |||
265 | return (old & (1 << nr)) != 0; | ||
266 | } | ||
267 | |||
268 | static inline int | ||
269 | test_and_change_bit(unsigned long nr, volatile unsigned long *m) | ||
270 | { | ||
271 | unsigned long old, flags; | ||
272 | m += nr >> 5; | ||
273 | |||
274 | if (__builtin_constant_p(nr)) | ||
275 | nr &= 0x1f; | ||
276 | |||
277 | bitops_lock(flags); | ||
278 | |||
279 | old = *m; | ||
280 | *m = old ^ (1 << nr); | ||
281 | |||
282 | bitops_unlock(flags); | ||
283 | |||
284 | return (old & (1 << nr)) != 0; | ||
285 | } | ||
286 | |||
287 | #endif /* CONFIG_ARC_HAS_LLSC */ | ||
288 | |||
289 | /*************************************** | ||
290 | * Non atomic variants | ||
291 | **************************************/ | ||
292 | |||
293 | static inline void __set_bit(unsigned long nr, volatile unsigned long *m) | ||
294 | { | ||
295 | unsigned long temp; | ||
296 | m += nr >> 5; | ||
297 | |||
298 | if (__builtin_constant_p(nr)) | ||
299 | nr &= 0x1f; | ||
300 | |||
301 | temp = *m; | ||
302 | *m = temp | (1UL << nr); | ||
303 | } | ||
304 | |||
305 | static inline void __clear_bit(unsigned long nr, volatile unsigned long *m) | ||
306 | { | ||
307 | unsigned long temp; | ||
308 | m += nr >> 5; | ||
309 | |||
310 | if (__builtin_constant_p(nr)) | ||
311 | nr &= 0x1f; | ||
312 | |||
313 | temp = *m; | ||
314 | *m = temp & ~(1UL << nr); | ||
315 | } | ||
316 | |||
317 | static inline void __change_bit(unsigned long nr, volatile unsigned long *m) | ||
318 | { | ||
319 | unsigned long temp; | ||
320 | m += nr >> 5; | ||
321 | |||
322 | if (__builtin_constant_p(nr)) | ||
323 | nr &= 0x1f; | ||
324 | |||
325 | temp = *m; | ||
326 | *m = temp ^ (1UL << nr); | ||
327 | } | ||
328 | |||
329 | static inline int | ||
330 | __test_and_set_bit(unsigned long nr, volatile unsigned long *m) | ||
331 | { | ||
332 | unsigned long old; | ||
333 | m += nr >> 5; | ||
334 | |||
335 | if (__builtin_constant_p(nr)) | ||
336 | nr &= 0x1f; | ||
337 | |||
338 | old = *m; | ||
339 | *m = old | (1 << nr); | ||
340 | |||
341 | return (old & (1 << nr)) != 0; | ||
342 | } | ||
343 | |||
344 | static inline int | ||
345 | __test_and_clear_bit(unsigned long nr, volatile unsigned long *m) | ||
346 | { | ||
347 | unsigned long old; | ||
348 | m += nr >> 5; | ||
349 | |||
350 | if (__builtin_constant_p(nr)) | ||
351 | nr &= 0x1f; | ||
352 | |||
353 | old = *m; | ||
354 | *m = old & ~(1 << nr); | ||
355 | |||
356 | return (old & (1 << nr)) != 0; | ||
357 | } | ||
358 | |||
359 | static inline int | ||
360 | __test_and_change_bit(unsigned long nr, volatile unsigned long *m) | ||
361 | { | ||
362 | unsigned long old; | ||
363 | m += nr >> 5; | ||
364 | |||
365 | if (__builtin_constant_p(nr)) | ||
366 | nr &= 0x1f; | ||
367 | |||
368 | old = *m; | ||
369 | *m = old ^ (1 << nr); | ||
370 | |||
371 | return (old & (1 << nr)) != 0; | ||
372 | } | ||
373 | |||
374 | /* | ||
375 | * This routine doesn't need to be atomic. | ||
376 | */ | ||
377 | static inline int | ||
378 | __constant_test_bit(unsigned int nr, const volatile unsigned long *addr) | ||
379 | { | ||
380 | return ((1UL << (nr & 31)) & | ||
381 | (((const volatile unsigned int *)addr)[nr >> 5])) != 0; | ||
382 | } | ||
383 | |||
384 | static inline int | ||
385 | __test_bit(unsigned int nr, const volatile unsigned long *addr) | ||
386 | { | ||
387 | unsigned long mask; | ||
388 | |||
389 | addr += nr >> 5; | ||
390 | |||
391 | /* ARC700 only considers 5 bits in bit-fiddling insn */ | ||
392 | mask = 1 << nr; | ||
393 | |||
394 | return ((mask & *addr) != 0); | ||
395 | } | ||
396 | |||
397 | #define test_bit(nr, addr) (__builtin_constant_p(nr) ? \ | ||
398 | __constant_test_bit((nr), (addr)) : \ | ||
399 | __test_bit((nr), (addr))) | ||
400 | |||
401 | /* | ||
402 | * Count the number of zeros, starting from MSB | ||
403 | * Helper for fls( ) friends | ||
404 | * This is a pure count, so (1-32) or (0-31) doesn't apply | ||
405 | * It could be 0 to 32, based on num of 0's in there | ||
406 | * clz(0x8000_0000) = 0, clz(0xFFFF_FFFF)=0, clz(0) = 32, clz(1) = 31 | ||
407 | */ | ||
408 | static inline __attribute__ ((const)) int clz(unsigned int x) | ||
409 | { | ||
410 | unsigned int res; | ||
411 | |||
412 | __asm__ __volatile__( | ||
413 | " norm.f %0, %1 \n" | ||
414 | " mov.n %0, 0 \n" | ||
415 | " add.p %0, %0, 1 \n" | ||
416 | : "=r"(res) | ||
417 | : "r"(x) | ||
418 | : "cc"); | ||
419 | |||
420 | return res; | ||
421 | } | ||
422 | |||
423 | static inline int constant_fls(int x) | ||
424 | { | ||
425 | int r = 32; | ||
426 | |||
427 | if (!x) | ||
428 | return 0; | ||
429 | if (!(x & 0xffff0000u)) { | ||
430 | x <<= 16; | ||
431 | r -= 16; | ||
432 | } | ||
433 | if (!(x & 0xff000000u)) { | ||
434 | x <<= 8; | ||
435 | r -= 8; | ||
436 | } | ||
437 | if (!(x & 0xf0000000u)) { | ||
438 | x <<= 4; | ||
439 | r -= 4; | ||
440 | } | ||
441 | if (!(x & 0xc0000000u)) { | ||
442 | x <<= 2; | ||
443 | r -= 2; | ||
444 | } | ||
445 | if (!(x & 0x80000000u)) { | ||
446 | x <<= 1; | ||
447 | r -= 1; | ||
448 | } | ||
449 | return r; | ||
450 | } | ||
451 | |||
452 | /* | ||
453 | * fls = Find Last Set in word | ||
454 | * @result: [1-32] | ||
455 | * fls(1) = 1, fls(0x80000000) = 32, fls(0) = 0 | ||
456 | */ | ||
457 | static inline __attribute__ ((const)) int fls(unsigned long x) | ||
458 | { | ||
459 | if (__builtin_constant_p(x)) | ||
460 | return constant_fls(x); | ||
461 | |||
462 | return 32 - clz(x); | ||
463 | } | ||
464 | |||
465 | /* | ||
466 | * __fls: Similar to fls, but zero based (0-31) | ||
467 | */ | ||
468 | static inline __attribute__ ((const)) int __fls(unsigned long x) | ||
469 | { | ||
470 | if (!x) | ||
471 | return 0; | ||
472 | else | ||
473 | return fls(x) - 1; | ||
474 | } | ||
475 | |||
476 | /* | ||
477 | * ffs = Find First Set in word (LSB to MSB) | ||
478 | * @result: [1-32], 0 if all 0's | ||
479 | */ | ||
480 | #define ffs(x) ({ unsigned long __t = (x); fls(__t & -__t); }) | ||
481 | |||
482 | /* | ||
483 | * __ffs: Similar to ffs, but zero based (0-31) | ||
484 | */ | ||
485 | static inline __attribute__ ((const)) int __ffs(unsigned long word) | ||
486 | { | ||
487 | if (!word) | ||
488 | return word; | ||
489 | |||
490 | return ffs(word) - 1; | ||
491 | } | ||
492 | |||
493 | /* | ||
494 | * ffz = Find First Zero in word. | ||
495 | * @return:[0-31], 32 if all 1's | ||
496 | */ | ||
497 | #define ffz(x) __ffs(~(x)) | ||
498 | |||
499 | /* TODO does this affect uni-processor code */ | ||
500 | #define smp_mb__before_clear_bit() barrier() | ||
501 | #define smp_mb__after_clear_bit() barrier() | ||
502 | |||
503 | #include <asm-generic/bitops/hweight.h> | ||
504 | #include <asm-generic/bitops/fls64.h> | ||
505 | #include <asm-generic/bitops/sched.h> | ||
506 | #include <asm-generic/bitops/lock.h> | ||
507 | |||
508 | #include <asm-generic/bitops/find.h> | ||
509 | #include <asm-generic/bitops/le.h> | ||
510 | #include <asm-generic/bitops/ext2-atomic-setbit.h> | ||
511 | |||
512 | #endif /* !__ASSEMBLY__ */ | ||
513 | |||
514 | #endif /* __KERNEL__ */ | ||
515 | |||
516 | #endif | ||
diff --git a/arch/arc/include/asm/bug.h b/arch/arc/include/asm/bug.h new file mode 100644 index 000000000000..2ad8f9b1c54b --- /dev/null +++ b/arch/arc/include/asm/bug.h | |||
@@ -0,0 +1,37 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_BUG_H | ||
10 | #define _ASM_ARC_BUG_H | ||
11 | |||
12 | #ifndef __ASSEMBLY__ | ||
13 | |||
14 | #include <asm/ptrace.h> | ||
15 | |||
16 | struct task_struct; | ||
17 | |||
18 | void show_regs(struct pt_regs *regs); | ||
19 | void show_stacktrace(struct task_struct *tsk, struct pt_regs *regs); | ||
20 | void show_kernel_fault_diag(const char *str, struct pt_regs *regs, | ||
21 | unsigned long address, unsigned long cause_reg); | ||
22 | void die(const char *str, struct pt_regs *regs, unsigned long address, | ||
23 | unsigned long cause_reg); | ||
24 | |||
25 | #define BUG() do { \ | ||
26 | dump_stack(); \ | ||
27 | pr_warn("Kernel BUG in %s: %s: %d!\n", \ | ||
28 | __FILE__, __func__, __LINE__); \ | ||
29 | } while (0) | ||
30 | |||
31 | #define HAVE_ARCH_BUG | ||
32 | |||
33 | #include <asm-generic/bug.h> | ||
34 | |||
35 | #endif /* !__ASSEMBLY__ */ | ||
36 | |||
37 | #endif | ||
diff --git a/arch/arc/include/asm/cache.h b/arch/arc/include/asm/cache.h new file mode 100644 index 000000000000..6632273861fd --- /dev/null +++ b/arch/arc/include/asm/cache.h | |||
@@ -0,0 +1,75 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ARC_ASM_CACHE_H | ||
10 | #define __ARC_ASM_CACHE_H | ||
11 | |||
12 | /* In case $$ not config, setup a dummy number for rest of kernel */ | ||
13 | #ifndef CONFIG_ARC_CACHE_LINE_SHIFT | ||
14 | #define L1_CACHE_SHIFT 6 | ||
15 | #else | ||
16 | #define L1_CACHE_SHIFT CONFIG_ARC_CACHE_LINE_SHIFT | ||
17 | #endif | ||
18 | |||
19 | #define L1_CACHE_BYTES (1 << L1_CACHE_SHIFT) | ||
20 | |||
21 | #define ARC_ICACHE_WAYS 2 | ||
22 | #define ARC_DCACHE_WAYS 4 | ||
23 | |||
24 | /* Helpers */ | ||
25 | #define ARC_ICACHE_LINE_LEN L1_CACHE_BYTES | ||
26 | #define ARC_DCACHE_LINE_LEN L1_CACHE_BYTES | ||
27 | |||
28 | #define ICACHE_LINE_MASK (~(ARC_ICACHE_LINE_LEN - 1)) | ||
29 | #define DCACHE_LINE_MASK (~(ARC_DCACHE_LINE_LEN - 1)) | ||
30 | |||
31 | #if ARC_ICACHE_LINE_LEN != ARC_DCACHE_LINE_LEN | ||
32 | #error "Need to fix some code as I/D cache lines not same" | ||
33 | #else | ||
34 | #define is_not_cache_aligned(p) ((unsigned long)p & (~DCACHE_LINE_MASK)) | ||
35 | #endif | ||
36 | |||
37 | #ifndef __ASSEMBLY__ | ||
38 | |||
39 | /* Uncached access macros */ | ||
40 | #define arc_read_uncached_32(ptr) \ | ||
41 | ({ \ | ||
42 | unsigned int __ret; \ | ||
43 | __asm__ __volatile__( \ | ||
44 | " ld.di %0, [%1] \n" \ | ||
45 | : "=r"(__ret) \ | ||
46 | : "r"(ptr)); \ | ||
47 | __ret; \ | ||
48 | }) | ||
49 | |||
50 | #define arc_write_uncached_32(ptr, data)\ | ||
51 | ({ \ | ||
52 | __asm__ __volatile__( \ | ||
53 | " st.di %0, [%1] \n" \ | ||
54 | : \ | ||
55 | : "r"(data), "r"(ptr)); \ | ||
56 | }) | ||
57 | |||
58 | /* used to give SHMLBA a value to avoid Cache Aliasing */ | ||
59 | extern unsigned int ARC_shmlba; | ||
60 | |||
61 | #define ARCH_DMA_MINALIGN L1_CACHE_BYTES | ||
62 | |||
63 | /* | ||
64 | * ARC700 doesn't cache any access in top 256M. | ||
65 | * Ideal for wiring memory mapped peripherals as we don't need to do | ||
66 | * explicit uncached accesses (LD.di/ST.di) hence more portable drivers | ||
67 | */ | ||
68 | #define ARC_UNCACHED_ADDR_SPACE 0xc0000000 | ||
69 | |||
70 | extern void arc_cache_init(void); | ||
71 | extern char *arc_cache_mumbojumbo(int cpu_id, char *buf, int len); | ||
72 | extern void __init read_decode_cache_bcr(void); | ||
73 | #endif | ||
74 | |||
75 | #endif /* _ASM_CACHE_H */ | ||
diff --git a/arch/arc/include/asm/cacheflush.h b/arch/arc/include/asm/cacheflush.h new file mode 100644 index 000000000000..97ee96f26505 --- /dev/null +++ b/arch/arc/include/asm/cacheflush.h | |||
@@ -0,0 +1,67 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * vineetg: May 2011: for Non-aliasing VIPT D-cache following can be NOPs | ||
9 | * -flush_cache_dup_mm (fork) | ||
10 | * -likewise for flush_cache_mm (exit/execve) | ||
11 | * -likewise for flush_cache_{range,page} (munmap, exit, COW-break) | ||
12 | * | ||
13 | * vineetg: April 2008 | ||
14 | * -Added a critical CacheLine flush to copy_to_user_page( ) which | ||
15 | * was causing gdbserver to not setup breakpoints consistently | ||
16 | */ | ||
17 | |||
18 | #ifndef _ASM_CACHEFLUSH_H | ||
19 | #define _ASM_CACHEFLUSH_H | ||
20 | |||
21 | #include <linux/mm.h> | ||
22 | |||
23 | void flush_cache_all(void); | ||
24 | |||
25 | void flush_icache_range(unsigned long start, unsigned long end); | ||
26 | void flush_icache_page(struct vm_area_struct *vma, struct page *page); | ||
27 | void flush_icache_range_vaddr(unsigned long paddr, unsigned long u_vaddr, | ||
28 | int len); | ||
29 | |||
30 | #define ARCH_IMPLEMENTS_FLUSH_DCACHE_PAGE 1 | ||
31 | |||
32 | void flush_dcache_page(struct page *page); | ||
33 | |||
34 | void dma_cache_wback_inv(unsigned long start, unsigned long sz); | ||
35 | void dma_cache_inv(unsigned long start, unsigned long sz); | ||
36 | void dma_cache_wback(unsigned long start, unsigned long sz); | ||
37 | |||
38 | #define flush_dcache_mmap_lock(mapping) do { } while (0) | ||
39 | #define flush_dcache_mmap_unlock(mapping) do { } while (0) | ||
40 | |||
41 | /* TBD: optimize this */ | ||
42 | #define flush_cache_vmap(start, end) flush_cache_all() | ||
43 | #define flush_cache_vunmap(start, end) flush_cache_all() | ||
44 | |||
45 | /* | ||
46 | * VM callbacks when entire/range of user-space V-P mappings are | ||
47 | * torn-down/get-invalidated | ||
48 | * | ||
49 | * Currently we don't support D$ aliasing configs for our VIPT caches | ||
50 | * NOPS for VIPT Cache with non-aliasing D$ configurations only | ||
51 | */ | ||
52 | #define flush_cache_dup_mm(mm) /* called on fork */ | ||
53 | #define flush_cache_mm(mm) /* called on munmap/exit */ | ||
54 | #define flush_cache_range(mm, u_vstart, u_vend) | ||
55 | #define flush_cache_page(vma, u_vaddr, pfn) /* PF handling/COW-break */ | ||
56 | |||
57 | #define copy_to_user_page(vma, page, vaddr, dst, src, len) \ | ||
58 | do { \ | ||
59 | memcpy(dst, src, len); \ | ||
60 | if (vma->vm_flags & VM_EXEC) \ | ||
61 | flush_icache_range_vaddr((unsigned long)(dst), vaddr, len);\ | ||
62 | } while (0) | ||
63 | |||
64 | #define copy_from_user_page(vma, page, vaddr, dst, src, len) \ | ||
65 | memcpy(dst, src, len); \ | ||
66 | |||
67 | #endif | ||
diff --git a/arch/arc/include/asm/checksum.h b/arch/arc/include/asm/checksum.h new file mode 100644 index 000000000000..10957298b7a3 --- /dev/null +++ b/arch/arc/include/asm/checksum.h | |||
@@ -0,0 +1,101 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Joern Rennecke <joern.rennecke@embecosm.com>: Jan 2012 | ||
9 | * -Insn Scheduling improvements to csum core routines. | ||
10 | * = csum_fold( ) largely derived from ARM version. | ||
11 | * = ip_fast_cum( ) to have module scheduling | ||
12 | * -gcc 4.4.x broke networking. Alias analysis needed to be primed. | ||
13 | * worked around by adding memory clobber to ip_fast_csum( ) | ||
14 | * | ||
15 | * vineetg: May 2010 | ||
16 | * -Rewrote ip_fast_cscum( ) and csum_fold( ) with fast inline asm | ||
17 | */ | ||
18 | |||
19 | #ifndef _ASM_ARC_CHECKSUM_H | ||
20 | #define _ASM_ARC_CHECKSUM_H | ||
21 | |||
22 | /* | ||
23 | * Fold a partial checksum | ||
24 | * | ||
25 | * The 2 swords comprising the 32bit sum are added, any carry to 16th bit | ||
26 | * added back and final sword result inverted. | ||
27 | */ | ||
28 | static inline __sum16 csum_fold(__wsum s) | ||
29 | { | ||
30 | unsigned r = s << 16 | s >> 16; /* ror */ | ||
31 | s = ~s; | ||
32 | s -= r; | ||
33 | return s >> 16; | ||
34 | } | ||
35 | |||
36 | /* | ||
37 | * This is a version of ip_compute_csum() optimized for IP headers, | ||
38 | * which always checksum on 4 octet boundaries. | ||
39 | */ | ||
40 | static inline __sum16 | ||
41 | ip_fast_csum(const void *iph, unsigned int ihl) | ||
42 | { | ||
43 | const void *ptr = iph; | ||
44 | unsigned int tmp, tmp2, sum; | ||
45 | |||
46 | __asm__( | ||
47 | " ld.ab %0, [%3, 4] \n" | ||
48 | " ld.ab %2, [%3, 4] \n" | ||
49 | " sub %1, %4, 2 \n" | ||
50 | " lsr.f lp_count, %1, 1 \n" | ||
51 | " bcc 0f \n" | ||
52 | " add.f %0, %0, %2 \n" | ||
53 | " ld.ab %2, [%3, 4] \n" | ||
54 | "0: lp 1f \n" | ||
55 | " ld.ab %1, [%3, 4] \n" | ||
56 | " adc.f %0, %0, %2 \n" | ||
57 | " ld.ab %2, [%3, 4] \n" | ||
58 | " adc.f %0, %0, %1 \n" | ||
59 | "1: adc.f %0, %0, %2 \n" | ||
60 | " add.cs %0,%0,1 \n" | ||
61 | : "=&r"(sum), "=r"(tmp), "=&r"(tmp2), "+&r" (ptr) | ||
62 | : "r"(ihl) | ||
63 | : "cc", "lp_count", "memory"); | ||
64 | |||
65 | return csum_fold(sum); | ||
66 | } | ||
67 | |||
68 | /* | ||
69 | * TCP pseudo Header is 12 bytes: | ||
70 | * SA [4], DA [4], zeroes [1], Proto[1], TCP Seg(hdr+data) Len [2] | ||
71 | */ | ||
72 | static inline __wsum | ||
73 | csum_tcpudp_nofold(__be32 saddr, __be32 daddr, unsigned short len, | ||
74 | unsigned short proto, __wsum sum) | ||
75 | { | ||
76 | __asm__ __volatile__( | ||
77 | " add.f %0, %0, %1 \n" | ||
78 | " adc.f %0, %0, %2 \n" | ||
79 | " adc.f %0, %0, %3 \n" | ||
80 | " adc.f %0, %0, %4 \n" | ||
81 | " adc %0, %0, 0 \n" | ||
82 | : "+&r"(sum) | ||
83 | : "r"(saddr), "r"(daddr), | ||
84 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
85 | "r"(len), | ||
86 | #else | ||
87 | "r"(len << 8), | ||
88 | #endif | ||
89 | "r"(htons(proto)) | ||
90 | : "cc"); | ||
91 | |||
92 | return sum; | ||
93 | } | ||
94 | |||
95 | #define csum_fold csum_fold | ||
96 | #define ip_fast_csum ip_fast_csum | ||
97 | #define csum_tcpudp_nofold csum_tcpudp_nofold | ||
98 | |||
99 | #include <asm-generic/checksum.h> | ||
100 | |||
101 | #endif /* _ASM_ARC_CHECKSUM_H */ | ||
diff --git a/arch/arc/include/asm/clk.h b/arch/arc/include/asm/clk.h new file mode 100644 index 000000000000..bf9d29f5bd53 --- /dev/null +++ b/arch/arc/include/asm/clk.h | |||
@@ -0,0 +1,22 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_CLK_H | ||
10 | #define _ASM_ARC_CLK_H | ||
11 | |||
12 | /* Although we can't really hide core_freq, the accessor is still better way */ | ||
13 | extern unsigned long core_freq; | ||
14 | |||
15 | static inline unsigned long arc_get_core_freq(void) | ||
16 | { | ||
17 | return core_freq; | ||
18 | } | ||
19 | |||
20 | extern int arc_set_core_freq(unsigned long); | ||
21 | |||
22 | #endif | ||
diff --git a/arch/arc/include/asm/cmpxchg.h b/arch/arc/include/asm/cmpxchg.h new file mode 100644 index 000000000000..03cd6894855d --- /dev/null +++ b/arch/arc/include/asm/cmpxchg.h | |||
@@ -0,0 +1,143 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_ARC_CMPXCHG_H | ||
10 | #define __ASM_ARC_CMPXCHG_H | ||
11 | |||
12 | #include <linux/types.h> | ||
13 | #include <asm/smp.h> | ||
14 | |||
15 | #ifdef CONFIG_ARC_HAS_LLSC | ||
16 | |||
17 | static inline unsigned long | ||
18 | __cmpxchg(volatile void *ptr, unsigned long expected, unsigned long new) | ||
19 | { | ||
20 | unsigned long prev; | ||
21 | |||
22 | __asm__ __volatile__( | ||
23 | "1: llock %0, [%1] \n" | ||
24 | " brne %0, %2, 2f \n" | ||
25 | " scond %3, [%1] \n" | ||
26 | " bnz 1b \n" | ||
27 | "2: \n" | ||
28 | : "=&r"(prev) | ||
29 | : "r"(ptr), "ir"(expected), | ||
30 | "r"(new) /* can't be "ir". scond can't take limm for "b" */ | ||
31 | : "cc"); | ||
32 | |||
33 | return prev; | ||
34 | } | ||
35 | |||
36 | #else | ||
37 | |||
38 | static inline unsigned long | ||
39 | __cmpxchg(volatile void *ptr, unsigned long expected, unsigned long new) | ||
40 | { | ||
41 | unsigned long flags; | ||
42 | int prev; | ||
43 | volatile unsigned long *p = ptr; | ||
44 | |||
45 | atomic_ops_lock(flags); | ||
46 | prev = *p; | ||
47 | if (prev == expected) | ||
48 | *p = new; | ||
49 | atomic_ops_unlock(flags); | ||
50 | return prev; | ||
51 | } | ||
52 | |||
53 | #endif /* CONFIG_ARC_HAS_LLSC */ | ||
54 | |||
55 | #define cmpxchg(ptr, o, n) ((typeof(*(ptr)))__cmpxchg((ptr), \ | ||
56 | (unsigned long)(o), (unsigned long)(n))) | ||
57 | |||
58 | /* | ||
59 | * Since not supported natively, ARC cmpxchg() uses atomic_ops_lock (UP/SMP) | ||
60 | * just to gaurantee semantics. | ||
61 | * atomic_cmpxchg() needs to use the same locks as it's other atomic siblings | ||
62 | * which also happens to be atomic_ops_lock. | ||
63 | * | ||
64 | * Thus despite semantically being different, implementation of atomic_cmpxchg() | ||
65 | * is same as cmpxchg(). | ||
66 | */ | ||
67 | #define atomic_cmpxchg(v, o, n) ((int)cmpxchg(&((v)->counter), (o), (n))) | ||
68 | |||
69 | |||
70 | /* | ||
71 | * xchg (reg with memory) based on "Native atomic" EX insn | ||
72 | */ | ||
73 | static inline unsigned long __xchg(unsigned long val, volatile void *ptr, | ||
74 | int size) | ||
75 | { | ||
76 | extern unsigned long __xchg_bad_pointer(void); | ||
77 | |||
78 | switch (size) { | ||
79 | case 4: | ||
80 | __asm__ __volatile__( | ||
81 | " ex %0, [%1] \n" | ||
82 | : "+r"(val) | ||
83 | : "r"(ptr) | ||
84 | : "memory"); | ||
85 | |||
86 | return val; | ||
87 | } | ||
88 | return __xchg_bad_pointer(); | ||
89 | } | ||
90 | |||
91 | #define _xchg(ptr, with) ((typeof(*(ptr)))__xchg((unsigned long)(with), (ptr), \ | ||
92 | sizeof(*(ptr)))) | ||
93 | |||
94 | /* | ||
95 | * On ARC700, EX insn is inherently atomic, so by default "vanilla" xchg() need | ||
96 | * not require any locking. However there's a quirk. | ||
97 | * ARC lacks native CMPXCHG, thus emulated (see above), using external locking - | ||
98 | * incidently it "reuses" the same atomic_ops_lock used by atomic APIs. | ||
99 | * Now, llist code uses cmpxchg() and xchg() on same data, so xchg() needs to | ||
100 | * abide by same serializing rules, thus ends up using atomic_ops_lock as well. | ||
101 | * | ||
102 | * This however is only relevant if SMP and/or ARC lacks LLSC | ||
103 | * if (UP or LLSC) | ||
104 | * xchg doesn't need serialization | ||
105 | * else <==> !(UP or LLSC) <==> (!UP and !LLSC) <==> (SMP and !LLSC) | ||
106 | * xchg needs serialization | ||
107 | */ | ||
108 | |||
109 | #if !defined(CONFIG_ARC_HAS_LLSC) && defined(CONFIG_SMP) | ||
110 | |||
111 | #define xchg(ptr, with) \ | ||
112 | ({ \ | ||
113 | unsigned long flags; \ | ||
114 | typeof(*(ptr)) old_val; \ | ||
115 | \ | ||
116 | atomic_ops_lock(flags); \ | ||
117 | old_val = _xchg(ptr, with); \ | ||
118 | atomic_ops_unlock(flags); \ | ||
119 | old_val; \ | ||
120 | }) | ||
121 | |||
122 | #else | ||
123 | |||
124 | #define xchg(ptr, with) _xchg(ptr, with) | ||
125 | |||
126 | #endif | ||
127 | |||
128 | /* | ||
129 | * "atomic" variant of xchg() | ||
130 | * REQ: It needs to follow the same serialization rules as other atomic_xxx() | ||
131 | * Since xchg() doesn't always do that, it would seem that following defintion | ||
132 | * is incorrect. But here's the rationale: | ||
133 | * SMP : Even xchg() takes the atomic_ops_lock, so OK. | ||
134 | * LLSC: atomic_ops_lock are not relevent at all (even if SMP, since LLSC | ||
135 | * is natively "SMP safe", no serialization required). | ||
136 | * UP : other atomics disable IRQ, so no way a difft ctxt atomic_xchg() | ||
137 | * could clobber them. atomic_xchg() itself would be 1 insn, so it | ||
138 | * can't be clobbered by others. Thus no serialization required when | ||
139 | * atomic_xchg is involved. | ||
140 | */ | ||
141 | #define atomic_xchg(v, new) (xchg(&((v)->counter), new)) | ||
142 | |||
143 | #endif | ||
diff --git a/arch/arc/include/asm/current.h b/arch/arc/include/asm/current.h new file mode 100644 index 000000000000..87b918585c4a --- /dev/null +++ b/arch/arc/include/asm/current.h | |||
@@ -0,0 +1,32 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Vineetg: May 16th, 2008 | ||
9 | * - Current macro is now implemented as "global register" r25 | ||
10 | */ | ||
11 | |||
12 | #ifndef _ASM_ARC_CURRENT_H | ||
13 | #define _ASM_ARC_CURRENT_H | ||
14 | |||
15 | #ifdef __KERNEL__ | ||
16 | |||
17 | #ifndef __ASSEMBLY__ | ||
18 | |||
19 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
20 | |||
21 | register struct task_struct *curr_arc asm("r25"); | ||
22 | #define current (curr_arc) | ||
23 | |||
24 | #else | ||
25 | #include <asm-generic/current.h> | ||
26 | #endif /* ! CONFIG_ARC_CURR_IN_REG */ | ||
27 | |||
28 | #endif /* ! __ASSEMBLY__ */ | ||
29 | |||
30 | #endif /* __KERNEL__ */ | ||
31 | |||
32 | #endif /* _ASM_ARC_CURRENT_H */ | ||
diff --git a/arch/arc/include/asm/defines.h b/arch/arc/include/asm/defines.h new file mode 100644 index 000000000000..6097bb439cc5 --- /dev/null +++ b/arch/arc/include/asm/defines.h | |||
@@ -0,0 +1,56 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ARC_ASM_DEFINES_H__ | ||
10 | #define __ARC_ASM_DEFINES_H__ | ||
11 | |||
12 | #if defined(CONFIG_ARC_MMU_V1) | ||
13 | #define CONFIG_ARC_MMU_VER 1 | ||
14 | #elif defined(CONFIG_ARC_MMU_V2) | ||
15 | #define CONFIG_ARC_MMU_VER 2 | ||
16 | #elif defined(CONFIG_ARC_MMU_V3) | ||
17 | #define CONFIG_ARC_MMU_VER 3 | ||
18 | #endif | ||
19 | |||
20 | #ifdef CONFIG_ARC_HAS_LLSC | ||
21 | #define __CONFIG_ARC_HAS_LLSC_VAL 1 | ||
22 | #else | ||
23 | #define __CONFIG_ARC_HAS_LLSC_VAL 0 | ||
24 | #endif | ||
25 | |||
26 | #ifdef CONFIG_ARC_HAS_SWAPE | ||
27 | #define __CONFIG_ARC_HAS_SWAPE_VAL 1 | ||
28 | #else | ||
29 | #define __CONFIG_ARC_HAS_SWAPE_VAL 0 | ||
30 | #endif | ||
31 | |||
32 | #ifdef CONFIG_ARC_HAS_RTSC | ||
33 | #define __CONFIG_ARC_HAS_RTSC_VAL 1 | ||
34 | #else | ||
35 | #define __CONFIG_ARC_HAS_RTSC_VAL 0 | ||
36 | #endif | ||
37 | |||
38 | #ifdef CONFIG_ARC_MMU_SASID | ||
39 | #define __CONFIG_ARC_MMU_SASID_VAL 1 | ||
40 | #else | ||
41 | #define __CONFIG_ARC_MMU_SASID_VAL 0 | ||
42 | #endif | ||
43 | |||
44 | #ifdef CONFIG_ARC_HAS_ICACHE | ||
45 | #define __CONFIG_ARC_HAS_ICACHE 1 | ||
46 | #else | ||
47 | #define __CONFIG_ARC_HAS_ICACHE 0 | ||
48 | #endif | ||
49 | |||
50 | #ifdef CONFIG_ARC_HAS_DCACHE | ||
51 | #define __CONFIG_ARC_HAS_DCACHE 1 | ||
52 | #else | ||
53 | #define __CONFIG_ARC_HAS_DCACHE 0 | ||
54 | #endif | ||
55 | |||
56 | #endif /* __ARC_ASM_DEFINES_H__ */ | ||
diff --git a/arch/arc/include/asm/delay.h b/arch/arc/include/asm/delay.h new file mode 100644 index 000000000000..442ce5d0f709 --- /dev/null +++ b/arch/arc/include/asm/delay.h | |||
@@ -0,0 +1,68 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Delay routines using pre computed loops_per_jiffy value. | ||
9 | * | ||
10 | * vineetg: Feb 2012 | ||
11 | * -Rewrote in "C" to avoid dealing with availability of H/w MPY | ||
12 | * -Also reduced the num of MPY operations from 3 to 2 | ||
13 | * | ||
14 | * Amit Bhor: Codito Technologies 2004 | ||
15 | */ | ||
16 | |||
17 | #ifndef __ASM_ARC_UDELAY_H | ||
18 | #define __ASM_ARC_UDELAY_H | ||
19 | |||
20 | #include <asm/param.h> /* HZ */ | ||
21 | |||
22 | static inline void __delay(unsigned long loops) | ||
23 | { | ||
24 | __asm__ __volatile__( | ||
25 | "1: sub.f %0, %0, 1 \n" | ||
26 | " jpnz 1b \n" | ||
27 | : "+r"(loops) | ||
28 | : | ||
29 | : "cc"); | ||
30 | } | ||
31 | |||
32 | extern void __bad_udelay(void); | ||
33 | |||
34 | /* | ||
35 | * Normal Math for computing loops in "N" usecs | ||
36 | * -we have precomputed @loops_per_jiffy | ||
37 | * -1 sec has HZ jiffies | ||
38 | * loops per "N" usecs = ((loops_per_jiffy * HZ / 1000000) * N) | ||
39 | * | ||
40 | * Approximate Division by multiplication: | ||
41 | * -Mathematically if we multiply and divide a number by same value the | ||
42 | * result remains unchanged: In this case, we use 2^32 | ||
43 | * -> (loops_per_N_usec * 2^32 ) / 2^32 | ||
44 | * -> (((loops_per_jiffy * HZ / 1000000) * N) * 2^32) / 2^32 | ||
45 | * -> (loops_per_jiffy * HZ * N * 4295) / 2^32 | ||
46 | * | ||
47 | * -Divide by 2^32 is very simply right shift by 32 | ||
48 | * -We simply need to ensure that the multiply per above eqn happens in | ||
49 | * 64-bit precision (if CPU doesn't support it - gcc can emaulate it) | ||
50 | */ | ||
51 | |||
52 | static inline void __udelay(unsigned long usecs) | ||
53 | { | ||
54 | unsigned long loops; | ||
55 | |||
56 | /* (long long) cast ensures 64 bit MPY - real or emulated | ||
57 | * HZ * 4295 is pre-evaluated by gcc - hence only 2 mpy ops | ||
58 | */ | ||
59 | loops = ((long long)(usecs * 4295 * HZ) * | ||
60 | (long long)(loops_per_jiffy)) >> 32; | ||
61 | |||
62 | __delay(loops); | ||
63 | } | ||
64 | |||
65 | #define udelay(n) (__builtin_constant_p(n) ? ((n) > 20000 ? __bad_udelay() \ | ||
66 | : __udelay(n)) : __udelay(n)) | ||
67 | |||
68 | #endif /* __ASM_ARC_UDELAY_H */ | ||
diff --git a/arch/arc/include/asm/disasm.h b/arch/arc/include/asm/disasm.h new file mode 100644 index 000000000000..f1cce3d059a1 --- /dev/null +++ b/arch/arc/include/asm/disasm.h | |||
@@ -0,0 +1,116 @@ | |||
1 | /* | ||
2 | * several functions that help interpret ARC instructions | ||
3 | * used for unaligned accesses, kprobes and kgdb | ||
4 | * | ||
5 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
6 | * | ||
7 | * This program is free software; you can redistribute it and/or modify | ||
8 | * it under the terms of the GNU General Public License version 2 as | ||
9 | * published by the Free Software Foundation. | ||
10 | */ | ||
11 | |||
12 | #ifndef __ARC_DISASM_H__ | ||
13 | #define __ARC_DISASM_H__ | ||
14 | |||
15 | enum { | ||
16 | op_Bcc = 0, op_BLcc = 1, op_LD = 2, op_ST = 3, op_MAJOR_4 = 4, | ||
17 | op_MAJOR_5 = 5, op_LD_ADD = 12, op_ADD_SUB_SHIFT = 13, | ||
18 | op_ADD_MOV_CMP = 14, op_S = 15, op_LD_S = 16, op_LDB_S = 17, | ||
19 | op_LDW_S = 18, op_LDWX_S = 19, op_ST_S = 20, op_STB_S = 21, | ||
20 | op_STW_S = 22, op_Su5 = 23, op_SP = 24, op_GP = 25, | ||
21 | op_Pcl = 26, op_MOV_S = 27, op_ADD_CMP = 28, op_BR_S = 29, | ||
22 | op_B_S = 30, op_BL_S = 31 | ||
23 | }; | ||
24 | |||
25 | enum flow { | ||
26 | noflow, | ||
27 | direct_jump, | ||
28 | direct_call, | ||
29 | indirect_jump, | ||
30 | indirect_call, | ||
31 | invalid_instr | ||
32 | }; | ||
33 | |||
34 | #define IS_BIT(word, n) ((word) & (1<<n)) | ||
35 | #define BITS(word, s, e) (((word) >> (s)) & (~((-2) << ((e) - (s))))) | ||
36 | |||
37 | #define MAJOR_OPCODE(word) (BITS((word), 27, 31)) | ||
38 | #define MINOR_OPCODE(word) (BITS((word), 16, 21)) | ||
39 | #define FIELD_A(word) (BITS((word), 0, 5)) | ||
40 | #define FIELD_B(word) ((BITS((word), 12, 14)<<3) | \ | ||
41 | (BITS((word), 24, 26))) | ||
42 | #define FIELD_C(word) (BITS((word), 6, 11)) | ||
43 | #define FIELD_u6(word) FIELDC(word) | ||
44 | #define FIELD_s12(word) sign_extend(((BITS((word), 0, 5) << 6) | \ | ||
45 | BITS((word), 6, 11)), 12) | ||
46 | |||
47 | /* note that for BL/BRcc these two macro's need another AND statement to mask | ||
48 | * out bit 1 (make the result a multiple of 4) */ | ||
49 | #define FIELD_s9(word) sign_extend(((BITS(word, 15, 15) << 8) | \ | ||
50 | BITS(word, 16, 23)), 9) | ||
51 | #define FIELD_s21(word) sign_extend(((BITS(word, 6, 15) << 11) | \ | ||
52 | (BITS(word, 17, 26) << 1)), 12) | ||
53 | #define FIELD_s25(word) sign_extend(((BITS(word, 0, 3) << 21) | \ | ||
54 | (BITS(word, 6, 15) << 11) | \ | ||
55 | (BITS(word, 17, 26) << 1)), 12) | ||
56 | |||
57 | /* note: these operate on 16 bits! */ | ||
58 | #define FIELD_S_A(word) ((BITS((word), 2, 2)<<3) | BITS((word), 0, 2)) | ||
59 | #define FIELD_S_B(word) ((BITS((word), 10, 10)<<3) | \ | ||
60 | BITS((word), 8, 10)) | ||
61 | #define FIELD_S_C(word) ((BITS((word), 7, 7)<<3) | BITS((word), 5, 7)) | ||
62 | #define FIELD_S_H(word) ((BITS((word), 0, 2)<<3) | BITS((word), 5, 8)) | ||
63 | #define FIELD_S_u5(word) (BITS((word), 0, 4)) | ||
64 | #define FIELD_S_u6(word) (BITS((word), 0, 4) << 1) | ||
65 | #define FIELD_S_u7(word) (BITS((word), 0, 4) << 2) | ||
66 | #define FIELD_S_u10(word) (BITS((word), 0, 7) << 2) | ||
67 | #define FIELD_S_s7(word) sign_extend(BITS((word), 0, 5) << 1, 9) | ||
68 | #define FIELD_S_s8(word) sign_extend(BITS((word), 0, 7) << 1, 9) | ||
69 | #define FIELD_S_s9(word) sign_extend(BITS((word), 0, 8), 9) | ||
70 | #define FIELD_S_s10(word) sign_extend(BITS((word), 0, 8) << 1, 10) | ||
71 | #define FIELD_S_s11(word) sign_extend(BITS((word), 0, 8) << 2, 11) | ||
72 | #define FIELD_S_s13(word) sign_extend(BITS((word), 0, 10) << 2, 13) | ||
73 | |||
74 | #define STATUS32_L 0x00000100 | ||
75 | #define REG_LIMM 62 | ||
76 | |||
77 | struct disasm_state { | ||
78 | /* generic info */ | ||
79 | unsigned long words[2]; | ||
80 | int instr_len; | ||
81 | int major_opcode; | ||
82 | /* info for branch/jump */ | ||
83 | int is_branch; | ||
84 | int target; | ||
85 | int delay_slot; | ||
86 | enum flow flow; | ||
87 | /* info for load/store */ | ||
88 | int src1, src2, src3, dest, wb_reg; | ||
89 | int zz, aa, x, pref, di; | ||
90 | int fault, write; | ||
91 | }; | ||
92 | |||
93 | static inline int sign_extend(int value, int bits) | ||
94 | { | ||
95 | if (IS_BIT(value, (bits - 1))) | ||
96 | value |= (0xffffffff << bits); | ||
97 | |||
98 | return value; | ||
99 | } | ||
100 | |||
101 | static inline int is_short_instr(unsigned long addr) | ||
102 | { | ||
103 | uint16_t word = *((uint16_t *)addr); | ||
104 | int opcode = (word >> 11) & 0x1F; | ||
105 | return (opcode >= 0x0B); | ||
106 | } | ||
107 | |||
108 | void disasm_instr(unsigned long addr, struct disasm_state *state, | ||
109 | int userspace, struct pt_regs *regs, struct callee_regs *cregs); | ||
110 | int disasm_next_pc(unsigned long pc, struct pt_regs *regs, struct callee_regs | ||
111 | *cregs, unsigned long *fall_thru, unsigned long *target); | ||
112 | long get_reg(int reg, struct pt_regs *regs, struct callee_regs *cregs); | ||
113 | void set_reg(int reg, long val, struct pt_regs *regs, | ||
114 | struct callee_regs *cregs); | ||
115 | |||
116 | #endif /* __ARC_DISASM_H__ */ | ||
diff --git a/arch/arc/include/asm/dma-mapping.h b/arch/arc/include/asm/dma-mapping.h new file mode 100644 index 000000000000..31f77aec0823 --- /dev/null +++ b/arch/arc/include/asm/dma-mapping.h | |||
@@ -0,0 +1,221 @@ | |||
1 | /* | ||
2 | * DMA Mapping glue for ARC | ||
3 | * | ||
4 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | */ | ||
10 | |||
11 | #ifndef ASM_ARC_DMA_MAPPING_H | ||
12 | #define ASM_ARC_DMA_MAPPING_H | ||
13 | |||
14 | #include <asm-generic/dma-coherent.h> | ||
15 | #include <asm/cacheflush.h> | ||
16 | |||
17 | #ifndef CONFIG_ARC_PLAT_NEEDS_CPU_TO_DMA | ||
18 | /* | ||
19 | * dma_map_* API take cpu addresses, which is kernel logical address in the | ||
20 | * untranslated address space (0x8000_0000) based. The dma address (bus addr) | ||
21 | * ideally needs to be 0x0000_0000 based hence these glue routines. | ||
22 | * However given that intermediate bus bridges can ignore the high bit, we can | ||
23 | * do with these routines being no-ops. | ||
24 | * If a platform/device comes up which sriclty requires 0 based bus addr | ||
25 | * (e.g. AHB-PCI bridge on Angel4 board), then it can provide it's own versions | ||
26 | */ | ||
27 | #define plat_dma_addr_to_kernel(dev, addr) ((unsigned long)(addr)) | ||
28 | #define plat_kernel_addr_to_dma(dev, ptr) ((dma_addr_t)(ptr)) | ||
29 | |||
30 | #else | ||
31 | #include <plat/dma_addr.h> | ||
32 | #endif | ||
33 | |||
34 | void *dma_alloc_noncoherent(struct device *dev, size_t size, | ||
35 | dma_addr_t *dma_handle, gfp_t gfp); | ||
36 | |||
37 | void dma_free_noncoherent(struct device *dev, size_t size, void *vaddr, | ||
38 | dma_addr_t dma_handle); | ||
39 | |||
40 | void *dma_alloc_coherent(struct device *dev, size_t size, | ||
41 | dma_addr_t *dma_handle, gfp_t gfp); | ||
42 | |||
43 | void dma_free_coherent(struct device *dev, size_t size, void *kvaddr, | ||
44 | dma_addr_t dma_handle); | ||
45 | |||
46 | /* drivers/base/dma-mapping.c */ | ||
47 | extern int dma_common_mmap(struct device *dev, struct vm_area_struct *vma, | ||
48 | void *cpu_addr, dma_addr_t dma_addr, size_t size); | ||
49 | extern int dma_common_get_sgtable(struct device *dev, struct sg_table *sgt, | ||
50 | void *cpu_addr, dma_addr_t dma_addr, | ||
51 | size_t size); | ||
52 | |||
53 | #define dma_mmap_coherent(d, v, c, h, s) dma_common_mmap(d, v, c, h, s) | ||
54 | #define dma_get_sgtable(d, t, v, h, s) dma_common_get_sgtable(d, t, v, h, s) | ||
55 | |||
56 | /* | ||
57 | * streaming DMA Mapping API... | ||
58 | * CPU accesses page via normal paddr, thus needs to explicitly made | ||
59 | * consistent before each use | ||
60 | */ | ||
61 | |||
62 | static inline void __inline_dma_cache_sync(unsigned long paddr, size_t size, | ||
63 | enum dma_data_direction dir) | ||
64 | { | ||
65 | switch (dir) { | ||
66 | case DMA_FROM_DEVICE: | ||
67 | dma_cache_inv(paddr, size); | ||
68 | break; | ||
69 | case DMA_TO_DEVICE: | ||
70 | dma_cache_wback(paddr, size); | ||
71 | break; | ||
72 | case DMA_BIDIRECTIONAL: | ||
73 | dma_cache_wback_inv(paddr, size); | ||
74 | break; | ||
75 | default: | ||
76 | pr_err("Invalid DMA dir [%d] for OP @ %lx\n", dir, paddr); | ||
77 | } | ||
78 | } | ||
79 | |||
80 | void __arc_dma_cache_sync(unsigned long paddr, size_t size, | ||
81 | enum dma_data_direction dir); | ||
82 | |||
83 | #define _dma_cache_sync(addr, sz, dir) \ | ||
84 | do { \ | ||
85 | if (__builtin_constant_p(dir)) \ | ||
86 | __inline_dma_cache_sync(addr, sz, dir); \ | ||
87 | else \ | ||
88 | __arc_dma_cache_sync(addr, sz, dir); \ | ||
89 | } \ | ||
90 | while (0); | ||
91 | |||
92 | static inline dma_addr_t | ||
93 | dma_map_single(struct device *dev, void *cpu_addr, size_t size, | ||
94 | enum dma_data_direction dir) | ||
95 | { | ||
96 | _dma_cache_sync((unsigned long)cpu_addr, size, dir); | ||
97 | return plat_kernel_addr_to_dma(dev, cpu_addr); | ||
98 | } | ||
99 | |||
100 | static inline void | ||
101 | dma_unmap_single(struct device *dev, dma_addr_t dma_addr, | ||
102 | size_t size, enum dma_data_direction dir) | ||
103 | { | ||
104 | } | ||
105 | |||
106 | static inline dma_addr_t | ||
107 | dma_map_page(struct device *dev, struct page *page, | ||
108 | unsigned long offset, size_t size, | ||
109 | enum dma_data_direction dir) | ||
110 | { | ||
111 | unsigned long paddr = page_to_phys(page) + offset; | ||
112 | return dma_map_single(dev, (void *)paddr, size, dir); | ||
113 | } | ||
114 | |||
115 | static inline void | ||
116 | dma_unmap_page(struct device *dev, dma_addr_t dma_handle, | ||
117 | size_t size, enum dma_data_direction dir) | ||
118 | { | ||
119 | } | ||
120 | |||
121 | static inline int | ||
122 | dma_map_sg(struct device *dev, struct scatterlist *sg, | ||
123 | int nents, enum dma_data_direction dir) | ||
124 | { | ||
125 | struct scatterlist *s; | ||
126 | int i; | ||
127 | |||
128 | for_each_sg(sg, s, nents, i) | ||
129 | sg->dma_address = dma_map_page(dev, sg_page(s), s->offset, | ||
130 | s->length, dir); | ||
131 | |||
132 | return nents; | ||
133 | } | ||
134 | |||
135 | static inline void | ||
136 | dma_unmap_sg(struct device *dev, struct scatterlist *sg, | ||
137 | int nents, enum dma_data_direction dir) | ||
138 | { | ||
139 | struct scatterlist *s; | ||
140 | int i; | ||
141 | |||
142 | for_each_sg(sg, s, nents, i) | ||
143 | dma_unmap_page(dev, sg_dma_address(s), sg_dma_len(s), dir); | ||
144 | } | ||
145 | |||
146 | static inline void | ||
147 | dma_sync_single_for_cpu(struct device *dev, dma_addr_t dma_handle, | ||
148 | size_t size, enum dma_data_direction dir) | ||
149 | { | ||
150 | _dma_cache_sync(plat_dma_addr_to_kernel(dev, dma_handle), size, | ||
151 | DMA_FROM_DEVICE); | ||
152 | } | ||
153 | |||
154 | static inline void | ||
155 | dma_sync_single_for_device(struct device *dev, dma_addr_t dma_handle, | ||
156 | size_t size, enum dma_data_direction dir) | ||
157 | { | ||
158 | _dma_cache_sync(plat_dma_addr_to_kernel(dev, dma_handle), size, | ||
159 | DMA_TO_DEVICE); | ||
160 | } | ||
161 | |||
162 | static inline void | ||
163 | dma_sync_single_range_for_cpu(struct device *dev, dma_addr_t dma_handle, | ||
164 | unsigned long offset, size_t size, | ||
165 | enum dma_data_direction direction) | ||
166 | { | ||
167 | _dma_cache_sync(plat_dma_addr_to_kernel(dev, dma_handle) + offset, | ||
168 | size, DMA_FROM_DEVICE); | ||
169 | } | ||
170 | |||
171 | static inline void | ||
172 | dma_sync_single_range_for_device(struct device *dev, dma_addr_t dma_handle, | ||
173 | unsigned long offset, size_t size, | ||
174 | enum dma_data_direction direction) | ||
175 | { | ||
176 | _dma_cache_sync(plat_dma_addr_to_kernel(dev, dma_handle) + offset, | ||
177 | size, DMA_TO_DEVICE); | ||
178 | } | ||
179 | |||
180 | static inline void | ||
181 | dma_sync_sg_for_cpu(struct device *dev, struct scatterlist *sg, int nelems, | ||
182 | enum dma_data_direction dir) | ||
183 | { | ||
184 | int i; | ||
185 | |||
186 | for (i = 0; i < nelems; i++, sg++) | ||
187 | _dma_cache_sync((unsigned int)sg_virt(sg), sg->length, dir); | ||
188 | } | ||
189 | |||
190 | static inline void | ||
191 | dma_sync_sg_for_device(struct device *dev, struct scatterlist *sg, int nelems, | ||
192 | enum dma_data_direction dir) | ||
193 | { | ||
194 | int i; | ||
195 | |||
196 | for (i = 0; i < nelems; i++, sg++) | ||
197 | _dma_cache_sync((unsigned int)sg_virt(sg), sg->length, dir); | ||
198 | } | ||
199 | |||
200 | static inline int dma_supported(struct device *dev, u64 dma_mask) | ||
201 | { | ||
202 | /* Support 32 bit DMA mask exclusively */ | ||
203 | return dma_mask == DMA_BIT_MASK(32); | ||
204 | } | ||
205 | |||
206 | static inline int dma_mapping_error(struct device *dev, dma_addr_t dma_addr) | ||
207 | { | ||
208 | return 0; | ||
209 | } | ||
210 | |||
211 | static inline int dma_set_mask(struct device *dev, u64 dma_mask) | ||
212 | { | ||
213 | if (!dev->dma_mask || !dma_supported(dev, dma_mask)) | ||
214 | return -EIO; | ||
215 | |||
216 | *dev->dma_mask = dma_mask; | ||
217 | |||
218 | return 0; | ||
219 | } | ||
220 | |||
221 | #endif | ||
diff --git a/arch/arc/include/asm/dma.h b/arch/arc/include/asm/dma.h new file mode 100644 index 000000000000..ca7c45181de9 --- /dev/null +++ b/arch/arc/include/asm/dma.h | |||
@@ -0,0 +1,14 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef ASM_ARC_DMA_H | ||
10 | #define ASM_ARC_DMA_H | ||
11 | |||
12 | #define MAX_DMA_ADDRESS 0xC0000000 | ||
13 | |||
14 | #endif | ||
diff --git a/arch/arc/include/asm/elf.h b/arch/arc/include/asm/elf.h new file mode 100644 index 000000000000..f4c8d36ebecb --- /dev/null +++ b/arch/arc/include/asm/elf.h | |||
@@ -0,0 +1,78 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_ARC_ELF_H | ||
10 | #define __ASM_ARC_ELF_H | ||
11 | |||
12 | #include <linux/types.h> | ||
13 | #include <uapi/asm/elf.h> | ||
14 | |||
15 | /* These ELF defines belong to uapi but libc elf.h already defines them */ | ||
16 | #define EM_ARCOMPACT 93 | ||
17 | |||
18 | /* ARC Relocations (kernel Modules only) */ | ||
19 | #define R_ARC_32 0x4 | ||
20 | #define R_ARC_32_ME 0x1B | ||
21 | #define R_ARC_S25H_PCREL 0x10 | ||
22 | #define R_ARC_S25W_PCREL 0x11 | ||
23 | |||
24 | /*to set parameters in the core dumps */ | ||
25 | #define ELF_ARCH EM_ARCOMPACT | ||
26 | #define ELF_CLASS ELFCLASS32 | ||
27 | |||
28 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
29 | #define ELF_DATA ELFDATA2MSB | ||
30 | #else | ||
31 | #define ELF_DATA ELFDATA2LSB | ||
32 | #endif | ||
33 | |||
34 | /* | ||
35 | * To ensure that | ||
36 | * -we don't load something for the wrong architecture. | ||
37 | * -The userspace is using the correct syscall ABI | ||
38 | */ | ||
39 | struct elf32_hdr; | ||
40 | extern int elf_check_arch(const struct elf32_hdr *); | ||
41 | #define elf_check_arch elf_check_arch | ||
42 | |||
43 | #define CORE_DUMP_USE_REGSET | ||
44 | |||
45 | #define ELF_EXEC_PAGESIZE PAGE_SIZE | ||
46 | |||
47 | /* | ||
48 | * This is the location that an ET_DYN program is loaded if exec'ed. Typical | ||
49 | * use of this is to invoke "./ld.so someprog" to test out a new version of | ||
50 | * the loader. We need to make sure that it is out of the way of the program | ||
51 | * that it will "exec", and that there is sufficient room for the brk. | ||
52 | */ | ||
53 | #define ELF_ET_DYN_BASE (2 * TASK_SIZE / 3) | ||
54 | |||
55 | /* | ||
56 | * When the program starts, a1 contains a pointer to a function to be | ||
57 | * registered with atexit, as per the SVR4 ABI. A value of 0 means we | ||
58 | * have no such handler. | ||
59 | */ | ||
60 | #define ELF_PLAT_INIT(_r, load_addr) ((_r)->r0 = 0) | ||
61 | |||
62 | /* | ||
63 | * This yields a mask that user programs can use to figure out what | ||
64 | * instruction set this cpu supports. | ||
65 | */ | ||
66 | #define ELF_HWCAP (0) | ||
67 | |||
68 | /* | ||
69 | * This yields a string that ld.so will use to load implementation | ||
70 | * specific libraries for optimization. This is more specific in | ||
71 | * intent than poking at uname or /proc/cpuinfo. | ||
72 | */ | ||
73 | #define ELF_PLATFORM (NULL) | ||
74 | |||
75 | #define SET_PERSONALITY(ex) \ | ||
76 | set_personality(PER_LINUX | (current->personality & (~PER_MASK))) | ||
77 | |||
78 | #endif | ||
diff --git a/arch/arc/include/asm/entry.h b/arch/arc/include/asm/entry.h new file mode 100644 index 000000000000..23daa326fc9b --- /dev/null +++ b/arch/arc/include/asm/entry.h | |||
@@ -0,0 +1,724 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Vineetg: March 2009 (Supporting 2 levels of Interrupts) | ||
9 | * Stack switching code can no longer reliably rely on the fact that | ||
10 | * if we are NOT in user mode, stack is switched to kernel mode. | ||
11 | * e.g. L2 IRQ interrupted a L1 ISR which had not yet completed | ||
12 | * it's prologue including stack switching from user mode | ||
13 | * | ||
14 | * Vineetg: Aug 28th 2008: Bug #94984 | ||
15 | * -Zero Overhead Loop Context shd be cleared when entering IRQ/EXcp/Trap | ||
16 | * Normally CPU does this automatically, however when doing FAKE rtie, | ||
17 | * we also need to explicitly do this. The problem in macros | ||
18 | * FAKE_RET_FROM_EXCPN and FAKE_RET_FROM_EXCPN_LOCK_IRQ was that this bit | ||
19 | * was being "CLEARED" rather then "SET". Actually "SET" clears ZOL context | ||
20 | * | ||
21 | * Vineetg: May 5th 2008 | ||
22 | * -Modified CALLEE_REG save/restore macros to handle the fact that | ||
23 | * r25 contains the kernel current task ptr | ||
24 | * - Defined Stack Switching Macro to be reused in all intr/excp hdlrs | ||
25 | * - Shaved off 11 instructions from RESTORE_ALL_INT1 by using the | ||
26 | * address Write back load ld.ab instead of seperate ld/add instn | ||
27 | * | ||
28 | * Amit Bhor, Sameer Dhavale: Codito Technologies 2004 | ||
29 | */ | ||
30 | |||
31 | #ifndef __ASM_ARC_ENTRY_H | ||
32 | #define __ASM_ARC_ENTRY_H | ||
33 | |||
34 | #ifdef __ASSEMBLY__ | ||
35 | #include <asm/unistd.h> /* For NR_syscalls defination */ | ||
36 | #include <asm/asm-offsets.h> | ||
37 | #include <asm/arcregs.h> | ||
38 | #include <asm/ptrace.h> | ||
39 | #include <asm/processor.h> /* For VMALLOC_START */ | ||
40 | #include <asm/thread_info.h> /* For THREAD_SIZE */ | ||
41 | |||
42 | /* Note on the LD/ST addr modes with addr reg wback | ||
43 | * | ||
44 | * LD.a same as LD.aw | ||
45 | * | ||
46 | * LD.a reg1, [reg2, x] => Pre Incr | ||
47 | * Eff Addr for load = [reg2 + x] | ||
48 | * | ||
49 | * LD.ab reg1, [reg2, x] => Post Incr | ||
50 | * Eff Addr for load = [reg2] | ||
51 | */ | ||
52 | |||
53 | /*-------------------------------------------------------------- | ||
54 | * Save caller saved registers (scratch registers) ( r0 - r12 ) | ||
55 | * Registers are pushed / popped in the order defined in struct ptregs | ||
56 | * in asm/ptrace.h | ||
57 | *-------------------------------------------------------------*/ | ||
58 | .macro SAVE_CALLER_SAVED | ||
59 | st.a r0, [sp, -4] | ||
60 | st.a r1, [sp, -4] | ||
61 | st.a r2, [sp, -4] | ||
62 | st.a r3, [sp, -4] | ||
63 | st.a r4, [sp, -4] | ||
64 | st.a r5, [sp, -4] | ||
65 | st.a r6, [sp, -4] | ||
66 | st.a r7, [sp, -4] | ||
67 | st.a r8, [sp, -4] | ||
68 | st.a r9, [sp, -4] | ||
69 | st.a r10, [sp, -4] | ||
70 | st.a r11, [sp, -4] | ||
71 | st.a r12, [sp, -4] | ||
72 | .endm | ||
73 | |||
74 | /*-------------------------------------------------------------- | ||
75 | * Restore caller saved registers (scratch registers) | ||
76 | *-------------------------------------------------------------*/ | ||
77 | .macro RESTORE_CALLER_SAVED | ||
78 | ld.ab r12, [sp, 4] | ||
79 | ld.ab r11, [sp, 4] | ||
80 | ld.ab r10, [sp, 4] | ||
81 | ld.ab r9, [sp, 4] | ||
82 | ld.ab r8, [sp, 4] | ||
83 | ld.ab r7, [sp, 4] | ||
84 | ld.ab r6, [sp, 4] | ||
85 | ld.ab r5, [sp, 4] | ||
86 | ld.ab r4, [sp, 4] | ||
87 | ld.ab r3, [sp, 4] | ||
88 | ld.ab r2, [sp, 4] | ||
89 | ld.ab r1, [sp, 4] | ||
90 | ld.ab r0, [sp, 4] | ||
91 | .endm | ||
92 | |||
93 | |||
94 | /*-------------------------------------------------------------- | ||
95 | * Save callee saved registers (non scratch registers) ( r13 - r25 ) | ||
96 | * on kernel stack. | ||
97 | * User mode callee regs need to be saved in case of | ||
98 | * -fork and friends for replicating from parent to child | ||
99 | * -before going into do_signal( ) for ptrace/core-dump | ||
100 | * Special case handling is required for r25 in case it is used by kernel | ||
101 | * for caching task ptr. Low level exception/ISR save user mode r25 | ||
102 | * into task->thread.user_r25. So it needs to be retrieved from there and | ||
103 | * saved into kernel stack with rest of callee reg-file | ||
104 | *-------------------------------------------------------------*/ | ||
105 | .macro SAVE_CALLEE_SAVED_USER | ||
106 | st.a r13, [sp, -4] | ||
107 | st.a r14, [sp, -4] | ||
108 | st.a r15, [sp, -4] | ||
109 | st.a r16, [sp, -4] | ||
110 | st.a r17, [sp, -4] | ||
111 | st.a r18, [sp, -4] | ||
112 | st.a r19, [sp, -4] | ||
113 | st.a r20, [sp, -4] | ||
114 | st.a r21, [sp, -4] | ||
115 | st.a r22, [sp, -4] | ||
116 | st.a r23, [sp, -4] | ||
117 | st.a r24, [sp, -4] | ||
118 | |||
119 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
120 | ; Retrieve orig r25 and save it on stack | ||
121 | ld r12, [r25, TASK_THREAD + THREAD_USER_R25] | ||
122 | st.a r12, [sp, -4] | ||
123 | #else | ||
124 | st.a r25, [sp, -4] | ||
125 | #endif | ||
126 | |||
127 | /* move up by 1 word to "create" callee_regs->"stack_place_holder" */ | ||
128 | sub sp, sp, 4 | ||
129 | .endm | ||
130 | |||
131 | /*-------------------------------------------------------------- | ||
132 | * Save callee saved registers (non scratch registers) ( r13 - r25 ) | ||
133 | * kernel mode callee regs needed to be saved in case of context switch | ||
134 | * If r25 is used for caching task pointer then that need not be saved | ||
135 | * as it can be re-created from current task global | ||
136 | *-------------------------------------------------------------*/ | ||
137 | .macro SAVE_CALLEE_SAVED_KERNEL | ||
138 | st.a r13, [sp, -4] | ||
139 | st.a r14, [sp, -4] | ||
140 | st.a r15, [sp, -4] | ||
141 | st.a r16, [sp, -4] | ||
142 | st.a r17, [sp, -4] | ||
143 | st.a r18, [sp, -4] | ||
144 | st.a r19, [sp, -4] | ||
145 | st.a r20, [sp, -4] | ||
146 | st.a r21, [sp, -4] | ||
147 | st.a r22, [sp, -4] | ||
148 | st.a r23, [sp, -4] | ||
149 | st.a r24, [sp, -4] | ||
150 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
151 | sub sp, sp, 8 | ||
152 | #else | ||
153 | st.a r25, [sp, -4] | ||
154 | sub sp, sp, 4 | ||
155 | #endif | ||
156 | .endm | ||
157 | |||
158 | /*-------------------------------------------------------------- | ||
159 | * RESTORE_CALLEE_SAVED_KERNEL: | ||
160 | * Loads callee (non scratch) Reg File by popping from Kernel mode stack. | ||
161 | * This is reverse of SAVE_CALLEE_SAVED, | ||
162 | * | ||
163 | * NOTE: | ||
164 | * Ideally this shd only be called in switch_to for loading | ||
165 | * switched-IN task's CALLEE Reg File. | ||
166 | * For all other cases RESTORE_CALLEE_SAVED_FAST must be used | ||
167 | * which simply pops the stack w/o touching regs. | ||
168 | *-------------------------------------------------------------*/ | ||
169 | .macro RESTORE_CALLEE_SAVED_KERNEL | ||
170 | |||
171 | |||
172 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
173 | add sp, sp, 8 /* skip callee_reg gutter and user r25 placeholder */ | ||
174 | #else | ||
175 | add sp, sp, 4 /* skip "callee_regs->stack_place_holder" */ | ||
176 | ld.ab r25, [sp, 4] | ||
177 | #endif | ||
178 | |||
179 | ld.ab r24, [sp, 4] | ||
180 | ld.ab r23, [sp, 4] | ||
181 | ld.ab r22, [sp, 4] | ||
182 | ld.ab r21, [sp, 4] | ||
183 | ld.ab r20, [sp, 4] | ||
184 | ld.ab r19, [sp, 4] | ||
185 | ld.ab r18, [sp, 4] | ||
186 | ld.ab r17, [sp, 4] | ||
187 | ld.ab r16, [sp, 4] | ||
188 | ld.ab r15, [sp, 4] | ||
189 | ld.ab r14, [sp, 4] | ||
190 | ld.ab r13, [sp, 4] | ||
191 | |||
192 | .endm | ||
193 | |||
194 | /*-------------------------------------------------------------- | ||
195 | * RESTORE_CALLEE_SAVED_USER: | ||
196 | * This is called after do_signal where tracer might have changed callee regs | ||
197 | * thus we need to restore the reg file. | ||
198 | * Special case handling is required for r25 in case it is used by kernel | ||
199 | * for caching task ptr. Ptrace would have modified on-kernel-stack value of | ||
200 | * r25, which needs to be shoved back into task->thread.user_r25 where from | ||
201 | * Low level exception/ISR return code will retrieve to populate with rest of | ||
202 | * callee reg-file. | ||
203 | *-------------------------------------------------------------*/ | ||
204 | .macro RESTORE_CALLEE_SAVED_USER | ||
205 | |||
206 | add sp, sp, 4 /* skip "callee_regs->stack_place_holder" */ | ||
207 | |||
208 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
209 | ld.ab r12, [sp, 4] | ||
210 | st r12, [r25, TASK_THREAD + THREAD_USER_R25] | ||
211 | #else | ||
212 | ld.ab r25, [sp, 4] | ||
213 | #endif | ||
214 | |||
215 | ld.ab r24, [sp, 4] | ||
216 | ld.ab r23, [sp, 4] | ||
217 | ld.ab r22, [sp, 4] | ||
218 | ld.ab r21, [sp, 4] | ||
219 | ld.ab r20, [sp, 4] | ||
220 | ld.ab r19, [sp, 4] | ||
221 | ld.ab r18, [sp, 4] | ||
222 | ld.ab r17, [sp, 4] | ||
223 | ld.ab r16, [sp, 4] | ||
224 | ld.ab r15, [sp, 4] | ||
225 | ld.ab r14, [sp, 4] | ||
226 | ld.ab r13, [sp, 4] | ||
227 | .endm | ||
228 | |||
229 | /*-------------------------------------------------------------- | ||
230 | * Super FAST Restore callee saved regs by simply re-adjusting SP | ||
231 | *-------------------------------------------------------------*/ | ||
232 | .macro DISCARD_CALLEE_SAVED_USER | ||
233 | add sp, sp, 14 * 4 | ||
234 | .endm | ||
235 | |||
236 | /*-------------------------------------------------------------- | ||
237 | * Restore User mode r25 saved in task_struct->thread.user_r25 | ||
238 | *-------------------------------------------------------------*/ | ||
239 | .macro RESTORE_USER_R25 | ||
240 | ld r25, [r25, TASK_THREAD + THREAD_USER_R25] | ||
241 | .endm | ||
242 | |||
243 | /*------------------------------------------------------------- | ||
244 | * given a tsk struct, get to the base of it's kernel mode stack | ||
245 | * tsk->thread_info is really a PAGE, whose bottom hoists stack | ||
246 | * which grows upwards towards thread_info | ||
247 | *------------------------------------------------------------*/ | ||
248 | |||
249 | .macro GET_TSK_STACK_BASE tsk, out | ||
250 | |||
251 | /* Get task->thread_info (this is essentially start of a PAGE) */ | ||
252 | ld \out, [\tsk, TASK_THREAD_INFO] | ||
253 | |||
254 | /* Go to end of page where stack begins (grows upwards) */ | ||
255 | add2 \out, \out, (THREAD_SIZE - 4)/4 /* one word GUTTER */ | ||
256 | |||
257 | .endm | ||
258 | |||
259 | /*-------------------------------------------------------------- | ||
260 | * Switch to Kernel Mode stack if SP points to User Mode stack | ||
261 | * | ||
262 | * Entry : r9 contains pre-IRQ/exception/trap status32 | ||
263 | * Exit : SP is set to kernel mode stack pointer | ||
264 | * If CURR_IN_REG, r25 set to "current" task pointer | ||
265 | * Clobbers: r9 | ||
266 | *-------------------------------------------------------------*/ | ||
267 | |||
268 | .macro SWITCH_TO_KERNEL_STK | ||
269 | |||
270 | /* User Mode when this happened ? Yes: Proceed to switch stack */ | ||
271 | bbit1 r9, STATUS_U_BIT, 88f | ||
272 | |||
273 | /* OK we were already in kernel mode when this event happened, thus can | ||
274 | * assume SP is kernel mode SP. _NO_ need to do any stack switching | ||
275 | */ | ||
276 | |||
277 | #ifdef CONFIG_ARC_COMPACT_IRQ_LEVELS | ||
278 | /* However.... | ||
279 | * If Level 2 Interrupts enabled, we may end up with a corner case: | ||
280 | * 1. User Task executing | ||
281 | * 2. L1 IRQ taken, ISR starts (CPU auto-switched to KERNEL mode) | ||
282 | * 3. But before it could switch SP from USER to KERNEL stack | ||
283 | * a L2 IRQ "Interrupts" L1 | ||
284 | * Thay way although L2 IRQ happened in Kernel mode, stack is still | ||
285 | * not switched. | ||
286 | * To handle this, we may need to switch stack even if in kernel mode | ||
287 | * provided SP has values in range of USER mode stack ( < 0x7000_0000 ) | ||
288 | */ | ||
289 | brlo sp, VMALLOC_START, 88f | ||
290 | |||
291 | /* TODO: vineetg: | ||
292 | * We need to be a bit more cautious here. What if a kernel bug in | ||
293 | * L1 ISR, caused SP to go whaco (some small value which looks like | ||
294 | * USER stk) and then we take L2 ISR. | ||
295 | * Above brlo alone would treat it as a valid L1-L2 sceanrio | ||
296 | * instead of shouting alound | ||
297 | * The only feasible way is to make sure this L2 happened in | ||
298 | * L1 prelogue ONLY i.e. ilink2 is less than a pre-set marker in | ||
299 | * L1 ISR before it switches stack | ||
300 | */ | ||
301 | |||
302 | #endif | ||
303 | |||
304 | /* Save Pre Intr/Exception KERNEL MODE SP on kernel stack | ||
305 | * safe-keeping not really needed, but it keeps the epilogue code | ||
306 | * (SP restore) simpler/uniform. | ||
307 | */ | ||
308 | b.d 77f | ||
309 | |||
310 | st.a sp, [sp, -12] ; Make room for orig_r0 and orig_r8 | ||
311 | |||
312 | 88: /*------Intr/Ecxp happened in user mode, "switch" stack ------ */ | ||
313 | |||
314 | GET_CURR_TASK_ON_CPU r9 | ||
315 | |||
316 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
317 | |||
318 | /* If current task pointer cached in r25, time to | ||
319 | * -safekeep USER r25 in task->thread_struct->user_r25 | ||
320 | * -load r25 with current task ptr | ||
321 | */ | ||
322 | st.as r25, [r9, (TASK_THREAD + THREAD_USER_R25)/4] | ||
323 | mov r25, r9 | ||
324 | #endif | ||
325 | |||
326 | /* With current tsk in r9, get it's kernel mode stack base */ | ||
327 | GET_TSK_STACK_BASE r9, r9 | ||
328 | |||
329 | #ifdef PT_REGS_CANARY | ||
330 | st 0xabcdabcd, [r9, 0] | ||
331 | #endif | ||
332 | |||
333 | /* Save Pre Intr/Exception User SP on kernel stack */ | ||
334 | st.a sp, [r9, -12] ; Make room for orig_r0 and orig_r8 | ||
335 | |||
336 | /* CAUTION: | ||
337 | * SP should be set at the very end when we are done with everything | ||
338 | * In case of 2 levels of interrupt we depend on value of SP to assume | ||
339 | * that everything else is done (loading r25 etc) | ||
340 | */ | ||
341 | |||
342 | /* set SP to point to kernel mode stack */ | ||
343 | mov sp, r9 | ||
344 | |||
345 | 77: /* ----- Stack Switched to kernel Mode, Now save REG FILE ----- */ | ||
346 | |||
347 | .endm | ||
348 | |||
349 | /*------------------------------------------------------------ | ||
350 | * "FAKE" a rtie to return from CPU Exception context | ||
351 | * This is to re-enable Exceptions within exception | ||
352 | * Look at EV_ProtV to see how this is actually used | ||
353 | *-------------------------------------------------------------*/ | ||
354 | |||
355 | .macro FAKE_RET_FROM_EXCPN reg | ||
356 | |||
357 | ld \reg, [sp, PT_status32] | ||
358 | bic \reg, \reg, (STATUS_U_MASK|STATUS_DE_MASK) | ||
359 | bset \reg, \reg, STATUS_L_BIT | ||
360 | sr \reg, [erstatus] | ||
361 | mov \reg, 55f | ||
362 | sr \reg, [eret] | ||
363 | |||
364 | rtie | ||
365 | 55: | ||
366 | .endm | ||
367 | |||
368 | /* | ||
369 | * @reg [OUT] &thread_info of "current" | ||
370 | */ | ||
371 | .macro GET_CURR_THR_INFO_FROM_SP reg | ||
372 | and \reg, sp, ~(THREAD_SIZE - 1) | ||
373 | .endm | ||
374 | |||
375 | /* | ||
376 | * @reg [OUT] thread_info->flags of "current" | ||
377 | */ | ||
378 | .macro GET_CURR_THR_INFO_FLAGS reg | ||
379 | GET_CURR_THR_INFO_FROM_SP \reg | ||
380 | ld \reg, [\reg, THREAD_INFO_FLAGS] | ||
381 | .endm | ||
382 | |||
383 | /*-------------------------------------------------------------- | ||
384 | * For early Exception Prologue, a core reg is temporarily needed to | ||
385 | * code the rest of prolog (stack switching). This is done by stashing | ||
386 | * it to memory (non-SMP case) or SCRATCH0 Aux Reg (SMP). | ||
387 | * | ||
388 | * Before saving the full regfile - this reg is restored back, only | ||
389 | * to be saved again on kernel mode stack, as part of ptregs. | ||
390 | *-------------------------------------------------------------*/ | ||
391 | .macro EXCPN_PROLOG_FREEUP_REG reg | ||
392 | #ifdef CONFIG_SMP | ||
393 | sr \reg, [ARC_REG_SCRATCH_DATA0] | ||
394 | #else | ||
395 | st \reg, [@ex_saved_reg1] | ||
396 | #endif | ||
397 | .endm | ||
398 | |||
399 | .macro EXCPN_PROLOG_RESTORE_REG reg | ||
400 | #ifdef CONFIG_SMP | ||
401 | lr \reg, [ARC_REG_SCRATCH_DATA0] | ||
402 | #else | ||
403 | ld \reg, [@ex_saved_reg1] | ||
404 | #endif | ||
405 | .endm | ||
406 | |||
407 | /*-------------------------------------------------------------- | ||
408 | * Save all registers used by Exceptions (TLB Miss, Prot-V, Mem err etc) | ||
409 | * Requires SP to be already switched to kernel mode Stack | ||
410 | * sp points to the next free element on the stack at exit of this macro. | ||
411 | * Registers are pushed / popped in the order defined in struct ptregs | ||
412 | * in asm/ptrace.h | ||
413 | * Note that syscalls are implemented via TRAP which is also a exception | ||
414 | * from CPU's point of view | ||
415 | *-------------------------------------------------------------*/ | ||
416 | .macro SAVE_ALL_EXCEPTION marker | ||
417 | |||
418 | st \marker, [sp, 8] | ||
419 | st r0, [sp, 4] /* orig_r0, needed only for sys calls */ | ||
420 | |||
421 | /* Restore r9 used to code the early prologue */ | ||
422 | EXCPN_PROLOG_RESTORE_REG r9 | ||
423 | |||
424 | SAVE_CALLER_SAVED | ||
425 | st.a r26, [sp, -4] /* gp */ | ||
426 | st.a fp, [sp, -4] | ||
427 | st.a blink, [sp, -4] | ||
428 | lr r9, [eret] | ||
429 | st.a r9, [sp, -4] | ||
430 | lr r9, [erstatus] | ||
431 | st.a r9, [sp, -4] | ||
432 | st.a lp_count, [sp, -4] | ||
433 | lr r9, [lp_end] | ||
434 | st.a r9, [sp, -4] | ||
435 | lr r9, [lp_start] | ||
436 | st.a r9, [sp, -4] | ||
437 | lr r9, [erbta] | ||
438 | st.a r9, [sp, -4] | ||
439 | |||
440 | #ifdef PT_REGS_CANARY | ||
441 | mov r9, 0xdeadbeef | ||
442 | st r9, [sp, -4] | ||
443 | #endif | ||
444 | |||
445 | /* move up by 1 word to "create" pt_regs->"stack_place_holder" */ | ||
446 | sub sp, sp, 4 | ||
447 | .endm | ||
448 | |||
449 | /*-------------------------------------------------------------- | ||
450 | * Save scratch regs for exceptions | ||
451 | *-------------------------------------------------------------*/ | ||
452 | .macro SAVE_ALL_SYS | ||
453 | SAVE_ALL_EXCEPTION orig_r8_IS_EXCPN | ||
454 | .endm | ||
455 | |||
456 | /*-------------------------------------------------------------- | ||
457 | * Save scratch regs for sys calls | ||
458 | *-------------------------------------------------------------*/ | ||
459 | .macro SAVE_ALL_TRAP | ||
460 | /* | ||
461 | * Setup pt_regs->orig_r8. | ||
462 | * Encode syscall number (r8) in upper short word of event type (r9) | ||
463 | * N.B. #1: This is already endian safe (see ptrace.h) | ||
464 | * #2: Only r9 can be used as scratch as it is already clobbered | ||
465 | * and it's contents are no longer needed by the latter part | ||
466 | * of exception prologue | ||
467 | */ | ||
468 | lsl r9, r8, 16 | ||
469 | or r9, r9, orig_r8_IS_SCALL | ||
470 | |||
471 | SAVE_ALL_EXCEPTION r9 | ||
472 | .endm | ||
473 | |||
474 | /*-------------------------------------------------------------- | ||
475 | * Restore all registers used by system call or Exceptions | ||
476 | * SP should always be pointing to the next free stack element | ||
477 | * when entering this macro. | ||
478 | * | ||
479 | * NOTE: | ||
480 | * | ||
481 | * It is recommended that lp_count/ilink1/ilink2 not be used as a dest reg | ||
482 | * for memory load operations. If used in that way interrupts are deffered | ||
483 | * by hardware and that is not good. | ||
484 | *-------------------------------------------------------------*/ | ||
485 | .macro RESTORE_ALL_SYS | ||
486 | |||
487 | add sp, sp, 4 /* hop over unused "pt_regs->stack_place_holder" */ | ||
488 | |||
489 | ld.ab r9, [sp, 4] | ||
490 | sr r9, [erbta] | ||
491 | ld.ab r9, [sp, 4] | ||
492 | sr r9, [lp_start] | ||
493 | ld.ab r9, [sp, 4] | ||
494 | sr r9, [lp_end] | ||
495 | ld.ab r9, [sp, 4] | ||
496 | mov lp_count, r9 | ||
497 | ld.ab r9, [sp, 4] | ||
498 | sr r9, [erstatus] | ||
499 | ld.ab r9, [sp, 4] | ||
500 | sr r9, [eret] | ||
501 | ld.ab blink, [sp, 4] | ||
502 | ld.ab fp, [sp, 4] | ||
503 | ld.ab r26, [sp, 4] /* gp */ | ||
504 | RESTORE_CALLER_SAVED | ||
505 | |||
506 | ld sp, [sp] /* restore original sp */ | ||
507 | /* orig_r0 and orig_r8 skipped automatically */ | ||
508 | .endm | ||
509 | |||
510 | |||
511 | /*-------------------------------------------------------------- | ||
512 | * Save all registers used by interrupt handlers. | ||
513 | *-------------------------------------------------------------*/ | ||
514 | .macro SAVE_ALL_INT1 | ||
515 | |||
516 | /* restore original r9 , saved in int1_saved_reg | ||
517 | * It will be saved on stack in macro: SAVE_CALLER_SAVED | ||
518 | */ | ||
519 | #ifdef CONFIG_SMP | ||
520 | lr r9, [ARC_REG_SCRATCH_DATA0] | ||
521 | #else | ||
522 | ld r9, [@int1_saved_reg] | ||
523 | #endif | ||
524 | |||
525 | /* now we are ready to save the remaining context :) */ | ||
526 | st orig_r8_IS_IRQ1, [sp, 8] /* Event Type */ | ||
527 | st 0, [sp, 4] /* orig_r0 , N/A for IRQ */ | ||
528 | SAVE_CALLER_SAVED | ||
529 | st.a r26, [sp, -4] /* gp */ | ||
530 | st.a fp, [sp, -4] | ||
531 | st.a blink, [sp, -4] | ||
532 | st.a ilink1, [sp, -4] | ||
533 | lr r9, [status32_l1] | ||
534 | st.a r9, [sp, -4] | ||
535 | st.a lp_count, [sp, -4] | ||
536 | lr r9, [lp_end] | ||
537 | st.a r9, [sp, -4] | ||
538 | lr r9, [lp_start] | ||
539 | st.a r9, [sp, -4] | ||
540 | lr r9, [bta_l1] | ||
541 | st.a r9, [sp, -4] | ||
542 | |||
543 | #ifdef PT_REGS_CANARY | ||
544 | mov r9, 0xdeadbee1 | ||
545 | st r9, [sp, -4] | ||
546 | #endif | ||
547 | /* move up by 1 word to "create" pt_regs->"stack_place_holder" */ | ||
548 | sub sp, sp, 4 | ||
549 | .endm | ||
550 | |||
551 | .macro SAVE_ALL_INT2 | ||
552 | |||
553 | /* TODO-vineetg: SMP we can't use global nor can we use | ||
554 | * SCRATCH0 as we do for int1 because while int1 is using | ||
555 | * it, int2 can come | ||
556 | */ | ||
557 | /* retsore original r9 , saved in sys_saved_r9 */ | ||
558 | ld r9, [@int2_saved_reg] | ||
559 | |||
560 | /* now we are ready to save the remaining context :) */ | ||
561 | st orig_r8_IS_IRQ2, [sp, 8] /* Event Type */ | ||
562 | st 0, [sp, 4] /* orig_r0 , N/A for IRQ */ | ||
563 | SAVE_CALLER_SAVED | ||
564 | st.a r26, [sp, -4] /* gp */ | ||
565 | st.a fp, [sp, -4] | ||
566 | st.a blink, [sp, -4] | ||
567 | st.a ilink2, [sp, -4] | ||
568 | lr r9, [status32_l2] | ||
569 | st.a r9, [sp, -4] | ||
570 | st.a lp_count, [sp, -4] | ||
571 | lr r9, [lp_end] | ||
572 | st.a r9, [sp, -4] | ||
573 | lr r9, [lp_start] | ||
574 | st.a r9, [sp, -4] | ||
575 | lr r9, [bta_l2] | ||
576 | st.a r9, [sp, -4] | ||
577 | |||
578 | #ifdef PT_REGS_CANARY | ||
579 | mov r9, 0xdeadbee2 | ||
580 | st r9, [sp, -4] | ||
581 | #endif | ||
582 | |||
583 | /* move up by 1 word to "create" pt_regs->"stack_place_holder" */ | ||
584 | sub sp, sp, 4 | ||
585 | .endm | ||
586 | |||
587 | /*-------------------------------------------------------------- | ||
588 | * Restore all registers used by interrupt handlers. | ||
589 | * | ||
590 | * NOTE: | ||
591 | * | ||
592 | * It is recommended that lp_count/ilink1/ilink2 not be used as a dest reg | ||
593 | * for memory load operations. If used in that way interrupts are deffered | ||
594 | * by hardware and that is not good. | ||
595 | *-------------------------------------------------------------*/ | ||
596 | |||
597 | .macro RESTORE_ALL_INT1 | ||
598 | add sp, sp, 4 /* hop over unused "pt_regs->stack_place_holder" */ | ||
599 | |||
600 | ld.ab r9, [sp, 4] /* Actual reg file */ | ||
601 | sr r9, [bta_l1] | ||
602 | ld.ab r9, [sp, 4] | ||
603 | sr r9, [lp_start] | ||
604 | ld.ab r9, [sp, 4] | ||
605 | sr r9, [lp_end] | ||
606 | ld.ab r9, [sp, 4] | ||
607 | mov lp_count, r9 | ||
608 | ld.ab r9, [sp, 4] | ||
609 | sr r9, [status32_l1] | ||
610 | ld.ab r9, [sp, 4] | ||
611 | mov ilink1, r9 | ||
612 | ld.ab blink, [sp, 4] | ||
613 | ld.ab fp, [sp, 4] | ||
614 | ld.ab r26, [sp, 4] /* gp */ | ||
615 | RESTORE_CALLER_SAVED | ||
616 | |||
617 | ld sp, [sp] /* restore original sp */ | ||
618 | /* orig_r0 and orig_r8 skipped automatically */ | ||
619 | .endm | ||
620 | |||
621 | .macro RESTORE_ALL_INT2 | ||
622 | add sp, sp, 4 /* hop over unused "pt_regs->stack_place_holder" */ | ||
623 | |||
624 | ld.ab r9, [sp, 4] | ||
625 | sr r9, [bta_l2] | ||
626 | ld.ab r9, [sp, 4] | ||
627 | sr r9, [lp_start] | ||
628 | ld.ab r9, [sp, 4] | ||
629 | sr r9, [lp_end] | ||
630 | ld.ab r9, [sp, 4] | ||
631 | mov lp_count, r9 | ||
632 | ld.ab r9, [sp, 4] | ||
633 | sr r9, [status32_l2] | ||
634 | ld.ab r9, [sp, 4] | ||
635 | mov ilink2, r9 | ||
636 | ld.ab blink, [sp, 4] | ||
637 | ld.ab fp, [sp, 4] | ||
638 | ld.ab r26, [sp, 4] /* gp */ | ||
639 | RESTORE_CALLER_SAVED | ||
640 | |||
641 | ld sp, [sp] /* restore original sp */ | ||
642 | /* orig_r0 and orig_r8 skipped automatically */ | ||
643 | |||
644 | .endm | ||
645 | |||
646 | |||
647 | /* Get CPU-ID of this core */ | ||
648 | .macro GET_CPU_ID reg | ||
649 | lr \reg, [identity] | ||
650 | lsr \reg, \reg, 8 | ||
651 | bmsk \reg, \reg, 7 | ||
652 | .endm | ||
653 | |||
654 | #ifdef CONFIG_SMP | ||
655 | |||
656 | /*------------------------------------------------- | ||
657 | * Retrieve the current running task on this CPU | ||
658 | * 1. Determine curr CPU id. | ||
659 | * 2. Use it to index into _current_task[ ] | ||
660 | */ | ||
661 | .macro GET_CURR_TASK_ON_CPU reg | ||
662 | GET_CPU_ID \reg | ||
663 | ld.as \reg, [@_current_task, \reg] | ||
664 | .endm | ||
665 | |||
666 | /*------------------------------------------------- | ||
667 | * Save a new task as the "current" task on this CPU | ||
668 | * 1. Determine curr CPU id. | ||
669 | * 2. Use it to index into _current_task[ ] | ||
670 | * | ||
671 | * Coded differently than GET_CURR_TASK_ON_CPU (which uses LD.AS) | ||
672 | * because ST r0, [r1, offset] can ONLY have s9 @offset | ||
673 | * while LD can take s9 (4 byte insn) or LIMM (8 byte insn) | ||
674 | */ | ||
675 | |||
676 | .macro SET_CURR_TASK_ON_CPU tsk, tmp | ||
677 | GET_CPU_ID \tmp | ||
678 | add2 \tmp, @_current_task, \tmp | ||
679 | st \tsk, [\tmp] | ||
680 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
681 | mov r25, \tsk | ||
682 | #endif | ||
683 | |||
684 | .endm | ||
685 | |||
686 | |||
687 | #else /* Uniprocessor implementation of macros */ | ||
688 | |||
689 | .macro GET_CURR_TASK_ON_CPU reg | ||
690 | ld \reg, [@_current_task] | ||
691 | .endm | ||
692 | |||
693 | .macro SET_CURR_TASK_ON_CPU tsk, tmp | ||
694 | st \tsk, [@_current_task] | ||
695 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
696 | mov r25, \tsk | ||
697 | #endif | ||
698 | .endm | ||
699 | |||
700 | #endif /* SMP / UNI */ | ||
701 | |||
702 | /* ------------------------------------------------------------------ | ||
703 | * Get the ptr to some field of Current Task at @off in task struct | ||
704 | * -Uses r25 for Current task ptr if that is enabled | ||
705 | */ | ||
706 | |||
707 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
708 | |||
709 | .macro GET_CURR_TASK_FIELD_PTR off, reg | ||
710 | add \reg, r25, \off | ||
711 | .endm | ||
712 | |||
713 | #else | ||
714 | |||
715 | .macro GET_CURR_TASK_FIELD_PTR off, reg | ||
716 | GET_CURR_TASK_ON_CPU \reg | ||
717 | add \reg, \reg, \off | ||
718 | .endm | ||
719 | |||
720 | #endif /* CONFIG_ARC_CURR_IN_REG */ | ||
721 | |||
722 | #endif /* __ASSEMBLY__ */ | ||
723 | |||
724 | #endif /* __ASM_ARC_ENTRY_H */ | ||
diff --git a/arch/arc/include/asm/exec.h b/arch/arc/include/asm/exec.h new file mode 100644 index 000000000000..28abc6905e07 --- /dev/null +++ b/arch/arc/include/asm/exec.h | |||
@@ -0,0 +1,15 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_ARC_EXEC_H | ||
10 | #define __ASM_ARC_EXEC_H | ||
11 | |||
12 | /* Align to 16b */ | ||
13 | #define arch_align_stack(p) ((unsigned long)(p) & ~0xf) | ||
14 | |||
15 | #endif | ||
diff --git a/arch/arc/include/asm/futex.h b/arch/arc/include/asm/futex.h new file mode 100644 index 000000000000..4dc64ddebece --- /dev/null +++ b/arch/arc/include/asm/futex.h | |||
@@ -0,0 +1,151 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Vineetg: August 2010: From Android kernel work | ||
9 | */ | ||
10 | |||
11 | #ifndef _ASM_FUTEX_H | ||
12 | #define _ASM_FUTEX_H | ||
13 | |||
14 | #include <linux/futex.h> | ||
15 | #include <linux/preempt.h> | ||
16 | #include <linux/uaccess.h> | ||
17 | #include <asm/errno.h> | ||
18 | |||
19 | #define __futex_atomic_op(insn, ret, oldval, uaddr, oparg)\ | ||
20 | \ | ||
21 | __asm__ __volatile__( \ | ||
22 | "1: ld %1, [%2] \n" \ | ||
23 | insn "\n" \ | ||
24 | "2: st %0, [%2] \n" \ | ||
25 | " mov %0, 0 \n" \ | ||
26 | "3: \n" \ | ||
27 | " .section .fixup,\"ax\" \n" \ | ||
28 | " .align 4 \n" \ | ||
29 | "4: mov %0, %4 \n" \ | ||
30 | " b 3b \n" \ | ||
31 | " .previous \n" \ | ||
32 | " .section __ex_table,\"a\" \n" \ | ||
33 | " .align 4 \n" \ | ||
34 | " .word 1b, 4b \n" \ | ||
35 | " .word 2b, 4b \n" \ | ||
36 | " .previous \n" \ | ||
37 | \ | ||
38 | : "=&r" (ret), "=&r" (oldval) \ | ||
39 | : "r" (uaddr), "r" (oparg), "ir" (-EFAULT) \ | ||
40 | : "cc", "memory") | ||
41 | |||
42 | static inline int futex_atomic_op_inuser(int encoded_op, u32 __user *uaddr) | ||
43 | { | ||
44 | int op = (encoded_op >> 28) & 7; | ||
45 | int cmp = (encoded_op >> 24) & 15; | ||
46 | int oparg = (encoded_op << 8) >> 20; | ||
47 | int cmparg = (encoded_op << 20) >> 20; | ||
48 | int oldval = 0, ret; | ||
49 | |||
50 | if (encoded_op & (FUTEX_OP_OPARG_SHIFT << 28)) | ||
51 | oparg = 1 << oparg; | ||
52 | |||
53 | if (!access_ok(VERIFY_WRITE, uaddr, sizeof(int))) | ||
54 | return -EFAULT; | ||
55 | |||
56 | pagefault_disable(); /* implies preempt_disable() */ | ||
57 | |||
58 | switch (op) { | ||
59 | case FUTEX_OP_SET: | ||
60 | __futex_atomic_op("mov %0, %3", ret, oldval, uaddr, oparg); | ||
61 | break; | ||
62 | case FUTEX_OP_ADD: | ||
63 | __futex_atomic_op("add %0, %1, %3", ret, oldval, uaddr, oparg); | ||
64 | break; | ||
65 | case FUTEX_OP_OR: | ||
66 | __futex_atomic_op("or %0, %1, %3", ret, oldval, uaddr, oparg); | ||
67 | break; | ||
68 | case FUTEX_OP_ANDN: | ||
69 | __futex_atomic_op("bic %0, %1, %3", ret, oldval, uaddr, oparg); | ||
70 | break; | ||
71 | case FUTEX_OP_XOR: | ||
72 | __futex_atomic_op("xor %0, %1, %3", ret, oldval, uaddr, oparg); | ||
73 | break; | ||
74 | default: | ||
75 | ret = -ENOSYS; | ||
76 | } | ||
77 | |||
78 | pagefault_enable(); /* subsumes preempt_enable() */ | ||
79 | |||
80 | if (!ret) { | ||
81 | switch (cmp) { | ||
82 | case FUTEX_OP_CMP_EQ: | ||
83 | ret = (oldval == cmparg); | ||
84 | break; | ||
85 | case FUTEX_OP_CMP_NE: | ||
86 | ret = (oldval != cmparg); | ||
87 | break; | ||
88 | case FUTEX_OP_CMP_LT: | ||
89 | ret = (oldval < cmparg); | ||
90 | break; | ||
91 | case FUTEX_OP_CMP_GE: | ||
92 | ret = (oldval >= cmparg); | ||
93 | break; | ||
94 | case FUTEX_OP_CMP_LE: | ||
95 | ret = (oldval <= cmparg); | ||
96 | break; | ||
97 | case FUTEX_OP_CMP_GT: | ||
98 | ret = (oldval > cmparg); | ||
99 | break; | ||
100 | default: | ||
101 | ret = -ENOSYS; | ||
102 | } | ||
103 | } | ||
104 | return ret; | ||
105 | } | ||
106 | |||
107 | /* Compare-xchg with preemption disabled. | ||
108 | * Notes: | ||
109 | * -Best-Effort: Exchg happens only if compare succeeds. | ||
110 | * If compare fails, returns; leaving retry/looping to upper layers | ||
111 | * -successful cmp-xchg: return orig value in @addr (same as cmp val) | ||
112 | * -Compare fails: return orig value in @addr | ||
113 | * -user access r/w fails: return -EFAULT | ||
114 | */ | ||
115 | static inline int | ||
116 | futex_atomic_cmpxchg_inatomic(u32 *uval, u32 __user *uaddr, u32 oldval, | ||
117 | u32 newval) | ||
118 | { | ||
119 | u32 val; | ||
120 | |||
121 | if (!access_ok(VERIFY_WRITE, uaddr, sizeof(int))) | ||
122 | return -EFAULT; | ||
123 | |||
124 | pagefault_disable(); /* implies preempt_disable() */ | ||
125 | |||
126 | /* TBD : can use llock/scond */ | ||
127 | __asm__ __volatile__( | ||
128 | "1: ld %0, [%3] \n" | ||
129 | " brne %0, %1, 3f \n" | ||
130 | "2: st %2, [%3] \n" | ||
131 | "3: \n" | ||
132 | " .section .fixup,\"ax\" \n" | ||
133 | "4: mov %0, %4 \n" | ||
134 | " b 3b \n" | ||
135 | " .previous \n" | ||
136 | " .section __ex_table,\"a\" \n" | ||
137 | " .align 4 \n" | ||
138 | " .word 1b, 4b \n" | ||
139 | " .word 2b, 4b \n" | ||
140 | " .previous\n" | ||
141 | : "=&r"(val) | ||
142 | : "r"(oldval), "r"(newval), "r"(uaddr), "ir"(-EFAULT) | ||
143 | : "cc", "memory"); | ||
144 | |||
145 | pagefault_enable(); /* subsumes preempt_enable() */ | ||
146 | |||
147 | *uval = val; | ||
148 | return val; | ||
149 | } | ||
150 | |||
151 | #endif | ||
diff --git a/arch/arc/include/asm/io.h b/arch/arc/include/asm/io.h new file mode 100644 index 000000000000..473424d7528b --- /dev/null +++ b/arch/arc/include/asm/io.h | |||
@@ -0,0 +1,105 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_IO_H | ||
10 | #define _ASM_ARC_IO_H | ||
11 | |||
12 | #include <linux/types.h> | ||
13 | #include <asm/byteorder.h> | ||
14 | #include <asm/page.h> | ||
15 | |||
16 | #define PCI_IOBASE ((void __iomem *)0) | ||
17 | |||
18 | extern void __iomem *ioremap(unsigned long physaddr, unsigned long size); | ||
19 | extern void __iomem *ioremap_prot(phys_addr_t offset, unsigned long size, | ||
20 | unsigned long flags); | ||
21 | extern void iounmap(const void __iomem *addr); | ||
22 | |||
23 | #define ioremap_nocache(phy, sz) ioremap(phy, sz) | ||
24 | #define ioremap_wc(phy, sz) ioremap(phy, sz) | ||
25 | |||
26 | /* Change struct page to physical address */ | ||
27 | #define page_to_phys(page) (page_to_pfn(page) << PAGE_SHIFT) | ||
28 | |||
29 | #define __raw_readb __raw_readb | ||
30 | static inline u8 __raw_readb(const volatile void __iomem *addr) | ||
31 | { | ||
32 | u8 b; | ||
33 | |||
34 | __asm__ __volatile__( | ||
35 | " ldb%U1 %0, %1 \n" | ||
36 | : "=r" (b) | ||
37 | : "m" (*(volatile u8 __force *)addr) | ||
38 | : "memory"); | ||
39 | |||
40 | return b; | ||
41 | } | ||
42 | |||
43 | #define __raw_readw __raw_readw | ||
44 | static inline u16 __raw_readw(const volatile void __iomem *addr) | ||
45 | { | ||
46 | u16 s; | ||
47 | |||
48 | __asm__ __volatile__( | ||
49 | " ldw%U1 %0, %1 \n" | ||
50 | : "=r" (s) | ||
51 | : "m" (*(volatile u16 __force *)addr) | ||
52 | : "memory"); | ||
53 | |||
54 | return s; | ||
55 | } | ||
56 | |||
57 | #define __raw_readl __raw_readl | ||
58 | static inline u32 __raw_readl(const volatile void __iomem *addr) | ||
59 | { | ||
60 | u32 w; | ||
61 | |||
62 | __asm__ __volatile__( | ||
63 | " ld%U1 %0, %1 \n" | ||
64 | : "=r" (w) | ||
65 | : "m" (*(volatile u32 __force *)addr) | ||
66 | : "memory"); | ||
67 | |||
68 | return w; | ||
69 | } | ||
70 | |||
71 | #define __raw_writeb __raw_writeb | ||
72 | static inline void __raw_writeb(u8 b, volatile void __iomem *addr) | ||
73 | { | ||
74 | __asm__ __volatile__( | ||
75 | " stb%U1 %0, %1 \n" | ||
76 | : | ||
77 | : "r" (b), "m" (*(volatile u8 __force *)addr) | ||
78 | : "memory"); | ||
79 | } | ||
80 | |||
81 | #define __raw_writew __raw_writew | ||
82 | static inline void __raw_writew(u16 s, volatile void __iomem *addr) | ||
83 | { | ||
84 | __asm__ __volatile__( | ||
85 | " stw%U1 %0, %1 \n" | ||
86 | : | ||
87 | : "r" (s), "m" (*(volatile u16 __force *)addr) | ||
88 | : "memory"); | ||
89 | |||
90 | } | ||
91 | |||
92 | #define __raw_writel __raw_writel | ||
93 | static inline void __raw_writel(u32 w, volatile void __iomem *addr) | ||
94 | { | ||
95 | __asm__ __volatile__( | ||
96 | " st%U1 %0, %1 \n" | ||
97 | : | ||
98 | : "r" (w), "m" (*(volatile u32 __force *)addr) | ||
99 | : "memory"); | ||
100 | |||
101 | } | ||
102 | |||
103 | #include <asm-generic/io.h> | ||
104 | |||
105 | #endif /* _ASM_ARC_IO_H */ | ||
diff --git a/arch/arc/include/asm/irq.h b/arch/arc/include/asm/irq.h new file mode 100644 index 000000000000..4c588f9820cf --- /dev/null +++ b/arch/arc/include/asm/irq.h | |||
@@ -0,0 +1,25 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_ARC_IRQ_H | ||
10 | #define __ASM_ARC_IRQ_H | ||
11 | |||
12 | #define NR_IRQS 32 | ||
13 | |||
14 | /* Platform Independent IRQs */ | ||
15 | #define TIMER0_IRQ 3 | ||
16 | #define TIMER1_IRQ 4 | ||
17 | |||
18 | #include <asm-generic/irq.h> | ||
19 | |||
20 | extern void __init arc_init_IRQ(void); | ||
21 | extern int __init get_hw_config_num_irq(void); | ||
22 | |||
23 | void __cpuinit arc_local_timer_setup(unsigned int cpu); | ||
24 | |||
25 | #endif | ||
diff --git a/arch/arc/include/asm/irqflags.h b/arch/arc/include/asm/irqflags.h new file mode 100644 index 000000000000..ccd84806b62f --- /dev/null +++ b/arch/arc/include/asm/irqflags.h | |||
@@ -0,0 +1,153 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_ARC_IRQFLAGS_H | ||
10 | #define __ASM_ARC_IRQFLAGS_H | ||
11 | |||
12 | /* vineetg: March 2010 : local_irq_save( ) optimisation | ||
13 | * -Remove explicit mov of current status32 into reg, that is not needed | ||
14 | * -Use BIC insn instead of INVERTED + AND | ||
15 | * -Conditionally disable interrupts (if they are not enabled, don't disable) | ||
16 | */ | ||
17 | |||
18 | #ifdef __KERNEL__ | ||
19 | |||
20 | #include <asm/arcregs.h> | ||
21 | |||
22 | #ifndef __ASSEMBLY__ | ||
23 | |||
24 | /****************************************************************** | ||
25 | * IRQ Control Macros | ||
26 | ******************************************************************/ | ||
27 | |||
28 | /* | ||
29 | * Save IRQ state and disable IRQs | ||
30 | */ | ||
31 | static inline long arch_local_irq_save(void) | ||
32 | { | ||
33 | unsigned long temp, flags; | ||
34 | |||
35 | __asm__ __volatile__( | ||
36 | " lr %1, [status32] \n" | ||
37 | " bic %0, %1, %2 \n" | ||
38 | " and.f 0, %1, %2 \n" | ||
39 | " flag.nz %0 \n" | ||
40 | : "=r"(temp), "=r"(flags) | ||
41 | : "n"((STATUS_E1_MASK | STATUS_E2_MASK)) | ||
42 | : "cc"); | ||
43 | |||
44 | return flags; | ||
45 | } | ||
46 | |||
47 | /* | ||
48 | * restore saved IRQ state | ||
49 | */ | ||
50 | static inline void arch_local_irq_restore(unsigned long flags) | ||
51 | { | ||
52 | |||
53 | __asm__ __volatile__( | ||
54 | " flag %0 \n" | ||
55 | : | ||
56 | : "r"(flags)); | ||
57 | } | ||
58 | |||
59 | /* | ||
60 | * Unconditionally Enable IRQs | ||
61 | */ | ||
62 | extern void arch_local_irq_enable(void); | ||
63 | |||
64 | /* | ||
65 | * Unconditionally Disable IRQs | ||
66 | */ | ||
67 | static inline void arch_local_irq_disable(void) | ||
68 | { | ||
69 | unsigned long temp; | ||
70 | |||
71 | __asm__ __volatile__( | ||
72 | " lr %0, [status32] \n" | ||
73 | " and %0, %0, %1 \n" | ||
74 | " flag %0 \n" | ||
75 | : "=&r"(temp) | ||
76 | : "n"(~(STATUS_E1_MASK | STATUS_E2_MASK))); | ||
77 | } | ||
78 | |||
79 | /* | ||
80 | * save IRQ state | ||
81 | */ | ||
82 | static inline long arch_local_save_flags(void) | ||
83 | { | ||
84 | unsigned long temp; | ||
85 | |||
86 | __asm__ __volatile__( | ||
87 | " lr %0, [status32] \n" | ||
88 | : "=&r"(temp)); | ||
89 | |||
90 | return temp; | ||
91 | } | ||
92 | |||
93 | /* | ||
94 | * Query IRQ state | ||
95 | */ | ||
96 | static inline int arch_irqs_disabled_flags(unsigned long flags) | ||
97 | { | ||
98 | return !(flags & (STATUS_E1_MASK | ||
99 | #ifdef CONFIG_ARC_COMPACT_IRQ_LEVELS | ||
100 | | STATUS_E2_MASK | ||
101 | #endif | ||
102 | )); | ||
103 | } | ||
104 | |||
105 | static inline int arch_irqs_disabled(void) | ||
106 | { | ||
107 | return arch_irqs_disabled_flags(arch_local_save_flags()); | ||
108 | } | ||
109 | |||
110 | static inline void arch_mask_irq(unsigned int irq) | ||
111 | { | ||
112 | unsigned int ienb; | ||
113 | |||
114 | ienb = read_aux_reg(AUX_IENABLE); | ||
115 | ienb &= ~(1 << irq); | ||
116 | write_aux_reg(AUX_IENABLE, ienb); | ||
117 | } | ||
118 | |||
119 | static inline void arch_unmask_irq(unsigned int irq) | ||
120 | { | ||
121 | unsigned int ienb; | ||
122 | |||
123 | ienb = read_aux_reg(AUX_IENABLE); | ||
124 | ienb |= (1 << irq); | ||
125 | write_aux_reg(AUX_IENABLE, ienb); | ||
126 | } | ||
127 | |||
128 | #else | ||
129 | |||
130 | .macro IRQ_DISABLE scratch | ||
131 | lr \scratch, [status32] | ||
132 | bic \scratch, \scratch, (STATUS_E1_MASK | STATUS_E2_MASK) | ||
133 | flag \scratch | ||
134 | .endm | ||
135 | |||
136 | .macro IRQ_DISABLE_SAVE scratch, save | ||
137 | lr \scratch, [status32] | ||
138 | mov \save, \scratch /* Make a copy */ | ||
139 | bic \scratch, \scratch, (STATUS_E1_MASK | STATUS_E2_MASK) | ||
140 | flag \scratch | ||
141 | .endm | ||
142 | |||
143 | .macro IRQ_ENABLE scratch | ||
144 | lr \scratch, [status32] | ||
145 | or \scratch, \scratch, (STATUS_E1_MASK | STATUS_E2_MASK) | ||
146 | flag \scratch | ||
147 | .endm | ||
148 | |||
149 | #endif /* __ASSEMBLY__ */ | ||
150 | |||
151 | #endif /* KERNEL */ | ||
152 | |||
153 | #endif | ||
diff --git a/arch/arc/include/asm/kdebug.h b/arch/arc/include/asm/kdebug.h new file mode 100644 index 000000000000..3fbe6c472c0a --- /dev/null +++ b/arch/arc/include/asm/kdebug.h | |||
@@ -0,0 +1,19 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_KDEBUG_H | ||
10 | #define _ASM_ARC_KDEBUG_H | ||
11 | |||
12 | enum die_val { | ||
13 | DIE_UNUSED, | ||
14 | DIE_TRAP, | ||
15 | DIE_IERR, | ||
16 | DIE_OOPS | ||
17 | }; | ||
18 | |||
19 | #endif | ||
diff --git a/arch/arc/include/asm/kgdb.h b/arch/arc/include/asm/kgdb.h new file mode 100644 index 000000000000..f3c4934f0ca9 --- /dev/null +++ b/arch/arc/include/asm/kgdb.h | |||
@@ -0,0 +1,61 @@ | |||
1 | /* | ||
2 | * kgdb support for ARC | ||
3 | * | ||
4 | * Copyright (C) 2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | */ | ||
10 | |||
11 | #ifndef __ARC_KGDB_H__ | ||
12 | #define __ARC_KGDB_H__ | ||
13 | |||
14 | #ifdef CONFIG_KGDB | ||
15 | |||
16 | #include <asm/user.h> | ||
17 | |||
18 | /* to ensure compatibility with Linux 2.6.35, we don't implement the get/set | ||
19 | * register API yet */ | ||
20 | #undef DBG_MAX_REG_NUM | ||
21 | |||
22 | #define GDB_MAX_REGS 39 | ||
23 | |||
24 | #define BREAK_INSTR_SIZE 2 | ||
25 | #define CACHE_FLUSH_IS_SAFE 1 | ||
26 | #define NUMREGBYTES (GDB_MAX_REGS * 4) | ||
27 | #define BUFMAX 2048 | ||
28 | |||
29 | static inline void arch_kgdb_breakpoint(void) | ||
30 | { | ||
31 | __asm__ __volatile__ ("trap_s 0x4\n"); | ||
32 | } | ||
33 | |||
34 | extern void kgdb_trap(struct pt_regs *regs, int param); | ||
35 | |||
36 | enum arc700_linux_regnums { | ||
37 | _R0 = 0, | ||
38 | _R1, _R2, _R3, _R4, _R5, _R6, _R7, _R8, _R9, _R10, _R11, _R12, _R13, | ||
39 | _R14, _R15, _R16, _R17, _R18, _R19, _R20, _R21, _R22, _R23, _R24, | ||
40 | _R25, _R26, | ||
41 | _BTA = 27, | ||
42 | _LP_START = 28, | ||
43 | _LP_END = 29, | ||
44 | _LP_COUNT = 30, | ||
45 | _STATUS32 = 31, | ||
46 | _BLINK = 32, | ||
47 | _FP = 33, | ||
48 | __SP = 34, | ||
49 | _EFA = 35, | ||
50 | _RET = 36, | ||
51 | _ORIG_R8 = 37, | ||
52 | _STOP_PC = 38 | ||
53 | }; | ||
54 | |||
55 | #else | ||
56 | static inline void kgdb_trap(struct pt_regs *regs, int param) | ||
57 | { | ||
58 | } | ||
59 | #endif | ||
60 | |||
61 | #endif /* __ARC_KGDB_H__ */ | ||
diff --git a/arch/arc/include/asm/kprobes.h b/arch/arc/include/asm/kprobes.h new file mode 100644 index 000000000000..4d9c211fce70 --- /dev/null +++ b/arch/arc/include/asm/kprobes.h | |||
@@ -0,0 +1,62 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ARC_KPROBES_H | ||
10 | #define _ARC_KPROBES_H | ||
11 | |||
12 | #ifdef CONFIG_KPROBES | ||
13 | |||
14 | typedef u16 kprobe_opcode_t; | ||
15 | |||
16 | #define UNIMP_S_INSTRUCTION 0x79e0 | ||
17 | #define TRAP_S_2_INSTRUCTION 0x785e | ||
18 | |||
19 | #define MAX_INSN_SIZE 8 | ||
20 | #define MAX_STACK_SIZE 64 | ||
21 | |||
22 | struct arch_specific_insn { | ||
23 | int is_short; | ||
24 | kprobe_opcode_t *t1_addr, *t2_addr; | ||
25 | kprobe_opcode_t t1_opcode, t2_opcode; | ||
26 | }; | ||
27 | |||
28 | #define flush_insn_slot(p) do { } while (0) | ||
29 | |||
30 | #define kretprobe_blacklist_size 0 | ||
31 | |||
32 | struct kprobe; | ||
33 | |||
34 | void arch_remove_kprobe(struct kprobe *p); | ||
35 | |||
36 | int kprobe_exceptions_notify(struct notifier_block *self, | ||
37 | unsigned long val, void *data); | ||
38 | |||
39 | struct prev_kprobe { | ||
40 | struct kprobe *kp; | ||
41 | unsigned long status; | ||
42 | }; | ||
43 | |||
44 | struct kprobe_ctlblk { | ||
45 | unsigned int kprobe_status; | ||
46 | struct pt_regs jprobe_saved_regs; | ||
47 | char jprobes_stack[MAX_STACK_SIZE]; | ||
48 | struct prev_kprobe prev_kprobe; | ||
49 | }; | ||
50 | |||
51 | int kprobe_fault_handler(struct pt_regs *regs, unsigned long cause); | ||
52 | void kretprobe_trampoline(void); | ||
53 | void trap_is_kprobe(unsigned long cause, unsigned long address, | ||
54 | struct pt_regs *regs); | ||
55 | #else | ||
56 | static void trap_is_kprobe(unsigned long cause, unsigned long address, | ||
57 | struct pt_regs *regs) | ||
58 | { | ||
59 | } | ||
60 | #endif | ||
61 | |||
62 | #endif | ||
diff --git a/arch/arc/include/asm/linkage.h b/arch/arc/include/asm/linkage.h new file mode 100644 index 000000000000..0283e9e44e0d --- /dev/null +++ b/arch/arc/include/asm/linkage.h | |||
@@ -0,0 +1,63 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_LINKAGE_H | ||
10 | #define __ASM_LINKAGE_H | ||
11 | |||
12 | #ifdef __ASSEMBLY__ | ||
13 | |||
14 | /* Can't use the ENTRY macro in linux/linkage.h | ||
15 | * gas considers ';' as comment vs. newline | ||
16 | */ | ||
17 | .macro ARC_ENTRY name | ||
18 | .global \name | ||
19 | .align 4 | ||
20 | \name: | ||
21 | .endm | ||
22 | |||
23 | .macro ARC_EXIT name | ||
24 | #define ASM_PREV_SYM_ADDR(name) .-##name | ||
25 | .size \ name, ASM_PREV_SYM_ADDR(\name) | ||
26 | .endm | ||
27 | |||
28 | /* annotation for data we want in DCCM - if enabled in .config */ | ||
29 | .macro ARCFP_DATA nm | ||
30 | #ifdef CONFIG_ARC_HAS_DCCM | ||
31 | .section .data.arcfp | ||
32 | #else | ||
33 | .section .data | ||
34 | #endif | ||
35 | .global \nm | ||
36 | .endm | ||
37 | |||
38 | /* annotation for data we want in DCCM - if enabled in .config */ | ||
39 | .macro ARCFP_CODE | ||
40 | #ifdef CONFIG_ARC_HAS_ICCM | ||
41 | .section .text.arcfp, "ax",@progbits | ||
42 | #else | ||
43 | .section .text, "ax",@progbits | ||
44 | #endif | ||
45 | .endm | ||
46 | |||
47 | #else /* !__ASSEMBLY__ */ | ||
48 | |||
49 | #ifdef CONFIG_ARC_HAS_ICCM | ||
50 | #define __arcfp_code __attribute__((__section__(".text.arcfp"))) | ||
51 | #else | ||
52 | #define __arcfp_code __attribute__((__section__(".text"))) | ||
53 | #endif | ||
54 | |||
55 | #ifdef CONFIG_ARC_HAS_DCCM | ||
56 | #define __arcfp_data __attribute__((__section__(".data.arcfp"))) | ||
57 | #else | ||
58 | #define __arcfp_data __attribute__((__section__(".data"))) | ||
59 | #endif | ||
60 | |||
61 | #endif /* __ASSEMBLY__ */ | ||
62 | |||
63 | #endif | ||
diff --git a/arch/arc/include/asm/mach_desc.h b/arch/arc/include/asm/mach_desc.h new file mode 100644 index 000000000000..9998dc846ebb --- /dev/null +++ b/arch/arc/include/asm/mach_desc.h | |||
@@ -0,0 +1,87 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * based on METAG mach/arch.h (which in turn was based on ARM) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | */ | ||
10 | |||
11 | #ifndef _ASM_ARC_MACH_DESC_H_ | ||
12 | #define _ASM_ARC_MACH_DESC_H_ | ||
13 | |||
14 | /** | ||
15 | * struct machine_desc - Board specific callbacks, called from ARC common code | ||
16 | * Provided by each ARC board using MACHINE_START()/MACHINE_END(), so | ||
17 | * a multi-platform kernel builds with array of such descriptors. | ||
18 | * We extend the early DT scan to also match the DT's "compatible" string | ||
19 | * against the @dt_compat of all such descriptors, and one with highest | ||
20 | * "DT score" is selected as global @machine_desc. | ||
21 | * | ||
22 | * @name: Board/SoC name | ||
23 | * @dt_compat: Array of device tree 'compatible' strings | ||
24 | * (XXX: although only 1st entry is looked at) | ||
25 | * @init_early: Very early callback [called from setup_arch()] | ||
26 | * @init_irq: setup external IRQ controllers [called from init_IRQ()] | ||
27 | * @init_smp: for each CPU (e.g. setup IPI) | ||
28 | * [(M):init_IRQ(), (o):start_kernel_secondary()] | ||
29 | * @init_time: platform specific clocksource/clockevent registration | ||
30 | * [called from time_init()] | ||
31 | * @init_machine: arch initcall level callback (e.g. populate static | ||
32 | * platform devices or parse Devicetree) | ||
33 | * @init_late: Late initcall level callback | ||
34 | * | ||
35 | */ | ||
36 | struct machine_desc { | ||
37 | const char *name; | ||
38 | const char **dt_compat; | ||
39 | |||
40 | void (*init_early)(void); | ||
41 | void (*init_irq)(void); | ||
42 | #ifdef CONFIG_SMP | ||
43 | void (*init_smp)(unsigned int); | ||
44 | #endif | ||
45 | void (*init_time)(void); | ||
46 | void (*init_machine)(void); | ||
47 | void (*init_late)(void); | ||
48 | |||
49 | }; | ||
50 | |||
51 | /* | ||
52 | * Current machine - only accessible during boot. | ||
53 | */ | ||
54 | extern struct machine_desc *machine_desc; | ||
55 | |||
56 | /* | ||
57 | * Machine type table - also only accessible during boot | ||
58 | */ | ||
59 | extern struct machine_desc __arch_info_begin[], __arch_info_end[]; | ||
60 | #define for_each_machine_desc(p) \ | ||
61 | for (p = __arch_info_begin; p < __arch_info_end; p++) | ||
62 | |||
63 | static inline struct machine_desc *default_machine_desc(void) | ||
64 | { | ||
65 | /* the default machine is the last one linked in */ | ||
66 | if (__arch_info_end - 1 < __arch_info_begin) | ||
67 | return NULL; | ||
68 | return __arch_info_end - 1; | ||
69 | } | ||
70 | |||
71 | /* | ||
72 | * Set of macros to define architecture features. | ||
73 | * This is built into a table by the linker. | ||
74 | */ | ||
75 | #define MACHINE_START(_type, _name) \ | ||
76 | static const struct machine_desc __mach_desc_##_type \ | ||
77 | __used \ | ||
78 | __attribute__((__section__(".arch.info.init"))) = { \ | ||
79 | .name = _name, | ||
80 | |||
81 | #define MACHINE_END \ | ||
82 | }; | ||
83 | |||
84 | extern struct machine_desc *setup_machine_fdt(void *dt); | ||
85 | extern void __init copy_devtree(void); | ||
86 | |||
87 | #endif | ||
diff --git a/arch/arc/include/asm/mmu.h b/arch/arc/include/asm/mmu.h new file mode 100644 index 000000000000..56b02320f1a9 --- /dev/null +++ b/arch/arc/include/asm/mmu.h | |||
@@ -0,0 +1,23 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_MMU_H | ||
10 | #define _ASM_ARC_MMU_H | ||
11 | |||
12 | #ifndef __ASSEMBLY__ | ||
13 | |||
14 | typedef struct { | ||
15 | unsigned long asid; /* Pvt Addr-Space ID for mm */ | ||
16 | #ifdef CONFIG_ARC_TLB_DBG | ||
17 | struct task_struct *tsk; | ||
18 | #endif | ||
19 | } mm_context_t; | ||
20 | |||
21 | #endif | ||
22 | |||
23 | #endif | ||
diff --git a/arch/arc/include/asm/mmu_context.h b/arch/arc/include/asm/mmu_context.h new file mode 100644 index 000000000000..0d71fb11b57c --- /dev/null +++ b/arch/arc/include/asm/mmu_context.h | |||
@@ -0,0 +1,213 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * vineetg: May 2011 | ||
9 | * -Refactored get_new_mmu_context( ) to only handle live-mm. | ||
10 | * retiring-mm handled in other hooks | ||
11 | * | ||
12 | * Vineetg: March 25th, 2008: Bug #92690 | ||
13 | * -Major rewrite of Core ASID allocation routine get_new_mmu_context | ||
14 | * | ||
15 | * Amit Bhor, Sameer Dhavale: Codito Technologies 2004 | ||
16 | */ | ||
17 | |||
18 | #ifndef _ASM_ARC_MMU_CONTEXT_H | ||
19 | #define _ASM_ARC_MMU_CONTEXT_H | ||
20 | |||
21 | #include <asm/arcregs.h> | ||
22 | #include <asm/tlb.h> | ||
23 | |||
24 | #include <asm-generic/mm_hooks.h> | ||
25 | |||
26 | /* ARC700 ASID Management | ||
27 | * | ||
28 | * ARC MMU provides 8-bit ASID (0..255) to TAG TLB entries, allowing entries | ||
29 | * with same vaddr (different tasks) to co-exit. This provides for | ||
30 | * "Fast Context Switch" i.e. no TLB flush on ctxt-switch | ||
31 | * | ||
32 | * Linux assigns each task a unique ASID. A simple round-robin allocation | ||
33 | * of H/w ASID is done using software tracker @asid_cache. | ||
34 | * When it reaches max 255, the allocation cycle starts afresh by flushing | ||
35 | * the entire TLB and wrapping ASID back to zero. | ||
36 | * | ||
37 | * For book-keeping, Linux uses a couple of data-structures: | ||
38 | * -mm_struct has an @asid field to keep a note of task's ASID (needed at the | ||
39 | * time of say switch_mm( ) | ||
40 | * -An array of mm structs @asid_mm_map[] for asid->mm the reverse mapping, | ||
41 | * given an ASID, finding the mm struct associated. | ||
42 | * | ||
43 | * The round-robin allocation algorithm allows for ASID stealing. | ||
44 | * If asid tracker is at "x-1", a new req will allocate "x", even if "x" was | ||
45 | * already assigned to another (switched-out) task. Obviously the prev owner | ||
46 | * is marked with an invalid ASID to make it request for a new ASID when it | ||
47 | * gets scheduled next time. However its TLB entries (with ASID "x") could | ||
48 | * exist, which must be cleared before the same ASID is used by the new owner. | ||
49 | * Flushing them would be plausible but costly solution. Instead we force a | ||
50 | * allocation policy quirk, which ensures that a stolen ASID won't have any | ||
51 | * TLB entries associates, alleviating the need to flush. | ||
52 | * The quirk essentially is not allowing ASID allocated in prev cycle | ||
53 | * to be used past a roll-over in the next cycle. | ||
54 | * When this happens (i.e. task ASID > asid tracker), task needs to refresh | ||
55 | * its ASID, aligning it to current value of tracker. If the task doesn't get | ||
56 | * scheduled past a roll-over, hence its ASID is not yet realigned with | ||
57 | * tracker, such ASID is anyways safely reusable because it is | ||
58 | * gauranteed that TLB entries with that ASID wont exist. | ||
59 | */ | ||
60 | |||
61 | #define FIRST_ASID 0 | ||
62 | #define MAX_ASID 255 /* 8 bit PID field in PID Aux reg */ | ||
63 | #define NO_ASID (MAX_ASID + 1) /* ASID Not alloc to mmu ctxt */ | ||
64 | #define NUM_ASID ((MAX_ASID - FIRST_ASID) + 1) | ||
65 | |||
66 | /* ASID to mm struct mapping */ | ||
67 | extern struct mm_struct *asid_mm_map[NUM_ASID + 1]; | ||
68 | |||
69 | extern int asid_cache; | ||
70 | |||
71 | /* | ||
72 | * Assign a new ASID to task. If the task already has an ASID, it is | ||
73 | * relinquished. | ||
74 | */ | ||
75 | static inline void get_new_mmu_context(struct mm_struct *mm) | ||
76 | { | ||
77 | struct mm_struct *prev_owner; | ||
78 | unsigned long flags; | ||
79 | |||
80 | local_irq_save(flags); | ||
81 | |||
82 | /* | ||
83 | * Relinquish the currently owned ASID (if any). | ||
84 | * Doing unconditionally saves a cmp-n-branch; for already unused | ||
85 | * ASID slot, the value was/remains NULL | ||
86 | */ | ||
87 | asid_mm_map[mm->context.asid] = (struct mm_struct *)NULL; | ||
88 | |||
89 | /* move to new ASID */ | ||
90 | if (++asid_cache > MAX_ASID) { /* ASID roll-over */ | ||
91 | asid_cache = FIRST_ASID; | ||
92 | flush_tlb_all(); | ||
93 | } | ||
94 | |||
95 | /* | ||
96 | * Is next ASID already owned by some-one else (we are stealing it). | ||
97 | * If so, let the orig owner be aware of this, so when it runs, it | ||
98 | * asks for a brand new ASID. This would only happen for a long-lived | ||
99 | * task with ASID from prev allocation cycle (before ASID roll-over). | ||
100 | * | ||
101 | * This might look wrong - if we are re-using some other task's ASID, | ||
102 | * won't we use it's stale TLB entries too. Actually switch_mm( ) takes | ||
103 | * care of such a case: it ensures that task with ASID from prev alloc | ||
104 | * cycle, when scheduled will refresh it's ASID: see switch_mm( ) below | ||
105 | * The stealing scenario described here will only happen if that task | ||
106 | * didn't get a chance to refresh it's ASID - implying stale entries | ||
107 | * won't exist. | ||
108 | */ | ||
109 | prev_owner = asid_mm_map[asid_cache]; | ||
110 | if (prev_owner) | ||
111 | prev_owner->context.asid = NO_ASID; | ||
112 | |||
113 | /* Assign new ASID to tsk */ | ||
114 | asid_mm_map[asid_cache] = mm; | ||
115 | mm->context.asid = asid_cache; | ||
116 | |||
117 | #ifdef CONFIG_ARC_TLB_DBG | ||
118 | pr_info("ARC_TLB_DBG: NewMM=0x%x OldMM=0x%x task_struct=0x%x Task: %s," | ||
119 | " pid:%u, assigned asid:%lu\n", | ||
120 | (unsigned int)mm, (unsigned int)prev_owner, | ||
121 | (unsigned int)(mm->context.tsk), (mm->context.tsk)->comm, | ||
122 | (mm->context.tsk)->pid, mm->context.asid); | ||
123 | #endif | ||
124 | |||
125 | write_aux_reg(ARC_REG_PID, asid_cache | MMU_ENABLE); | ||
126 | |||
127 | local_irq_restore(flags); | ||
128 | } | ||
129 | |||
130 | /* | ||
131 | * Initialize the context related info for a new mm_struct | ||
132 | * instance. | ||
133 | */ | ||
134 | static inline int | ||
135 | init_new_context(struct task_struct *tsk, struct mm_struct *mm) | ||
136 | { | ||
137 | mm->context.asid = NO_ASID; | ||
138 | #ifdef CONFIG_ARC_TLB_DBG | ||
139 | mm->context.tsk = tsk; | ||
140 | #endif | ||
141 | return 0; | ||
142 | } | ||
143 | |||
144 | /* Prepare the MMU for task: setup PID reg with allocated ASID | ||
145 | If task doesn't have an ASID (never alloc or stolen, get a new ASID) | ||
146 | */ | ||
147 | static inline void switch_mm(struct mm_struct *prev, struct mm_struct *next, | ||
148 | struct task_struct *tsk) | ||
149 | { | ||
150 | #ifndef CONFIG_SMP | ||
151 | /* PGD cached in MMU reg to avoid 3 mem lookups: task->mm->pgd */ | ||
152 | write_aux_reg(ARC_REG_SCRATCH_DATA0, next->pgd); | ||
153 | #endif | ||
154 | |||
155 | /* | ||
156 | * Get a new ASID if task doesn't have a valid one. Possible when | ||
157 | * -task never had an ASID (fresh after fork) | ||
158 | * -it's ASID was stolen - past an ASID roll-over. | ||
159 | * -There's a third obscure scenario (if this task is running for the | ||
160 | * first time afer an ASID rollover), where despite having a valid | ||
161 | * ASID, we force a get for new ASID - see comments at top. | ||
162 | * | ||
163 | * Both the non-alloc scenario and first-use-after-rollover can be | ||
164 | * detected using the single condition below: NO_ASID = 256 | ||
165 | * while asid_cache is always a valid ASID value (0-255). | ||
166 | */ | ||
167 | if (next->context.asid > asid_cache) { | ||
168 | get_new_mmu_context(next); | ||
169 | } else { | ||
170 | /* | ||
171 | * XXX: This will never happen given the chks above | ||
172 | * BUG_ON(next->context.asid > MAX_ASID); | ||
173 | */ | ||
174 | write_aux_reg(ARC_REG_PID, next->context.asid | MMU_ENABLE); | ||
175 | } | ||
176 | |||
177 | } | ||
178 | |||
179 | static inline void destroy_context(struct mm_struct *mm) | ||
180 | { | ||
181 | unsigned long flags; | ||
182 | |||
183 | local_irq_save(flags); | ||
184 | |||
185 | asid_mm_map[mm->context.asid] = NULL; | ||
186 | mm->context.asid = NO_ASID; | ||
187 | |||
188 | local_irq_restore(flags); | ||
189 | } | ||
190 | |||
191 | /* it seemed that deactivate_mm( ) is a reasonable place to do book-keeping | ||
192 | * for retiring-mm. However destroy_context( ) still needs to do that because | ||
193 | * between mm_release( ) = >deactive_mm( ) and | ||
194 | * mmput => .. => __mmdrop( ) => destroy_context( ) | ||
195 | * there is a good chance that task gets sched-out/in, making it's ASID valid | ||
196 | * again (this teased me for a whole day). | ||
197 | */ | ||
198 | #define deactivate_mm(tsk, mm) do { } while (0) | ||
199 | |||
200 | static inline void activate_mm(struct mm_struct *prev, struct mm_struct *next) | ||
201 | { | ||
202 | #ifndef CONFIG_SMP | ||
203 | write_aux_reg(ARC_REG_SCRATCH_DATA0, next->pgd); | ||
204 | #endif | ||
205 | |||
206 | /* Unconditionally get a new ASID */ | ||
207 | get_new_mmu_context(next); | ||
208 | |||
209 | } | ||
210 | |||
211 | #define enter_lazy_tlb(mm, tsk) | ||
212 | |||
213 | #endif /* __ASM_ARC_MMU_CONTEXT_H */ | ||
diff --git a/arch/arc/include/asm/module.h b/arch/arc/include/asm/module.h new file mode 100644 index 000000000000..518222bb3f8e --- /dev/null +++ b/arch/arc/include/asm/module.h | |||
@@ -0,0 +1,28 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Amit Bhor, Sameer Dhavale: Codito Technologies 2004 | ||
9 | |||
10 | */ | ||
11 | |||
12 | #ifndef _ASM_ARC_MODULE_H | ||
13 | #define _ASM_ARC_MODULE_H | ||
14 | |||
15 | #include <asm-generic/module.h> | ||
16 | |||
17 | #ifdef CONFIG_ARC_DW2_UNWIND | ||
18 | struct mod_arch_specific { | ||
19 | void *unw_info; | ||
20 | int unw_sec_idx; | ||
21 | }; | ||
22 | #endif | ||
23 | |||
24 | #define MODULE_PROC_FAMILY "ARC700" | ||
25 | |||
26 | #define MODULE_ARCH_VERMAGIC MODULE_PROC_FAMILY | ||
27 | |||
28 | #endif /* _ASM_ARC_MODULE_H */ | ||
diff --git a/arch/arc/include/asm/mutex.h b/arch/arc/include/asm/mutex.h new file mode 100644 index 000000000000..a2f88ff9f506 --- /dev/null +++ b/arch/arc/include/asm/mutex.h | |||
@@ -0,0 +1,18 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | /* | ||
10 | * xchg() based mutex fast path maintains a state of 0 or 1, as opposed to | ||
11 | * atomic dec based which can "count" any number of lock contenders. | ||
12 | * This ideally needs to be fixed in core, but for now switching to dec ver. | ||
13 | */ | ||
14 | #if defined(CONFIG_SMP) && (CONFIG_NR_CPUS > 2) | ||
15 | #include <asm-generic/mutex-dec.h> | ||
16 | #else | ||
17 | #include <asm-generic/mutex-xchg.h> | ||
18 | #endif | ||
diff --git a/arch/arc/include/asm/page.h b/arch/arc/include/asm/page.h new file mode 100644 index 000000000000..bdf546104551 --- /dev/null +++ b/arch/arc/include/asm/page.h | |||
@@ -0,0 +1,109 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | #ifndef __ASM_ARC_PAGE_H | ||
9 | #define __ASM_ARC_PAGE_H | ||
10 | |||
11 | #include <uapi/asm/page.h> | ||
12 | |||
13 | |||
14 | #ifndef __ASSEMBLY__ | ||
15 | |||
16 | #define get_user_page(vaddr) __get_free_page(GFP_KERNEL) | ||
17 | #define free_user_page(page, addr) free_page(addr) | ||
18 | |||
19 | /* TBD: for now don't worry about VIPT D$ aliasing */ | ||
20 | #define clear_page(paddr) memset((paddr), 0, PAGE_SIZE) | ||
21 | #define copy_page(to, from) memcpy((to), (from), PAGE_SIZE) | ||
22 | |||
23 | #define clear_user_page(addr, vaddr, pg) clear_page(addr) | ||
24 | #define copy_user_page(vto, vfrom, vaddr, pg) copy_page(vto, vfrom) | ||
25 | |||
26 | #undef STRICT_MM_TYPECHECKS | ||
27 | |||
28 | #ifdef STRICT_MM_TYPECHECKS | ||
29 | /* | ||
30 | * These are used to make use of C type-checking.. | ||
31 | */ | ||
32 | typedef struct { | ||
33 | unsigned long pte; | ||
34 | } pte_t; | ||
35 | typedef struct { | ||
36 | unsigned long pgd; | ||
37 | } pgd_t; | ||
38 | typedef struct { | ||
39 | unsigned long pgprot; | ||
40 | } pgprot_t; | ||
41 | typedef unsigned long pgtable_t; | ||
42 | |||
43 | #define pte_val(x) ((x).pte) | ||
44 | #define pgd_val(x) ((x).pgd) | ||
45 | #define pgprot_val(x) ((x).pgprot) | ||
46 | |||
47 | #define __pte(x) ((pte_t) { (x) }) | ||
48 | #define __pgd(x) ((pgd_t) { (x) }) | ||
49 | #define __pgprot(x) ((pgprot_t) { (x) }) | ||
50 | |||
51 | #define pte_pgprot(x) __pgprot(pte_val(x)) | ||
52 | |||
53 | #else /* !STRICT_MM_TYPECHECKS */ | ||
54 | |||
55 | typedef unsigned long pte_t; | ||
56 | typedef unsigned long pgd_t; | ||
57 | typedef unsigned long pgprot_t; | ||
58 | typedef unsigned long pgtable_t; | ||
59 | |||
60 | #define pte_val(x) (x) | ||
61 | #define pgd_val(x) (x) | ||
62 | #define pgprot_val(x) (x) | ||
63 | #define __pte(x) (x) | ||
64 | #define __pgprot(x) (x) | ||
65 | #define pte_pgprot(x) (x) | ||
66 | |||
67 | #endif | ||
68 | |||
69 | #define ARCH_PFN_OFFSET (CONFIG_LINUX_LINK_BASE >> PAGE_SHIFT) | ||
70 | |||
71 | #define pfn_valid(pfn) (((pfn) - ARCH_PFN_OFFSET) < max_mapnr) | ||
72 | |||
73 | /* | ||
74 | * __pa, __va, virt_to_page (ALERT: deprecated, don't use them) | ||
75 | * | ||
76 | * These macros have historically been misnamed | ||
77 | * virt here means link-address/program-address as embedded in object code. | ||
78 | * So if kernel img is linked at 0x8000_0000 onwards, 0x8010_0000 will be | ||
79 | * 128th page, and virt_to_page( ) will return the struct page corresp to it. | ||
80 | * mem_map[ ] is an array of struct page for each page frame in the system | ||
81 | * | ||
82 | * Independent of where linux is linked at, link-addr = physical address | ||
83 | * So the old macro __pa = vaddr + PAGE_OFFSET - CONFIG_LINUX_LINK_BASE | ||
84 | * would have been wrong in case kernel is not at 0x8zs | ||
85 | */ | ||
86 | #define __pa(vaddr) ((unsigned long)vaddr) | ||
87 | #define __va(paddr) ((void *)((unsigned long)(paddr))) | ||
88 | |||
89 | #define virt_to_page(kaddr) \ | ||
90 | (mem_map + ((__pa(kaddr) - CONFIG_LINUX_LINK_BASE) >> PAGE_SHIFT)) | ||
91 | |||
92 | #define virt_addr_valid(kaddr) pfn_valid(__pa(kaddr) >> PAGE_SHIFT) | ||
93 | |||
94 | /* Default Permissions for page, used in mmap.c */ | ||
95 | #ifdef CONFIG_ARC_STACK_NONEXEC | ||
96 | #define VM_DATA_DEFAULT_FLAGS (VM_READ | VM_WRITE | VM_MAYREAD | VM_MAYWRITE) | ||
97 | #else | ||
98 | #define VM_DATA_DEFAULT_FLAGS (VM_READ | VM_WRITE | VM_EXEC | \ | ||
99 | VM_MAYREAD | VM_MAYWRITE | VM_MAYEXEC) | ||
100 | #endif | ||
101 | |||
102 | #define WANT_PAGE_VIRTUAL 1 | ||
103 | |||
104 | #include <asm-generic/memory_model.h> /* page_to_pfn, pfn_to_page */ | ||
105 | #include <asm-generic/getorder.h> | ||
106 | |||
107 | #endif /* !__ASSEMBLY__ */ | ||
108 | |||
109 | #endif | ||
diff --git a/arch/arc/include/asm/perf_event.h b/arch/arc/include/asm/perf_event.h new file mode 100644 index 000000000000..115ad96480e6 --- /dev/null +++ b/arch/arc/include/asm/perf_event.h | |||
@@ -0,0 +1,13 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | */ | ||
9 | |||
10 | #ifndef __ASM_PERF_EVENT_H | ||
11 | #define __ASM_PERF_EVENT_H | ||
12 | |||
13 | #endif /* __ASM_PERF_EVENT_H */ | ||
diff --git a/arch/arc/include/asm/pgalloc.h b/arch/arc/include/asm/pgalloc.h new file mode 100644 index 000000000000..36a9f20c21a3 --- /dev/null +++ b/arch/arc/include/asm/pgalloc.h | |||
@@ -0,0 +1,134 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * vineetg: June 2011 | ||
9 | * -"/proc/meminfo | grep PageTables" kept on increasing | ||
10 | * Recently added pgtable dtor was not getting called. | ||
11 | * | ||
12 | * vineetg: May 2011 | ||
13 | * -Variable pg-sz means that Page Tables could be variable sized themselves | ||
14 | * So calculate it based on addr traversal split [pgd-bits:pte-bits:xxx] | ||
15 | * -Page Table size capped to max 1 to save memory - hence verified. | ||
16 | * -Since these deal with constants, gcc compile-time optimizes them. | ||
17 | * | ||
18 | * vineetg: Nov 2010 | ||
19 | * -Added pgtable ctor/dtor used for pgtable mem accounting | ||
20 | * | ||
21 | * vineetg: April 2010 | ||
22 | * -Switched pgtable_t from being struct page * to unsigned long | ||
23 | * =Needed so that Page Table allocator (pte_alloc_one) is not forced to | ||
24 | * to deal with struct page. Thay way in future we can make it allocate | ||
25 | * multiple PG Tbls in one Page Frame | ||
26 | * =sweet side effect is avoiding calls to ugly page_address( ) from the | ||
27 | * pg-tlb allocator sub-sys (pte_alloc_one, ptr_free, pmd_populate | ||
28 | * | ||
29 | * Amit Bhor, Sameer Dhavale: Codito Technologies 2004 | ||
30 | */ | ||
31 | |||
32 | #ifndef _ASM_ARC_PGALLOC_H | ||
33 | #define _ASM_ARC_PGALLOC_H | ||
34 | |||
35 | #include <linux/mm.h> | ||
36 | #include <linux/log2.h> | ||
37 | |||
38 | static inline void | ||
39 | pmd_populate_kernel(struct mm_struct *mm, pmd_t *pmd, pte_t *pte) | ||
40 | { | ||
41 | pmd_set(pmd, pte); | ||
42 | } | ||
43 | |||
44 | static inline void | ||
45 | pmd_populate(struct mm_struct *mm, pmd_t *pmd, pgtable_t ptep) | ||
46 | { | ||
47 | pmd_set(pmd, (pte_t *) ptep); | ||
48 | } | ||
49 | |||
50 | static inline int __get_order_pgd(void) | ||
51 | { | ||
52 | return get_order(PTRS_PER_PGD * 4); | ||
53 | } | ||
54 | |||
55 | static inline pgd_t *pgd_alloc(struct mm_struct *mm) | ||
56 | { | ||
57 | int num, num2; | ||
58 | pgd_t *ret = (pgd_t *) __get_free_pages(GFP_KERNEL, __get_order_pgd()); | ||
59 | |||
60 | if (ret) { | ||
61 | num = USER_PTRS_PER_PGD + USER_KERNEL_GUTTER / PGDIR_SIZE; | ||
62 | memzero(ret, num * sizeof(pgd_t)); | ||
63 | |||
64 | num2 = VMALLOC_SIZE / PGDIR_SIZE; | ||
65 | memcpy(ret + num, swapper_pg_dir + num, num2 * sizeof(pgd_t)); | ||
66 | |||
67 | memzero(ret + num + num2, | ||
68 | (PTRS_PER_PGD - num - num2) * sizeof(pgd_t)); | ||
69 | |||
70 | } | ||
71 | return ret; | ||
72 | } | ||
73 | |||
74 | static inline void pgd_free(struct mm_struct *mm, pgd_t *pgd) | ||
75 | { | ||
76 | free_pages((unsigned long)pgd, __get_order_pgd()); | ||
77 | } | ||
78 | |||
79 | |||
80 | /* | ||
81 | * With software-only page-tables, addr-split for traversal is tweakable and | ||
82 | * that directly governs how big tables would be at each level. | ||
83 | * Further, the MMU page size is configurable. | ||
84 | * Thus we need to programatically assert the size constraint | ||
85 | * All of this is const math, allowing gcc to do constant folding/propagation. | ||
86 | */ | ||
87 | |||
88 | static inline int __get_order_pte(void) | ||
89 | { | ||
90 | return get_order(PTRS_PER_PTE * 4); | ||
91 | } | ||
92 | |||
93 | static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm, | ||
94 | unsigned long address) | ||
95 | { | ||
96 | pte_t *pte; | ||
97 | |||
98 | pte = (pte_t *) __get_free_pages(GFP_KERNEL | __GFP_REPEAT | __GFP_ZERO, | ||
99 | __get_order_pte()); | ||
100 | |||
101 | return pte; | ||
102 | } | ||
103 | |||
104 | static inline pgtable_t | ||
105 | pte_alloc_one(struct mm_struct *mm, unsigned long address) | ||
106 | { | ||
107 | pgtable_t pte_pg; | ||
108 | |||
109 | pte_pg = __get_free_pages(GFP_KERNEL | __GFP_REPEAT, __get_order_pte()); | ||
110 | if (pte_pg) { | ||
111 | memzero((void *)pte_pg, PTRS_PER_PTE * 4); | ||
112 | pgtable_page_ctor(virt_to_page(pte_pg)); | ||
113 | } | ||
114 | |||
115 | return pte_pg; | ||
116 | } | ||
117 | |||
118 | static inline void pte_free_kernel(struct mm_struct *mm, pte_t *pte) | ||
119 | { | ||
120 | free_pages((unsigned long)pte, __get_order_pte()); /* takes phy addr */ | ||
121 | } | ||
122 | |||
123 | static inline void pte_free(struct mm_struct *mm, pgtable_t ptep) | ||
124 | { | ||
125 | pgtable_page_dtor(virt_to_page(ptep)); | ||
126 | free_pages(ptep, __get_order_pte()); | ||
127 | } | ||
128 | |||
129 | #define __pte_free_tlb(tlb, pte, addr) pte_free((tlb)->mm, pte) | ||
130 | |||
131 | #define check_pgt_cache() do { } while (0) | ||
132 | #define pmd_pgtable(pmd) pmd_page_vaddr(pmd) | ||
133 | |||
134 | #endif /* _ASM_ARC_PGALLOC_H */ | ||
diff --git a/arch/arc/include/asm/pgtable.h b/arch/arc/include/asm/pgtable.h new file mode 100644 index 000000000000..b7e36684c091 --- /dev/null +++ b/arch/arc/include/asm/pgtable.h | |||
@@ -0,0 +1,405 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * vineetg: May 2011 | ||
9 | * -Folded PAGE_PRESENT (used by VM) and PAGE_VALID (used by MMU) into 1. | ||
10 | * They are semantically the same although in different contexts | ||
11 | * VALID marks a TLB entry exists and it will only happen if PRESENT | ||
12 | * - Utilise some unused free bits to confine PTE flags to 12 bits | ||
13 | * This is a must for 4k pg-sz | ||
14 | * | ||
15 | * vineetg: Mar 2011 - changes to accomodate MMU TLB Page Descriptor mods | ||
16 | * -TLB Locking never really existed, except for initial specs | ||
17 | * -SILENT_xxx not needed for our port | ||
18 | * -Per my request, MMU V3 changes the layout of some of the bits | ||
19 | * to avoid a few shifts in TLB Miss handlers. | ||
20 | * | ||
21 | * vineetg: April 2010 | ||
22 | * -PGD entry no longer contains any flags. If empty it is 0, otherwise has | ||
23 | * Pg-Tbl ptr. Thus pmd_present(), pmd_valid(), pmd_set( ) become simpler | ||
24 | * | ||
25 | * vineetg: April 2010 | ||
26 | * -Switched form 8:11:13 split for page table lookup to 11:8:13 | ||
27 | * -this speeds up page table allocation itself as we now have to memset 1K | ||
28 | * instead of 8k per page table. | ||
29 | * -TODO: Right now page table alloc is 8K and rest 7K is unused | ||
30 | * need to optimise it | ||
31 | * | ||
32 | * Amit Bhor, Sameer Dhavale: Codito Technologies 2004 | ||
33 | */ | ||
34 | |||
35 | #ifndef _ASM_ARC_PGTABLE_H | ||
36 | #define _ASM_ARC_PGTABLE_H | ||
37 | |||
38 | #include <asm/page.h> | ||
39 | #include <asm/mmu.h> | ||
40 | #include <asm-generic/pgtable-nopmd.h> | ||
41 | |||
42 | /************************************************************************** | ||
43 | * Page Table Flags | ||
44 | * | ||
45 | * ARC700 MMU only deals with softare managed TLB entries. | ||
46 | * Page Tables are purely for Linux VM's consumption and the bits below are | ||
47 | * suited to that (uniqueness). Hence some are not implemented in the TLB and | ||
48 | * some have different value in TLB. | ||
49 | * e.g. MMU v2: K_READ bit is 8 and so is GLOBAL (possible becoz they live in | ||
50 | * seperate PD0 and PD1, which combined forms a translation entry) | ||
51 | * while for PTE perspective, they are 8 and 9 respectively | ||
52 | * with MMU v3: Most bits (except SHARED) represent the exact hardware pos | ||
53 | * (saves some bit shift ops in TLB Miss hdlrs) | ||
54 | */ | ||
55 | |||
56 | #if (CONFIG_ARC_MMU_VER <= 2) | ||
57 | |||
58 | #define _PAGE_ACCESSED (1<<1) /* Page is accessed (S) */ | ||
59 | #define _PAGE_CACHEABLE (1<<2) /* Page is cached (H) */ | ||
60 | #define _PAGE_EXECUTE (1<<3) /* Page has user execute perm (H) */ | ||
61 | #define _PAGE_WRITE (1<<4) /* Page has user write perm (H) */ | ||
62 | #define _PAGE_READ (1<<5) /* Page has user read perm (H) */ | ||
63 | #define _PAGE_K_EXECUTE (1<<6) /* Page has kernel execute perm (H) */ | ||
64 | #define _PAGE_K_WRITE (1<<7) /* Page has kernel write perm (H) */ | ||
65 | #define _PAGE_K_READ (1<<8) /* Page has kernel perm (H) */ | ||
66 | #define _PAGE_GLOBAL (1<<9) /* Page is global (H) */ | ||
67 | #define _PAGE_MODIFIED (1<<10) /* Page modified (dirty) (S) */ | ||
68 | #define _PAGE_FILE (1<<10) /* page cache/ swap (S) */ | ||
69 | #define _PAGE_PRESENT (1<<11) /* TLB entry is valid (H) */ | ||
70 | |||
71 | #else | ||
72 | |||
73 | /* PD1 */ | ||
74 | #define _PAGE_CACHEABLE (1<<0) /* Page is cached (H) */ | ||
75 | #define _PAGE_EXECUTE (1<<1) /* Page has user execute perm (H) */ | ||
76 | #define _PAGE_WRITE (1<<2) /* Page has user write perm (H) */ | ||
77 | #define _PAGE_READ (1<<3) /* Page has user read perm (H) */ | ||
78 | #define _PAGE_K_EXECUTE (1<<4) /* Page has kernel execute perm (H) */ | ||
79 | #define _PAGE_K_WRITE (1<<5) /* Page has kernel write perm (H) */ | ||
80 | #define _PAGE_K_READ (1<<6) /* Page has kernel perm (H) */ | ||
81 | #define _PAGE_ACCESSED (1<<7) /* Page is accessed (S) */ | ||
82 | |||
83 | /* PD0 */ | ||
84 | #define _PAGE_GLOBAL (1<<8) /* Page is global (H) */ | ||
85 | #define _PAGE_PRESENT (1<<9) /* TLB entry is valid (H) */ | ||
86 | #define _PAGE_SHARED_CODE (1<<10) /* Shared Code page with cmn vaddr | ||
87 | usable for shared TLB entries (H) */ | ||
88 | |||
89 | #define _PAGE_MODIFIED (1<<11) /* Page modified (dirty) (S) */ | ||
90 | #define _PAGE_FILE (1<<12) /* page cache/ swap (S) */ | ||
91 | |||
92 | #define _PAGE_SHARED_CODE_H (1<<31) /* Hardware counterpart of above */ | ||
93 | #endif | ||
94 | |||
95 | /* Kernel allowed all permissions for all pages */ | ||
96 | #define _K_PAGE_PERMS (_PAGE_K_EXECUTE | _PAGE_K_WRITE | _PAGE_K_READ) | ||
97 | |||
98 | #ifdef CONFIG_ARC_CACHE_PAGES | ||
99 | #define _PAGE_DEF_CACHEABLE _PAGE_CACHEABLE | ||
100 | #else | ||
101 | #define _PAGE_DEF_CACHEABLE (0) | ||
102 | #endif | ||
103 | |||
104 | /* Helper for every "user" page | ||
105 | * -kernel can R/W/X | ||
106 | * -by default cached, unless config otherwise | ||
107 | * -present in memory | ||
108 | */ | ||
109 | #define ___DEF (_PAGE_PRESENT | _K_PAGE_PERMS | _PAGE_DEF_CACHEABLE) | ||
110 | |||
111 | /* Set of bits not changed in pte_modify */ | ||
112 | #define _PAGE_CHG_MASK (PAGE_MASK | _PAGE_ACCESSED | _PAGE_MODIFIED) | ||
113 | |||
114 | /* More Abbrevaited helpers */ | ||
115 | #define PAGE_U_NONE __pgprot(___DEF) | ||
116 | #define PAGE_U_R __pgprot(___DEF | _PAGE_READ) | ||
117 | #define PAGE_U_W_R __pgprot(___DEF | _PAGE_READ | _PAGE_WRITE) | ||
118 | #define PAGE_U_X_R __pgprot(___DEF | _PAGE_READ | _PAGE_EXECUTE) | ||
119 | #define PAGE_U_X_W_R __pgprot(___DEF | _PAGE_READ | _PAGE_WRITE | \ | ||
120 | _PAGE_EXECUTE) | ||
121 | |||
122 | #define PAGE_SHARED PAGE_U_W_R | ||
123 | |||
124 | /* While kernel runs out of unstrslated space, vmalloc/modules use a chunk of | ||
125 | * kernel vaddr space - visible in all addr spaces, but kernel mode only | ||
126 | * Thus Global, all-kernel-access, no-user-access, cached | ||
127 | */ | ||
128 | #define PAGE_KERNEL __pgprot(___DEF | _PAGE_GLOBAL) | ||
129 | |||
130 | /* ioremap */ | ||
131 | #define PAGE_KERNEL_NO_CACHE __pgprot(_PAGE_PRESENT | _K_PAGE_PERMS | \ | ||
132 | _PAGE_GLOBAL) | ||
133 | |||
134 | /************************************************************************** | ||
135 | * Mapping of vm_flags (Generic VM) to PTE flags (arch specific) | ||
136 | * | ||
137 | * Certain cases have 1:1 mapping | ||
138 | * e.g. __P101 means VM_READ, VM_EXEC and !VM_SHARED | ||
139 | * which directly corresponds to PAGE_U_X_R | ||
140 | * | ||
141 | * Other rules which cause the divergence from 1:1 mapping | ||
142 | * | ||
143 | * 1. Although ARC700 can do exclusive execute/write protection (meaning R | ||
144 | * can be tracked independet of X/W unlike some other CPUs), still to | ||
145 | * keep things consistent with other archs: | ||
146 | * -Write implies Read: W => R | ||
147 | * -Execute implies Read: X => R | ||
148 | * | ||
149 | * 2. Pvt Writable doesn't have Write Enabled initially: Pvt-W => !W | ||
150 | * This is to enable COW mechanism | ||
151 | */ | ||
152 | /* xwr */ | ||
153 | #define __P000 PAGE_U_NONE | ||
154 | #define __P001 PAGE_U_R | ||
155 | #define __P010 PAGE_U_R /* Pvt-W => !W */ | ||
156 | #define __P011 PAGE_U_R /* Pvt-W => !W */ | ||
157 | #define __P100 PAGE_U_X_R /* X => R */ | ||
158 | #define __P101 PAGE_U_X_R | ||
159 | #define __P110 PAGE_U_X_R /* Pvt-W => !W and X => R */ | ||
160 | #define __P111 PAGE_U_X_R /* Pvt-W => !W */ | ||
161 | |||
162 | #define __S000 PAGE_U_NONE | ||
163 | #define __S001 PAGE_U_R | ||
164 | #define __S010 PAGE_U_W_R /* W => R */ | ||
165 | #define __S011 PAGE_U_W_R | ||
166 | #define __S100 PAGE_U_X_R /* X => R */ | ||
167 | #define __S101 PAGE_U_X_R | ||
168 | #define __S110 PAGE_U_X_W_R /* X => R */ | ||
169 | #define __S111 PAGE_U_X_W_R | ||
170 | |||
171 | /**************************************************************** | ||
172 | * Page Table Lookup split | ||
173 | * | ||
174 | * We implement 2 tier paging and since this is all software, we are free | ||
175 | * to customize the span of a PGD / PTE entry to suit us | ||
176 | * | ||
177 | * 32 bit virtual address | ||
178 | * ------------------------------------------------------- | ||
179 | * | BITS_FOR_PGD | BITS_FOR_PTE | BITS_IN_PAGE | | ||
180 | * ------------------------------------------------------- | ||
181 | * | | | | ||
182 | * | | --> off in page frame | ||
183 | * | | | ||
184 | * | ---> index into Page Table | ||
185 | * | | ||
186 | * ----> index into Page Directory | ||
187 | */ | ||
188 | |||
189 | #define BITS_IN_PAGE PAGE_SHIFT | ||
190 | |||
191 | /* Optimal Sizing of Pg Tbl - based on MMU page size */ | ||
192 | #if defined(CONFIG_ARC_PAGE_SIZE_8K) | ||
193 | #define BITS_FOR_PTE 8 | ||
194 | #elif defined(CONFIG_ARC_PAGE_SIZE_16K) | ||
195 | #define BITS_FOR_PTE 8 | ||
196 | #elif defined(CONFIG_ARC_PAGE_SIZE_4K) | ||
197 | #define BITS_FOR_PTE 9 | ||
198 | #endif | ||
199 | |||
200 | #define BITS_FOR_PGD (32 - BITS_FOR_PTE - BITS_IN_PAGE) | ||
201 | |||
202 | #define PGDIR_SHIFT (BITS_FOR_PTE + BITS_IN_PAGE) | ||
203 | #define PGDIR_SIZE (1UL << PGDIR_SHIFT) /* vaddr span, not PDG sz */ | ||
204 | #define PGDIR_MASK (~(PGDIR_SIZE-1)) | ||
205 | |||
206 | #ifdef __ASSEMBLY__ | ||
207 | #define PTRS_PER_PTE (1 << BITS_FOR_PTE) | ||
208 | #define PTRS_PER_PGD (1 << BITS_FOR_PGD) | ||
209 | #else | ||
210 | #define PTRS_PER_PTE (1UL << BITS_FOR_PTE) | ||
211 | #define PTRS_PER_PGD (1UL << BITS_FOR_PGD) | ||
212 | #endif | ||
213 | /* | ||
214 | * Number of entries a user land program use. | ||
215 | * TASK_SIZE is the maximum vaddr that can be used by a userland program. | ||
216 | */ | ||
217 | #define USER_PTRS_PER_PGD (TASK_SIZE / PGDIR_SIZE) | ||
218 | |||
219 | /* | ||
220 | * No special requirements for lowest virtual address we permit any user space | ||
221 | * mapping to be mapped at. | ||
222 | */ | ||
223 | #define FIRST_USER_ADDRESS 0 | ||
224 | |||
225 | |||
226 | /**************************************************************** | ||
227 | * Bucket load of VM Helpers | ||
228 | */ | ||
229 | |||
230 | #ifndef __ASSEMBLY__ | ||
231 | |||
232 | #define pte_ERROR(e) \ | ||
233 | pr_crit("%s:%d: bad pte %08lx.\n", __FILE__, __LINE__, pte_val(e)) | ||
234 | #define pgd_ERROR(e) \ | ||
235 | pr_crit("%s:%d: bad pgd %08lx.\n", __FILE__, __LINE__, pgd_val(e)) | ||
236 | |||
237 | /* the zero page used for uninitialized and anonymous pages */ | ||
238 | extern char empty_zero_page[PAGE_SIZE]; | ||
239 | #define ZERO_PAGE(vaddr) (virt_to_page(empty_zero_page)) | ||
240 | |||
241 | #define pte_unmap(pte) do { } while (0) | ||
242 | #define pte_unmap_nested(pte) do { } while (0) | ||
243 | |||
244 | #define set_pte(pteptr, pteval) ((*(pteptr)) = (pteval)) | ||
245 | #define set_pmd(pmdptr, pmdval) (*(pmdptr) = pmdval) | ||
246 | |||
247 | /* find the page descriptor of the Page Tbl ref by PMD entry */ | ||
248 | #define pmd_page(pmd) virt_to_page(pmd_val(pmd) & PAGE_MASK) | ||
249 | |||
250 | /* find the logical addr (phy for ARC) of the Page Tbl ref by PMD entry */ | ||
251 | #define pmd_page_vaddr(pmd) (pmd_val(pmd) & PAGE_MASK) | ||
252 | |||
253 | /* In a 2 level sys, setup the PGD entry with PTE value */ | ||
254 | static inline void pmd_set(pmd_t *pmdp, pte_t *ptep) | ||
255 | { | ||
256 | pmd_val(*pmdp) = (unsigned long)ptep; | ||
257 | } | ||
258 | |||
259 | #define pte_none(x) (!pte_val(x)) | ||
260 | #define pte_present(x) (pte_val(x) & _PAGE_PRESENT) | ||
261 | #define pte_clear(mm, addr, ptep) set_pte_at(mm, addr, ptep, __pte(0)) | ||
262 | |||
263 | #define pmd_none(x) (!pmd_val(x)) | ||
264 | #define pmd_bad(x) ((pmd_val(x) & ~PAGE_MASK)) | ||
265 | #define pmd_present(x) (pmd_val(x)) | ||
266 | #define pmd_clear(xp) do { pmd_val(*(xp)) = 0; } while (0) | ||
267 | |||
268 | #define pte_page(x) (mem_map + \ | ||
269 | (unsigned long)(((pte_val(x) - PAGE_OFFSET) >> PAGE_SHIFT))) | ||
270 | |||
271 | #define mk_pte(page, pgprot) \ | ||
272 | ({ \ | ||
273 | pte_t pte; \ | ||
274 | pte_val(pte) = __pa(page_address(page)) + pgprot_val(pgprot); \ | ||
275 | pte; \ | ||
276 | }) | ||
277 | |||
278 | /* TBD: Non linear mapping stuff */ | ||
279 | static inline int pte_file(pte_t pte) | ||
280 | { | ||
281 | return pte_val(pte) & _PAGE_FILE; | ||
282 | } | ||
283 | |||
284 | #define PTE_FILE_MAX_BITS 30 | ||
285 | #define pgoff_to_pte(x) __pte(x) | ||
286 | #define pte_to_pgoff(x) (pte_val(x) >> 2) | ||
287 | #define pte_pfn(pte) (pte_val(pte) >> PAGE_SHIFT) | ||
288 | #define pfn_pte(pfn, prot) (__pte(((pfn) << PAGE_SHIFT) | pgprot_val(prot))) | ||
289 | #define __pte_index(addr) (((addr) >> PAGE_SHIFT) & (PTRS_PER_PTE - 1)) | ||
290 | |||
291 | /* | ||
292 | * pte_offset gets a @ptr to PMD entry (PGD in our 2-tier paging system) | ||
293 | * and returns ptr to PTE entry corresponding to @addr | ||
294 | */ | ||
295 | #define pte_offset(dir, addr) ((pte_t *)(pmd_page_vaddr(*dir)) +\ | ||
296 | __pte_index(addr)) | ||
297 | |||
298 | /* No mapping of Page Tables in high mem etc, so following same as above */ | ||
299 | #define pte_offset_kernel(dir, addr) pte_offset(dir, addr) | ||
300 | #define pte_offset_map(dir, addr) pte_offset(dir, addr) | ||
301 | |||
302 | /* Zoo of pte_xxx function */ | ||
303 | #define pte_read(pte) (pte_val(pte) & _PAGE_READ) | ||
304 | #define pte_write(pte) (pte_val(pte) & _PAGE_WRITE) | ||
305 | #define pte_dirty(pte) (pte_val(pte) & _PAGE_MODIFIED) | ||
306 | #define pte_young(pte) (pte_val(pte) & _PAGE_ACCESSED) | ||
307 | #define pte_special(pte) (0) | ||
308 | |||
309 | #define PTE_BIT_FUNC(fn, op) \ | ||
310 | static inline pte_t pte_##fn(pte_t pte) { pte_val(pte) op; return pte; } | ||
311 | |||
312 | PTE_BIT_FUNC(wrprotect, &= ~(_PAGE_WRITE)); | ||
313 | PTE_BIT_FUNC(mkwrite, |= (_PAGE_WRITE)); | ||
314 | PTE_BIT_FUNC(mkclean, &= ~(_PAGE_MODIFIED)); | ||
315 | PTE_BIT_FUNC(mkdirty, |= (_PAGE_MODIFIED)); | ||
316 | PTE_BIT_FUNC(mkold, &= ~(_PAGE_ACCESSED)); | ||
317 | PTE_BIT_FUNC(mkyoung, |= (_PAGE_ACCESSED)); | ||
318 | PTE_BIT_FUNC(exprotect, &= ~(_PAGE_EXECUTE)); | ||
319 | PTE_BIT_FUNC(mkexec, |= (_PAGE_EXECUTE)); | ||
320 | |||
321 | static inline pte_t pte_mkspecial(pte_t pte) { return pte; } | ||
322 | |||
323 | static inline pte_t pte_modify(pte_t pte, pgprot_t newprot) | ||
324 | { | ||
325 | return __pte((pte_val(pte) & _PAGE_CHG_MASK) | pgprot_val(newprot)); | ||
326 | } | ||
327 | |||
328 | /* Macro to mark a page protection as uncacheable */ | ||
329 | #define pgprot_noncached(prot) (__pgprot(pgprot_val(prot) & ~_PAGE_CACHEABLE)) | ||
330 | |||
331 | static inline void set_pte_at(struct mm_struct *mm, unsigned long addr, | ||
332 | pte_t *ptep, pte_t pteval) | ||
333 | { | ||
334 | set_pte(ptep, pteval); | ||
335 | } | ||
336 | |||
337 | /* | ||
338 | * All kernel related VM pages are in init's mm. | ||
339 | */ | ||
340 | #define pgd_offset_k(address) pgd_offset(&init_mm, address) | ||
341 | #define pgd_index(addr) ((addr) >> PGDIR_SHIFT) | ||
342 | #define pgd_offset(mm, addr) (((mm)->pgd)+pgd_index(addr)) | ||
343 | |||
344 | /* | ||
345 | * Macro to quickly access the PGD entry, utlising the fact that some | ||
346 | * arch may cache the pointer to Page Directory of "current" task | ||
347 | * in a MMU register | ||
348 | * | ||
349 | * Thus task->mm->pgd (3 pointer dereferences, cache misses etc simply | ||
350 | * becomes read a register | ||
351 | * | ||
352 | * ********CAUTION*******: | ||
353 | * Kernel code might be dealing with some mm_struct of NON "current" | ||
354 | * Thus use this macro only when you are certain that "current" is current | ||
355 | * e.g. when dealing with signal frame setup code etc | ||
356 | */ | ||
357 | #ifndef CONFIG_SMP | ||
358 | #define pgd_offset_fast(mm, addr) \ | ||
359 | ({ \ | ||
360 | pgd_t *pgd_base = (pgd_t *) read_aux_reg(ARC_REG_SCRATCH_DATA0); \ | ||
361 | pgd_base + pgd_index(addr); \ | ||
362 | }) | ||
363 | #else | ||
364 | #define pgd_offset_fast(mm, addr) pgd_offset(mm, addr) | ||
365 | #endif | ||
366 | |||
367 | extern void paging_init(void); | ||
368 | extern pgd_t swapper_pg_dir[] __aligned(PAGE_SIZE); | ||
369 | void update_mmu_cache(struct vm_area_struct *vma, unsigned long address, | ||
370 | pte_t *ptep); | ||
371 | |||
372 | /* Encode swap {type,off} tuple into PTE | ||
373 | * We reserve 13 bits for 5-bit @type, keeping bits 12-5 zero, ensuring that | ||
374 | * both PAGE_FILE and PAGE_PRESENT are zero in a PTE holding swap "identifier" | ||
375 | */ | ||
376 | #define __swp_entry(type, off) ((swp_entry_t) { \ | ||
377 | ((type) & 0x1f) | ((off) << 13) }) | ||
378 | |||
379 | /* Decode a PTE containing swap "identifier "into constituents */ | ||
380 | #define __swp_type(pte_lookalike) (((pte_lookalike).val) & 0x1f) | ||
381 | #define __swp_offset(pte_lookalike) ((pte_lookalike).val << 13) | ||
382 | |||
383 | /* NOPs, to keep generic kernel happy */ | ||
384 | #define __pte_to_swp_entry(pte) ((swp_entry_t) { pte_val(pte) }) | ||
385 | #define __swp_entry_to_pte(x) ((pte_t) { (x).val }) | ||
386 | |||
387 | #define kern_addr_valid(addr) (1) | ||
388 | |||
389 | /* | ||
390 | * remap a physical page `pfn' of size `size' with page protection `prot' | ||
391 | * into virtual address `from' | ||
392 | */ | ||
393 | #define io_remap_pfn_range(vma, from, pfn, size, prot) \ | ||
394 | remap_pfn_range(vma, from, pfn, size, prot) | ||
395 | |||
396 | #include <asm-generic/pgtable.h> | ||
397 | |||
398 | /* | ||
399 | * No page table caches to initialise | ||
400 | */ | ||
401 | #define pgtable_cache_init() do { } while (0) | ||
402 | |||
403 | #endif /* __ASSEMBLY__ */ | ||
404 | |||
405 | #endif | ||
diff --git a/arch/arc/include/asm/processor.h b/arch/arc/include/asm/processor.h new file mode 100644 index 000000000000..5f26b2c1cba0 --- /dev/null +++ b/arch/arc/include/asm/processor.h | |||
@@ -0,0 +1,151 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * vineetg: March 2009 | ||
9 | * -Implemented task_pt_regs( ) | ||
10 | * | ||
11 | * Amit Bhor, Sameer Dhavale, Ashwin Chaugule: Codito Technologies 2004 | ||
12 | */ | ||
13 | |||
14 | #ifndef __ASM_ARC_PROCESSOR_H | ||
15 | #define __ASM_ARC_PROCESSOR_H | ||
16 | |||
17 | #ifdef __KERNEL__ | ||
18 | |||
19 | #ifndef __ASSEMBLY__ | ||
20 | |||
21 | #include <asm/arcregs.h> /* for STATUS_E1_MASK et all */ | ||
22 | |||
23 | /* Arch specific stuff which needs to be saved per task. | ||
24 | * However these items are not so important so as to earn a place in | ||
25 | * struct thread_info | ||
26 | */ | ||
27 | struct thread_struct { | ||
28 | unsigned long ksp; /* kernel mode stack pointer */ | ||
29 | unsigned long callee_reg; /* pointer to callee regs */ | ||
30 | unsigned long fault_address; /* dbls as brkpt holder as well */ | ||
31 | unsigned long cause_code; /* Exception Cause Code (ECR) */ | ||
32 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
33 | unsigned long user_r25; | ||
34 | #endif | ||
35 | #ifdef CONFIG_ARC_FPU_SAVE_RESTORE | ||
36 | struct arc_fpu fpu; | ||
37 | #endif | ||
38 | }; | ||
39 | |||
40 | #define INIT_THREAD { \ | ||
41 | .ksp = sizeof(init_stack) + (unsigned long) init_stack, \ | ||
42 | } | ||
43 | |||
44 | /* Forward declaration, a strange C thing */ | ||
45 | struct task_struct; | ||
46 | |||
47 | /* | ||
48 | * Return saved PC of a blocked thread. | ||
49 | */ | ||
50 | unsigned long thread_saved_pc(struct task_struct *t); | ||
51 | |||
52 | #define task_pt_regs(p) \ | ||
53 | ((struct pt_regs *)(THREAD_SIZE - 4 + (void *)task_stack_page(p)) - 1) | ||
54 | |||
55 | /* Free all resources held by a thread. */ | ||
56 | #define release_thread(thread) do { } while (0) | ||
57 | |||
58 | /* Prepare to copy thread state - unlazy all lazy status */ | ||
59 | #define prepare_to_copy(tsk) do { } while (0) | ||
60 | |||
61 | /* | ||
62 | * A lot of busy-wait loops in SMP are based off of non-volatile data otherwise | ||
63 | * get optimised away by gcc | ||
64 | */ | ||
65 | #ifdef CONFIG_SMP | ||
66 | #define cpu_relax() __asm__ __volatile__ ("" : : : "memory") | ||
67 | #else | ||
68 | #define cpu_relax() do { } while (0) | ||
69 | #endif | ||
70 | |||
71 | #define copy_segments(tsk, mm) do { } while (0) | ||
72 | #define release_segments(mm) do { } while (0) | ||
73 | |||
74 | #define KSTK_EIP(tsk) (task_pt_regs(tsk)->ret) | ||
75 | |||
76 | /* | ||
77 | * Where abouts of Task's sp, fp, blink when it was last seen in kernel mode. | ||
78 | * These can't be derived from pt_regs as that would give correp user-mode val | ||
79 | */ | ||
80 | #define KSTK_ESP(tsk) (tsk->thread.ksp) | ||
81 | #define KSTK_BLINK(tsk) (*((unsigned int *)((KSTK_ESP(tsk)) + (13+1+1)*4))) | ||
82 | #define KSTK_FP(tsk) (*((unsigned int *)((KSTK_ESP(tsk)) + (13+1)*4))) | ||
83 | |||
84 | /* | ||
85 | * Do necessary setup to start up a newly executed thread. | ||
86 | * | ||
87 | * E1,E2 so that Interrupts are enabled in user mode | ||
88 | * L set, so Loop inhibited to begin with | ||
89 | * lp_start and lp_end seeded with bogus non-zero values so to easily catch | ||
90 | * the ARC700 sr to lp_start hardware bug | ||
91 | */ | ||
92 | #define start_thread(_regs, _pc, _usp) \ | ||
93 | do { \ | ||
94 | set_fs(USER_DS); /* reads from user space */ \ | ||
95 | (_regs)->ret = (_pc); \ | ||
96 | /* Interrupts enabled in User Mode */ \ | ||
97 | (_regs)->status32 = STATUS_U_MASK | STATUS_L_MASK \ | ||
98 | | STATUS_E1_MASK | STATUS_E2_MASK; \ | ||
99 | (_regs)->sp = (_usp); \ | ||
100 | /* bogus seed values for debugging */ \ | ||
101 | (_regs)->lp_start = 0x10; \ | ||
102 | (_regs)->lp_end = 0x80; \ | ||
103 | } while (0) | ||
104 | |||
105 | extern unsigned int get_wchan(struct task_struct *p); | ||
106 | |||
107 | /* | ||
108 | * Default implementation of macro that returns current | ||
109 | * instruction pointer ("program counter"). | ||
110 | * Should the PC register be read instead ? This macro does not seem to | ||
111 | * be used in many places so this wont be all that bad. | ||
112 | */ | ||
113 | #define current_text_addr() ({ __label__ _l; _l: &&_l; }) | ||
114 | |||
115 | #endif /* !__ASSEMBLY__ */ | ||
116 | |||
117 | /* Kernels Virtual memory area. | ||
118 | * Unlike other architectures(MIPS, sh, cris ) ARC 700 does not have a | ||
119 | * "kernel translated" region (like KSEG2 in MIPS). So we use a upper part | ||
120 | * of the translated bottom 2GB for kernel virtual memory and protect | ||
121 | * these pages from user accesses by disabling Ru, Eu and Wu. | ||
122 | */ | ||
123 | #define VMALLOC_SIZE (0x10000000) /* 256M */ | ||
124 | #define VMALLOC_START (PAGE_OFFSET - VMALLOC_SIZE) | ||
125 | #define VMALLOC_END (PAGE_OFFSET) | ||
126 | |||
127 | /* Most of the architectures seem to be keeping some kind of padding between | ||
128 | * userspace TASK_SIZE and PAGE_OFFSET. i.e TASK_SIZE != PAGE_OFFSET. | ||
129 | */ | ||
130 | #define USER_KERNEL_GUTTER 0x10000000 | ||
131 | |||
132 | /* User address space: | ||
133 | * On ARC700, CPU allows the entire lower half of 32 bit address space to be | ||
134 | * translated. Thus potentially 2G (0:0x7FFF_FFFF) could be User vaddr space. | ||
135 | * However we steal 256M for kernel addr (0x7000_0000:0x7FFF_FFFF) and another | ||
136 | * 256M (0x6000_0000:0x6FFF_FFFF) is gutter between user/kernel spaces | ||
137 | * Thus total User vaddr space is (0:0x5FFF_FFFF) | ||
138 | */ | ||
139 | #define TASK_SIZE (PAGE_OFFSET - VMALLOC_SIZE - USER_KERNEL_GUTTER) | ||
140 | |||
141 | #define STACK_TOP TASK_SIZE | ||
142 | #define STACK_TOP_MAX STACK_TOP | ||
143 | |||
144 | /* This decides where the kernel will search for a free chunk of vm | ||
145 | * space during mmap's. | ||
146 | */ | ||
147 | #define TASK_UNMAPPED_BASE (TASK_SIZE / 3) | ||
148 | |||
149 | #endif /* __KERNEL__ */ | ||
150 | |||
151 | #endif /* __ASM_ARC_PROCESSOR_H */ | ||
diff --git a/arch/arc/include/asm/prom.h b/arch/arc/include/asm/prom.h new file mode 100644 index 000000000000..692d0d0789a7 --- /dev/null +++ b/arch/arc/include/asm/prom.h | |||
@@ -0,0 +1,14 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_PROM_H_ | ||
10 | #define _ASM_ARC_PROM_H_ | ||
11 | |||
12 | #define HAVE_ARCH_DEVTREE_FIXUPS | ||
13 | |||
14 | #endif | ||
diff --git a/arch/arc/include/asm/ptrace.h b/arch/arc/include/asm/ptrace.h new file mode 100644 index 000000000000..8ae783d20a81 --- /dev/null +++ b/arch/arc/include/asm/ptrace.h | |||
@@ -0,0 +1,130 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Amit Bhor, Sameer Dhavale: Codito Technologies 2004 | ||
9 | */ | ||
10 | #ifndef __ASM_ARC_PTRACE_H | ||
11 | #define __ASM_ARC_PTRACE_H | ||
12 | |||
13 | #include <uapi/asm/ptrace.h> | ||
14 | |||
15 | #ifndef __ASSEMBLY__ | ||
16 | |||
17 | /* THE pt_regs: Defines how regs are saved during entry into kernel */ | ||
18 | |||
19 | struct pt_regs { | ||
20 | /* | ||
21 | * 1 word gutter after reg-file has been saved | ||
22 | * Technically not needed, Since SP always points to a "full" location | ||
23 | * (vs. "empty"). But pt_regs is shared with tools.... | ||
24 | */ | ||
25 | long res; | ||
26 | |||
27 | /* Real registers */ | ||
28 | long bta; /* bta_l1, bta_l2, erbta */ | ||
29 | long lp_start; | ||
30 | long lp_end; | ||
31 | long lp_count; | ||
32 | long status32; /* status32_l1, status32_l2, erstatus */ | ||
33 | long ret; /* ilink1, ilink2 or eret */ | ||
34 | long blink; | ||
35 | long fp; | ||
36 | long r26; /* gp */ | ||
37 | long r12; | ||
38 | long r11; | ||
39 | long r10; | ||
40 | long r9; | ||
41 | long r8; | ||
42 | long r7; | ||
43 | long r6; | ||
44 | long r5; | ||
45 | long r4; | ||
46 | long r3; | ||
47 | long r2; | ||
48 | long r1; | ||
49 | long r0; | ||
50 | long sp; /* user/kernel sp depending on where we came from */ | ||
51 | long orig_r0; | ||
52 | |||
53 | /*to distinguish bet excp, syscall, irq */ | ||
54 | union { | ||
55 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
56 | /* so that assembly code is same for LE/BE */ | ||
57 | unsigned long orig_r8:16, event:16; | ||
58 | #else | ||
59 | unsigned long event:16, orig_r8:16; | ||
60 | #endif | ||
61 | long orig_r8_word; | ||
62 | }; | ||
63 | }; | ||
64 | |||
65 | /* Callee saved registers - need to be saved only when you are scheduled out */ | ||
66 | |||
67 | struct callee_regs { | ||
68 | long res; /* Again this is not needed */ | ||
69 | long r25; | ||
70 | long r24; | ||
71 | long r23; | ||
72 | long r22; | ||
73 | long r21; | ||
74 | long r20; | ||
75 | long r19; | ||
76 | long r18; | ||
77 | long r17; | ||
78 | long r16; | ||
79 | long r15; | ||
80 | long r14; | ||
81 | long r13; | ||
82 | }; | ||
83 | |||
84 | #define instruction_pointer(regs) ((regs)->ret) | ||
85 | #define profile_pc(regs) instruction_pointer(regs) | ||
86 | |||
87 | /* return 1 if user mode or 0 if kernel mode */ | ||
88 | #define user_mode(regs) (regs->status32 & STATUS_U_MASK) | ||
89 | |||
90 | #define user_stack_pointer(regs)\ | ||
91 | ({ unsigned int sp; \ | ||
92 | if (user_mode(regs)) \ | ||
93 | sp = (regs)->sp;\ | ||
94 | else \ | ||
95 | sp = -1; \ | ||
96 | sp; \ | ||
97 | }) | ||
98 | |||
99 | /* return 1 if PC in delay slot */ | ||
100 | #define delay_mode(regs) ((regs->status32 & STATUS_DE_MASK) == STATUS_DE_MASK) | ||
101 | |||
102 | #define in_syscall(regs) (regs->event & orig_r8_IS_SCALL) | ||
103 | #define in_brkpt_trap(regs) (regs->event & orig_r8_IS_BRKPT) | ||
104 | |||
105 | #define syscall_wont_restart(regs) (regs->event |= orig_r8_IS_SCALL_RESTARTED) | ||
106 | #define syscall_restartable(regs) !(regs->event & orig_r8_IS_SCALL_RESTARTED) | ||
107 | |||
108 | #define current_pt_regs() \ | ||
109 | ({ \ | ||
110 | /* open-coded current_thread_info() */ \ | ||
111 | register unsigned long sp asm ("sp"); \ | ||
112 | unsigned long pg_start = (sp & ~(THREAD_SIZE - 1)); \ | ||
113 | (struct pt_regs *)(pg_start + THREAD_SIZE - 4) - 1; \ | ||
114 | }) | ||
115 | |||
116 | static inline long regs_return_value(struct pt_regs *regs) | ||
117 | { | ||
118 | return regs->r0; | ||
119 | } | ||
120 | |||
121 | #endif /* !__ASSEMBLY__ */ | ||
122 | |||
123 | #define orig_r8_IS_SCALL 0x0001 | ||
124 | #define orig_r8_IS_SCALL_RESTARTED 0x0002 | ||
125 | #define orig_r8_IS_BRKPT 0x0004 | ||
126 | #define orig_r8_IS_EXCPN 0x0004 | ||
127 | #define orig_r8_IS_IRQ1 0x0010 | ||
128 | #define orig_r8_IS_IRQ2 0x0020 | ||
129 | |||
130 | #endif /* __ASM_PTRACE_H */ | ||
diff --git a/arch/arc/include/asm/sections.h b/arch/arc/include/asm/sections.h new file mode 100644 index 000000000000..6fc1159dfefe --- /dev/null +++ b/arch/arc/include/asm/sections.h | |||
@@ -0,0 +1,18 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_SECTIONS_H | ||
10 | #define _ASM_ARC_SECTIONS_H | ||
11 | |||
12 | #include <asm-generic/sections.h> | ||
13 | |||
14 | extern char _int_vec_base_lds[]; | ||
15 | extern char __arc_dccm_base[]; | ||
16 | extern char __dtb_start[]; | ||
17 | |||
18 | #endif | ||
diff --git a/arch/arc/include/asm/segment.h b/arch/arc/include/asm/segment.h new file mode 100644 index 000000000000..da2c45979817 --- /dev/null +++ b/arch/arc/include/asm/segment.h | |||
@@ -0,0 +1,24 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASMARC_SEGMENT_H | ||
10 | #define __ASMARC_SEGMENT_H | ||
11 | |||
12 | #ifndef __ASSEMBLY__ | ||
13 | |||
14 | typedef unsigned long mm_segment_t; | ||
15 | |||
16 | #define MAKE_MM_SEG(s) ((mm_segment_t) { (s) }) | ||
17 | |||
18 | #define KERNEL_DS MAKE_MM_SEG(0) | ||
19 | #define USER_DS MAKE_MM_SEG(TASK_SIZE) | ||
20 | |||
21 | #define segment_eq(a, b) ((a) == (b)) | ||
22 | |||
23 | #endif /* __ASSEMBLY__ */ | ||
24 | #endif /* __ASMARC_SEGMENT_H */ | ||
diff --git a/arch/arc/include/asm/serial.h b/arch/arc/include/asm/serial.h new file mode 100644 index 000000000000..4dff5a1e4128 --- /dev/null +++ b/arch/arc/include/asm/serial.h | |||
@@ -0,0 +1,25 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_SERIAL_H | ||
10 | #define _ASM_ARC_SERIAL_H | ||
11 | |||
12 | /* | ||
13 | * early-8250 requires BASE_BAUD to be defined and includes this header. | ||
14 | * We put in a typical value: | ||
15 | * (core clk / 16) - i.e. UART samples 16 times per sec. | ||
16 | * Athough in multi-platform-image this might not work, specially if the | ||
17 | * clk driving the UART is different. | ||
18 | * We can't use DeviceTree as this is typically for early serial. | ||
19 | */ | ||
20 | |||
21 | #include <asm/clk.h> | ||
22 | |||
23 | #define BASE_BAUD (arc_get_core_freq() / 16) | ||
24 | |||
25 | #endif /* _ASM_ARC_SERIAL_H */ | ||
diff --git a/arch/arc/include/asm/setup.h b/arch/arc/include/asm/setup.h new file mode 100644 index 000000000000..229e50681497 --- /dev/null +++ b/arch/arc/include/asm/setup.h | |||
@@ -0,0 +1,37 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | #ifndef __ASMARC_SETUP_H | ||
9 | #define __ASMARC_SETUP_H | ||
10 | |||
11 | |||
12 | #include <linux/types.h> | ||
13 | #include <uapi/asm/setup.h> | ||
14 | |||
15 | #define COMMAND_LINE_SIZE 256 | ||
16 | |||
17 | /* | ||
18 | * Data structure to map a ID to string | ||
19 | * Used a lot for bootup reporting of hardware diversity | ||
20 | */ | ||
21 | struct id_to_str { | ||
22 | int id; | ||
23 | const char *str; | ||
24 | }; | ||
25 | |||
26 | struct cpuinfo_data { | ||
27 | struct id_to_str info; | ||
28 | int up_range; | ||
29 | }; | ||
30 | |||
31 | extern int root_mountflags, end_mem; | ||
32 | extern int running_on_hw; | ||
33 | |||
34 | void __init setup_processor(void); | ||
35 | void __init setup_arch_memory(void); | ||
36 | |||
37 | #endif /* __ASMARC_SETUP_H */ | ||
diff --git a/arch/arc/include/asm/smp.h b/arch/arc/include/asm/smp.h new file mode 100644 index 000000000000..c4fb211dcd25 --- /dev/null +++ b/arch/arc/include/asm/smp.h | |||
@@ -0,0 +1,130 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_ARC_SMP_H | ||
10 | #define __ASM_ARC_SMP_H | ||
11 | |||
12 | #ifdef CONFIG_SMP | ||
13 | |||
14 | #include <linux/types.h> | ||
15 | #include <linux/init.h> | ||
16 | #include <linux/threads.h> | ||
17 | |||
18 | #define raw_smp_processor_id() (current_thread_info()->cpu) | ||
19 | |||
20 | /* including cpumask.h leads to cyclic deps hence this Forward declaration */ | ||
21 | struct cpumask; | ||
22 | |||
23 | /* | ||
24 | * APIs provided by arch SMP code to generic code | ||
25 | */ | ||
26 | extern void arch_send_call_function_single_ipi(int cpu); | ||
27 | extern void arch_send_call_function_ipi_mask(const struct cpumask *mask); | ||
28 | |||
29 | /* | ||
30 | * APIs provided by arch SMP code to rest of arch code | ||
31 | */ | ||
32 | extern void __init smp_init_cpus(void); | ||
33 | extern void __init first_lines_of_secondary(void); | ||
34 | extern const char *arc_platform_smp_cpuinfo(void); | ||
35 | |||
36 | /* | ||
37 | * API expected BY platform smp code (FROM arch smp code) | ||
38 | * | ||
39 | * smp_ipi_irq_setup: | ||
40 | * Takes @cpu and @irq to which the arch-common ISR is hooked up | ||
41 | */ | ||
42 | extern int smp_ipi_irq_setup(int cpu, int irq); | ||
43 | |||
44 | /* | ||
45 | * struct plat_smp_ops - SMP callbacks provided by platform to ARC SMP | ||
46 | * | ||
47 | * @info: SoC SMP specific info for /proc/cpuinfo etc | ||
48 | * @cpu_kick: For Master to kickstart a cpu (optionally at a PC) | ||
49 | * @ipi_send: To send IPI to a @cpumask | ||
50 | * @ips_clear: To clear IPI received by @cpu at @irq | ||
51 | */ | ||
52 | struct plat_smp_ops { | ||
53 | const char *info; | ||
54 | void (*cpu_kick)(int cpu, unsigned long pc); | ||
55 | void (*ipi_send)(void *callmap); | ||
56 | void (*ipi_clear)(int cpu, int irq); | ||
57 | }; | ||
58 | |||
59 | /* TBD: stop exporting it for direct population by platform */ | ||
60 | extern struct plat_smp_ops plat_smp_ops; | ||
61 | |||
62 | #endif /* CONFIG_SMP */ | ||
63 | |||
64 | /* | ||
65 | * ARC700 doesn't support atomic Read-Modify-Write ops. | ||
66 | * Originally Interrupts had to be disabled around code to gaurantee atomicity. | ||
67 | * The LLOCK/SCOND insns allow writing interrupt-hassle-free based atomic ops | ||
68 | * based on retry-if-irq-in-atomic (with hardware assist). | ||
69 | * However despite these, we provide the IRQ disabling variant | ||
70 | * | ||
71 | * (1) These insn were introduced only in 4.10 release. So for older released | ||
72 | * support needed. | ||
73 | * | ||
74 | * (2) In a SMP setup, the LLOCK/SCOND atomiticity across CPUs needs to be | ||
75 | * gaurantted by the platform (not something which core handles). | ||
76 | * Assuming a platform won't, SMP Linux needs to use spinlocks + local IRQ | ||
77 | * disabling for atomicity. | ||
78 | * | ||
79 | * However exported spinlock API is not usable due to cyclic hdr deps | ||
80 | * (even after system.h disintegration upstream) | ||
81 | * asm/bitops.h -> linux/spinlock.h -> linux/preempt.h | ||
82 | * -> linux/thread_info.h -> linux/bitops.h -> asm/bitops.h | ||
83 | * | ||
84 | * So the workaround is to use the lowest level arch spinlock API. | ||
85 | * The exported spinlock API is smart enough to be NOP for !CONFIG_SMP, | ||
86 | * but same is not true for ARCH backend, hence the need for 2 variants | ||
87 | */ | ||
88 | #ifndef CONFIG_ARC_HAS_LLSC | ||
89 | |||
90 | #include <linux/irqflags.h> | ||
91 | #ifdef CONFIG_SMP | ||
92 | |||
93 | #include <asm/spinlock.h> | ||
94 | |||
95 | extern arch_spinlock_t smp_atomic_ops_lock; | ||
96 | extern arch_spinlock_t smp_bitops_lock; | ||
97 | |||
98 | #define atomic_ops_lock(flags) do { \ | ||
99 | local_irq_save(flags); \ | ||
100 | arch_spin_lock(&smp_atomic_ops_lock); \ | ||
101 | } while (0) | ||
102 | |||
103 | #define atomic_ops_unlock(flags) do { \ | ||
104 | arch_spin_unlock(&smp_atomic_ops_lock); \ | ||
105 | local_irq_restore(flags); \ | ||
106 | } while (0) | ||
107 | |||
108 | #define bitops_lock(flags) do { \ | ||
109 | local_irq_save(flags); \ | ||
110 | arch_spin_lock(&smp_bitops_lock); \ | ||
111 | } while (0) | ||
112 | |||
113 | #define bitops_unlock(flags) do { \ | ||
114 | arch_spin_unlock(&smp_bitops_lock); \ | ||
115 | local_irq_restore(flags); \ | ||
116 | } while (0) | ||
117 | |||
118 | #else /* !CONFIG_SMP */ | ||
119 | |||
120 | #define atomic_ops_lock(flags) local_irq_save(flags) | ||
121 | #define atomic_ops_unlock(flags) local_irq_restore(flags) | ||
122 | |||
123 | #define bitops_lock(flags) local_irq_save(flags) | ||
124 | #define bitops_unlock(flags) local_irq_restore(flags) | ||
125 | |||
126 | #endif /* !CONFIG_SMP */ | ||
127 | |||
128 | #endif /* !CONFIG_ARC_HAS_LLSC */ | ||
129 | |||
130 | #endif | ||
diff --git a/arch/arc/include/asm/spinlock.h b/arch/arc/include/asm/spinlock.h new file mode 100644 index 000000000000..f158197ac5b0 --- /dev/null +++ b/arch/arc/include/asm/spinlock.h | |||
@@ -0,0 +1,144 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_SPINLOCK_H | ||
10 | #define __ASM_SPINLOCK_H | ||
11 | |||
12 | #include <asm/spinlock_types.h> | ||
13 | #include <asm/processor.h> | ||
14 | #include <asm/barrier.h> | ||
15 | |||
16 | #define arch_spin_is_locked(x) ((x)->slock != __ARCH_SPIN_LOCK_UNLOCKED__) | ||
17 | #define arch_spin_lock_flags(lock, flags) arch_spin_lock(lock) | ||
18 | #define arch_spin_unlock_wait(x) \ | ||
19 | do { while (arch_spin_is_locked(x)) cpu_relax(); } while (0) | ||
20 | |||
21 | static inline void arch_spin_lock(arch_spinlock_t *lock) | ||
22 | { | ||
23 | unsigned int tmp = __ARCH_SPIN_LOCK_LOCKED__; | ||
24 | |||
25 | __asm__ __volatile__( | ||
26 | "1: ex %0, [%1] \n" | ||
27 | " breq %0, %2, 1b \n" | ||
28 | : "+&r" (tmp) | ||
29 | : "r"(&(lock->slock)), "ir"(__ARCH_SPIN_LOCK_LOCKED__) | ||
30 | : "memory"); | ||
31 | } | ||
32 | |||
33 | static inline int arch_spin_trylock(arch_spinlock_t *lock) | ||
34 | { | ||
35 | unsigned int tmp = __ARCH_SPIN_LOCK_LOCKED__; | ||
36 | |||
37 | __asm__ __volatile__( | ||
38 | "1: ex %0, [%1] \n" | ||
39 | : "+r" (tmp) | ||
40 | : "r"(&(lock->slock)) | ||
41 | : "memory"); | ||
42 | |||
43 | return (tmp == __ARCH_SPIN_LOCK_UNLOCKED__); | ||
44 | } | ||
45 | |||
46 | static inline void arch_spin_unlock(arch_spinlock_t *lock) | ||
47 | { | ||
48 | lock->slock = __ARCH_SPIN_LOCK_UNLOCKED__; | ||
49 | smp_mb(); | ||
50 | } | ||
51 | |||
52 | /* | ||
53 | * Read-write spinlocks, allowing multiple readers but only one writer. | ||
54 | * | ||
55 | * The spinlock itself is contained in @counter and access to it is | ||
56 | * serialized with @lock_mutex. | ||
57 | * | ||
58 | * Unfair locking as Writers could be starved indefinitely by Reader(s) | ||
59 | */ | ||
60 | |||
61 | /* Would read_trylock() succeed? */ | ||
62 | #define arch_read_can_lock(x) ((x)->counter > 0) | ||
63 | |||
64 | /* Would write_trylock() succeed? */ | ||
65 | #define arch_write_can_lock(x) ((x)->counter == __ARCH_RW_LOCK_UNLOCKED__) | ||
66 | |||
67 | /* 1 - lock taken successfully */ | ||
68 | static inline int arch_read_trylock(arch_rwlock_t *rw) | ||
69 | { | ||
70 | int ret = 0; | ||
71 | |||
72 | arch_spin_lock(&(rw->lock_mutex)); | ||
73 | |||
74 | /* | ||
75 | * zero means writer holds the lock exclusively, deny Reader. | ||
76 | * Otherwise grant lock to first/subseq reader | ||
77 | */ | ||
78 | if (rw->counter > 0) { | ||
79 | rw->counter--; | ||
80 | ret = 1; | ||
81 | } | ||
82 | |||
83 | arch_spin_unlock(&(rw->lock_mutex)); | ||
84 | |||
85 | smp_mb(); | ||
86 | return ret; | ||
87 | } | ||
88 | |||
89 | /* 1 - lock taken successfully */ | ||
90 | static inline int arch_write_trylock(arch_rwlock_t *rw) | ||
91 | { | ||
92 | int ret = 0; | ||
93 | |||
94 | arch_spin_lock(&(rw->lock_mutex)); | ||
95 | |||
96 | /* | ||
97 | * If reader(s) hold lock (lock < __ARCH_RW_LOCK_UNLOCKED__), | ||
98 | * deny writer. Otherwise if unlocked grant to writer | ||
99 | * Hence the claim that Linux rwlocks are unfair to writers. | ||
100 | * (can be starved for an indefinite time by readers). | ||
101 | */ | ||
102 | if (rw->counter == __ARCH_RW_LOCK_UNLOCKED__) { | ||
103 | rw->counter = 0; | ||
104 | ret = 1; | ||
105 | } | ||
106 | arch_spin_unlock(&(rw->lock_mutex)); | ||
107 | |||
108 | return ret; | ||
109 | } | ||
110 | |||
111 | static inline void arch_read_lock(arch_rwlock_t *rw) | ||
112 | { | ||
113 | while (!arch_read_trylock(rw)) | ||
114 | cpu_relax(); | ||
115 | } | ||
116 | |||
117 | static inline void arch_write_lock(arch_rwlock_t *rw) | ||
118 | { | ||
119 | while (!arch_write_trylock(rw)) | ||
120 | cpu_relax(); | ||
121 | } | ||
122 | |||
123 | static inline void arch_read_unlock(arch_rwlock_t *rw) | ||
124 | { | ||
125 | arch_spin_lock(&(rw->lock_mutex)); | ||
126 | rw->counter++; | ||
127 | arch_spin_unlock(&(rw->lock_mutex)); | ||
128 | } | ||
129 | |||
130 | static inline void arch_write_unlock(arch_rwlock_t *rw) | ||
131 | { | ||
132 | arch_spin_lock(&(rw->lock_mutex)); | ||
133 | rw->counter = __ARCH_RW_LOCK_UNLOCKED__; | ||
134 | arch_spin_unlock(&(rw->lock_mutex)); | ||
135 | } | ||
136 | |||
137 | #define arch_read_lock_flags(lock, flags) arch_read_lock(lock) | ||
138 | #define arch_write_lock_flags(lock, flags) arch_write_lock(lock) | ||
139 | |||
140 | #define arch_spin_relax(lock) cpu_relax() | ||
141 | #define arch_read_relax(lock) cpu_relax() | ||
142 | #define arch_write_relax(lock) cpu_relax() | ||
143 | |||
144 | #endif /* __ASM_SPINLOCK_H */ | ||
diff --git a/arch/arc/include/asm/spinlock_types.h b/arch/arc/include/asm/spinlock_types.h new file mode 100644 index 000000000000..8276bfd61704 --- /dev/null +++ b/arch/arc/include/asm/spinlock_types.h | |||
@@ -0,0 +1,35 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_SPINLOCK_TYPES_H | ||
10 | #define __ASM_SPINLOCK_TYPES_H | ||
11 | |||
12 | typedef struct { | ||
13 | volatile unsigned int slock; | ||
14 | } arch_spinlock_t; | ||
15 | |||
16 | #define __ARCH_SPIN_LOCK_UNLOCKED__ 0 | ||
17 | #define __ARCH_SPIN_LOCK_LOCKED__ 1 | ||
18 | |||
19 | #define __ARCH_SPIN_LOCK_UNLOCKED { __ARCH_SPIN_LOCK_UNLOCKED__ } | ||
20 | #define __ARCH_SPIN_LOCK_LOCKED { __ARCH_SPIN_LOCK_LOCKED__ } | ||
21 | |||
22 | /* | ||
23 | * Unlocked: 0x01_00_00_00 | ||
24 | * Read lock(s): 0x00_FF_00_00 to say 0x01 | ||
25 | * Write lock: 0x0, but only possible if prior value "unlocked" 0x0100_0000 | ||
26 | */ | ||
27 | typedef struct { | ||
28 | volatile unsigned int counter; | ||
29 | arch_spinlock_t lock_mutex; | ||
30 | } arch_rwlock_t; | ||
31 | |||
32 | #define __ARCH_RW_LOCK_UNLOCKED__ 0x01000000 | ||
33 | #define __ARCH_RW_LOCK_UNLOCKED { .counter = __ARCH_RW_LOCK_UNLOCKED__ } | ||
34 | |||
35 | #endif | ||
diff --git a/arch/arc/include/asm/string.h b/arch/arc/include/asm/string.h new file mode 100644 index 000000000000..87676c8f1412 --- /dev/null +++ b/arch/arc/include/asm/string.h | |||
@@ -0,0 +1,40 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * vineetg: May 2011 | ||
9 | * -We had half-optimised memset/memcpy, got better versions of those | ||
10 | * -Added memcmp, strchr, strcpy, strcmp, strlen | ||
11 | * | ||
12 | * Amit Bhor: Codito Technologies 2004 | ||
13 | */ | ||
14 | |||
15 | #ifndef _ASM_ARC_STRING_H | ||
16 | #define _ASM_ARC_STRING_H | ||
17 | |||
18 | #include <linux/types.h> | ||
19 | |||
20 | #ifdef __KERNEL__ | ||
21 | |||
22 | #define __HAVE_ARCH_MEMSET | ||
23 | #define __HAVE_ARCH_MEMCPY | ||
24 | #define __HAVE_ARCH_MEMCMP | ||
25 | #define __HAVE_ARCH_STRCHR | ||
26 | #define __HAVE_ARCH_STRCPY | ||
27 | #define __HAVE_ARCH_STRCMP | ||
28 | #define __HAVE_ARCH_STRLEN | ||
29 | |||
30 | extern void *memset(void *ptr, int, __kernel_size_t); | ||
31 | extern void *memcpy(void *, const void *, __kernel_size_t); | ||
32 | extern void memzero(void *ptr, __kernel_size_t n); | ||
33 | extern int memcmp(const void *, const void *, __kernel_size_t); | ||
34 | extern char *strchr(const char *s, int c); | ||
35 | extern char *strcpy(char *dest, const char *src); | ||
36 | extern int strcmp(const char *cs, const char *ct); | ||
37 | extern __kernel_size_t strlen(const char *); | ||
38 | |||
39 | #endif /* __KERNEL__ */ | ||
40 | #endif /* _ASM_ARC_STRING_H */ | ||
diff --git a/arch/arc/include/asm/switch_to.h b/arch/arc/include/asm/switch_to.h new file mode 100644 index 000000000000..1b171ab5fec0 --- /dev/null +++ b/arch/arc/include/asm/switch_to.h | |||
@@ -0,0 +1,41 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_SWITCH_TO_H | ||
10 | #define _ASM_ARC_SWITCH_TO_H | ||
11 | |||
12 | #ifndef __ASSEMBLY__ | ||
13 | |||
14 | #include <linux/sched.h> | ||
15 | |||
16 | #ifdef CONFIG_ARC_FPU_SAVE_RESTORE | ||
17 | |||
18 | extern void fpu_save_restore(struct task_struct *p, struct task_struct *n); | ||
19 | #define ARC_FPU_PREV(p, n) fpu_save_restore(p, n) | ||
20 | #define ARC_FPU_NEXT(t) | ||
21 | |||
22 | #else | ||
23 | |||
24 | #define ARC_FPU_PREV(p, n) | ||
25 | #define ARC_FPU_NEXT(n) | ||
26 | |||
27 | #endif /* !CONFIG_ARC_FPU_SAVE_RESTORE */ | ||
28 | |||
29 | struct task_struct *__switch_to(struct task_struct *p, struct task_struct *n); | ||
30 | |||
31 | #define switch_to(prev, next, last) \ | ||
32 | do { \ | ||
33 | ARC_FPU_PREV(prev, next); \ | ||
34 | last = __switch_to(prev, next);\ | ||
35 | ARC_FPU_NEXT(next); \ | ||
36 | mb(); \ | ||
37 | } while (0) | ||
38 | |||
39 | #endif | ||
40 | |||
41 | #endif | ||
diff --git a/arch/arc/include/asm/syscall.h b/arch/arc/include/asm/syscall.h new file mode 100644 index 000000000000..33ab3048e9b2 --- /dev/null +++ b/arch/arc/include/asm/syscall.h | |||
@@ -0,0 +1,72 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_SYSCALL_H | ||
10 | #define _ASM_ARC_SYSCALL_H 1 | ||
11 | |||
12 | #include <linux/err.h> | ||
13 | #include <linux/sched.h> | ||
14 | #include <asm/unistd.h> | ||
15 | #include <asm/ptrace.h> /* in_syscall() */ | ||
16 | |||
17 | static inline long | ||
18 | syscall_get_nr(struct task_struct *task, struct pt_regs *regs) | ||
19 | { | ||
20 | if (user_mode(regs) && in_syscall(regs)) | ||
21 | return regs->orig_r8; | ||
22 | else | ||
23 | return -1; | ||
24 | } | ||
25 | |||
26 | static inline void | ||
27 | syscall_rollback(struct task_struct *task, struct pt_regs *regs) | ||
28 | { | ||
29 | /* XXX: I can't fathom how pt_regs->r8 will be clobbered ? */ | ||
30 | regs->r8 = regs->orig_r8; | ||
31 | } | ||
32 | |||
33 | static inline long | ||
34 | syscall_get_error(struct task_struct *task, struct pt_regs *regs) | ||
35 | { | ||
36 | /* 0 if syscall succeeded, otherwise -Errorcode */ | ||
37 | return IS_ERR_VALUE(regs->r0) ? regs->r0 : 0; | ||
38 | } | ||
39 | |||
40 | static inline long | ||
41 | syscall_get_return_value(struct task_struct *task, struct pt_regs *regs) | ||
42 | { | ||
43 | return regs->r0; | ||
44 | } | ||
45 | |||
46 | static inline void | ||
47 | syscall_set_return_value(struct task_struct *task, struct pt_regs *regs, | ||
48 | int error, long val) | ||
49 | { | ||
50 | regs->r0 = (long) error ?: val; | ||
51 | } | ||
52 | |||
53 | /* | ||
54 | * @i: argument index [0,5] | ||
55 | * @n: number of arguments; n+i must be [1,6]. | ||
56 | */ | ||
57 | static inline void | ||
58 | syscall_get_arguments(struct task_struct *task, struct pt_regs *regs, | ||
59 | unsigned int i, unsigned int n, unsigned long *args) | ||
60 | { | ||
61 | unsigned long *inside_ptregs = &(regs->r0); | ||
62 | inside_ptregs -= i; | ||
63 | |||
64 | BUG_ON((i + n) > 6); | ||
65 | |||
66 | while (n--) { | ||
67 | args[i++] = (*inside_ptregs); | ||
68 | inside_ptregs--; | ||
69 | } | ||
70 | } | ||
71 | |||
72 | #endif | ||
diff --git a/arch/arc/include/asm/syscalls.h b/arch/arc/include/asm/syscalls.h new file mode 100644 index 000000000000..e53a5340ba4f --- /dev/null +++ b/arch/arc/include/asm/syscalls.h | |||
@@ -0,0 +1,29 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_SYSCALLS_H | ||
10 | #define _ASM_ARC_SYSCALLS_H 1 | ||
11 | |||
12 | #ifdef __KERNEL__ | ||
13 | |||
14 | #include <linux/compiler.h> | ||
15 | #include <linux/linkage.h> | ||
16 | #include <linux/types.h> | ||
17 | |||
18 | int sys_clone_wrapper(int, int, int, int, int); | ||
19 | int sys_fork_wrapper(void); | ||
20 | int sys_vfork_wrapper(void); | ||
21 | int sys_cacheflush(uint32_t, uint32_t uint32_t); | ||
22 | int sys_arc_settls(void *); | ||
23 | int sys_arc_gettls(void); | ||
24 | |||
25 | #include <asm-generic/syscalls.h> | ||
26 | |||
27 | #endif /* __KERNEL__ */ | ||
28 | |||
29 | #endif | ||
diff --git a/arch/arc/include/asm/thread_info.h b/arch/arc/include/asm/thread_info.h new file mode 100644 index 000000000000..2d50a4cdd7f3 --- /dev/null +++ b/arch/arc/include/asm/thread_info.h | |||
@@ -0,0 +1,121 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Vineetg: Oct 2009 | ||
9 | * No need for ARC specific thread_info allocator (kmalloc/free). This is | ||
10 | * anyways one page allocation, thus slab alloc can be short-circuited and | ||
11 | * the generic version (get_free_page) would be loads better. | ||
12 | * | ||
13 | * Sameer Dhavale: Codito Technologies 2004 | ||
14 | */ | ||
15 | |||
16 | #ifndef _ASM_THREAD_INFO_H | ||
17 | #define _ASM_THREAD_INFO_H | ||
18 | |||
19 | #ifdef __KERNEL__ | ||
20 | |||
21 | #include <asm/page.h> | ||
22 | |||
23 | #ifdef CONFIG_16KSTACKS | ||
24 | #define THREAD_SIZE_ORDER 1 | ||
25 | #else | ||
26 | #define THREAD_SIZE_ORDER 0 | ||
27 | #endif | ||
28 | |||
29 | #define THREAD_SIZE (PAGE_SIZE << THREAD_SIZE_ORDER) | ||
30 | |||
31 | #ifndef __ASSEMBLY__ | ||
32 | |||
33 | #include <linux/thread_info.h> | ||
34 | #include <asm/segment.h> | ||
35 | |||
36 | /* | ||
37 | * low level task data that entry.S needs immediate access to | ||
38 | * - this struct should fit entirely inside of one cache line | ||
39 | * - this struct shares the supervisor stack pages | ||
40 | * - if the contents of this structure are changed, the assembly constants | ||
41 | * must also be changed | ||
42 | */ | ||
43 | struct thread_info { | ||
44 | unsigned long flags; /* low level flags */ | ||
45 | int preempt_count; /* 0 => preemptable, <0 => BUG */ | ||
46 | struct task_struct *task; /* main task structure */ | ||
47 | mm_segment_t addr_limit; /* thread address space */ | ||
48 | struct exec_domain *exec_domain;/* execution domain */ | ||
49 | __u32 cpu; /* current CPU */ | ||
50 | unsigned long thr_ptr; /* TLS ptr */ | ||
51 | struct restart_block restart_block; | ||
52 | }; | ||
53 | |||
54 | /* | ||
55 | * macros/functions for gaining access to the thread information structure | ||
56 | * | ||
57 | * preempt_count needs to be 1 initially, until the scheduler is functional. | ||
58 | */ | ||
59 | #define INIT_THREAD_INFO(tsk) \ | ||
60 | { \ | ||
61 | .task = &tsk, \ | ||
62 | .exec_domain = &default_exec_domain, \ | ||
63 | .flags = 0, \ | ||
64 | .cpu = 0, \ | ||
65 | .preempt_count = INIT_PREEMPT_COUNT, \ | ||
66 | .addr_limit = KERNEL_DS, \ | ||
67 | .restart_block = { \ | ||
68 | .fn = do_no_restart_syscall, \ | ||
69 | }, \ | ||
70 | } | ||
71 | |||
72 | #define init_thread_info (init_thread_union.thread_info) | ||
73 | #define init_stack (init_thread_union.stack) | ||
74 | |||
75 | static inline __attribute_const__ struct thread_info *current_thread_info(void) | ||
76 | { | ||
77 | register unsigned long sp asm("sp"); | ||
78 | return (struct thread_info *)(sp & ~(THREAD_SIZE - 1)); | ||
79 | } | ||
80 | |||
81 | #endif /* !__ASSEMBLY__ */ | ||
82 | |||
83 | #define PREEMPT_ACTIVE 0x10000000 | ||
84 | |||
85 | /* | ||
86 | * thread information flags | ||
87 | * - these are process state flags that various assembly files may need to | ||
88 | * access | ||
89 | * - pending work-to-be-done flags are in LSW | ||
90 | * - other flags in MSW | ||
91 | */ | ||
92 | #define TIF_RESTORE_SIGMASK 0 /* restore sig mask in do_signal() */ | ||
93 | #define TIF_NOTIFY_RESUME 1 /* resumption notification requested */ | ||
94 | #define TIF_SIGPENDING 2 /* signal pending */ | ||
95 | #define TIF_NEED_RESCHED 3 /* rescheduling necessary */ | ||
96 | #define TIF_SYSCALL_AUDIT 4 /* syscall auditing active */ | ||
97 | #define TIF_SYSCALL_TRACE 15 /* syscall trace active */ | ||
98 | |||
99 | /* true if poll_idle() is polling TIF_NEED_RESCHED */ | ||
100 | #define TIF_MEMDIE 16 | ||
101 | |||
102 | #define _TIF_SYSCALL_TRACE (1<<TIF_SYSCALL_TRACE) | ||
103 | #define _TIF_NOTIFY_RESUME (1<<TIF_NOTIFY_RESUME) | ||
104 | #define _TIF_SIGPENDING (1<<TIF_SIGPENDING) | ||
105 | #define _TIF_NEED_RESCHED (1<<TIF_NEED_RESCHED) | ||
106 | #define _TIF_SYSCALL_AUDIT (1<<TIF_SYSCALL_AUDIT) | ||
107 | #define _TIF_MEMDIE (1<<TIF_MEMDIE) | ||
108 | |||
109 | /* work to do on interrupt/exception return */ | ||
110 | #define _TIF_WORK_MASK (_TIF_NEED_RESCHED | _TIF_SIGPENDING | \ | ||
111 | _TIF_NOTIFY_RESUME) | ||
112 | |||
113 | /* | ||
114 | * _TIF_ALLWORK_MASK includes SYSCALL_TRACE, but we don't need it. | ||
115 | * SYSCALL_TRACE is anways seperately/unconditionally tested right after a | ||
116 | * syscall, so all that reamins to be tested is _TIF_WORK_MASK | ||
117 | */ | ||
118 | |||
119 | #endif /* __KERNEL__ */ | ||
120 | |||
121 | #endif /* _ASM_THREAD_INFO_H */ | ||
diff --git a/arch/arc/include/asm/timex.h b/arch/arc/include/asm/timex.h new file mode 100644 index 000000000000..0a82960a75e9 --- /dev/null +++ b/arch/arc/include/asm/timex.h | |||
@@ -0,0 +1,18 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_TIMEX_H | ||
10 | #define _ASM_ARC_TIMEX_H | ||
11 | |||
12 | #define CLOCK_TICK_RATE 80000000 /* slated to be removed */ | ||
13 | |||
14 | #include <asm-generic/timex.h> | ||
15 | |||
16 | /* XXX: get_cycles() to be implemented with RTSC insn */ | ||
17 | |||
18 | #endif /* _ASM_ARC_TIMEX_H */ | ||
diff --git a/arch/arc/include/asm/tlb-mmu1.h b/arch/arc/include/asm/tlb-mmu1.h new file mode 100644 index 000000000000..a5ff961b1efc --- /dev/null +++ b/arch/arc/include/asm/tlb-mmu1.h | |||
@@ -0,0 +1,104 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_TLB_MMU_V1_H__ | ||
10 | #define __ASM_TLB_MMU_V1_H__ | ||
11 | |||
12 | #if defined(__ASSEMBLY__) && defined(CONFIG_ARC_MMU_VER == 1) | ||
13 | |||
14 | #include <asm/tlb.h> | ||
15 | |||
16 | .macro TLB_WRITE_HEURISTICS | ||
17 | |||
18 | #define JH_HACK1 | ||
19 | #undef JH_HACK2 | ||
20 | #undef JH_HACK3 | ||
21 | |||
22 | #ifdef JH_HACK3 | ||
23 | ; Calculate set index for 2-way MMU | ||
24 | ; -avoiding use of GetIndex from MMU | ||
25 | ; and its unpleasant LFSR pseudo-random sequence | ||
26 | ; | ||
27 | ; r1 = TLBPD0 from TLB_RELOAD above | ||
28 | ; | ||
29 | ; -- jh_ex_way_set not cleared on startup | ||
30 | ; didn't want to change setup.c | ||
31 | ; hence extra instruction to clean | ||
32 | ; | ||
33 | ; -- should be in cache since in same line | ||
34 | ; as r0/r1 saves above | ||
35 | ; | ||
36 | ld r0,[jh_ex_way_sel] ; victim pointer | ||
37 | and r0,r0,1 ; clean | ||
38 | xor.f r0,r0,1 ; flip | ||
39 | st r0,[jh_ex_way_sel] ; store back | ||
40 | asr r0,r1,12 ; get set # <<1, note bit 12=R=0 | ||
41 | or.nz r0,r0,1 ; set way bit | ||
42 | and r0,r0,0xff ; clean | ||
43 | sr r0,[ARC_REG_TLBINDEX] | ||
44 | #endif | ||
45 | |||
46 | #ifdef JH_HACK2 | ||
47 | ; JH hack #2 | ||
48 | ; Faster than hack #1 in non-thrash case, but hard-coded for 2-way MMU | ||
49 | ; Slower in thrash case (where it matters) because more code is executed | ||
50 | ; Inefficient due to two-register paradigm of this miss handler | ||
51 | ; | ||
52 | /* r1 = data TLBPD0 at this point */ | ||
53 | lr r0,[eret] /* instruction address */ | ||
54 | xor r0,r0,r1 /* compare set # */ | ||
55 | and.f r0,r0,0x000fe000 /* 2-way MMU mask */ | ||
56 | bne 88f /* not in same set - no need to probe */ | ||
57 | |||
58 | lr r0,[eret] /* instruction address */ | ||
59 | and r0,r0,PAGE_MASK /* VPN of instruction address */ | ||
60 | ; lr r1,[ARC_REG_TLBPD0] /* Data VPN+ASID - already in r1 from TLB_RELOAD*/ | ||
61 | and r1,r1,0xff /* Data ASID */ | ||
62 | or r0,r0,r1 /* Instruction address + Data ASID */ | ||
63 | |||
64 | lr r1,[ARC_REG_TLBPD0] /* save TLBPD0 containing data TLB*/ | ||
65 | sr r0,[ARC_REG_TLBPD0] /* write instruction address to TLBPD0 */ | ||
66 | sr TLBProbe, [ARC_REG_TLBCOMMAND] /* Look for instruction */ | ||
67 | lr r0,[ARC_REG_TLBINDEX] /* r0 = index where instruction is, if at all */ | ||
68 | sr r1,[ARC_REG_TLBPD0] /* restore TLBPD0 */ | ||
69 | |||
70 | xor r0,r0,1 /* flip bottom bit of data index */ | ||
71 | b.d 89f | ||
72 | sr r0,[ARC_REG_TLBINDEX] /* and put it back */ | ||
73 | 88: | ||
74 | sr TLBGetIndex, [ARC_REG_TLBCOMMAND] | ||
75 | 89: | ||
76 | #endif | ||
77 | |||
78 | #ifdef JH_HACK1 | ||
79 | ; | ||
80 | ; Always checks whether instruction will be kicked out by dtlb miss | ||
81 | ; | ||
82 | mov_s r3, r1 ; save PD0 prepared by TLB_RELOAD in r3 | ||
83 | lr r0,[eret] /* instruction address */ | ||
84 | and r0,r0,PAGE_MASK /* VPN of instruction address */ | ||
85 | bmsk r1,r3,7 /* Data ASID, bits 7-0 */ | ||
86 | or_s r0,r0,r1 /* Instruction address + Data ASID */ | ||
87 | |||
88 | sr r0,[ARC_REG_TLBPD0] /* write instruction address to TLBPD0 */ | ||
89 | sr TLBProbe, [ARC_REG_TLBCOMMAND] /* Look for instruction */ | ||
90 | lr r0,[ARC_REG_TLBINDEX] /* r0 = index where instruction is, if at all */ | ||
91 | sr r3,[ARC_REG_TLBPD0] /* restore TLBPD0 */ | ||
92 | |||
93 | sr TLBGetIndex, [ARC_REG_TLBCOMMAND] | ||
94 | lr r1,[ARC_REG_TLBINDEX] /* r1 = index where MMU wants to put data */ | ||
95 | cmp r0,r1 /* if no match on indices, go around */ | ||
96 | xor.eq r1,r1,1 /* flip bottom bit of data index */ | ||
97 | sr r1,[ARC_REG_TLBINDEX] /* and put it back */ | ||
98 | #endif | ||
99 | |||
100 | .endm | ||
101 | |||
102 | #endif | ||
103 | |||
104 | #endif | ||
diff --git a/arch/arc/include/asm/tlb.h b/arch/arc/include/asm/tlb.h new file mode 100644 index 000000000000..3eb2ce0bdfa3 --- /dev/null +++ b/arch/arc/include/asm/tlb.h | |||
@@ -0,0 +1,58 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_TLB_H | ||
10 | #define _ASM_ARC_TLB_H | ||
11 | |||
12 | #ifdef __KERNEL__ | ||
13 | |||
14 | #include <asm/pgtable.h> | ||
15 | |||
16 | /* Masks for actual TLB "PD"s */ | ||
17 | #define PTE_BITS_IN_PD0 (_PAGE_GLOBAL | _PAGE_PRESENT) | ||
18 | #define PTE_BITS_IN_PD1 (PAGE_MASK | _PAGE_CACHEABLE | \ | ||
19 | _PAGE_EXECUTE | _PAGE_WRITE | _PAGE_READ | \ | ||
20 | _PAGE_K_EXECUTE | _PAGE_K_WRITE | _PAGE_K_READ) | ||
21 | |||
22 | #ifndef __ASSEMBLY__ | ||
23 | |||
24 | #define tlb_flush(tlb) local_flush_tlb_mm((tlb)->mm) | ||
25 | |||
26 | /* | ||
27 | * This pair is called at time of munmap/exit to flush cache and TLB entries | ||
28 | * for mappings being torn down. | ||
29 | * 1) cache-flush part -implemented via tlb_start_vma( ) can be NOP (for now) | ||
30 | * as we don't support aliasing configs in our VIPT D$. | ||
31 | * 2) tlb-flush part - implemted via tlb_end_vma( ) can be NOP as well- | ||
32 | * albiet for difft reasons - its better handled by moving to new ASID | ||
33 | * | ||
34 | * Note, read http://lkml.org/lkml/2004/1/15/6 | ||
35 | */ | ||
36 | #define tlb_start_vma(tlb, vma) | ||
37 | #define tlb_end_vma(tlb, vma) | ||
38 | |||
39 | #define __tlb_remove_tlb_entry(tlb, ptep, address) | ||
40 | |||
41 | #include <linux/pagemap.h> | ||
42 | #include <asm-generic/tlb.h> | ||
43 | |||
44 | #ifdef CONFIG_ARC_DBG_TLB_PARANOIA | ||
45 | void tlb_paranoid_check(unsigned int pid_sw, unsigned long address); | ||
46 | #else | ||
47 | #define tlb_paranoid_check(a, b) | ||
48 | #endif | ||
49 | |||
50 | void arc_mmu_init(void); | ||
51 | extern char *arc_mmu_mumbojumbo(int cpu_id, char *buf, int len); | ||
52 | void __init read_decode_mmu_bcr(void); | ||
53 | |||
54 | #endif /* __ASSEMBLY__ */ | ||
55 | |||
56 | #endif /* __KERNEL__ */ | ||
57 | |||
58 | #endif /* _ASM_ARC_TLB_H */ | ||
diff --git a/arch/arc/include/asm/tlbflush.h b/arch/arc/include/asm/tlbflush.h new file mode 100644 index 000000000000..b2f9bc7f68c8 --- /dev/null +++ b/arch/arc/include/asm/tlbflush.h | |||
@@ -0,0 +1,28 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_ARC_TLBFLUSH__ | ||
10 | #define __ASM_ARC_TLBFLUSH__ | ||
11 | |||
12 | #include <linux/mm.h> | ||
13 | |||
14 | void local_flush_tlb_all(void); | ||
15 | void local_flush_tlb_mm(struct mm_struct *mm); | ||
16 | void local_flush_tlb_page(struct vm_area_struct *vma, unsigned long page); | ||
17 | void local_flush_tlb_kernel_range(unsigned long start, unsigned long end); | ||
18 | void local_flush_tlb_range(struct vm_area_struct *vma, | ||
19 | unsigned long start, unsigned long end); | ||
20 | |||
21 | /* XXX: Revisit for SMP */ | ||
22 | #define flush_tlb_range(vma, s, e) local_flush_tlb_range(vma, s, e) | ||
23 | #define flush_tlb_page(vma, page) local_flush_tlb_page(vma, page) | ||
24 | #define flush_tlb_kernel_range(s, e) local_flush_tlb_kernel_range(s, e) | ||
25 | #define flush_tlb_all() local_flush_tlb_all() | ||
26 | #define flush_tlb_mm(mm) local_flush_tlb_mm(mm) | ||
27 | |||
28 | #endif | ||
diff --git a/arch/arc/include/asm/uaccess.h b/arch/arc/include/asm/uaccess.h new file mode 100644 index 000000000000..32420824375b --- /dev/null +++ b/arch/arc/include/asm/uaccess.h | |||
@@ -0,0 +1,751 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * vineetg: June 2010 | ||
9 | * -__clear_user( ) called multiple times during elf load was byte loop | ||
10 | * converted to do as much word clear as possible. | ||
11 | * | ||
12 | * vineetg: Dec 2009 | ||
13 | * -Hand crafted constant propagation for "constant" copy sizes | ||
14 | * -stock kernel shrunk by 33K at -O3 | ||
15 | * | ||
16 | * vineetg: Sept 2009 | ||
17 | * -Added option to (UN)inline copy_(to|from)_user to reduce code sz | ||
18 | * -kernel shrunk by 200K even at -O3 (gcc 4.2.1) | ||
19 | * -Enabled when doing -Os | ||
20 | * | ||
21 | * Amit Bhor, Sameer Dhavale: Codito Technologies 2004 | ||
22 | */ | ||
23 | |||
24 | #ifndef _ASM_ARC_UACCESS_H | ||
25 | #define _ASM_ARC_UACCESS_H | ||
26 | |||
27 | #include <linux/sched.h> | ||
28 | #include <asm/errno.h> | ||
29 | #include <linux/string.h> /* for generic string functions */ | ||
30 | |||
31 | |||
32 | #define __kernel_ok (segment_eq(get_fs(), KERNEL_DS)) | ||
33 | |||
34 | /* | ||
35 | * Algorthmically, for __user_ok() we want do: | ||
36 | * (start < TASK_SIZE) && (start+len < TASK_SIZE) | ||
37 | * where TASK_SIZE could either be retrieved from thread_info->addr_limit or | ||
38 | * emitted directly in code. | ||
39 | * | ||
40 | * This can however be rewritten as follows: | ||
41 | * (len <= TASK_SIZE) && (start+len < TASK_SIZE) | ||
42 | * | ||
43 | * Because it essentially checks if buffer end is within limit and @len is | ||
44 | * non-ngeative, which implies that buffer start will be within limit too. | ||
45 | * | ||
46 | * The reason for rewriting being, for majorit yof cases, @len is generally | ||
47 | * compile time constant, causing first sub-expression to be compile time | ||
48 | * subsumed. | ||
49 | * | ||
50 | * The second part would generate weird large LIMMs e.g. (0x6000_0000 - 0x10), | ||
51 | * so we check for TASK_SIZE using get_fs() since the addr_limit load from mem | ||
52 | * would already have been done at this call site for __kernel_ok() | ||
53 | * | ||
54 | */ | ||
55 | #define __user_ok(addr, sz) (((sz) <= TASK_SIZE) && \ | ||
56 | (((addr)+(sz)) <= get_fs())) | ||
57 | #define __access_ok(addr, sz) (unlikely(__kernel_ok) || \ | ||
58 | likely(__user_ok((addr), (sz)))) | ||
59 | |||
60 | /*********** Single byte/hword/word copies ******************/ | ||
61 | |||
62 | #define __get_user_fn(sz, u, k) \ | ||
63 | ({ \ | ||
64 | long __ret = 0; /* success by default */ \ | ||
65 | switch (sz) { \ | ||
66 | case 1: __arc_get_user_one(*(k), u, "ldb", __ret); break; \ | ||
67 | case 2: __arc_get_user_one(*(k), u, "ldw", __ret); break; \ | ||
68 | case 4: __arc_get_user_one(*(k), u, "ld", __ret); break; \ | ||
69 | case 8: __arc_get_user_one_64(*(k), u, __ret); break; \ | ||
70 | } \ | ||
71 | __ret; \ | ||
72 | }) | ||
73 | |||
74 | /* | ||
75 | * Returns 0 on success, -EFAULT if not. | ||
76 | * @ret already contains 0 - given that errors will be less likely | ||
77 | * (hence +r asm constraint below). | ||
78 | * In case of error, fixup code will make it -EFAULT | ||
79 | */ | ||
80 | #define __arc_get_user_one(dst, src, op, ret) \ | ||
81 | __asm__ __volatile__( \ | ||
82 | "1: "op" %1,[%2]\n" \ | ||
83 | "2: ;nop\n" \ | ||
84 | " .section .fixup, \"ax\"\n" \ | ||
85 | " .align 4\n" \ | ||
86 | "3: mov %0, %3\n" \ | ||
87 | " j 2b\n" \ | ||
88 | " .previous\n" \ | ||
89 | " .section __ex_table, \"a\"\n" \ | ||
90 | " .align 4\n" \ | ||
91 | " .word 1b,3b\n" \ | ||
92 | " .previous\n" \ | ||
93 | \ | ||
94 | : "+r" (ret), "=r" (dst) \ | ||
95 | : "r" (src), "ir" (-EFAULT)) | ||
96 | |||
97 | #define __arc_get_user_one_64(dst, src, ret) \ | ||
98 | __asm__ __volatile__( \ | ||
99 | "1: ld %1,[%2]\n" \ | ||
100 | "4: ld %R1,[%2, 4]\n" \ | ||
101 | "2: ;nop\n" \ | ||
102 | " .section .fixup, \"ax\"\n" \ | ||
103 | " .align 4\n" \ | ||
104 | "3: mov %0, %3\n" \ | ||
105 | " j 2b\n" \ | ||
106 | " .previous\n" \ | ||
107 | " .section __ex_table, \"a\"\n" \ | ||
108 | " .align 4\n" \ | ||
109 | " .word 1b,3b\n" \ | ||
110 | " .word 4b,3b\n" \ | ||
111 | " .previous\n" \ | ||
112 | \ | ||
113 | : "+r" (ret), "=r" (dst) \ | ||
114 | : "r" (src), "ir" (-EFAULT)) | ||
115 | |||
116 | #define __put_user_fn(sz, u, k) \ | ||
117 | ({ \ | ||
118 | long __ret = 0; /* success by default */ \ | ||
119 | switch (sz) { \ | ||
120 | case 1: __arc_put_user_one(*(k), u, "stb", __ret); break; \ | ||
121 | case 2: __arc_put_user_one(*(k), u, "stw", __ret); break; \ | ||
122 | case 4: __arc_put_user_one(*(k), u, "st", __ret); break; \ | ||
123 | case 8: __arc_put_user_one_64(*(k), u, __ret); break; \ | ||
124 | } \ | ||
125 | __ret; \ | ||
126 | }) | ||
127 | |||
128 | #define __arc_put_user_one(src, dst, op, ret) \ | ||
129 | __asm__ __volatile__( \ | ||
130 | "1: "op" %1,[%2]\n" \ | ||
131 | "2: ;nop\n" \ | ||
132 | " .section .fixup, \"ax\"\n" \ | ||
133 | " .align 4\n" \ | ||
134 | "3: mov %0, %3\n" \ | ||
135 | " j 2b\n" \ | ||
136 | " .previous\n" \ | ||
137 | " .section __ex_table, \"a\"\n" \ | ||
138 | " .align 4\n" \ | ||
139 | " .word 1b,3b\n" \ | ||
140 | " .previous\n" \ | ||
141 | \ | ||
142 | : "+r" (ret) \ | ||
143 | : "r" (src), "r" (dst), "ir" (-EFAULT)) | ||
144 | |||
145 | #define __arc_put_user_one_64(src, dst, ret) \ | ||
146 | __asm__ __volatile__( \ | ||
147 | "1: st %1,[%2]\n" \ | ||
148 | "4: st %R1,[%2, 4]\n" \ | ||
149 | "2: ;nop\n" \ | ||
150 | " .section .fixup, \"ax\"\n" \ | ||
151 | " .align 4\n" \ | ||
152 | "3: mov %0, %3\n" \ | ||
153 | " j 2b\n" \ | ||
154 | " .previous\n" \ | ||
155 | " .section __ex_table, \"a\"\n" \ | ||
156 | " .align 4\n" \ | ||
157 | " .word 1b,3b\n" \ | ||
158 | " .word 4b,3b\n" \ | ||
159 | " .previous\n" \ | ||
160 | \ | ||
161 | : "+r" (ret) \ | ||
162 | : "r" (src), "r" (dst), "ir" (-EFAULT)) | ||
163 | |||
164 | |||
165 | static inline unsigned long | ||
166 | __arc_copy_from_user(void *to, const void __user *from, unsigned long n) | ||
167 | { | ||
168 | long res = 0; | ||
169 | char val; | ||
170 | unsigned long tmp1, tmp2, tmp3, tmp4; | ||
171 | unsigned long orig_n = n; | ||
172 | |||
173 | if (n == 0) | ||
174 | return 0; | ||
175 | |||
176 | /* unaligned */ | ||
177 | if (((unsigned long)to & 0x3) || ((unsigned long)from & 0x3)) { | ||
178 | |||
179 | unsigned char tmp; | ||
180 | |||
181 | __asm__ __volatile__ ( | ||
182 | " mov.f lp_count, %0 \n" | ||
183 | " lpnz 2f \n" | ||
184 | "1: ldb.ab %1, [%3, 1] \n" | ||
185 | " stb.ab %1, [%2, 1] \n" | ||
186 | " sub %0,%0,1 \n" | ||
187 | "2: ;nop \n" | ||
188 | " .section .fixup, \"ax\" \n" | ||
189 | " .align 4 \n" | ||
190 | "3: j 2b \n" | ||
191 | " .previous \n" | ||
192 | " .section __ex_table, \"a\" \n" | ||
193 | " .align 4 \n" | ||
194 | " .word 1b, 3b \n" | ||
195 | " .previous \n" | ||
196 | |||
197 | : "+r" (n), | ||
198 | /* | ||
199 | * Note as an '&' earlyclobber operand to make sure the | ||
200 | * temporary register inside the loop is not the same as | ||
201 | * FROM or TO. | ||
202 | */ | ||
203 | "=&r" (tmp), "+r" (to), "+r" (from) | ||
204 | : | ||
205 | : "lp_count", "lp_start", "lp_end", "memory"); | ||
206 | |||
207 | return n; | ||
208 | } | ||
209 | |||
210 | /* | ||
211 | * Hand-crafted constant propagation to reduce code sz of the | ||
212 | * laddered copy 16x,8,4,2,1 | ||
213 | */ | ||
214 | if (__builtin_constant_p(orig_n)) { | ||
215 | res = orig_n; | ||
216 | |||
217 | if (orig_n / 16) { | ||
218 | orig_n = orig_n % 16; | ||
219 | |||
220 | __asm__ __volatile__( | ||
221 | " lsr lp_count, %7,4 \n" | ||
222 | " lp 3f \n" | ||
223 | "1: ld.ab %3, [%2, 4] \n" | ||
224 | "11: ld.ab %4, [%2, 4] \n" | ||
225 | "12: ld.ab %5, [%2, 4] \n" | ||
226 | "13: ld.ab %6, [%2, 4] \n" | ||
227 | " st.ab %3, [%1, 4] \n" | ||
228 | " st.ab %4, [%1, 4] \n" | ||
229 | " st.ab %5, [%1, 4] \n" | ||
230 | " st.ab %6, [%1, 4] \n" | ||
231 | " sub %0,%0,16 \n" | ||
232 | "3: ;nop \n" | ||
233 | " .section .fixup, \"ax\" \n" | ||
234 | " .align 4 \n" | ||
235 | "4: j 3b \n" | ||
236 | " .previous \n" | ||
237 | " .section __ex_table, \"a\" \n" | ||
238 | " .align 4 \n" | ||
239 | " .word 1b, 4b \n" | ||
240 | " .word 11b,4b \n" | ||
241 | " .word 12b,4b \n" | ||
242 | " .word 13b,4b \n" | ||
243 | " .previous \n" | ||
244 | : "+r" (res), "+r"(to), "+r"(from), | ||
245 | "=r"(tmp1), "=r"(tmp2), "=r"(tmp3), "=r"(tmp4) | ||
246 | : "ir"(n) | ||
247 | : "lp_count", "memory"); | ||
248 | } | ||
249 | if (orig_n / 8) { | ||
250 | orig_n = orig_n % 8; | ||
251 | |||
252 | __asm__ __volatile__( | ||
253 | "14: ld.ab %3, [%2,4] \n" | ||
254 | "15: ld.ab %4, [%2,4] \n" | ||
255 | " st.ab %3, [%1,4] \n" | ||
256 | " st.ab %4, [%1,4] \n" | ||
257 | " sub %0,%0,8 \n" | ||
258 | "31: ;nop \n" | ||
259 | " .section .fixup, \"ax\" \n" | ||
260 | " .align 4 \n" | ||
261 | "4: j 31b \n" | ||
262 | " .previous \n" | ||
263 | " .section __ex_table, \"a\" \n" | ||
264 | " .align 4 \n" | ||
265 | " .word 14b,4b \n" | ||
266 | " .word 15b,4b \n" | ||
267 | " .previous \n" | ||
268 | : "+r" (res), "+r"(to), "+r"(from), | ||
269 | "=r"(tmp1), "=r"(tmp2) | ||
270 | : | ||
271 | : "memory"); | ||
272 | } | ||
273 | if (orig_n / 4) { | ||
274 | orig_n = orig_n % 4; | ||
275 | |||
276 | __asm__ __volatile__( | ||
277 | "16: ld.ab %3, [%2,4] \n" | ||
278 | " st.ab %3, [%1,4] \n" | ||
279 | " sub %0,%0,4 \n" | ||
280 | "32: ;nop \n" | ||
281 | " .section .fixup, \"ax\" \n" | ||
282 | " .align 4 \n" | ||
283 | "4: j 32b \n" | ||
284 | " .previous \n" | ||
285 | " .section __ex_table, \"a\" \n" | ||
286 | " .align 4 \n" | ||
287 | " .word 16b,4b \n" | ||
288 | " .previous \n" | ||
289 | : "+r" (res), "+r"(to), "+r"(from), "=r"(tmp1) | ||
290 | : | ||
291 | : "memory"); | ||
292 | } | ||
293 | if (orig_n / 2) { | ||
294 | orig_n = orig_n % 2; | ||
295 | |||
296 | __asm__ __volatile__( | ||
297 | "17: ldw.ab %3, [%2,2] \n" | ||
298 | " stw.ab %3, [%1,2] \n" | ||
299 | " sub %0,%0,2 \n" | ||
300 | "33: ;nop \n" | ||
301 | " .section .fixup, \"ax\" \n" | ||
302 | " .align 4 \n" | ||
303 | "4: j 33b \n" | ||
304 | " .previous \n" | ||
305 | " .section __ex_table, \"a\" \n" | ||
306 | " .align 4 \n" | ||
307 | " .word 17b,4b \n" | ||
308 | " .previous \n" | ||
309 | : "+r" (res), "+r"(to), "+r"(from), "=r"(tmp1) | ||
310 | : | ||
311 | : "memory"); | ||
312 | } | ||
313 | if (orig_n & 1) { | ||
314 | __asm__ __volatile__( | ||
315 | "18: ldb.ab %3, [%2,2] \n" | ||
316 | " stb.ab %3, [%1,2] \n" | ||
317 | " sub %0,%0,1 \n" | ||
318 | "34: ; nop \n" | ||
319 | " .section .fixup, \"ax\" \n" | ||
320 | " .align 4 \n" | ||
321 | "4: j 34b \n" | ||
322 | " .previous \n" | ||
323 | " .section __ex_table, \"a\" \n" | ||
324 | " .align 4 \n" | ||
325 | " .word 18b,4b \n" | ||
326 | " .previous \n" | ||
327 | : "+r" (res), "+r"(to), "+r"(from), "=r"(tmp1) | ||
328 | : | ||
329 | : "memory"); | ||
330 | } | ||
331 | } else { /* n is NOT constant, so laddered copy of 16x,8,4,2,1 */ | ||
332 | |||
333 | __asm__ __volatile__( | ||
334 | " mov %0,%3 \n" | ||
335 | " lsr.f lp_count, %3,4 \n" /* 16x bytes */ | ||
336 | " lpnz 3f \n" | ||
337 | "1: ld.ab %5, [%2, 4] \n" | ||
338 | "11: ld.ab %6, [%2, 4] \n" | ||
339 | "12: ld.ab %7, [%2, 4] \n" | ||
340 | "13: ld.ab %8, [%2, 4] \n" | ||
341 | " st.ab %5, [%1, 4] \n" | ||
342 | " st.ab %6, [%1, 4] \n" | ||
343 | " st.ab %7, [%1, 4] \n" | ||
344 | " st.ab %8, [%1, 4] \n" | ||
345 | " sub %0,%0,16 \n" | ||
346 | "3: and.f %3,%3,0xf \n" /* stragglers */ | ||
347 | " bz 34f \n" | ||
348 | " bbit0 %3,3,31f \n" /* 8 bytes left */ | ||
349 | "14: ld.ab %5, [%2,4] \n" | ||
350 | "15: ld.ab %6, [%2,4] \n" | ||
351 | " st.ab %5, [%1,4] \n" | ||
352 | " st.ab %6, [%1,4] \n" | ||
353 | " sub.f %0,%0,8 \n" | ||
354 | "31: bbit0 %3,2,32f \n" /* 4 bytes left */ | ||
355 | "16: ld.ab %5, [%2,4] \n" | ||
356 | " st.ab %5, [%1,4] \n" | ||
357 | " sub.f %0,%0,4 \n" | ||
358 | "32: bbit0 %3,1,33f \n" /* 2 bytes left */ | ||
359 | "17: ldw.ab %5, [%2,2] \n" | ||
360 | " stw.ab %5, [%1,2] \n" | ||
361 | " sub.f %0,%0,2 \n" | ||
362 | "33: bbit0 %3,0,34f \n" | ||
363 | "18: ldb.ab %5, [%2,1] \n" /* 1 byte left */ | ||
364 | " stb.ab %5, [%1,1] \n" | ||
365 | " sub.f %0,%0,1 \n" | ||
366 | "34: ;nop \n" | ||
367 | " .section .fixup, \"ax\" \n" | ||
368 | " .align 4 \n" | ||
369 | "4: j 34b \n" | ||
370 | " .previous \n" | ||
371 | " .section __ex_table, \"a\" \n" | ||
372 | " .align 4 \n" | ||
373 | " .word 1b, 4b \n" | ||
374 | " .word 11b,4b \n" | ||
375 | " .word 12b,4b \n" | ||
376 | " .word 13b,4b \n" | ||
377 | " .word 14b,4b \n" | ||
378 | " .word 15b,4b \n" | ||
379 | " .word 16b,4b \n" | ||
380 | " .word 17b,4b \n" | ||
381 | " .word 18b,4b \n" | ||
382 | " .previous \n" | ||
383 | : "=r" (res), "+r"(to), "+r"(from), "+r"(n), "=r"(val), | ||
384 | "=r"(tmp1), "=r"(tmp2), "=r"(tmp3), "=r"(tmp4) | ||
385 | : | ||
386 | : "lp_count", "memory"); | ||
387 | } | ||
388 | |||
389 | return res; | ||
390 | } | ||
391 | |||
392 | extern unsigned long slowpath_copy_to_user(void __user *to, const void *from, | ||
393 | unsigned long n); | ||
394 | |||
395 | static inline unsigned long | ||
396 | __arc_copy_to_user(void __user *to, const void *from, unsigned long n) | ||
397 | { | ||
398 | long res = 0; | ||
399 | char val; | ||
400 | unsigned long tmp1, tmp2, tmp3, tmp4; | ||
401 | unsigned long orig_n = n; | ||
402 | |||
403 | if (n == 0) | ||
404 | return 0; | ||
405 | |||
406 | /* unaligned */ | ||
407 | if (((unsigned long)to & 0x3) || ((unsigned long)from & 0x3)) { | ||
408 | |||
409 | unsigned char tmp; | ||
410 | |||
411 | __asm__ __volatile__( | ||
412 | " mov.f lp_count, %0 \n" | ||
413 | " lpnz 3f \n" | ||
414 | " ldb.ab %1, [%3, 1] \n" | ||
415 | "1: stb.ab %1, [%2, 1] \n" | ||
416 | " sub %0, %0, 1 \n" | ||
417 | "3: ;nop \n" | ||
418 | " .section .fixup, \"ax\" \n" | ||
419 | " .align 4 \n" | ||
420 | "4: j 3b \n" | ||
421 | " .previous \n" | ||
422 | " .section __ex_table, \"a\" \n" | ||
423 | " .align 4 \n" | ||
424 | " .word 1b, 4b \n" | ||
425 | " .previous \n" | ||
426 | |||
427 | : "+r" (n), | ||
428 | /* Note as an '&' earlyclobber operand to make sure the | ||
429 | * temporary register inside the loop is not the same as | ||
430 | * FROM or TO. | ||
431 | */ | ||
432 | "=&r" (tmp), "+r" (to), "+r" (from) | ||
433 | : | ||
434 | : "lp_count", "lp_start", "lp_end", "memory"); | ||
435 | |||
436 | return n; | ||
437 | } | ||
438 | |||
439 | if (__builtin_constant_p(orig_n)) { | ||
440 | res = orig_n; | ||
441 | |||
442 | if (orig_n / 16) { | ||
443 | orig_n = orig_n % 16; | ||
444 | |||
445 | __asm__ __volatile__( | ||
446 | " lsr lp_count, %7,4 \n" | ||
447 | " lp 3f \n" | ||
448 | " ld.ab %3, [%2, 4] \n" | ||
449 | " ld.ab %4, [%2, 4] \n" | ||
450 | " ld.ab %5, [%2, 4] \n" | ||
451 | " ld.ab %6, [%2, 4] \n" | ||
452 | "1: st.ab %3, [%1, 4] \n" | ||
453 | "11: st.ab %4, [%1, 4] \n" | ||
454 | "12: st.ab %5, [%1, 4] \n" | ||
455 | "13: st.ab %6, [%1, 4] \n" | ||
456 | " sub %0, %0, 16 \n" | ||
457 | "3:;nop \n" | ||
458 | " .section .fixup, \"ax\" \n" | ||
459 | " .align 4 \n" | ||
460 | "4: j 3b \n" | ||
461 | " .previous \n" | ||
462 | " .section __ex_table, \"a\" \n" | ||
463 | " .align 4 \n" | ||
464 | " .word 1b, 4b \n" | ||
465 | " .word 11b,4b \n" | ||
466 | " .word 12b,4b \n" | ||
467 | " .word 13b,4b \n" | ||
468 | " .previous \n" | ||
469 | : "+r" (res), "+r"(to), "+r"(from), | ||
470 | "=r"(tmp1), "=r"(tmp2), "=r"(tmp3), "=r"(tmp4) | ||
471 | : "ir"(n) | ||
472 | : "lp_count", "memory"); | ||
473 | } | ||
474 | if (orig_n / 8) { | ||
475 | orig_n = orig_n % 8; | ||
476 | |||
477 | __asm__ __volatile__( | ||
478 | " ld.ab %3, [%2,4] \n" | ||
479 | " ld.ab %4, [%2,4] \n" | ||
480 | "14: st.ab %3, [%1,4] \n" | ||
481 | "15: st.ab %4, [%1,4] \n" | ||
482 | " sub %0, %0, 8 \n" | ||
483 | "31:;nop \n" | ||
484 | " .section .fixup, \"ax\" \n" | ||
485 | " .align 4 \n" | ||
486 | "4: j 31b \n" | ||
487 | " .previous \n" | ||
488 | " .section __ex_table, \"a\" \n" | ||
489 | " .align 4 \n" | ||
490 | " .word 14b,4b \n" | ||
491 | " .word 15b,4b \n" | ||
492 | " .previous \n" | ||
493 | : "+r" (res), "+r"(to), "+r"(from), | ||
494 | "=r"(tmp1), "=r"(tmp2) | ||
495 | : | ||
496 | : "memory"); | ||
497 | } | ||
498 | if (orig_n / 4) { | ||
499 | orig_n = orig_n % 4; | ||
500 | |||
501 | __asm__ __volatile__( | ||
502 | " ld.ab %3, [%2,4] \n" | ||
503 | "16: st.ab %3, [%1,4] \n" | ||
504 | " sub %0, %0, 4 \n" | ||
505 | "32:;nop \n" | ||
506 | " .section .fixup, \"ax\" \n" | ||
507 | " .align 4 \n" | ||
508 | "4: j 32b \n" | ||
509 | " .previous \n" | ||
510 | " .section __ex_table, \"a\" \n" | ||
511 | " .align 4 \n" | ||
512 | " .word 16b,4b \n" | ||
513 | " .previous \n" | ||
514 | : "+r" (res), "+r"(to), "+r"(from), "=r"(tmp1) | ||
515 | : | ||
516 | : "memory"); | ||
517 | } | ||
518 | if (orig_n / 2) { | ||
519 | orig_n = orig_n % 2; | ||
520 | |||
521 | __asm__ __volatile__( | ||
522 | " ldw.ab %3, [%2,2] \n" | ||
523 | "17: stw.ab %3, [%1,2] \n" | ||
524 | " sub %0, %0, 2 \n" | ||
525 | "33:;nop \n" | ||
526 | " .section .fixup, \"ax\" \n" | ||
527 | " .align 4 \n" | ||
528 | "4: j 33b \n" | ||
529 | " .previous \n" | ||
530 | " .section __ex_table, \"a\" \n" | ||
531 | " .align 4 \n" | ||
532 | " .word 17b,4b \n" | ||
533 | " .previous \n" | ||
534 | : "+r" (res), "+r"(to), "+r"(from), "=r"(tmp1) | ||
535 | : | ||
536 | : "memory"); | ||
537 | } | ||
538 | if (orig_n & 1) { | ||
539 | __asm__ __volatile__( | ||
540 | " ldb.ab %3, [%2,1] \n" | ||
541 | "18: stb.ab %3, [%1,1] \n" | ||
542 | " sub %0, %0, 1 \n" | ||
543 | "34: ;nop \n" | ||
544 | " .section .fixup, \"ax\" \n" | ||
545 | " .align 4 \n" | ||
546 | "4: j 34b \n" | ||
547 | " .previous \n" | ||
548 | " .section __ex_table, \"a\" \n" | ||
549 | " .align 4 \n" | ||
550 | " .word 18b,4b \n" | ||
551 | " .previous \n" | ||
552 | : "+r" (res), "+r"(to), "+r"(from), "=r"(tmp1) | ||
553 | : | ||
554 | : "memory"); | ||
555 | } | ||
556 | } else { /* n is NOT constant, so laddered copy of 16x,8,4,2,1 */ | ||
557 | |||
558 | __asm__ __volatile__( | ||
559 | " mov %0,%3 \n" | ||
560 | " lsr.f lp_count, %3,4 \n" /* 16x bytes */ | ||
561 | " lpnz 3f \n" | ||
562 | " ld.ab %5, [%2, 4] \n" | ||
563 | " ld.ab %6, [%2, 4] \n" | ||
564 | " ld.ab %7, [%2, 4] \n" | ||
565 | " ld.ab %8, [%2, 4] \n" | ||
566 | "1: st.ab %5, [%1, 4] \n" | ||
567 | "11: st.ab %6, [%1, 4] \n" | ||
568 | "12: st.ab %7, [%1, 4] \n" | ||
569 | "13: st.ab %8, [%1, 4] \n" | ||
570 | " sub %0, %0, 16 \n" | ||
571 | "3: and.f %3,%3,0xf \n" /* stragglers */ | ||
572 | " bz 34f \n" | ||
573 | " bbit0 %3,3,31f \n" /* 8 bytes left */ | ||
574 | " ld.ab %5, [%2,4] \n" | ||
575 | " ld.ab %6, [%2,4] \n" | ||
576 | "14: st.ab %5, [%1,4] \n" | ||
577 | "15: st.ab %6, [%1,4] \n" | ||
578 | " sub.f %0, %0, 8 \n" | ||
579 | "31: bbit0 %3,2,32f \n" /* 4 bytes left */ | ||
580 | " ld.ab %5, [%2,4] \n" | ||
581 | "16: st.ab %5, [%1,4] \n" | ||
582 | " sub.f %0, %0, 4 \n" | ||
583 | "32: bbit0 %3,1,33f \n" /* 2 bytes left */ | ||
584 | " ldw.ab %5, [%2,2] \n" | ||
585 | "17: stw.ab %5, [%1,2] \n" | ||
586 | " sub.f %0, %0, 2 \n" | ||
587 | "33: bbit0 %3,0,34f \n" | ||
588 | " ldb.ab %5, [%2,1] \n" /* 1 byte left */ | ||
589 | "18: stb.ab %5, [%1,1] \n" | ||
590 | " sub.f %0, %0, 1 \n" | ||
591 | "34: ;nop \n" | ||
592 | " .section .fixup, \"ax\" \n" | ||
593 | " .align 4 \n" | ||
594 | "4: j 34b \n" | ||
595 | " .previous \n" | ||
596 | " .section __ex_table, \"a\" \n" | ||
597 | " .align 4 \n" | ||
598 | " .word 1b, 4b \n" | ||
599 | " .word 11b,4b \n" | ||
600 | " .word 12b,4b \n" | ||
601 | " .word 13b,4b \n" | ||
602 | " .word 14b,4b \n" | ||
603 | " .word 15b,4b \n" | ||
604 | " .word 16b,4b \n" | ||
605 | " .word 17b,4b \n" | ||
606 | " .word 18b,4b \n" | ||
607 | " .previous \n" | ||
608 | : "=r" (res), "+r"(to), "+r"(from), "+r"(n), "=r"(val), | ||
609 | "=r"(tmp1), "=r"(tmp2), "=r"(tmp3), "=r"(tmp4) | ||
610 | : | ||
611 | : "lp_count", "memory"); | ||
612 | } | ||
613 | |||
614 | return res; | ||
615 | } | ||
616 | |||
617 | static inline unsigned long __arc_clear_user(void __user *to, unsigned long n) | ||
618 | { | ||
619 | long res = n; | ||
620 | unsigned char *d_char = to; | ||
621 | |||
622 | __asm__ __volatile__( | ||
623 | " bbit0 %0, 0, 1f \n" | ||
624 | "75: stb.ab %2, [%0,1] \n" | ||
625 | " sub %1, %1, 1 \n" | ||
626 | "1: bbit0 %0, 1, 2f \n" | ||
627 | "76: stw.ab %2, [%0,2] \n" | ||
628 | " sub %1, %1, 2 \n" | ||
629 | "2: asr.f lp_count, %1, 2 \n" | ||
630 | " lpnz 3f \n" | ||
631 | "77: st.ab %2, [%0,4] \n" | ||
632 | " sub %1, %1, 4 \n" | ||
633 | "3: bbit0 %1, 1, 4f \n" | ||
634 | "78: stw.ab %2, [%0,2] \n" | ||
635 | " sub %1, %1, 2 \n" | ||
636 | "4: bbit0 %1, 0, 5f \n" | ||
637 | "79: stb.ab %2, [%0,1] \n" | ||
638 | " sub %1, %1, 1 \n" | ||
639 | "5: \n" | ||
640 | " .section .fixup, \"ax\" \n" | ||
641 | " .align 4 \n" | ||
642 | "3: j 5b \n" | ||
643 | " .previous \n" | ||
644 | " .section __ex_table, \"a\" \n" | ||
645 | " .align 4 \n" | ||
646 | " .word 75b, 3b \n" | ||
647 | " .word 76b, 3b \n" | ||
648 | " .word 77b, 3b \n" | ||
649 | " .word 78b, 3b \n" | ||
650 | " .word 79b, 3b \n" | ||
651 | " .previous \n" | ||
652 | : "+r"(d_char), "+r"(res) | ||
653 | : "i"(0) | ||
654 | : "lp_count", "lp_start", "lp_end", "memory"); | ||
655 | |||
656 | return res; | ||
657 | } | ||
658 | |||
659 | static inline long | ||
660 | __arc_strncpy_from_user(char *dst, const char __user *src, long count) | ||
661 | { | ||
662 | long res = count; | ||
663 | char val; | ||
664 | unsigned int hw_count; | ||
665 | |||
666 | if (count == 0) | ||
667 | return 0; | ||
668 | |||
669 | __asm__ __volatile__( | ||
670 | " lp 2f \n" | ||
671 | "1: ldb.ab %3, [%2, 1] \n" | ||
672 | " breq.d %3, 0, 2f \n" | ||
673 | " stb.ab %3, [%1, 1] \n" | ||
674 | "2: sub %0, %6, %4 \n" | ||
675 | "3: ;nop \n" | ||
676 | " .section .fixup, \"ax\" \n" | ||
677 | " .align 4 \n" | ||
678 | "4: mov %0, %5 \n" | ||
679 | " j 3b \n" | ||
680 | " .previous \n" | ||
681 | " .section __ex_table, \"a\" \n" | ||
682 | " .align 4 \n" | ||
683 | " .word 1b, 4b \n" | ||
684 | " .previous \n" | ||
685 | : "=r"(res), "+r"(dst), "+r"(src), "=&r"(val), "=l"(hw_count) | ||
686 | : "g"(-EFAULT), "ir"(count), "4"(count) /* this "4" seeds lp_count */ | ||
687 | : "memory"); | ||
688 | |||
689 | return res; | ||
690 | } | ||
691 | |||
692 | static inline long __arc_strnlen_user(const char __user *s, long n) | ||
693 | { | ||
694 | long res, tmp1, cnt; | ||
695 | char val; | ||
696 | |||
697 | __asm__ __volatile__( | ||
698 | " mov %2, %1 \n" | ||
699 | "1: ldb.ab %3, [%0, 1] \n" | ||
700 | " breq.d %3, 0, 2f \n" | ||
701 | " sub.f %2, %2, 1 \n" | ||
702 | " bnz 1b \n" | ||
703 | " sub %2, %2, 1 \n" | ||
704 | "2: sub %0, %1, %2 \n" | ||
705 | "3: ;nop \n" | ||
706 | " .section .fixup, \"ax\" \n" | ||
707 | " .align 4 \n" | ||
708 | "4: mov %0, 0 \n" | ||
709 | " j 3b \n" | ||
710 | " .previous \n" | ||
711 | " .section __ex_table, \"a\" \n" | ||
712 | " .align 4 \n" | ||
713 | " .word 1b, 4b \n" | ||
714 | " .previous \n" | ||
715 | : "=r"(res), "=r"(tmp1), "=r"(cnt), "=r"(val) | ||
716 | : "0"(s), "1"(n) | ||
717 | : "memory"); | ||
718 | |||
719 | return res; | ||
720 | } | ||
721 | |||
722 | #ifndef CONFIG_CC_OPTIMIZE_FOR_SIZE | ||
723 | #define __copy_from_user(t, f, n) __arc_copy_from_user(t, f, n) | ||
724 | #define __copy_to_user(t, f, n) __arc_copy_to_user(t, f, n) | ||
725 | #define __clear_user(d, n) __arc_clear_user(d, n) | ||
726 | #define __strncpy_from_user(d, s, n) __arc_strncpy_from_user(d, s, n) | ||
727 | #define __strnlen_user(s, n) __arc_strnlen_user(s, n) | ||
728 | #else | ||
729 | extern long arc_copy_from_user_noinline(void *to, const void __user * from, | ||
730 | unsigned long n); | ||
731 | extern long arc_copy_to_user_noinline(void __user *to, const void *from, | ||
732 | unsigned long n); | ||
733 | extern unsigned long arc_clear_user_noinline(void __user *to, | ||
734 | unsigned long n); | ||
735 | extern long arc_strncpy_from_user_noinline (char *dst, const char __user *src, | ||
736 | long count); | ||
737 | extern long arc_strnlen_user_noinline(const char __user *src, long n); | ||
738 | |||
739 | #define __copy_from_user(t, f, n) arc_copy_from_user_noinline(t, f, n) | ||
740 | #define __copy_to_user(t, f, n) arc_copy_to_user_noinline(t, f, n) | ||
741 | #define __clear_user(d, n) arc_clear_user_noinline(d, n) | ||
742 | #define __strncpy_from_user(d, s, n) arc_strncpy_from_user_noinline(d, s, n) | ||
743 | #define __strnlen_user(s, n) arc_strnlen_user_noinline(s, n) | ||
744 | |||
745 | #endif | ||
746 | |||
747 | #include <asm-generic/uaccess.h> | ||
748 | |||
749 | extern int fixup_exception(struct pt_regs *regs); | ||
750 | |||
751 | #endif | ||
diff --git a/arch/arc/include/asm/unaligned.h b/arch/arc/include/asm/unaligned.h new file mode 100644 index 000000000000..5dbe63f17b66 --- /dev/null +++ b/arch/arc/include/asm/unaligned.h | |||
@@ -0,0 +1,29 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_UNALIGNED_H | ||
10 | #define _ASM_ARC_UNALIGNED_H | ||
11 | |||
12 | /* ARC700 can't handle unaligned Data accesses. */ | ||
13 | |||
14 | #include <asm-generic/unaligned.h> | ||
15 | #include <asm/ptrace.h> | ||
16 | |||
17 | #ifdef CONFIG_ARC_MISALIGN_ACCESS | ||
18 | int misaligned_fixup(unsigned long address, struct pt_regs *regs, | ||
19 | unsigned long cause, struct callee_regs *cregs); | ||
20 | #else | ||
21 | static inline int | ||
22 | misaligned_fixup(unsigned long address, struct pt_regs *regs, | ||
23 | unsigned long cause, struct callee_regs *cregs) | ||
24 | { | ||
25 | return 0; | ||
26 | } | ||
27 | #endif | ||
28 | |||
29 | #endif /* _ASM_ARC_UNALIGNED_H */ | ||
diff --git a/arch/arc/include/asm/unwind.h b/arch/arc/include/asm/unwind.h new file mode 100644 index 000000000000..7ca628b6ee2a --- /dev/null +++ b/arch/arc/include/asm/unwind.h | |||
@@ -0,0 +1,163 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_UNWIND_H | ||
10 | #define _ASM_ARC_UNWIND_H | ||
11 | |||
12 | #ifdef CONFIG_ARC_DW2_UNWIND | ||
13 | |||
14 | #include <linux/sched.h> | ||
15 | |||
16 | struct arc700_regs { | ||
17 | unsigned long r0; | ||
18 | unsigned long r1; | ||
19 | unsigned long r2; | ||
20 | unsigned long r3; | ||
21 | unsigned long r4; | ||
22 | unsigned long r5; | ||
23 | unsigned long r6; | ||
24 | unsigned long r7; | ||
25 | unsigned long r8; | ||
26 | unsigned long r9; | ||
27 | unsigned long r10; | ||
28 | unsigned long r11; | ||
29 | unsigned long r12; | ||
30 | unsigned long r13; | ||
31 | unsigned long r14; | ||
32 | unsigned long r15; | ||
33 | unsigned long r16; | ||
34 | unsigned long r17; | ||
35 | unsigned long r18; | ||
36 | unsigned long r19; | ||
37 | unsigned long r20; | ||
38 | unsigned long r21; | ||
39 | unsigned long r22; | ||
40 | unsigned long r23; | ||
41 | unsigned long r24; | ||
42 | unsigned long r25; | ||
43 | unsigned long r26; | ||
44 | unsigned long r27; /* fp */ | ||
45 | unsigned long r28; /* sp */ | ||
46 | unsigned long r29; | ||
47 | unsigned long r30; | ||
48 | unsigned long r31; /* blink */ | ||
49 | unsigned long r63; /* pc */ | ||
50 | }; | ||
51 | |||
52 | struct unwind_frame_info { | ||
53 | struct arc700_regs regs; | ||
54 | struct task_struct *task; | ||
55 | unsigned call_frame:1; | ||
56 | }; | ||
57 | |||
58 | #define UNW_PC(frame) ((frame)->regs.r63) | ||
59 | #define UNW_SP(frame) ((frame)->regs.r28) | ||
60 | #define UNW_BLINK(frame) ((frame)->regs.r31) | ||
61 | |||
62 | /* Rajesh FIXME */ | ||
63 | #ifdef CONFIG_FRAME_POINTER | ||
64 | #define UNW_FP(frame) ((frame)->regs.r27) | ||
65 | #define FRAME_RETADDR_OFFSET 4 | ||
66 | #define FRAME_LINK_OFFSET 0 | ||
67 | #define STACK_BOTTOM_UNW(tsk) STACK_LIMIT((tsk)->thread.ksp) | ||
68 | #define STACK_TOP_UNW(tsk) ((tsk)->thread.ksp) | ||
69 | #else | ||
70 | #define UNW_FP(frame) ((void)(frame), 0) | ||
71 | #endif | ||
72 | |||
73 | #define STACK_LIMIT(ptr) (((ptr) - 1) & ~(THREAD_SIZE - 1)) | ||
74 | |||
75 | #define UNW_REGISTER_INFO \ | ||
76 | PTREGS_INFO(r0), \ | ||
77 | PTREGS_INFO(r1), \ | ||
78 | PTREGS_INFO(r2), \ | ||
79 | PTREGS_INFO(r3), \ | ||
80 | PTREGS_INFO(r4), \ | ||
81 | PTREGS_INFO(r5), \ | ||
82 | PTREGS_INFO(r6), \ | ||
83 | PTREGS_INFO(r7), \ | ||
84 | PTREGS_INFO(r8), \ | ||
85 | PTREGS_INFO(r9), \ | ||
86 | PTREGS_INFO(r10), \ | ||
87 | PTREGS_INFO(r11), \ | ||
88 | PTREGS_INFO(r12), \ | ||
89 | PTREGS_INFO(r13), \ | ||
90 | PTREGS_INFO(r14), \ | ||
91 | PTREGS_INFO(r15), \ | ||
92 | PTREGS_INFO(r16), \ | ||
93 | PTREGS_INFO(r17), \ | ||
94 | PTREGS_INFO(r18), \ | ||
95 | PTREGS_INFO(r19), \ | ||
96 | PTREGS_INFO(r20), \ | ||
97 | PTREGS_INFO(r21), \ | ||
98 | PTREGS_INFO(r22), \ | ||
99 | PTREGS_INFO(r23), \ | ||
100 | PTREGS_INFO(r24), \ | ||
101 | PTREGS_INFO(r25), \ | ||
102 | PTREGS_INFO(r26), \ | ||
103 | PTREGS_INFO(r27), \ | ||
104 | PTREGS_INFO(r28), \ | ||
105 | PTREGS_INFO(r29), \ | ||
106 | PTREGS_INFO(r30), \ | ||
107 | PTREGS_INFO(r31), \ | ||
108 | PTREGS_INFO(r63) | ||
109 | |||
110 | #define UNW_DEFAULT_RA(raItem, dataAlign) \ | ||
111 | ((raItem).where == Memory && !((raItem).value * (dataAlign) + 4)) | ||
112 | |||
113 | extern int arc_unwind(struct unwind_frame_info *frame); | ||
114 | extern void arc_unwind_init(void); | ||
115 | extern void arc_unwind_setup(void); | ||
116 | extern void *unwind_add_table(struct module *module, const void *table_start, | ||
117 | unsigned long table_size); | ||
118 | extern void unwind_remove_table(void *handle, int init_only); | ||
119 | |||
120 | static inline int | ||
121 | arch_unwind_init_running(struct unwind_frame_info *info, | ||
122 | int (*callback) (struct unwind_frame_info *info, | ||
123 | void *arg), | ||
124 | void *arg) | ||
125 | { | ||
126 | return 0; | ||
127 | } | ||
128 | |||
129 | static inline int arch_unw_user_mode(const struct unwind_frame_info *info) | ||
130 | { | ||
131 | return 0; | ||
132 | } | ||
133 | |||
134 | static inline void arch_unw_init_blocked(struct unwind_frame_info *info) | ||
135 | { | ||
136 | return; | ||
137 | } | ||
138 | |||
139 | static inline void arch_unw_init_frame_info(struct unwind_frame_info *info, | ||
140 | struct pt_regs *regs) | ||
141 | { | ||
142 | return; | ||
143 | } | ||
144 | |||
145 | #else | ||
146 | |||
147 | #define UNW_PC(frame) ((void)(frame), 0) | ||
148 | #define UNW_SP(frame) ((void)(frame), 0) | ||
149 | #define UNW_FP(frame) ((void)(frame), 0) | ||
150 | |||
151 | static inline void arc_unwind_init(void) | ||
152 | { | ||
153 | } | ||
154 | |||
155 | static inline void arc_unwind_setup(void) | ||
156 | { | ||
157 | } | ||
158 | #define unwind_add_table(a, b, c) | ||
159 | #define unwind_remove_table(a, b) | ||
160 | |||
161 | #endif /* CONFIG_ARC_DW2_UNWIND */ | ||
162 | |||
163 | #endif /* _ASM_ARC_UNWIND_H */ | ||
diff --git a/arch/arc/include/uapi/asm/Kbuild b/arch/arc/include/uapi/asm/Kbuild new file mode 100644 index 000000000000..18fefaea73fd --- /dev/null +++ b/arch/arc/include/uapi/asm/Kbuild | |||
@@ -0,0 +1,12 @@ | |||
1 | # UAPI Header export list | ||
2 | include include/uapi/asm-generic/Kbuild.asm | ||
3 | header-y += elf.h | ||
4 | header-y += page.h | ||
5 | header-y += setup.h | ||
6 | header-y += byteorder.h | ||
7 | header-y += cachectl.h | ||
8 | header-y += ptrace.h | ||
9 | header-y += sigcontext.h | ||
10 | header-y += signal.h | ||
11 | header-y += swab.h | ||
12 | header-y += unistd.h | ||
diff --git a/arch/arc/include/uapi/asm/byteorder.h b/arch/arc/include/uapi/asm/byteorder.h new file mode 100644 index 000000000000..9da71d415c38 --- /dev/null +++ b/arch/arc/include/uapi/asm/byteorder.h | |||
@@ -0,0 +1,18 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_ARC_BYTEORDER_H | ||
10 | #define __ASM_ARC_BYTEORDER_H | ||
11 | |||
12 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
13 | #include <linux/byteorder/big_endian.h> | ||
14 | #else | ||
15 | #include <linux/byteorder/little_endian.h> | ||
16 | #endif | ||
17 | |||
18 | #endif /* ASM_ARC_BYTEORDER_H */ | ||
diff --git a/arch/arc/include/uapi/asm/cachectl.h b/arch/arc/include/uapi/asm/cachectl.h new file mode 100644 index 000000000000..51c73f0255b3 --- /dev/null +++ b/arch/arc/include/uapi/asm/cachectl.h | |||
@@ -0,0 +1,28 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ARC_ASM_CACHECTL_H | ||
10 | #define __ARC_ASM_CACHECTL_H | ||
11 | |||
12 | /* | ||
13 | * ARC ABI flags defined for Android's finegrained cacheflush requirements | ||
14 | */ | ||
15 | #define CF_I_INV 0x0002 | ||
16 | #define CF_D_FLUSH 0x0010 | ||
17 | #define CF_D_FLUSH_INV 0x0020 | ||
18 | |||
19 | #define CF_DEFAULT (CF_I_INV | CF_D_FLUSH) | ||
20 | |||
21 | /* | ||
22 | * Standard flags expected by cacheflush system call users | ||
23 | */ | ||
24 | #define ICACHE CF_I_INV | ||
25 | #define DCACHE CF_D_FLUSH | ||
26 | #define BCACHE (CF_I_INV | CF_D_FLUSH) | ||
27 | |||
28 | #endif | ||
diff --git a/arch/arc/include/uapi/asm/elf.h b/arch/arc/include/uapi/asm/elf.h new file mode 100644 index 000000000000..0f99ac8fcbb2 --- /dev/null +++ b/arch/arc/include/uapi/asm/elf.h | |||
@@ -0,0 +1,26 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _UAPI__ASM_ARC_ELF_H | ||
10 | #define _UAPI__ASM_ARC_ELF_H | ||
11 | |||
12 | #include <asm/ptrace.h> /* for user_regs_struct */ | ||
13 | |||
14 | /* Machine specific ELF Hdr flags */ | ||
15 | #define EF_ARC_OSABI_MSK 0x00000f00 | ||
16 | #define EF_ARC_OSABI_ORIG 0x00000000 /* MUST be zero for back-compat */ | ||
17 | #define EF_ARC_OSABI_CURRENT 0x00000300 /* v3 (no legacy syscalls) */ | ||
18 | |||
19 | typedef unsigned long elf_greg_t; | ||
20 | typedef unsigned long elf_fpregset_t; | ||
21 | |||
22 | #define ELF_NGREG (sizeof(struct user_regs_struct) / sizeof(elf_greg_t)) | ||
23 | |||
24 | typedef elf_greg_t elf_gregset_t[ELF_NGREG]; | ||
25 | |||
26 | #endif | ||
diff --git a/arch/arc/include/uapi/asm/page.h b/arch/arc/include/uapi/asm/page.h new file mode 100644 index 000000000000..e5d41e08240c --- /dev/null +++ b/arch/arc/include/uapi/asm/page.h | |||
@@ -0,0 +1,39 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _UAPI__ASM_ARC_PAGE_H | ||
10 | #define _UAPI__ASM_ARC_PAGE_H | ||
11 | |||
12 | /* PAGE_SHIFT determines the page size */ | ||
13 | #if defined(CONFIG_ARC_PAGE_SIZE_16K) | ||
14 | #define PAGE_SHIFT 14 | ||
15 | #elif defined(CONFIG_ARC_PAGE_SIZE_4K) | ||
16 | #define PAGE_SHIFT 12 | ||
17 | #else | ||
18 | /* | ||
19 | * Default 8k | ||
20 | * done this way (instead of under CONFIG_ARC_PAGE_SIZE_8K) because adhoc | ||
21 | * user code (busybox appletlib.h) expects PAGE_SHIFT to be defined w/o | ||
22 | * using the correct uClibc header and in their build our autoconf.h is | ||
23 | * not available | ||
24 | */ | ||
25 | #define PAGE_SHIFT 13 | ||
26 | #endif | ||
27 | |||
28 | #ifdef __ASSEMBLY__ | ||
29 | #define PAGE_SIZE (1 << PAGE_SHIFT) | ||
30 | #define PAGE_OFFSET (0x80000000) | ||
31 | #else | ||
32 | #define PAGE_SIZE (1UL << PAGE_SHIFT) /* Default 8K */ | ||
33 | #define PAGE_OFFSET (0x80000000UL) /* Kernel starts at 2G onwards */ | ||
34 | #endif | ||
35 | |||
36 | #define PAGE_MASK (~(PAGE_SIZE-1)) | ||
37 | |||
38 | |||
39 | #endif /* _UAPI__ASM_ARC_PAGE_H */ | ||
diff --git a/arch/arc/include/uapi/asm/ptrace.h b/arch/arc/include/uapi/asm/ptrace.h new file mode 100644 index 000000000000..6afa4f702075 --- /dev/null +++ b/arch/arc/include/uapi/asm/ptrace.h | |||
@@ -0,0 +1,48 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Amit Bhor, Sameer Dhavale: Codito Technologies 2004 | ||
9 | */ | ||
10 | |||
11 | #ifndef _UAPI__ASM_ARC_PTRACE_H | ||
12 | #define _UAPI__ASM_ARC_PTRACE_H | ||
13 | |||
14 | |||
15 | #ifndef __ASSEMBLY__ | ||
16 | /* | ||
17 | * Userspace ABI: Register state needed by | ||
18 | * -ptrace (gdbserver) | ||
19 | * -sigcontext (SA_SIGNINFO signal frame) | ||
20 | * | ||
21 | * This is to decouple pt_regs from user-space ABI, to be able to change it | ||
22 | * w/o affecting the ABI. | ||
23 | * Although the layout (initial padding) is similar to pt_regs to have some | ||
24 | * optimizations when copying pt_regs to/from user_regs_struct. | ||
25 | * | ||
26 | * Also, sigcontext only care about the scratch regs as that is what we really | ||
27 | * save/restore for signal handling. | ||
28 | */ | ||
29 | struct user_regs_struct { | ||
30 | |||
31 | struct scratch { | ||
32 | long pad; | ||
33 | long bta, lp_start, lp_end, lp_count; | ||
34 | long status32, ret, blink, fp, gp; | ||
35 | long r12, r11, r10, r9, r8, r7, r6, r5, r4, r3, r2, r1, r0; | ||
36 | long sp; | ||
37 | } scratch; | ||
38 | struct callee { | ||
39 | long pad; | ||
40 | long r25, r24, r23, r22, r21, r20; | ||
41 | long r19, r18, r17, r16, r15, r14, r13; | ||
42 | } callee; | ||
43 | long efa; /* break pt addr, for break points in delay slots */ | ||
44 | long stop_pc; /* give dbg stop_pc directly after checking orig_r8 */ | ||
45 | }; | ||
46 | #endif /* !__ASSEMBLY__ */ | ||
47 | |||
48 | #endif /* _UAPI__ASM_ARC_PTRACE_H */ | ||
diff --git a/arch/arc/include/uapi/asm/setup.h b/arch/arc/include/uapi/asm/setup.h new file mode 100644 index 000000000000..a6d4e44938be --- /dev/null +++ b/arch/arc/include/uapi/asm/setup.h | |||
@@ -0,0 +1,6 @@ | |||
1 | /* | ||
2 | * setup.h is part of userspace header ABI so UAPI scripts have to generate it | ||
3 | * even if there's nothing to export - causing empty <uapi/asm/setup.h> | ||
4 | * However to prevent "patch" from discarding it we add this placeholder | ||
5 | * comment | ||
6 | */ | ||
diff --git a/arch/arc/include/uapi/asm/sigcontext.h b/arch/arc/include/uapi/asm/sigcontext.h new file mode 100644 index 000000000000..9678a11fc158 --- /dev/null +++ b/arch/arc/include/uapi/asm/sigcontext.h | |||
@@ -0,0 +1,22 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #ifndef _ASM_ARC_SIGCONTEXT_H | ||
10 | #define _ASM_ARC_SIGCONTEXT_H | ||
11 | |||
12 | #include <asm/ptrace.h> | ||
13 | |||
14 | /* | ||
15 | * Signal context structure - contains all info to do with the state | ||
16 | * before the signal handler was invoked. | ||
17 | */ | ||
18 | struct sigcontext { | ||
19 | struct user_regs_struct regs; | ||
20 | }; | ||
21 | |||
22 | #endif /* _ASM_ARC_SIGCONTEXT_H */ | ||
diff --git a/arch/arc/include/uapi/asm/signal.h b/arch/arc/include/uapi/asm/signal.h new file mode 100644 index 000000000000..fad62f7f42d6 --- /dev/null +++ b/arch/arc/include/uapi/asm/signal.h | |||
@@ -0,0 +1,27 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Amit Bhor, Sameer Dhavale: Codito Technologies 2004 | ||
9 | */ | ||
10 | |||
11 | #ifndef _ASM_ARC_SIGNAL_H | ||
12 | #define _ASM_ARC_SIGNAL_H | ||
13 | |||
14 | /* | ||
15 | * This is much needed for ARC sigreturn optimization. | ||
16 | * This allows uClibc to piggback the addr of a sigreturn stub in sigaction, | ||
17 | * which allows sigreturn based re-entry into kernel after handling signal. | ||
18 | * W/o this kernel needs to "synthesize" the sigreturn trampoline on user | ||
19 | * mode stack which in turn forces the following: | ||
20 | * -TLB Flush (after making the stack page executable) | ||
21 | * -Cache line Flush (to make I/D Cache lines coherent) | ||
22 | */ | ||
23 | #define SA_RESTORER 0x04000000 | ||
24 | |||
25 | #include <asm-generic/signal.h> | ||
26 | |||
27 | #endif /* _ASM_ARC_SIGNAL_H */ | ||
diff --git a/arch/arc/include/uapi/asm/swab.h b/arch/arc/include/uapi/asm/swab.h new file mode 100644 index 000000000000..095599a73195 --- /dev/null +++ b/arch/arc/include/uapi/asm/swab.h | |||
@@ -0,0 +1,98 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * vineetg: May 2011 | ||
9 | * -Support single cycle endian-swap insn in ARC700 4.10 | ||
10 | * | ||
11 | * vineetg: June 2009 | ||
12 | * -Better htonl implementation (5 instead of 9 ALU instructions) | ||
13 | * -Hardware assisted single cycle bswap (Use Case of ARC custom instrn) | ||
14 | */ | ||
15 | |||
16 | #ifndef __ASM_ARC_SWAB_H | ||
17 | #define __ASM_ARC_SWAB_H | ||
18 | |||
19 | #include <linux/types.h> | ||
20 | |||
21 | /* Native single cycle endian swap insn */ | ||
22 | #ifdef CONFIG_ARC_HAS_SWAPE | ||
23 | |||
24 | #define __arch_swab32(x) \ | ||
25 | ({ \ | ||
26 | unsigned int tmp = x; \ | ||
27 | __asm__( \ | ||
28 | " swape %0, %1 \n" \ | ||
29 | : "=r" (tmp) \ | ||
30 | : "r" (tmp)); \ | ||
31 | tmp; \ | ||
32 | }) | ||
33 | |||
34 | #else | ||
35 | |||
36 | /* Several ways of Endian-Swap Emulation for ARC | ||
37 | * 0: kernel generic | ||
38 | * 1: ARC optimised "C" | ||
39 | * 2: ARC Custom instruction | ||
40 | */ | ||
41 | #define ARC_BSWAP_TYPE 1 | ||
42 | |||
43 | #if (ARC_BSWAP_TYPE == 1) /******* Software only ********/ | ||
44 | |||
45 | /* The kernel default implementation of htonl is | ||
46 | * return x<<24 | x>>24 | | ||
47 | * (x & (__u32)0x0000ff00UL)<<8 | (x & (__u32)0x00ff0000UL)>>8; | ||
48 | * | ||
49 | * This generates 9 instructions on ARC (excluding the ld/st) | ||
50 | * | ||
51 | * 8051fd8c: ld r3,[r7,20] ; Mem op : Get the value to be swapped | ||
52 | * 8051fd98: asl r5,r3,24 ; get 3rd Byte | ||
53 | * 8051fd9c: lsr r2,r3,24 ; get 0th Byte | ||
54 | * 8051fda0: and r4,r3,0xff00 | ||
55 | * 8051fda8: asl r4,r4,8 ; get 1st Byte | ||
56 | * 8051fdac: and r3,r3,0x00ff0000 | ||
57 | * 8051fdb4: or r2,r2,r5 ; combine 0th and 3rd Bytes | ||
58 | * 8051fdb8: lsr r3,r3,8 ; 2nd Byte at correct place in Dst Reg | ||
59 | * 8051fdbc: or r2,r2,r4 ; combine 0,3 Bytes with 1st Byte | ||
60 | * 8051fdc0: or r2,r2,r3 ; combine 0,3,1 Bytes with 2nd Byte | ||
61 | * 8051fdc4: st r2,[r1,20] ; Mem op : save result back to mem | ||
62 | * | ||
63 | * Joern suggested a better "C" algorithm which is great since | ||
64 | * (1) It is portable to any architecure | ||
65 | * (2) At the same time it takes advantage of ARC ISA (rotate intrns) | ||
66 | */ | ||
67 | |||
68 | #define __arch_swab32(x) \ | ||
69 | ({ unsigned long __in = (x), __tmp; \ | ||
70 | __tmp = __in << 8 | __in >> 24; /* ror tmp,in,24 */ \ | ||
71 | __in = __in << 24 | __in >> 8; /* ror in,in,8 */ \ | ||
72 | __tmp ^= __in; \ | ||
73 | __tmp &= 0xff00ff; \ | ||
74 | __tmp ^ __in; \ | ||
75 | }) | ||
76 | |||
77 | #elif (ARC_BSWAP_TYPE == 2) /* Custom single cycle bwap instruction */ | ||
78 | |||
79 | #define __arch_swab32(x) \ | ||
80 | ({ \ | ||
81 | unsigned int tmp = x; \ | ||
82 | __asm__( \ | ||
83 | " .extInstruction bswap, 7, 0x00, SUFFIX_NONE, SYNTAX_2OP \n"\ | ||
84 | " bswap %0, %1 \n"\ | ||
85 | : "=r" (tmp) \ | ||
86 | : "r" (tmp)); \ | ||
87 | tmp; \ | ||
88 | }) | ||
89 | |||
90 | #endif /* ARC_BSWAP_TYPE=zzz */ | ||
91 | |||
92 | #endif /* CONFIG_ARC_HAS_SWAPE */ | ||
93 | |||
94 | #if !defined(__STRICT_ANSI__) || defined(__KERNEL__) | ||
95 | #define __SWAB_64_THRU_32__ | ||
96 | #endif | ||
97 | |||
98 | #endif | ||
diff --git a/arch/arc/include/uapi/asm/unistd.h b/arch/arc/include/uapi/asm/unistd.h new file mode 100644 index 000000000000..6f30484f34b7 --- /dev/null +++ b/arch/arc/include/uapi/asm/unistd.h | |||
@@ -0,0 +1,34 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | /******** no-legacy-syscalls-ABI *******/ | ||
10 | |||
11 | #define __ARCH_WANT_SYS_EXECVE | ||
12 | #define __ARCH_WANT_SYS_CLONE | ||
13 | #define __ARCH_WANT_SYS_VFORK | ||
14 | #define __ARCH_WANT_SYS_FORK | ||
15 | |||
16 | #define sys_mmap2 sys_mmap_pgoff | ||
17 | |||
18 | #include <asm-generic/unistd.h> | ||
19 | |||
20 | #define NR_syscalls __NR_syscalls | ||
21 | |||
22 | /* ARC specific syscall */ | ||
23 | #define __NR_cacheflush (__NR_arch_specific_syscall + 0) | ||
24 | #define __NR_arc_settls (__NR_arch_specific_syscall + 1) | ||
25 | #define __NR_arc_gettls (__NR_arch_specific_syscall + 2) | ||
26 | |||
27 | __SYSCALL(__NR_cacheflush, sys_cacheflush) | ||
28 | __SYSCALL(__NR_arc_settls, sys_arc_settls) | ||
29 | __SYSCALL(__NR_arc_gettls, sys_arc_gettls) | ||
30 | |||
31 | |||
32 | /* Generic syscall (fs/filesystems.c - lost in asm-generic/unistd.h */ | ||
33 | #define __NR_sysfs (__NR_arch_specific_syscall + 3) | ||
34 | __SYSCALL(__NR_sysfs, sys_sysfs) | ||
diff --git a/arch/arc/kernel/Makefile b/arch/arc/kernel/Makefile new file mode 100644 index 000000000000..c242ef07ba70 --- /dev/null +++ b/arch/arc/kernel/Makefile | |||
@@ -0,0 +1,33 @@ | |||
1 | # | ||
2 | # Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | # | ||
4 | # This program is free software; you can redistribute it and/or modify | ||
5 | # it under the terms of the GNU General Public License version 2 as | ||
6 | # published by the Free Software Foundation. | ||
7 | |||
8 | # Pass UTS_MACHINE for user_regset definition | ||
9 | CFLAGS_ptrace.o += -DUTS_MACHINE='"$(UTS_MACHINE)"' | ||
10 | |||
11 | obj-y := arcksyms.o setup.o irq.o time.o reset.o ptrace.o entry.o process.o | ||
12 | obj-y += signal.o traps.o sys.o troubleshoot.o stacktrace.o disasm.o clk.o | ||
13 | obj-y += devtree.o | ||
14 | |||
15 | obj-$(CONFIG_MODULES) += arcksyms.o module.o | ||
16 | obj-$(CONFIG_SMP) += smp.o | ||
17 | obj-$(CONFIG_ARC_DW2_UNWIND) += unwind.o | ||
18 | obj-$(CONFIG_KPROBES) += kprobes.o | ||
19 | obj-$(CONFIG_ARC_MISALIGN_ACCESS) += unaligned.o | ||
20 | obj-$(CONFIG_KGDB) += kgdb.o | ||
21 | obj-$(CONFIG_ARC_METAWARE_HLINK) += arc_hostlink.o | ||
22 | |||
23 | obj-$(CONFIG_ARC_FPU_SAVE_RESTORE) += fpu.o | ||
24 | CFLAGS_fpu.o += -mdpfp | ||
25 | |||
26 | ifdef CONFIG_ARC_DW2_UNWIND | ||
27 | CFLAGS_ctx_sw.o += -fno-omit-frame-pointer | ||
28 | obj-y += ctx_sw.o | ||
29 | else | ||
30 | obj-y += ctx_sw_asm.o | ||
31 | endif | ||
32 | |||
33 | extra-y := vmlinux.lds head.o | ||
diff --git a/arch/arc/kernel/arc_hostlink.c b/arch/arc/kernel/arc_hostlink.c new file mode 100644 index 000000000000..47b2a17cc52a --- /dev/null +++ b/arch/arc/kernel/arc_hostlink.c | |||
@@ -0,0 +1,58 @@ | |||
1 | /* | ||
2 | * arc_hostlink.c: Pseudo-driver for Metaware provided "hostlink" facility | ||
3 | * | ||
4 | * Allows Linux userland access to host in absence of any peripherals. | ||
5 | * | ||
6 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
7 | * | ||
8 | * This program is free software; you can redistribute it and/or modify | ||
9 | * it under the terms of the GNU General Public License version 2 as | ||
10 | * published by the Free Software Foundation. | ||
11 | */ | ||
12 | |||
13 | #include <linux/fs.h> /* file_operations */ | ||
14 | #include <linux/miscdevice.h> | ||
15 | #include <linux/mm.h> /* VM_IO */ | ||
16 | #include <linux/module.h> | ||
17 | #include <linux/uaccess.h> | ||
18 | |||
19 | static unsigned char __HOSTLINK__[4 * PAGE_SIZE] __aligned(PAGE_SIZE); | ||
20 | |||
21 | static int arc_hl_mmap(struct file *fp, struct vm_area_struct *vma) | ||
22 | { | ||
23 | vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot); | ||
24 | |||
25 | if (io_remap_pfn_range(vma, vma->vm_start, vma->vm_pgoff, | ||
26 | vma->vm_end - vma->vm_start, | ||
27 | vma->vm_page_prot)) { | ||
28 | pr_warn("Hostlink buffer mmap ERROR\n"); | ||
29 | return -EAGAIN; | ||
30 | } | ||
31 | return 0; | ||
32 | } | ||
33 | |||
34 | static long arc_hl_ioctl(struct file *file, unsigned int cmd, | ||
35 | unsigned long arg) | ||
36 | { | ||
37 | /* we only support, returning the physical addr to mmap in user space */ | ||
38 | put_user((unsigned int)__HOSTLINK__, (int __user *)arg); | ||
39 | return 0; | ||
40 | } | ||
41 | |||
42 | static const struct file_operations arc_hl_fops = { | ||
43 | .unlocked_ioctl = arc_hl_ioctl, | ||
44 | .mmap = arc_hl_mmap, | ||
45 | }; | ||
46 | |||
47 | static struct miscdevice arc_hl_dev = { | ||
48 | .minor = MISC_DYNAMIC_MINOR, | ||
49 | .name = "hostlink", | ||
50 | .fops = &arc_hl_fops | ||
51 | }; | ||
52 | |||
53 | static int __init arc_hl_init(void) | ||
54 | { | ||
55 | pr_info("ARC Hostlink driver mmap at 0x%p\n", __HOSTLINK__); | ||
56 | return misc_register(&arc_hl_dev); | ||
57 | } | ||
58 | module_init(arc_hl_init); | ||
diff --git a/arch/arc/kernel/arcksyms.c b/arch/arc/kernel/arcksyms.c new file mode 100644 index 000000000000..4d9e77724bed --- /dev/null +++ b/arch/arc/kernel/arcksyms.c | |||
@@ -0,0 +1,56 @@ | |||
1 | /* | ||
2 | * arcksyms.c - Exporting symbols not exportable from their own sources | ||
3 | * | ||
4 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | * | ||
10 | */ | ||
11 | |||
12 | #include <linux/module.h> | ||
13 | |||
14 | /* libgcc functions, not part of kernel sources */ | ||
15 | extern void __ashldi3(void); | ||
16 | extern void __ashrdi3(void); | ||
17 | extern void __divsi3(void); | ||
18 | extern void __divsf3(void); | ||
19 | extern void __lshrdi3(void); | ||
20 | extern void __modsi3(void); | ||
21 | extern void __muldi3(void); | ||
22 | extern void __ucmpdi2(void); | ||
23 | extern void __udivsi3(void); | ||
24 | extern void __umodsi3(void); | ||
25 | extern void __cmpdi2(void); | ||
26 | extern void __fixunsdfsi(void); | ||
27 | extern void __muldf3(void); | ||
28 | extern void __divdf3(void); | ||
29 | extern void __floatunsidf(void); | ||
30 | extern void __floatunsisf(void); | ||
31 | |||
32 | EXPORT_SYMBOL(__ashldi3); | ||
33 | EXPORT_SYMBOL(__ashrdi3); | ||
34 | EXPORT_SYMBOL(__divsi3); | ||
35 | EXPORT_SYMBOL(__divsf3); | ||
36 | EXPORT_SYMBOL(__lshrdi3); | ||
37 | EXPORT_SYMBOL(__modsi3); | ||
38 | EXPORT_SYMBOL(__muldi3); | ||
39 | EXPORT_SYMBOL(__ucmpdi2); | ||
40 | EXPORT_SYMBOL(__udivsi3); | ||
41 | EXPORT_SYMBOL(__umodsi3); | ||
42 | EXPORT_SYMBOL(__cmpdi2); | ||
43 | EXPORT_SYMBOL(__fixunsdfsi); | ||
44 | EXPORT_SYMBOL(__muldf3); | ||
45 | EXPORT_SYMBOL(__divdf3); | ||
46 | EXPORT_SYMBOL(__floatunsidf); | ||
47 | EXPORT_SYMBOL(__floatunsisf); | ||
48 | |||
49 | /* ARC optimised assembler routines */ | ||
50 | EXPORT_SYMBOL(memset); | ||
51 | EXPORT_SYMBOL(memcpy); | ||
52 | EXPORT_SYMBOL(memcmp); | ||
53 | EXPORT_SYMBOL(strchr); | ||
54 | EXPORT_SYMBOL(strcpy); | ||
55 | EXPORT_SYMBOL(strcmp); | ||
56 | EXPORT_SYMBOL(strlen); | ||
diff --git a/arch/arc/kernel/asm-offsets.c b/arch/arc/kernel/asm-offsets.c new file mode 100644 index 000000000000..0dc148ebce74 --- /dev/null +++ b/arch/arc/kernel/asm-offsets.c | |||
@@ -0,0 +1,64 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <linux/sched.h> | ||
10 | #include <linux/mm.h> | ||
11 | #include <linux/interrupt.h> | ||
12 | #include <linux/thread_info.h> | ||
13 | #include <linux/kbuild.h> | ||
14 | #include <asm/hardirq.h> | ||
15 | #include <asm/page.h> | ||
16 | #include <asm/ptrace.h> | ||
17 | |||
18 | int main(void) | ||
19 | { | ||
20 | DEFINE(TASK_THREAD, offsetof(struct task_struct, thread)); | ||
21 | DEFINE(TASK_THREAD_INFO, offsetof(struct task_struct, stack)); | ||
22 | |||
23 | BLANK(); | ||
24 | |||
25 | DEFINE(THREAD_KSP, offsetof(struct thread_struct, ksp)); | ||
26 | DEFINE(THREAD_CALLEE_REG, offsetof(struct thread_struct, callee_reg)); | ||
27 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
28 | DEFINE(THREAD_USER_R25, offsetof(struct thread_struct, user_r25)); | ||
29 | #endif | ||
30 | DEFINE(THREAD_FAULT_ADDR, | ||
31 | offsetof(struct thread_struct, fault_address)); | ||
32 | |||
33 | BLANK(); | ||
34 | |||
35 | DEFINE(THREAD_INFO_FLAGS, offsetof(struct thread_info, flags)); | ||
36 | DEFINE(THREAD_INFO_PREEMPT_COUNT, | ||
37 | offsetof(struct thread_info, preempt_count)); | ||
38 | |||
39 | BLANK(); | ||
40 | |||
41 | DEFINE(TASK_ACT_MM, offsetof(struct task_struct, active_mm)); | ||
42 | DEFINE(TASK_TGID, offsetof(struct task_struct, tgid)); | ||
43 | |||
44 | DEFINE(MM_CTXT, offsetof(struct mm_struct, context)); | ||
45 | DEFINE(MM_PGD, offsetof(struct mm_struct, pgd)); | ||
46 | |||
47 | DEFINE(MM_CTXT_ASID, offsetof(mm_context_t, asid)); | ||
48 | |||
49 | BLANK(); | ||
50 | |||
51 | DEFINE(PT_status32, offsetof(struct pt_regs, status32)); | ||
52 | DEFINE(PT_orig_r8, offsetof(struct pt_regs, orig_r8_word)); | ||
53 | DEFINE(PT_sp, offsetof(struct pt_regs, sp)); | ||
54 | DEFINE(PT_r0, offsetof(struct pt_regs, r0)); | ||
55 | DEFINE(PT_r1, offsetof(struct pt_regs, r1)); | ||
56 | DEFINE(PT_r2, offsetof(struct pt_regs, r2)); | ||
57 | DEFINE(PT_r3, offsetof(struct pt_regs, r3)); | ||
58 | DEFINE(PT_r4, offsetof(struct pt_regs, r4)); | ||
59 | DEFINE(PT_r5, offsetof(struct pt_regs, r5)); | ||
60 | DEFINE(PT_r6, offsetof(struct pt_regs, r6)); | ||
61 | DEFINE(PT_r7, offsetof(struct pt_regs, r7)); | ||
62 | |||
63 | return 0; | ||
64 | } | ||
diff --git a/arch/arc/kernel/clk.c b/arch/arc/kernel/clk.c new file mode 100644 index 000000000000..66ce0dc917fb --- /dev/null +++ b/arch/arc/kernel/clk.c | |||
@@ -0,0 +1,21 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <asm/clk.h> | ||
10 | |||
11 | unsigned long core_freq = 800000000; | ||
12 | |||
13 | /* | ||
14 | * As of now we default to device-tree provided clock | ||
15 | * In future we can determine this in early boot | ||
16 | */ | ||
17 | int arc_set_core_freq(unsigned long freq) | ||
18 | { | ||
19 | core_freq = freq; | ||
20 | return 0; | ||
21 | } | ||
diff --git a/arch/arc/kernel/ctx_sw.c b/arch/arc/kernel/ctx_sw.c new file mode 100644 index 000000000000..60844dac6132 --- /dev/null +++ b/arch/arc/kernel/ctx_sw.c | |||
@@ -0,0 +1,109 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Vineetg: Aug 2009 | ||
9 | * -"C" version of lowest level context switch asm macro called by schedular | ||
10 | * gcc doesn't generate the dward CFI info for hand written asm, hence can't | ||
11 | * backtrace out of it (e.g. tasks sleeping in kernel). | ||
12 | * So we cheat a bit by writing almost similar code in inline-asm. | ||
13 | * -This is a hacky way of doing things, but there is no other simple way. | ||
14 | * I don't want/intend to extend unwinding code to understand raw asm | ||
15 | */ | ||
16 | |||
17 | #include <asm/asm-offsets.h> | ||
18 | #include <linux/sched.h> | ||
19 | |||
20 | struct task_struct *__sched | ||
21 | __switch_to(struct task_struct *prev_task, struct task_struct *next_task) | ||
22 | { | ||
23 | unsigned int tmp; | ||
24 | unsigned int prev = (unsigned int)prev_task; | ||
25 | unsigned int next = (unsigned int)next_task; | ||
26 | int num_words_to_skip = 1; | ||
27 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
28 | num_words_to_skip++; | ||
29 | #endif | ||
30 | |||
31 | __asm__ __volatile__( | ||
32 | /* FP/BLINK save generated by gcc (standard function prologue */ | ||
33 | "st.a r13, [sp, -4] \n\t" | ||
34 | "st.a r14, [sp, -4] \n\t" | ||
35 | "st.a r15, [sp, -4] \n\t" | ||
36 | "st.a r16, [sp, -4] \n\t" | ||
37 | "st.a r17, [sp, -4] \n\t" | ||
38 | "st.a r18, [sp, -4] \n\t" | ||
39 | "st.a r19, [sp, -4] \n\t" | ||
40 | "st.a r20, [sp, -4] \n\t" | ||
41 | "st.a r21, [sp, -4] \n\t" | ||
42 | "st.a r22, [sp, -4] \n\t" | ||
43 | "st.a r23, [sp, -4] \n\t" | ||
44 | "st.a r24, [sp, -4] \n\t" | ||
45 | #ifndef CONFIG_ARC_CURR_IN_REG | ||
46 | "st.a r25, [sp, -4] \n\t" | ||
47 | #endif | ||
48 | "sub sp, sp, %4 \n\t" /* create gutter at top */ | ||
49 | |||
50 | /* set ksp of outgoing task in tsk->thread.ksp */ | ||
51 | "st.as sp, [%3, %1] \n\t" | ||
52 | |||
53 | "sync \n\t" | ||
54 | |||
55 | /* | ||
56 | * setup _current_task with incoming tsk. | ||
57 | * optionally, set r25 to that as well | ||
58 | * For SMP extra work to get to &_current_task[cpu] | ||
59 | * (open coded SET_CURR_TASK_ON_CPU) | ||
60 | */ | ||
61 | #ifndef CONFIG_SMP | ||
62 | "st %2, [@_current_task] \n\t" | ||
63 | #else | ||
64 | "lr r24, [identity] \n\t" | ||
65 | "lsr r24, r24, 8 \n\t" | ||
66 | "bmsk r24, r24, 7 \n\t" | ||
67 | "add2 r24, @_current_task, r24 \n\t" | ||
68 | "st %2, [r24] \n\t" | ||
69 | #endif | ||
70 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
71 | "mov r25, %2 \n\t" | ||
72 | #endif | ||
73 | |||
74 | /* get ksp of incoming task from tsk->thread.ksp */ | ||
75 | "ld.as sp, [%2, %1] \n\t" | ||
76 | |||
77 | /* start loading it's CALLEE reg file */ | ||
78 | |||
79 | "add sp, sp, %4 \n\t" /* skip gutter at top */ | ||
80 | |||
81 | #ifndef CONFIG_ARC_CURR_IN_REG | ||
82 | "ld.ab r25, [sp, 4] \n\t" | ||
83 | #endif | ||
84 | "ld.ab r24, [sp, 4] \n\t" | ||
85 | "ld.ab r23, [sp, 4] \n\t" | ||
86 | "ld.ab r22, [sp, 4] \n\t" | ||
87 | "ld.ab r21, [sp, 4] \n\t" | ||
88 | "ld.ab r20, [sp, 4] \n\t" | ||
89 | "ld.ab r19, [sp, 4] \n\t" | ||
90 | "ld.ab r18, [sp, 4] \n\t" | ||
91 | "ld.ab r17, [sp, 4] \n\t" | ||
92 | "ld.ab r16, [sp, 4] \n\t" | ||
93 | "ld.ab r15, [sp, 4] \n\t" | ||
94 | "ld.ab r14, [sp, 4] \n\t" | ||
95 | "ld.ab r13, [sp, 4] \n\t" | ||
96 | |||
97 | /* last (ret value) = prev : although for ARC it mov r0, r0 */ | ||
98 | "mov %0, %3 \n\t" | ||
99 | |||
100 | /* FP/BLINK restore generated by gcc (standard func epilogue */ | ||
101 | |||
102 | : "=r"(tmp) | ||
103 | : "n"((TASK_THREAD + THREAD_KSP) / 4), "r"(next), "r"(prev), | ||
104 | "n"(num_words_to_skip * 4) | ||
105 | : "blink" | ||
106 | ); | ||
107 | |||
108 | return (struct task_struct *)tmp; | ||
109 | } | ||
diff --git a/arch/arc/kernel/ctx_sw_asm.S b/arch/arc/kernel/ctx_sw_asm.S new file mode 100644 index 000000000000..d8972345e4c2 --- /dev/null +++ b/arch/arc/kernel/ctx_sw_asm.S | |||
@@ -0,0 +1,58 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Vineetg: Aug 2009 | ||
9 | * -Moved core context switch macro out of entry.S into this file. | ||
10 | * -This is the more "natural" hand written assembler | ||
11 | */ | ||
12 | |||
13 | #include <asm/entry.h> /* For the SAVE_* macros */ | ||
14 | #include <asm/asm-offsets.h> | ||
15 | #include <asm/linkage.h> | ||
16 | |||
17 | ;################### Low Level Context Switch ########################## | ||
18 | |||
19 | .section .sched.text,"ax",@progbits | ||
20 | .align 4 | ||
21 | .global __switch_to | ||
22 | .type __switch_to, @function | ||
23 | __switch_to: | ||
24 | |||
25 | /* Save regs on kernel mode stack of task */ | ||
26 | st.a blink, [sp, -4] | ||
27 | st.a fp, [sp, -4] | ||
28 | SAVE_CALLEE_SAVED_KERNEL | ||
29 | |||
30 | /* Save the now KSP in task->thread.ksp */ | ||
31 | st.as sp, [r0, (TASK_THREAD + THREAD_KSP)/4] | ||
32 | |||
33 | /* | ||
34 | * Return last task in r0 (return reg) | ||
35 | * On ARC, Return reg = First Arg reg = r0. | ||
36 | * Since we already have last task in r0, | ||
37 | * don't need to do anything special to return it | ||
38 | */ | ||
39 | |||
40 | /* hardware memory barrier */ | ||
41 | sync | ||
42 | |||
43 | /* | ||
44 | * switch to new task, contained in r1 | ||
45 | * Temp reg r3 is required to get the ptr to store val | ||
46 | */ | ||
47 | SET_CURR_TASK_ON_CPU r1, r3 | ||
48 | |||
49 | /* reload SP with kernel mode stack pointer in task->thread.ksp */ | ||
50 | ld.as sp, [r1, (TASK_THREAD + THREAD_KSP)/4] | ||
51 | |||
52 | /* restore the registers */ | ||
53 | RESTORE_CALLEE_SAVED_KERNEL | ||
54 | ld.ab fp, [sp, 4] | ||
55 | ld.ab blink, [sp, 4] | ||
56 | j [blink] | ||
57 | |||
58 | ARC_EXIT __switch_to | ||
diff --git a/arch/arc/kernel/devtree.c b/arch/arc/kernel/devtree.c new file mode 100644 index 000000000000..bdee3a812052 --- /dev/null +++ b/arch/arc/kernel/devtree.c | |||
@@ -0,0 +1,123 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * Based on reduced version of METAG | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | */ | ||
10 | |||
11 | |||
12 | #include <linux/init.h> | ||
13 | #include <linux/reboot.h> | ||
14 | #include <linux/memblock.h> | ||
15 | #include <linux/of.h> | ||
16 | #include <linux/of_fdt.h> | ||
17 | #include <asm/prom.h> | ||
18 | #include <asm/clk.h> | ||
19 | #include <asm/mach_desc.h> | ||
20 | |||
21 | /* called from unflatten_device_tree() to bootstrap devicetree itself */ | ||
22 | void * __init early_init_dt_alloc_memory_arch(u64 size, u64 align) | ||
23 | { | ||
24 | return __va(memblock_alloc(size, align)); | ||
25 | } | ||
26 | |||
27 | /** | ||
28 | * setup_machine_fdt - Machine setup when an dtb was passed to the kernel | ||
29 | * @dt: virtual address pointer to dt blob | ||
30 | * | ||
31 | * If a dtb was passed to the kernel, then use it to choose the correct | ||
32 | * machine_desc and to setup the system. | ||
33 | */ | ||
34 | struct machine_desc * __init setup_machine_fdt(void *dt) | ||
35 | { | ||
36 | struct boot_param_header *devtree = dt; | ||
37 | struct machine_desc *mdesc = NULL, *mdesc_best = NULL; | ||
38 | unsigned int score, mdesc_score = ~1; | ||
39 | unsigned long dt_root; | ||
40 | const char *model, *compat; | ||
41 | void *clk; | ||
42 | char manufacturer[16]; | ||
43 | unsigned long len; | ||
44 | |||
45 | /* check device tree validity */ | ||
46 | if (be32_to_cpu(devtree->magic) != OF_DT_HEADER) | ||
47 | return NULL; | ||
48 | |||
49 | initial_boot_params = devtree; | ||
50 | dt_root = of_get_flat_dt_root(); | ||
51 | |||
52 | /* | ||
53 | * The kernel could be multi-platform enabled, thus could have many | ||
54 | * "baked-in" machine descriptors. Search thru all for the best | ||
55 | * "compatible" string match. | ||
56 | */ | ||
57 | for_each_machine_desc(mdesc) { | ||
58 | score = of_flat_dt_match(dt_root, mdesc->dt_compat); | ||
59 | if (score > 0 && score < mdesc_score) { | ||
60 | mdesc_best = mdesc; | ||
61 | mdesc_score = score; | ||
62 | } | ||
63 | } | ||
64 | if (!mdesc_best) { | ||
65 | const char *prop; | ||
66 | long size; | ||
67 | |||
68 | pr_err("\n unrecognized device tree list:\n[ "); | ||
69 | |||
70 | prop = of_get_flat_dt_prop(dt_root, "compatible", &size); | ||
71 | if (prop) { | ||
72 | while (size > 0) { | ||
73 | printk("'%s' ", prop); | ||
74 | size -= strlen(prop) + 1; | ||
75 | prop += strlen(prop) + 1; | ||
76 | } | ||
77 | } | ||
78 | printk("]\n\n"); | ||
79 | |||
80 | machine_halt(); | ||
81 | } | ||
82 | |||
83 | /* compat = "<manufacturer>,<model>" */ | ||
84 | compat = mdesc_best->dt_compat[0]; | ||
85 | |||
86 | model = strchr(compat, ','); | ||
87 | if (model) | ||
88 | model++; | ||
89 | |||
90 | strlcpy(manufacturer, compat, model ? model - compat : strlen(compat)); | ||
91 | |||
92 | pr_info("Board \"%s\" from %s (Manufacturer)\n", model, manufacturer); | ||
93 | |||
94 | /* Retrieve various information from the /chosen node */ | ||
95 | of_scan_flat_dt(early_init_dt_scan_chosen, boot_command_line); | ||
96 | |||
97 | /* Initialize {size,address}-cells info */ | ||
98 | of_scan_flat_dt(early_init_dt_scan_root, NULL); | ||
99 | |||
100 | /* Setup memory, calling early_init_dt_add_memory_arch */ | ||
101 | of_scan_flat_dt(early_init_dt_scan_memory, NULL); | ||
102 | |||
103 | clk = of_get_flat_dt_prop(dt_root, "clock-frequency", &len); | ||
104 | if (clk) | ||
105 | arc_set_core_freq(of_read_ulong(clk, len/4)); | ||
106 | |||
107 | return mdesc_best; | ||
108 | } | ||
109 | |||
110 | /* | ||
111 | * Copy the flattened DT out of .init since unflattening doesn't copy strings | ||
112 | * and the normal DT APIs refs them from orig flat DT | ||
113 | */ | ||
114 | void __init copy_devtree(void) | ||
115 | { | ||
116 | void *alloc = early_init_dt_alloc_memory_arch( | ||
117 | be32_to_cpu(initial_boot_params->totalsize), 64); | ||
118 | if (alloc) { | ||
119 | memcpy(alloc, initial_boot_params, | ||
120 | be32_to_cpu(initial_boot_params->totalsize)); | ||
121 | initial_boot_params = alloc; | ||
122 | } | ||
123 | } | ||
diff --git a/arch/arc/kernel/disasm.c b/arch/arc/kernel/disasm.c new file mode 100644 index 000000000000..2f390289a792 --- /dev/null +++ b/arch/arc/kernel/disasm.c | |||
@@ -0,0 +1,538 @@ | |||
1 | /* | ||
2 | * several functions that help interpret ARC instructions | ||
3 | * used for unaligned accesses, kprobes and kgdb | ||
4 | * | ||
5 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
6 | * | ||
7 | * This program is free software; you can redistribute it and/or modify | ||
8 | * it under the terms of the GNU General Public License version 2 as | ||
9 | * published by the Free Software Foundation. | ||
10 | */ | ||
11 | |||
12 | #include <linux/types.h> | ||
13 | #include <linux/kprobes.h> | ||
14 | #include <linux/slab.h> | ||
15 | #include <asm/disasm.h> | ||
16 | #include <asm/uaccess.h> | ||
17 | |||
18 | #if defined(CONFIG_KGDB) || defined(CONFIG_ARC_MISALIGN_ACCESS) || \ | ||
19 | defined(CONFIG_KPROBES) | ||
20 | |||
21 | /* disasm_instr: Analyses instruction at addr, stores | ||
22 | * findings in *state | ||
23 | */ | ||
24 | void __kprobes disasm_instr(unsigned long addr, struct disasm_state *state, | ||
25 | int userspace, struct pt_regs *regs, struct callee_regs *cregs) | ||
26 | { | ||
27 | int fieldA = 0; | ||
28 | int fieldC = 0, fieldCisReg = 0; | ||
29 | uint16_t word1 = 0, word0 = 0; | ||
30 | int subopcode, is_linked, op_format; | ||
31 | uint16_t *ins_ptr; | ||
32 | uint16_t ins_buf[4]; | ||
33 | int bytes_not_copied = 0; | ||
34 | |||
35 | memset(state, 0, sizeof(struct disasm_state)); | ||
36 | |||
37 | /* This fetches the upper part of the 32 bit instruction | ||
38 | * in both the cases of Little Endian or Big Endian configurations. */ | ||
39 | if (userspace) { | ||
40 | bytes_not_copied = copy_from_user(ins_buf, | ||
41 | (const void __user *) addr, 8); | ||
42 | if (bytes_not_copied > 6) | ||
43 | goto fault; | ||
44 | ins_ptr = ins_buf; | ||
45 | } else { | ||
46 | ins_ptr = (uint16_t *) addr; | ||
47 | } | ||
48 | |||
49 | word1 = *((uint16_t *)addr); | ||
50 | |||
51 | state->major_opcode = (word1 >> 11) & 0x1F; | ||
52 | |||
53 | /* Check if the instruction is 32 bit or 16 bit instruction */ | ||
54 | if (state->major_opcode < 0x0B) { | ||
55 | if (bytes_not_copied > 4) | ||
56 | goto fault; | ||
57 | state->instr_len = 4; | ||
58 | word0 = *((uint16_t *)(addr+2)); | ||
59 | state->words[0] = (word1 << 16) | word0; | ||
60 | } else { | ||
61 | state->instr_len = 2; | ||
62 | state->words[0] = word1; | ||
63 | } | ||
64 | |||
65 | /* Read the second word in case of limm */ | ||
66 | word1 = *((uint16_t *)(addr + state->instr_len)); | ||
67 | word0 = *((uint16_t *)(addr + state->instr_len + 2)); | ||
68 | state->words[1] = (word1 << 16) | word0; | ||
69 | |||
70 | switch (state->major_opcode) { | ||
71 | case op_Bcc: | ||
72 | state->is_branch = 1; | ||
73 | |||
74 | /* unconditional branch s25, conditional branch s21 */ | ||
75 | fieldA = (IS_BIT(state->words[0], 16)) ? | ||
76 | FIELD_s25(state->words[0]) : | ||
77 | FIELD_s21(state->words[0]); | ||
78 | |||
79 | state->delay_slot = IS_BIT(state->words[0], 5); | ||
80 | state->target = fieldA + (addr & ~0x3); | ||
81 | state->flow = direct_jump; | ||
82 | break; | ||
83 | |||
84 | case op_BLcc: | ||
85 | if (IS_BIT(state->words[0], 16)) { | ||
86 | /* Branch and Link*/ | ||
87 | /* unconditional branch s25, conditional branch s21 */ | ||
88 | fieldA = (IS_BIT(state->words[0], 17)) ? | ||
89 | (FIELD_s25(state->words[0]) & ~0x3) : | ||
90 | FIELD_s21(state->words[0]); | ||
91 | |||
92 | state->flow = direct_call; | ||
93 | } else { | ||
94 | /*Branch On Compare */ | ||
95 | fieldA = FIELD_s9(state->words[0]) & ~0x3; | ||
96 | state->flow = direct_jump; | ||
97 | } | ||
98 | |||
99 | state->delay_slot = IS_BIT(state->words[0], 5); | ||
100 | state->target = fieldA + (addr & ~0x3); | ||
101 | state->is_branch = 1; | ||
102 | break; | ||
103 | |||
104 | case op_LD: /* LD<zz> a,[b,s9] */ | ||
105 | state->write = 0; | ||
106 | state->di = BITS(state->words[0], 11, 11); | ||
107 | if (state->di) | ||
108 | break; | ||
109 | state->x = BITS(state->words[0], 6, 6); | ||
110 | state->zz = BITS(state->words[0], 7, 8); | ||
111 | state->aa = BITS(state->words[0], 9, 10); | ||
112 | state->wb_reg = FIELD_B(state->words[0]); | ||
113 | if (state->wb_reg == REG_LIMM) { | ||
114 | state->instr_len += 4; | ||
115 | state->aa = 0; | ||
116 | state->src1 = state->words[1]; | ||
117 | } else { | ||
118 | state->src1 = get_reg(state->wb_reg, regs, cregs); | ||
119 | } | ||
120 | state->src2 = FIELD_s9(state->words[0]); | ||
121 | state->dest = FIELD_A(state->words[0]); | ||
122 | state->pref = (state->dest == REG_LIMM); | ||
123 | break; | ||
124 | |||
125 | case op_ST: | ||
126 | state->write = 1; | ||
127 | state->di = BITS(state->words[0], 5, 5); | ||
128 | if (state->di) | ||
129 | break; | ||
130 | state->aa = BITS(state->words[0], 3, 4); | ||
131 | state->zz = BITS(state->words[0], 1, 2); | ||
132 | state->src1 = FIELD_C(state->words[0]); | ||
133 | if (state->src1 == REG_LIMM) { | ||
134 | state->instr_len += 4; | ||
135 | state->src1 = state->words[1]; | ||
136 | } else { | ||
137 | state->src1 = get_reg(state->src1, regs, cregs); | ||
138 | } | ||
139 | state->wb_reg = FIELD_B(state->words[0]); | ||
140 | if (state->wb_reg == REG_LIMM) { | ||
141 | state->aa = 0; | ||
142 | state->instr_len += 4; | ||
143 | state->src2 = state->words[1]; | ||
144 | } else { | ||
145 | state->src2 = get_reg(state->wb_reg, regs, cregs); | ||
146 | } | ||
147 | state->src3 = FIELD_s9(state->words[0]); | ||
148 | break; | ||
149 | |||
150 | case op_MAJOR_4: | ||
151 | subopcode = MINOR_OPCODE(state->words[0]); | ||
152 | switch (subopcode) { | ||
153 | case 32: /* Jcc */ | ||
154 | case 33: /* Jcc.D */ | ||
155 | case 34: /* JLcc */ | ||
156 | case 35: /* JLcc.D */ | ||
157 | is_linked = 0; | ||
158 | |||
159 | if (subopcode == 33 || subopcode == 35) | ||
160 | state->delay_slot = 1; | ||
161 | |||
162 | if (subopcode == 34 || subopcode == 35) | ||
163 | is_linked = 1; | ||
164 | |||
165 | fieldCisReg = 0; | ||
166 | op_format = BITS(state->words[0], 22, 23); | ||
167 | if (op_format == 0 || ((op_format == 3) && | ||
168 | (!IS_BIT(state->words[0], 5)))) { | ||
169 | fieldC = FIELD_C(state->words[0]); | ||
170 | |||
171 | if (fieldC == REG_LIMM) { | ||
172 | fieldC = state->words[1]; | ||
173 | state->instr_len += 4; | ||
174 | } else { | ||
175 | fieldCisReg = 1; | ||
176 | } | ||
177 | } else if (op_format == 1 || ((op_format == 3) | ||
178 | && (IS_BIT(state->words[0], 5)))) { | ||
179 | fieldC = FIELD_C(state->words[0]); | ||
180 | } else { | ||
181 | /* op_format == 2 */ | ||
182 | fieldC = FIELD_s12(state->words[0]); | ||
183 | } | ||
184 | |||
185 | if (!fieldCisReg) { | ||
186 | state->target = fieldC; | ||
187 | state->flow = is_linked ? | ||
188 | direct_call : direct_jump; | ||
189 | } else { | ||
190 | state->target = get_reg(fieldC, regs, cregs); | ||
191 | state->flow = is_linked ? | ||
192 | indirect_call : indirect_jump; | ||
193 | } | ||
194 | state->is_branch = 1; | ||
195 | break; | ||
196 | |||
197 | case 40: /* LPcc */ | ||
198 | if (BITS(state->words[0], 22, 23) == 3) { | ||
199 | /* Conditional LPcc u7 */ | ||
200 | fieldC = FIELD_C(state->words[0]); | ||
201 | |||
202 | fieldC = fieldC << 1; | ||
203 | fieldC += (addr & ~0x03); | ||
204 | state->is_branch = 1; | ||
205 | state->flow = direct_jump; | ||
206 | state->target = fieldC; | ||
207 | } | ||
208 | /* For Unconditional lp, next pc is the fall through | ||
209 | * which is updated */ | ||
210 | break; | ||
211 | |||
212 | case 48 ... 55: /* LD a,[b,c] */ | ||
213 | state->di = BITS(state->words[0], 15, 15); | ||
214 | if (state->di) | ||
215 | break; | ||
216 | state->x = BITS(state->words[0], 16, 16); | ||
217 | state->zz = BITS(state->words[0], 17, 18); | ||
218 | state->aa = BITS(state->words[0], 22, 23); | ||
219 | state->wb_reg = FIELD_B(state->words[0]); | ||
220 | if (state->wb_reg == REG_LIMM) { | ||
221 | state->instr_len += 4; | ||
222 | state->src1 = state->words[1]; | ||
223 | } else { | ||
224 | state->src1 = get_reg(state->wb_reg, regs, | ||
225 | cregs); | ||
226 | } | ||
227 | state->src2 = FIELD_C(state->words[0]); | ||
228 | if (state->src2 == REG_LIMM) { | ||
229 | state->instr_len += 4; | ||
230 | state->src2 = state->words[1]; | ||
231 | } else { | ||
232 | state->src2 = get_reg(state->src2, regs, | ||
233 | cregs); | ||
234 | } | ||
235 | state->dest = FIELD_A(state->words[0]); | ||
236 | if (state->dest == REG_LIMM) | ||
237 | state->pref = 1; | ||
238 | break; | ||
239 | |||
240 | case 10: /* MOV */ | ||
241 | /* still need to check for limm to extract instr len */ | ||
242 | /* MOV is special case because it only takes 2 args */ | ||
243 | switch (BITS(state->words[0], 22, 23)) { | ||
244 | case 0: /* OP a,b,c */ | ||
245 | if (FIELD_C(state->words[0]) == REG_LIMM) | ||
246 | state->instr_len += 4; | ||
247 | break; | ||
248 | case 1: /* OP a,b,u6 */ | ||
249 | break; | ||
250 | case 2: /* OP b,b,s12 */ | ||
251 | break; | ||
252 | case 3: /* OP.cc b,b,c/u6 */ | ||
253 | if ((!IS_BIT(state->words[0], 5)) && | ||
254 | (FIELD_C(state->words[0]) == REG_LIMM)) | ||
255 | state->instr_len += 4; | ||
256 | break; | ||
257 | } | ||
258 | break; | ||
259 | |||
260 | |||
261 | default: | ||
262 | /* Not a Load, Jump or Loop instruction */ | ||
263 | /* still need to check for limm to extract instr len */ | ||
264 | switch (BITS(state->words[0], 22, 23)) { | ||
265 | case 0: /* OP a,b,c */ | ||
266 | if ((FIELD_B(state->words[0]) == REG_LIMM) || | ||
267 | (FIELD_C(state->words[0]) == REG_LIMM)) | ||
268 | state->instr_len += 4; | ||
269 | break; | ||
270 | case 1: /* OP a,b,u6 */ | ||
271 | break; | ||
272 | case 2: /* OP b,b,s12 */ | ||
273 | break; | ||
274 | case 3: /* OP.cc b,b,c/u6 */ | ||
275 | if ((!IS_BIT(state->words[0], 5)) && | ||
276 | ((FIELD_B(state->words[0]) == REG_LIMM) || | ||
277 | (FIELD_C(state->words[0]) == REG_LIMM))) | ||
278 | state->instr_len += 4; | ||
279 | break; | ||
280 | } | ||
281 | break; | ||
282 | } | ||
283 | break; | ||
284 | |||
285 | /* 16 Bit Instructions */ | ||
286 | case op_LD_ADD: /* LD_S|LDB_S|LDW_S a,[b,c] */ | ||
287 | state->zz = BITS(state->words[0], 3, 4); | ||
288 | state->src1 = get_reg(FIELD_S_B(state->words[0]), regs, cregs); | ||
289 | state->src2 = get_reg(FIELD_S_C(state->words[0]), regs, cregs); | ||
290 | state->dest = FIELD_S_A(state->words[0]); | ||
291 | break; | ||
292 | |||
293 | case op_ADD_MOV_CMP: | ||
294 | /* check for limm, ignore mov_s h,b (== mov_s 0,b) */ | ||
295 | if ((BITS(state->words[0], 3, 4) < 3) && | ||
296 | (FIELD_S_H(state->words[0]) == REG_LIMM)) | ||
297 | state->instr_len += 4; | ||
298 | break; | ||
299 | |||
300 | case op_S: | ||
301 | subopcode = BITS(state->words[0], 5, 7); | ||
302 | switch (subopcode) { | ||
303 | case 0: /* j_s */ | ||
304 | case 1: /* j_s.d */ | ||
305 | case 2: /* jl_s */ | ||
306 | case 3: /* jl_s.d */ | ||
307 | state->target = get_reg(FIELD_S_B(state->words[0]), | ||
308 | regs, cregs); | ||
309 | state->delay_slot = subopcode & 1; | ||
310 | state->flow = (subopcode >= 2) ? | ||
311 | direct_call : indirect_jump; | ||
312 | break; | ||
313 | case 7: | ||
314 | switch (BITS(state->words[0], 8, 10)) { | ||
315 | case 4: /* jeq_s [blink] */ | ||
316 | case 5: /* jne_s [blink] */ | ||
317 | case 6: /* j_s [blink] */ | ||
318 | case 7: /* j_s.d [blink] */ | ||
319 | state->delay_slot = (subopcode == 7); | ||
320 | state->flow = indirect_jump; | ||
321 | state->target = get_reg(31, regs, cregs); | ||
322 | default: | ||
323 | break; | ||
324 | } | ||
325 | default: | ||
326 | break; | ||
327 | } | ||
328 | break; | ||
329 | |||
330 | case op_LD_S: /* LD_S c, [b, u7] */ | ||
331 | state->src1 = get_reg(FIELD_S_B(state->words[0]), regs, cregs); | ||
332 | state->src2 = FIELD_S_u7(state->words[0]); | ||
333 | state->dest = FIELD_S_C(state->words[0]); | ||
334 | break; | ||
335 | |||
336 | case op_LDB_S: | ||
337 | case op_STB_S: | ||
338 | /* no further handling required as byte accesses should not | ||
339 | * cause an unaligned access exception */ | ||
340 | state->zz = 1; | ||
341 | break; | ||
342 | |||
343 | case op_LDWX_S: /* LDWX_S c, [b, u6] */ | ||
344 | state->x = 1; | ||
345 | /* intentional fall-through */ | ||
346 | |||
347 | case op_LDW_S: /* LDW_S c, [b, u6] */ | ||
348 | state->zz = 2; | ||
349 | state->src1 = get_reg(FIELD_S_B(state->words[0]), regs, cregs); | ||
350 | state->src2 = FIELD_S_u6(state->words[0]); | ||
351 | state->dest = FIELD_S_C(state->words[0]); | ||
352 | break; | ||
353 | |||
354 | case op_ST_S: /* ST_S c, [b, u7] */ | ||
355 | state->write = 1; | ||
356 | state->src1 = get_reg(FIELD_S_C(state->words[0]), regs, cregs); | ||
357 | state->src2 = get_reg(FIELD_S_B(state->words[0]), regs, cregs); | ||
358 | state->src3 = FIELD_S_u7(state->words[0]); | ||
359 | break; | ||
360 | |||
361 | case op_STW_S: /* STW_S c,[b,u6] */ | ||
362 | state->write = 1; | ||
363 | state->zz = 2; | ||
364 | state->src1 = get_reg(FIELD_S_C(state->words[0]), regs, cregs); | ||
365 | state->src2 = get_reg(FIELD_S_B(state->words[0]), regs, cregs); | ||
366 | state->src3 = FIELD_S_u6(state->words[0]); | ||
367 | break; | ||
368 | |||
369 | case op_SP: /* LD_S|LDB_S b,[sp,u7], ST_S|STB_S b,[sp,u7] */ | ||
370 | /* note: we are ignoring possibility of: | ||
371 | * ADD_S, SUB_S, PUSH_S, POP_S as these should not | ||
372 | * cause unaliged exception anyway */ | ||
373 | state->write = BITS(state->words[0], 6, 6); | ||
374 | state->zz = BITS(state->words[0], 5, 5); | ||
375 | if (state->zz) | ||
376 | break; /* byte accesses should not come here */ | ||
377 | if (!state->write) { | ||
378 | state->src1 = get_reg(28, regs, cregs); | ||
379 | state->src2 = FIELD_S_u7(state->words[0]); | ||
380 | state->dest = FIELD_S_B(state->words[0]); | ||
381 | } else { | ||
382 | state->src1 = get_reg(FIELD_S_B(state->words[0]), regs, | ||
383 | cregs); | ||
384 | state->src2 = get_reg(28, regs, cregs); | ||
385 | state->src3 = FIELD_S_u7(state->words[0]); | ||
386 | } | ||
387 | break; | ||
388 | |||
389 | case op_GP: /* LD_S|LDB_S|LDW_S r0,[gp,s11/s9/s10] */ | ||
390 | /* note: ADD_S r0, gp, s11 is ignored */ | ||
391 | state->zz = BITS(state->words[0], 9, 10); | ||
392 | state->src1 = get_reg(26, regs, cregs); | ||
393 | state->src2 = state->zz ? FIELD_S_s10(state->words[0]) : | ||
394 | FIELD_S_s11(state->words[0]); | ||
395 | state->dest = 0; | ||
396 | break; | ||
397 | |||
398 | case op_Pcl: /* LD_S b,[pcl,u10] */ | ||
399 | state->src1 = regs->ret & ~3; | ||
400 | state->src2 = FIELD_S_u10(state->words[0]); | ||
401 | state->dest = FIELD_S_B(state->words[0]); | ||
402 | break; | ||
403 | |||
404 | case op_BR_S: | ||
405 | state->target = FIELD_S_s8(state->words[0]) + (addr & ~0x03); | ||
406 | state->flow = direct_jump; | ||
407 | state->is_branch = 1; | ||
408 | break; | ||
409 | |||
410 | case op_B_S: | ||
411 | fieldA = (BITS(state->words[0], 9, 10) == 3) ? | ||
412 | FIELD_S_s7(state->words[0]) : | ||
413 | FIELD_S_s10(state->words[0]); | ||
414 | state->target = fieldA + (addr & ~0x03); | ||
415 | state->flow = direct_jump; | ||
416 | state->is_branch = 1; | ||
417 | break; | ||
418 | |||
419 | case op_BL_S: | ||
420 | state->target = FIELD_S_s13(state->words[0]) + (addr & ~0x03); | ||
421 | state->flow = direct_call; | ||
422 | state->is_branch = 1; | ||
423 | break; | ||
424 | |||
425 | default: | ||
426 | break; | ||
427 | } | ||
428 | |||
429 | if (bytes_not_copied <= (8 - state->instr_len)) | ||
430 | return; | ||
431 | |||
432 | fault: state->fault = 1; | ||
433 | } | ||
434 | |||
435 | long __kprobes get_reg(int reg, struct pt_regs *regs, | ||
436 | struct callee_regs *cregs) | ||
437 | { | ||
438 | long *p; | ||
439 | |||
440 | if (reg <= 12) { | ||
441 | p = ®s->r0; | ||
442 | return p[-reg]; | ||
443 | } | ||
444 | |||
445 | if (cregs && (reg <= 25)) { | ||
446 | p = &cregs->r13; | ||
447 | return p[13-reg]; | ||
448 | } | ||
449 | |||
450 | if (reg == 26) | ||
451 | return regs->r26; | ||
452 | if (reg == 27) | ||
453 | return regs->fp; | ||
454 | if (reg == 28) | ||
455 | return regs->sp; | ||
456 | if (reg == 31) | ||
457 | return regs->blink; | ||
458 | |||
459 | return 0; | ||
460 | } | ||
461 | |||
462 | void __kprobes set_reg(int reg, long val, struct pt_regs *regs, | ||
463 | struct callee_regs *cregs) | ||
464 | { | ||
465 | long *p; | ||
466 | |||
467 | switch (reg) { | ||
468 | case 0 ... 12: | ||
469 | p = ®s->r0; | ||
470 | p[-reg] = val; | ||
471 | break; | ||
472 | case 13 ... 25: | ||
473 | if (cregs) { | ||
474 | p = &cregs->r13; | ||
475 | p[13-reg] = val; | ||
476 | } | ||
477 | break; | ||
478 | case 26: | ||
479 | regs->r26 = val; | ||
480 | break; | ||
481 | case 27: | ||
482 | regs->fp = val; | ||
483 | break; | ||
484 | case 28: | ||
485 | regs->sp = val; | ||
486 | break; | ||
487 | case 31: | ||
488 | regs->blink = val; | ||
489 | break; | ||
490 | default: | ||
491 | break; | ||
492 | } | ||
493 | } | ||
494 | |||
495 | /* | ||
496 | * Disassembles the insn at @pc and sets @next_pc to next PC (which could be | ||
497 | * @pc +2/4/6 (ARCompact ISA allows free intermixing of 16/32 bit insns). | ||
498 | * | ||
499 | * If @pc is a branch | ||
500 | * -@tgt_if_br is set to branch target. | ||
501 | * -If branch has delay slot, @next_pc updated with actual next PC. | ||
502 | */ | ||
503 | int __kprobes disasm_next_pc(unsigned long pc, struct pt_regs *regs, | ||
504 | struct callee_regs *cregs, | ||
505 | unsigned long *next_pc, unsigned long *tgt_if_br) | ||
506 | { | ||
507 | struct disasm_state instr; | ||
508 | |||
509 | memset(&instr, 0, sizeof(struct disasm_state)); | ||
510 | disasm_instr(pc, &instr, 0, regs, cregs); | ||
511 | |||
512 | *next_pc = pc + instr.instr_len; | ||
513 | |||
514 | /* Instruction with possible two targets branch, jump and loop */ | ||
515 | if (instr.is_branch) | ||
516 | *tgt_if_br = instr.target; | ||
517 | |||
518 | /* For the instructions with delay slots, the fall through is the | ||
519 | * instruction following the instruction in delay slot. | ||
520 | */ | ||
521 | if (instr.delay_slot) { | ||
522 | struct disasm_state instr_d; | ||
523 | |||
524 | disasm_instr(*next_pc, &instr_d, 0, regs, cregs); | ||
525 | |||
526 | *next_pc += instr_d.instr_len; | ||
527 | } | ||
528 | |||
529 | /* Zero Overhead Loop - end of the loop */ | ||
530 | if (!(regs->status32 & STATUS32_L) && (*next_pc == regs->lp_end) | ||
531 | && (regs->lp_count > 1)) { | ||
532 | *next_pc = regs->lp_start; | ||
533 | } | ||
534 | |||
535 | return instr.is_branch; | ||
536 | } | ||
537 | |||
538 | #endif /* CONFIG_KGDB || CONFIG_MISALIGN_ACCESS || CONFIG_KPROBES */ | ||
diff --git a/arch/arc/kernel/entry.S b/arch/arc/kernel/entry.S new file mode 100644 index 000000000000..ef6800ba2f03 --- /dev/null +++ b/arch/arc/kernel/entry.S | |||
@@ -0,0 +1,839 @@ | |||
1 | /* | ||
2 | * Low Level Interrupts/Traps/Exceptions(non-TLB) Handling for ARC | ||
3 | * | ||
4 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | * | ||
10 | * vineetg: May 2011 | ||
11 | * -Userspace unaligned access emulation | ||
12 | * | ||
13 | * vineetg: Feb 2011 (ptrace low level code fixes) | ||
14 | * -traced syscall return code (r0) was not saved into pt_regs for restoring | ||
15 | * into user reg-file when traded task rets to user space. | ||
16 | * -syscalls needing arch-wrappers (mainly for passing sp as pt_regs) | ||
17 | * were not invoking post-syscall trace hook (jumping directly into | ||
18 | * ret_from_system_call) | ||
19 | * | ||
20 | * vineetg: Nov 2010: | ||
21 | * -Vector table jumps (@8 bytes) converted into branches (@4 bytes) | ||
22 | * -To maintain the slot size of 8 bytes/vector, added nop, which is | ||
23 | * not executed at runtime. | ||
24 | * | ||
25 | * vineetg: Nov 2009 (Everything needed for TIF_RESTORE_SIGMASK) | ||
26 | * -do_signal()invoked upon TIF_RESTORE_SIGMASK as well | ||
27 | * -Wrappers for sys_{,rt_}sigsuspend() nolonger needed as they don't | ||
28 | * need ptregs anymore | ||
29 | * | ||
30 | * Vineetg: Oct 2009 | ||
31 | * -In a rare scenario, Process gets a Priv-V exception and gets scheduled | ||
32 | * out. Since we don't do FAKE RTIE for Priv-V, CPU excpetion state remains | ||
33 | * active (AE bit enabled). This causes a double fault for a subseq valid | ||
34 | * exception. Thus FAKE RTIE needed in low level Priv-Violation handler. | ||
35 | * Instr Error could also cause similar scenario, so same there as well. | ||
36 | * | ||
37 | * Vineetg: March 2009 (Supporting 2 levels of Interrupts) | ||
38 | * | ||
39 | * Vineetg: Aug 28th 2008: Bug #94984 | ||
40 | * -Zero Overhead Loop Context shd be cleared when entering IRQ/EXcp/Trap | ||
41 | * Normally CPU does this automatically, however when doing FAKE rtie, | ||
42 | * we need to explicitly do this. The problem in macros | ||
43 | * FAKE_RET_FROM_EXCPN and FAKE_RET_FROM_EXCPN_LOCK_IRQ was that this bit | ||
44 | * was being "CLEARED" rather then "SET". Since it is Loop INHIBIT Bit, | ||
45 | * setting it and not clearing it clears ZOL context | ||
46 | * | ||
47 | * Vineetg: May 16th, 2008 | ||
48 | * - r25 now contains the Current Task when in kernel | ||
49 | * | ||
50 | * Vineetg: Dec 22, 2007 | ||
51 | * Minor Surgery of Low Level ISR to make it SMP safe | ||
52 | * - MMU_SCRATCH0 Reg used for freeing up r9 in Level 1 ISR | ||
53 | * - _current_task is made an array of NR_CPUS | ||
54 | * - Access of _current_task wrapped inside a macro so that if hardware | ||
55 | * team agrees for a dedicated reg, no other code is touched | ||
56 | * | ||
57 | * Amit Bhor, Rahul Trivedi, Kanika Nema, Sameer Dhavale : Codito Tech 2004 | ||
58 | */ | ||
59 | |||
60 | /*------------------------------------------------------------------ | ||
61 | * Function ABI | ||
62 | *------------------------------------------------------------------ | ||
63 | * | ||
64 | * Arguments r0 - r7 | ||
65 | * Caller Saved Registers r0 - r12 | ||
66 | * Callee Saved Registers r13- r25 | ||
67 | * Global Pointer (gp) r26 | ||
68 | * Frame Pointer (fp) r27 | ||
69 | * Stack Pointer (sp) r28 | ||
70 | * Interrupt link register (ilink1) r29 | ||
71 | * Interrupt link register (ilink2) r30 | ||
72 | * Branch link register (blink) r31 | ||
73 | *------------------------------------------------------------------ | ||
74 | */ | ||
75 | |||
76 | .cpu A7 | ||
77 | |||
78 | ;############################ Vector Table ################################# | ||
79 | |||
80 | .macro VECTOR lbl | ||
81 | #if 1 /* Just in case, build breaks */ | ||
82 | j \lbl | ||
83 | #else | ||
84 | b \lbl | ||
85 | nop | ||
86 | #endif | ||
87 | .endm | ||
88 | |||
89 | .section .vector, "ax",@progbits | ||
90 | .align 4 | ||
91 | |||
92 | /* Each entry in the vector table must occupy 2 words. Since it is a jump | ||
93 | * across sections (.vector to .text) we are gauranteed that 'j somewhere' | ||
94 | * will use the 'j limm' form of the intrsuction as long as somewhere is in | ||
95 | * a section other than .vector. | ||
96 | */ | ||
97 | |||
98 | ; ********* Critical System Events ********************** | ||
99 | VECTOR res_service ; 0x0, Restart Vector (0x0) | ||
100 | VECTOR mem_service ; 0x8, Mem exception (0x1) | ||
101 | VECTOR instr_service ; 0x10, Instrn Error (0x2) | ||
102 | |||
103 | ; ******************** Device ISRs ********************** | ||
104 | #ifdef CONFIG_ARC_IRQ3_LV2 | ||
105 | VECTOR handle_interrupt_level2 | ||
106 | #else | ||
107 | VECTOR handle_interrupt_level1 | ||
108 | #endif | ||
109 | |||
110 | VECTOR handle_interrupt_level1 | ||
111 | |||
112 | #ifdef CONFIG_ARC_IRQ5_LV2 | ||
113 | VECTOR handle_interrupt_level2 | ||
114 | #else | ||
115 | VECTOR handle_interrupt_level1 | ||
116 | #endif | ||
117 | |||
118 | #ifdef CONFIG_ARC_IRQ6_LV2 | ||
119 | VECTOR handle_interrupt_level2 | ||
120 | #else | ||
121 | VECTOR handle_interrupt_level1 | ||
122 | #endif | ||
123 | |||
124 | .rept 25 | ||
125 | VECTOR handle_interrupt_level1 ; Other devices | ||
126 | .endr | ||
127 | |||
128 | /* FOR ARC600: timer = 0x3, uart = 0x8, emac = 0x10 */ | ||
129 | |||
130 | ; ******************** Exceptions ********************** | ||
131 | VECTOR EV_MachineCheck ; 0x100, Fatal Machine check (0x20) | ||
132 | VECTOR EV_TLBMissI ; 0x108, Intruction TLB miss (0x21) | ||
133 | VECTOR EV_TLBMissD ; 0x110, Data TLB miss (0x22) | ||
134 | VECTOR EV_TLBProtV ; 0x118, Protection Violation (0x23) | ||
135 | ; or Misaligned Access | ||
136 | VECTOR EV_PrivilegeV ; 0x120, Privilege Violation (0x24) | ||
137 | VECTOR EV_Trap ; 0x128, Trap exception (0x25) | ||
138 | VECTOR EV_Extension ; 0x130, Extn Intruction Excp (0x26) | ||
139 | |||
140 | .rept 24 | ||
141 | VECTOR reserved ; Reserved Exceptions | ||
142 | .endr | ||
143 | |||
144 | #include <linux/linkage.h> /* ARC_{EXTRY,EXIT} */ | ||
145 | #include <asm/entry.h> /* SAVE_ALL_{INT1,INT2,TRAP...} */ | ||
146 | #include <asm/errno.h> | ||
147 | #include <asm/arcregs.h> | ||
148 | #include <asm/irqflags.h> | ||
149 | |||
150 | ;##################### Scratch Mem for IRQ stack switching ############# | ||
151 | |||
152 | ARCFP_DATA int1_saved_reg | ||
153 | .align 32 | ||
154 | .type int1_saved_reg, @object | ||
155 | .size int1_saved_reg, 4 | ||
156 | int1_saved_reg: | ||
157 | .zero 4 | ||
158 | |||
159 | /* Each Interrupt level needs it's own scratch */ | ||
160 | #ifdef CONFIG_ARC_COMPACT_IRQ_LEVELS | ||
161 | |||
162 | ARCFP_DATA int2_saved_reg | ||
163 | .type int2_saved_reg, @object | ||
164 | .size int2_saved_reg, 4 | ||
165 | int2_saved_reg: | ||
166 | .zero 4 | ||
167 | |||
168 | #endif | ||
169 | |||
170 | ; --------------------------------------------- | ||
171 | .section .text, "ax",@progbits | ||
172 | |||
173 | res_service: ; processor restart | ||
174 | flag 0x1 ; not implemented | ||
175 | nop | ||
176 | nop | ||
177 | |||
178 | reserved: ; processor restart | ||
179 | rtie ; jump to processor initializations | ||
180 | |||
181 | ;##################### Interrupt Handling ############################## | ||
182 | |||
183 | #ifdef CONFIG_ARC_COMPACT_IRQ_LEVELS | ||
184 | ; --------------------------------------------- | ||
185 | ; Level 2 ISR: Can interrupt a Level 1 ISR | ||
186 | ; --------------------------------------------- | ||
187 | ARC_ENTRY handle_interrupt_level2 | ||
188 | |||
189 | ; TODO-vineetg for SMP this wont work | ||
190 | ; free up r9 as scratchpad | ||
191 | st r9, [@int2_saved_reg] | ||
192 | |||
193 | ;Which mode (user/kernel) was the system in when intr occured | ||
194 | lr r9, [status32_l2] | ||
195 | |||
196 | SWITCH_TO_KERNEL_STK | ||
197 | SAVE_ALL_INT2 | ||
198 | |||
199 | ;------------------------------------------------------ | ||
200 | ; if L2 IRQ interrupted a L1 ISR, disable preemption | ||
201 | ;------------------------------------------------------ | ||
202 | |||
203 | ld r9, [sp, PT_status32] ; get statu32_l2 (saved in pt_regs) | ||
204 | bbit0 r9, STATUS_A1_BIT, 1f ; L1 not active when L2 IRQ, so normal | ||
205 | |||
206 | ; A1 is set in status32_l2 | ||
207 | ; bump thread_info->preempt_count (Disable preemption) | ||
208 | GET_CURR_THR_INFO_FROM_SP r10 | ||
209 | ld r9, [r10, THREAD_INFO_PREEMPT_COUNT] | ||
210 | add r9, r9, 1 | ||
211 | st r9, [r10, THREAD_INFO_PREEMPT_COUNT] | ||
212 | |||
213 | 1: | ||
214 | ;------------------------------------------------------ | ||
215 | ; setup params for Linux common ISR and invoke it | ||
216 | ;------------------------------------------------------ | ||
217 | lr r0, [icause2] | ||
218 | and r0, r0, 0x1f | ||
219 | |||
220 | bl.d @arch_do_IRQ | ||
221 | mov r1, sp | ||
222 | |||
223 | mov r8,0x2 | ||
224 | sr r8, [AUX_IRQ_LV12] ; clear bit in Sticky Status Reg | ||
225 | |||
226 | b ret_from_exception | ||
227 | |||
228 | ARC_EXIT handle_interrupt_level2 | ||
229 | |||
230 | #endif | ||
231 | |||
232 | ; --------------------------------------------- | ||
233 | ; Level 1 ISR | ||
234 | ; --------------------------------------------- | ||
235 | ARC_ENTRY handle_interrupt_level1 | ||
236 | |||
237 | /* free up r9 as scratchpad */ | ||
238 | #ifdef CONFIG_SMP | ||
239 | sr r9, [ARC_REG_SCRATCH_DATA0] | ||
240 | #else | ||
241 | st r9, [@int1_saved_reg] | ||
242 | #endif | ||
243 | |||
244 | ;Which mode (user/kernel) was the system in when intr occured | ||
245 | lr r9, [status32_l1] | ||
246 | |||
247 | SWITCH_TO_KERNEL_STK | ||
248 | SAVE_ALL_INT1 | ||
249 | |||
250 | lr r0, [icause1] | ||
251 | and r0, r0, 0x1f | ||
252 | |||
253 | bl.d @arch_do_IRQ | ||
254 | mov r1, sp | ||
255 | |||
256 | mov r8,0x1 | ||
257 | sr r8, [AUX_IRQ_LV12] ; clear bit in Sticky Status Reg | ||
258 | |||
259 | b ret_from_exception | ||
260 | ARC_EXIT handle_interrupt_level1 | ||
261 | |||
262 | ;################### Non TLB Exception Handling ############################# | ||
263 | |||
264 | ; --------------------------------------------- | ||
265 | ; Instruction Error Exception Handler | ||
266 | ; --------------------------------------------- | ||
267 | |||
268 | ARC_ENTRY instr_service | ||
269 | |||
270 | EXCPN_PROLOG_FREEUP_REG r9 | ||
271 | |||
272 | lr r9, [erstatus] | ||
273 | |||
274 | SWITCH_TO_KERNEL_STK | ||
275 | SAVE_ALL_SYS | ||
276 | |||
277 | lr r0, [ecr] | ||
278 | lr r1, [efa] | ||
279 | |||
280 | mov r2, sp | ||
281 | |||
282 | FAKE_RET_FROM_EXCPN r9 | ||
283 | |||
284 | bl do_insterror_or_kprobe | ||
285 | b ret_from_exception | ||
286 | ARC_EXIT instr_service | ||
287 | |||
288 | ; --------------------------------------------- | ||
289 | ; Memory Error Exception Handler | ||
290 | ; --------------------------------------------- | ||
291 | |||
292 | ARC_ENTRY mem_service | ||
293 | |||
294 | EXCPN_PROLOG_FREEUP_REG r9 | ||
295 | |||
296 | lr r9, [erstatus] | ||
297 | |||
298 | SWITCH_TO_KERNEL_STK | ||
299 | SAVE_ALL_SYS | ||
300 | |||
301 | lr r0, [ecr] | ||
302 | lr r1, [efa] | ||
303 | mov r2, sp | ||
304 | bl do_memory_error | ||
305 | b ret_from_exception | ||
306 | ARC_EXIT mem_service | ||
307 | |||
308 | ; --------------------------------------------- | ||
309 | ; Machine Check Exception Handler | ||
310 | ; --------------------------------------------- | ||
311 | |||
312 | ARC_ENTRY EV_MachineCheck | ||
313 | |||
314 | EXCPN_PROLOG_FREEUP_REG r9 | ||
315 | lr r9, [erstatus] | ||
316 | |||
317 | SWITCH_TO_KERNEL_STK | ||
318 | SAVE_ALL_SYS | ||
319 | |||
320 | lr r0, [ecr] | ||
321 | lr r1, [efa] | ||
322 | mov r2, sp | ||
323 | |||
324 | brne r0, 0x200100, 1f | ||
325 | bl do_tlb_overlap_fault | ||
326 | b ret_from_exception | ||
327 | |||
328 | 1: | ||
329 | ; DEAD END: can't do much, display Regs and HALT | ||
330 | SAVE_CALLEE_SAVED_USER | ||
331 | |||
332 | GET_CURR_TASK_FIELD_PTR TASK_THREAD, r10 | ||
333 | st sp, [r10, THREAD_CALLEE_REG] | ||
334 | |||
335 | j do_machine_check_fault | ||
336 | |||
337 | ARC_EXIT EV_MachineCheck | ||
338 | |||
339 | ; --------------------------------------------- | ||
340 | ; Protection Violation Exception Handler | ||
341 | ; --------------------------------------------- | ||
342 | |||
343 | ARC_ENTRY EV_TLBProtV | ||
344 | |||
345 | EXCPN_PROLOG_FREEUP_REG r9 | ||
346 | |||
347 | ;Which mode (user/kernel) was the system in when Exception occured | ||
348 | lr r9, [erstatus] | ||
349 | |||
350 | SWITCH_TO_KERNEL_STK | ||
351 | SAVE_ALL_SYS | ||
352 | |||
353 | ;---------(3) Save some more regs----------------- | ||
354 | ; vineetg: Mar 6th: Random Seg Fault issue #1 | ||
355 | ; ecr and efa were not saved in case an Intr sneaks in | ||
356 | ; after fake rtie | ||
357 | ; | ||
358 | lr r3, [ecr] | ||
359 | lr r4, [efa] | ||
360 | |||
361 | ; --------(4) Return from CPU Exception Mode --------- | ||
362 | ; Fake a rtie, but rtie to next label | ||
363 | ; That way, subsequently, do_page_fault ( ) executes in pure kernel | ||
364 | ; mode with further Exceptions enabled | ||
365 | |||
366 | FAKE_RET_FROM_EXCPN r9 | ||
367 | |||
368 | ;------ (5) Type of Protection Violation? ---------- | ||
369 | ; | ||
370 | ; ProtV Hardware Exception is triggered for Access Faults of 2 types | ||
371 | ; -Access Violaton (WRITE to READ ONLY Page) - for linux COW | ||
372 | ; -Unaligned Access (READ/WRITE on odd boundary) | ||
373 | ; | ||
374 | cmp r3, 0x230400 ; Misaligned data access ? | ||
375 | beq 4f | ||
376 | |||
377 | ;========= (6a) Access Violation Processing ======== | ||
378 | cmp r3, 0x230100 | ||
379 | mov r1, 0x0 ; if LD exception ? write = 0 | ||
380 | mov.ne r1, 0x1 ; else write = 1 | ||
381 | |||
382 | mov r2, r4 ; faulting address | ||
383 | mov r0, sp ; pt_regs | ||
384 | bl do_page_fault | ||
385 | b ret_from_exception | ||
386 | |||
387 | ;========== (6b) Non aligned access ============ | ||
388 | 4: | ||
389 | mov r0, r3 ; cause code | ||
390 | mov r1, r4 ; faulting address | ||
391 | mov r2, sp ; pt_regs | ||
392 | |||
393 | #ifdef CONFIG_ARC_MISALIGN_ACCESS | ||
394 | SAVE_CALLEE_SAVED_USER | ||
395 | mov r3, sp ; callee_regs | ||
396 | #endif | ||
397 | |||
398 | bl do_misaligned_access | ||
399 | |||
400 | #ifdef CONFIG_ARC_MISALIGN_ACCESS | ||
401 | DISCARD_CALLEE_SAVED_USER | ||
402 | #endif | ||
403 | |||
404 | b ret_from_exception | ||
405 | |||
406 | ARC_EXIT EV_TLBProtV | ||
407 | |||
408 | ; --------------------------------------------- | ||
409 | ; Privilege Violation Exception Handler | ||
410 | ; --------------------------------------------- | ||
411 | ARC_ENTRY EV_PrivilegeV | ||
412 | |||
413 | EXCPN_PROLOG_FREEUP_REG r9 | ||
414 | |||
415 | lr r9, [erstatus] | ||
416 | |||
417 | SWITCH_TO_KERNEL_STK | ||
418 | SAVE_ALL_SYS | ||
419 | |||
420 | lr r0, [ecr] | ||
421 | lr r1, [efa] | ||
422 | mov r2, sp | ||
423 | |||
424 | FAKE_RET_FROM_EXCPN r9 | ||
425 | |||
426 | bl do_privilege_fault | ||
427 | b ret_from_exception | ||
428 | ARC_EXIT EV_PrivilegeV | ||
429 | |||
430 | ; --------------------------------------------- | ||
431 | ; Extension Instruction Exception Handler | ||
432 | ; --------------------------------------------- | ||
433 | ARC_ENTRY EV_Extension | ||
434 | |||
435 | EXCPN_PROLOG_FREEUP_REG r9 | ||
436 | lr r9, [erstatus] | ||
437 | |||
438 | SWITCH_TO_KERNEL_STK | ||
439 | SAVE_ALL_SYS | ||
440 | |||
441 | lr r0, [ecr] | ||
442 | lr r1, [efa] | ||
443 | mov r2, sp | ||
444 | bl do_extension_fault | ||
445 | b ret_from_exception | ||
446 | ARC_EXIT EV_Extension | ||
447 | |||
448 | ;######################### System Call Tracing ######################### | ||
449 | |||
450 | tracesys: | ||
451 | ; save EFA in case tracer wants the PC of traced task | ||
452 | ; using ERET won't work since next-PC has already committed | ||
453 | lr r12, [efa] | ||
454 | GET_CURR_TASK_FIELD_PTR TASK_THREAD, r11 | ||
455 | st r12, [r11, THREAD_FAULT_ADDR] | ||
456 | |||
457 | ; PRE Sys Call Ptrace hook | ||
458 | mov r0, sp ; pt_regs needed | ||
459 | bl @syscall_trace_entry | ||
460 | |||
461 | ; Tracing code now returns the syscall num (orig or modif) | ||
462 | mov r8, r0 | ||
463 | |||
464 | ; Do the Sys Call as we normally would. | ||
465 | ; Validate the Sys Call number | ||
466 | cmp r8, NR_syscalls | ||
467 | mov.hi r0, -ENOSYS | ||
468 | bhi tracesys_exit | ||
469 | |||
470 | ; Restore the sys-call args. Mere invocation of the hook abv could have | ||
471 | ; clobbered them (since they are in scratch regs). The tracer could also | ||
472 | ; have deliberately changed the syscall args: r0-r7 | ||
473 | ld r0, [sp, PT_r0] | ||
474 | ld r1, [sp, PT_r1] | ||
475 | ld r2, [sp, PT_r2] | ||
476 | ld r3, [sp, PT_r3] | ||
477 | ld r4, [sp, PT_r4] | ||
478 | ld r5, [sp, PT_r5] | ||
479 | ld r6, [sp, PT_r6] | ||
480 | ld r7, [sp, PT_r7] | ||
481 | ld.as r9, [sys_call_table, r8] | ||
482 | jl [r9] ; Entry into Sys Call Handler | ||
483 | |||
484 | tracesys_exit: | ||
485 | st r0, [sp, PT_r0] ; sys call return value in pt_regs | ||
486 | |||
487 | ;POST Sys Call Ptrace Hook | ||
488 | bl @syscall_trace_exit | ||
489 | b ret_from_exception ; NOT ret_from_system_call at is saves r0 which | ||
490 | ; we'd done before calling post hook above | ||
491 | |||
492 | ;################### Break Point TRAP ########################## | ||
493 | |||
494 | ; ======= (5b) Trap is due to Break-Point ========= | ||
495 | |||
496 | trap_with_param: | ||
497 | |||
498 | ; stop_pc info by gdb needs this info | ||
499 | stw orig_r8_IS_BRKPT, [sp, PT_orig_r8] | ||
500 | |||
501 | mov r0, r12 | ||
502 | lr r1, [efa] | ||
503 | mov r2, sp | ||
504 | |||
505 | ; Now that we have read EFA, its safe to do "fake" rtie | ||
506 | ; and get out of CPU exception mode | ||
507 | FAKE_RET_FROM_EXCPN r11 | ||
508 | |||
509 | ; Save callee regs in case gdb wants to have a look | ||
510 | ; SP will grow up by size of CALLEE Reg-File | ||
511 | ; NOTE: clobbers r12 | ||
512 | SAVE_CALLEE_SAVED_USER | ||
513 | |||
514 | ; save location of saved Callee Regs @ thread_struct->pc | ||
515 | GET_CURR_TASK_FIELD_PTR TASK_THREAD, r10 | ||
516 | st sp, [r10, THREAD_CALLEE_REG] | ||
517 | |||
518 | ; Call the trap handler | ||
519 | bl do_non_swi_trap | ||
520 | |||
521 | ; unwind stack to discard Callee saved Regs | ||
522 | DISCARD_CALLEE_SAVED_USER | ||
523 | |||
524 | b ret_from_exception | ||
525 | |||
526 | ;##################### Trap Handling ############################## | ||
527 | ; | ||
528 | ; EV_Trap caused by TRAP_S and TRAP0 instructions. | ||
529 | ;------------------------------------------------------------------ | ||
530 | ; (1) System Calls | ||
531 | ; :parameters in r0-r7. | ||
532 | ; :r8 has the system call number | ||
533 | ; (2) Break Points | ||
534 | ;------------------------------------------------------------------ | ||
535 | |||
536 | ARC_ENTRY EV_Trap | ||
537 | |||
538 | ; Need at least 1 reg to code the early exception prolog | ||
539 | EXCPN_PROLOG_FREEUP_REG r9 | ||
540 | |||
541 | ;Which mode (user/kernel) was the system in when intr occured | ||
542 | lr r9, [erstatus] | ||
543 | |||
544 | SWITCH_TO_KERNEL_STK | ||
545 | SAVE_ALL_TRAP | ||
546 | |||
547 | ;------- (4) What caused the Trap -------------- | ||
548 | lr r12, [ecr] | ||
549 | and.f 0, r12, ECR_PARAM_MASK | ||
550 | bnz trap_with_param | ||
551 | |||
552 | ; ======= (5a) Trap is due to System Call ======== | ||
553 | |||
554 | ; Before doing anything, return from CPU Exception Mode | ||
555 | FAKE_RET_FROM_EXCPN r11 | ||
556 | |||
557 | ; If syscall tracing ongoing, invoke pre-pos-hooks | ||
558 | GET_CURR_THR_INFO_FLAGS r10 | ||
559 | btst r10, TIF_SYSCALL_TRACE | ||
560 | bnz tracesys ; this never comes back | ||
561 | |||
562 | ;============ This is normal System Call case ========== | ||
563 | ; Sys-call num shd not exceed the total system calls avail | ||
564 | cmp r8, NR_syscalls | ||
565 | mov.hi r0, -ENOSYS | ||
566 | bhi ret_from_system_call | ||
567 | |||
568 | ; Offset into the syscall_table and call handler | ||
569 | ld.as r9,[sys_call_table, r8] | ||
570 | jl [r9] ; Entry into Sys Call Handler | ||
571 | |||
572 | ; fall through to ret_from_system_call | ||
573 | ARC_EXIT EV_Trap | ||
574 | |||
575 | ARC_ENTRY ret_from_system_call | ||
576 | |||
577 | st r0, [sp, PT_r0] ; sys call return value in pt_regs | ||
578 | |||
579 | ; fall through yet again to ret_from_exception | ||
580 | |||
581 | ;############# Return from Intr/Excp/Trap (Linux Specifics) ############## | ||
582 | ; | ||
583 | ; If ret to user mode do we need to handle signals, schedule() et al. | ||
584 | |||
585 | ARC_ENTRY ret_from_exception | ||
586 | |||
587 | ; Pre-{IRQ,Trap,Exception} K/U mode from pt_regs->status32 | ||
588 | ld r8, [sp, PT_status32] ; returning to User/Kernel Mode | ||
589 | |||
590 | #ifdef CONFIG_PREEMPT | ||
591 | bbit0 r8, STATUS_U_BIT, resume_kernel_mode | ||
592 | #else | ||
593 | bbit0 r8, STATUS_U_BIT, restore_regs | ||
594 | #endif | ||
595 | |||
596 | ; Before returning to User mode check-for-and-complete any pending work | ||
597 | ; such as rescheduling/signal-delivery etc. | ||
598 | resume_user_mode_begin: | ||
599 | |||
600 | ; Disable IRQs to ensures that chk for pending work itself is atomic | ||
601 | ; (and we don't end up missing a NEED_RESCHED/SIGPENDING due to an | ||
602 | ; interim IRQ). | ||
603 | IRQ_DISABLE r10 | ||
604 | |||
605 | ; Fast Path return to user mode if no pending work | ||
606 | GET_CURR_THR_INFO_FLAGS r9 | ||
607 | and.f 0, r9, _TIF_WORK_MASK | ||
608 | bz restore_regs | ||
609 | |||
610 | ; --- (Slow Path #1) task preemption --- | ||
611 | bbit0 r9, TIF_NEED_RESCHED, .Lchk_pend_signals | ||
612 | mov blink, resume_user_mode_begin ; tail-call to U mode ret chks | ||
613 | b @schedule ; BTST+Bnz causes relo error in link | ||
614 | |||
615 | .Lchk_pend_signals: | ||
616 | IRQ_ENABLE r10 | ||
617 | |||
618 | ; --- (Slow Path #2) pending signal --- | ||
619 | mov r0, sp ; pt_regs for arg to do_signal()/do_notify_resume() | ||
620 | |||
621 | bbit0 r9, TIF_SIGPENDING, .Lchk_notify_resume | ||
622 | |||
623 | ; Normal Trap/IRQ entry only saves Scratch (caller-saved) regs | ||
624 | ; in pt_reg since the "C" ABI (kernel code) will automatically | ||
625 | ; save/restore callee-saved regs. | ||
626 | ; | ||
627 | ; However, here we need to explicitly save callee regs because | ||
628 | ; (i) If this signal causes coredump - full regfile needed | ||
629 | ; (ii) If signal is SIGTRAP/SIGSTOP, task is being traced thus | ||
630 | ; tracer might call PEEKUSR(CALLEE reg) | ||
631 | ; | ||
632 | ; NOTE: SP will grow up by size of CALLEE Reg-File | ||
633 | SAVE_CALLEE_SAVED_USER ; clobbers r12 | ||
634 | |||
635 | ; save location of saved Callee Regs @ thread_struct->callee | ||
636 | GET_CURR_TASK_FIELD_PTR TASK_THREAD, r10 | ||
637 | st sp, [r10, THREAD_CALLEE_REG] | ||
638 | |||
639 | bl @do_signal | ||
640 | |||
641 | ; Ideally we want to discard the Callee reg above, however if this was | ||
642 | ; a tracing signal, tracer could have done a POKEUSR(CALLEE reg) | ||
643 | RESTORE_CALLEE_SAVED_USER | ||
644 | |||
645 | b resume_user_mode_begin ; loop back to start of U mode ret | ||
646 | |||
647 | ; --- (Slow Path #3) notify_resume --- | ||
648 | .Lchk_notify_resume: | ||
649 | btst r9, TIF_NOTIFY_RESUME | ||
650 | blnz @do_notify_resume | ||
651 | b resume_user_mode_begin ; unconditionally back to U mode ret chks | ||
652 | ; for single exit point from this block | ||
653 | |||
654 | #ifdef CONFIG_PREEMPT | ||
655 | |||
656 | resume_kernel_mode: | ||
657 | |||
658 | ; Can't preempt if preemption disabled | ||
659 | GET_CURR_THR_INFO_FROM_SP r10 | ||
660 | ld r8, [r10, THREAD_INFO_PREEMPT_COUNT] | ||
661 | brne r8, 0, restore_regs | ||
662 | |||
663 | ; check if this task's NEED_RESCHED flag set | ||
664 | ld r9, [r10, THREAD_INFO_FLAGS] | ||
665 | bbit0 r9, TIF_NEED_RESCHED, restore_regs | ||
666 | |||
667 | IRQ_DISABLE r9 | ||
668 | |||
669 | ; Invoke PREEMPTION | ||
670 | bl preempt_schedule_irq | ||
671 | |||
672 | ; preempt_schedule_irq() always returns with IRQ disabled | ||
673 | #endif | ||
674 | |||
675 | ; fall through | ||
676 | |||
677 | ;############# Return from Intr/Excp/Trap (ARC Specifics) ############## | ||
678 | ; | ||
679 | ; Restore the saved sys context (common exit-path for EXCPN/IRQ/Trap) | ||
680 | ; IRQ shd definitely not happen between now and rtie | ||
681 | |||
682 | restore_regs : | ||
683 | |||
684 | ; Disable Interrupts while restoring reg-file back | ||
685 | ; XXX can this be optimised out | ||
686 | IRQ_DISABLE_SAVE r9, r10 ;@r10 has prisitine (pre-disable) copy | ||
687 | |||
688 | #ifdef CONFIG_ARC_CURR_IN_REG | ||
689 | ; Restore User R25 | ||
690 | ; Earlier this used to be only for returning to user mode | ||
691 | ; However with 2 levels of IRQ this can also happen even if | ||
692 | ; in kernel mode | ||
693 | ld r9, [sp, PT_sp] | ||
694 | brhs r9, VMALLOC_START, 8f | ||
695 | RESTORE_USER_R25 | ||
696 | 8: | ||
697 | #endif | ||
698 | |||
699 | ; Restore REG File. In case multiple Events outstanding, | ||
700 | ; use the same priorty as rtie: EXCPN, L2 IRQ, L1 IRQ, None | ||
701 | ; Note that we use realtime STATUS32 (not pt_regs->status32) to | ||
702 | ; decide that. | ||
703 | |||
704 | ; if Returning from Exception | ||
705 | bbit0 r10, STATUS_AE_BIT, not_exception | ||
706 | RESTORE_ALL_SYS | ||
707 | rtie | ||
708 | |||
709 | ; Not Exception so maybe Interrupts (Level 1 or 2) | ||
710 | |||
711 | not_exception: | ||
712 | |||
713 | #ifdef CONFIG_ARC_COMPACT_IRQ_LEVELS | ||
714 | |||
715 | bbit0 r10, STATUS_A2_BIT, not_level2_interrupt | ||
716 | |||
717 | ;------------------------------------------------------------------ | ||
718 | ; if L2 IRQ interrupted a L1 ISR, we'd disbaled preemption earlier | ||
719 | ; so that sched doesnt move to new task, causing L1 to be delayed | ||
720 | ; undeterministically. Now that we've achieved that, lets reset | ||
721 | ; things to what they were, before returning from L2 context | ||
722 | ;---------------------------------------------------------------- | ||
723 | |||
724 | ldw r9, [sp, PT_orig_r8] ; get orig_r8 to make sure it is | ||
725 | brne r9, orig_r8_IS_IRQ2, 149f ; infact a L2 ISR ret path | ||
726 | |||
727 | ld r9, [sp, PT_status32] ; get statu32_l2 (saved in pt_regs) | ||
728 | bbit0 r9, STATUS_A1_BIT, 149f ; L1 not active when L2 IRQ, so normal | ||
729 | |||
730 | ; A1 is set in status32_l2 | ||
731 | ; decrement thread_info->preempt_count (re-enable preemption) | ||
732 | GET_CURR_THR_INFO_FROM_SP r10 | ||
733 | ld r9, [r10, THREAD_INFO_PREEMPT_COUNT] | ||
734 | |||
735 | ; paranoid check, given A1 was active when A2 happened, preempt count | ||
736 | ; must not be 0 beccause we would have incremented it. | ||
737 | ; If this does happen we simply HALT as it means a BUG !!! | ||
738 | cmp r9, 0 | ||
739 | bnz 2f | ||
740 | flag 1 | ||
741 | |||
742 | 2: | ||
743 | sub r9, r9, 1 | ||
744 | st r9, [r10, THREAD_INFO_PREEMPT_COUNT] | ||
745 | |||
746 | 149: | ||
747 | ;return from level 2 | ||
748 | RESTORE_ALL_INT2 | ||
749 | debug_marker_l2: | ||
750 | rtie | ||
751 | |||
752 | not_level2_interrupt: | ||
753 | |||
754 | #endif | ||
755 | |||
756 | bbit0 r10, STATUS_A1_BIT, not_level1_interrupt | ||
757 | |||
758 | ;return from level 1 | ||
759 | |||
760 | RESTORE_ALL_INT1 | ||
761 | debug_marker_l1: | ||
762 | rtie | ||
763 | |||
764 | not_level1_interrupt: | ||
765 | |||
766 | ;this case is for syscalls or Exceptions (with fake rtie) | ||
767 | |||
768 | RESTORE_ALL_SYS | ||
769 | debug_marker_syscall: | ||
770 | rtie | ||
771 | |||
772 | ARC_EXIT ret_from_exception | ||
773 | |||
774 | ARC_ENTRY ret_from_fork | ||
775 | ; when the forked child comes here from the __switch_to function | ||
776 | ; r0 has the last task pointer. | ||
777 | ; put last task in scheduler queue | ||
778 | bl @schedule_tail | ||
779 | |||
780 | ; If kernel thread, jump to it's entry-point | ||
781 | ld r9, [sp, PT_status32] | ||
782 | brne r9, 0, 1f | ||
783 | |||
784 | jl.d [r14] | ||
785 | mov r0, r13 ; arg to payload | ||
786 | |||
787 | 1: | ||
788 | ; special case of kernel_thread entry point returning back due to | ||
789 | ; kernel_execve() - pretend return from syscall to ret to userland | ||
790 | b ret_from_exception | ||
791 | ARC_EXIT ret_from_fork | ||
792 | |||
793 | ;################### Special Sys Call Wrappers ########################## | ||
794 | |||
795 | ; TBD: call do_fork directly from here | ||
796 | ARC_ENTRY sys_fork_wrapper | ||
797 | SAVE_CALLEE_SAVED_USER | ||
798 | bl @sys_fork | ||
799 | DISCARD_CALLEE_SAVED_USER | ||
800 | |||
801 | GET_CURR_THR_INFO_FLAGS r10 | ||
802 | btst r10, TIF_SYSCALL_TRACE | ||
803 | bnz tracesys_exit | ||
804 | |||
805 | b ret_from_system_call | ||
806 | ARC_EXIT sys_fork_wrapper | ||
807 | |||
808 | ARC_ENTRY sys_vfork_wrapper | ||
809 | SAVE_CALLEE_SAVED_USER | ||
810 | bl @sys_vfork | ||
811 | DISCARD_CALLEE_SAVED_USER | ||
812 | |||
813 | GET_CURR_THR_INFO_FLAGS r10 | ||
814 | btst r10, TIF_SYSCALL_TRACE | ||
815 | bnz tracesys_exit | ||
816 | |||
817 | b ret_from_system_call | ||
818 | ARC_EXIT sys_vfork_wrapper | ||
819 | |||
820 | ARC_ENTRY sys_clone_wrapper | ||
821 | SAVE_CALLEE_SAVED_USER | ||
822 | bl @sys_clone | ||
823 | DISCARD_CALLEE_SAVED_USER | ||
824 | |||
825 | GET_CURR_THR_INFO_FLAGS r10 | ||
826 | btst r10, TIF_SYSCALL_TRACE | ||
827 | bnz tracesys_exit | ||
828 | |||
829 | b ret_from_system_call | ||
830 | ARC_EXIT sys_clone_wrapper | ||
831 | |||
832 | #ifdef CONFIG_ARC_DW2_UNWIND | ||
833 | ; Workaround for bug 94179 (STAR ): | ||
834 | ; Despite -fasynchronous-unwind-tables, linker is not making dwarf2 unwinder | ||
835 | ; section (.debug_frame) as loadable. So we force it here. | ||
836 | ; This also fixes STAR 9000487933 where the prev-workaround (objcopy --setflag) | ||
837 | ; would not work after a clean build due to kernel build system dependencies. | ||
838 | .section .debug_frame, "wa",@progbits | ||
839 | #endif | ||
diff --git a/arch/arc/kernel/fpu.c b/arch/arc/kernel/fpu.c new file mode 100644 index 000000000000..f352e512cbd1 --- /dev/null +++ b/arch/arc/kernel/fpu.c | |||
@@ -0,0 +1,55 @@ | |||
1 | /* | ||
2 | * fpu.c - save/restore of Floating Point Unit Registers on task switch | ||
3 | * | ||
4 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | */ | ||
10 | |||
11 | #include <linux/sched.h> | ||
12 | #include <asm/switch_to.h> | ||
13 | |||
14 | /* | ||
15 | * To save/restore FPU regs, simplest scheme would use LR/SR insns. | ||
16 | * However since SR serializes the pipeline, an alternate "hack" can be used | ||
17 | * which uses the FPU Exchange insn (DEXCL) to r/w FPU regs. | ||
18 | * | ||
19 | * Store to 64bit dpfp1 reg from a pair of core regs: | ||
20 | * dexcl1 0, r1, r0 ; where r1:r0 is the 64 bit val | ||
21 | * | ||
22 | * Read from dpfp1 into pair of core regs (w/o clobbering dpfp1) | ||
23 | * mov_s r3, 0 | ||
24 | * daddh11 r1, r3, r3 ; get "hi" into r1 (dpfp1 unchanged) | ||
25 | * dexcl1 r0, r1, r3 ; get "low" into r0 (dpfp1 low clobbered) | ||
26 | * dexcl1 0, r1, r0 ; restore dpfp1 to orig value | ||
27 | * | ||
28 | * However we can tweak the read, so that read-out of outgoing task's FPU regs | ||
29 | * and write of incoming task's regs happen in one shot. So all the work is | ||
30 | * done before context switch | ||
31 | */ | ||
32 | |||
33 | void fpu_save_restore(struct task_struct *prev, struct task_struct *next) | ||
34 | { | ||
35 | unsigned int *saveto = &prev->thread.fpu.aux_dpfp[0].l; | ||
36 | unsigned int *readfrom = &next->thread.fpu.aux_dpfp[0].l; | ||
37 | |||
38 | const unsigned int zero = 0; | ||
39 | |||
40 | __asm__ __volatile__( | ||
41 | "daddh11 %0, %2, %2\n" | ||
42 | "dexcl1 %1, %3, %4\n" | ||
43 | : "=&r" (*(saveto + 1)), /* early clobber must here */ | ||
44 | "=&r" (*(saveto)) | ||
45 | : "r" (zero), "r" (*(readfrom + 1)), "r" (*(readfrom)) | ||
46 | ); | ||
47 | |||
48 | __asm__ __volatile__( | ||
49 | "daddh22 %0, %2, %2\n" | ||
50 | "dexcl2 %1, %3, %4\n" | ||
51 | : "=&r"(*(saveto + 3)), /* early clobber must here */ | ||
52 | "=&r"(*(saveto + 2)) | ||
53 | : "r" (zero), "r" (*(readfrom + 3)), "r" (*(readfrom + 2)) | ||
54 | ); | ||
55 | } | ||
diff --git a/arch/arc/kernel/head.S b/arch/arc/kernel/head.S new file mode 100644 index 000000000000..006dec3fc353 --- /dev/null +++ b/arch/arc/kernel/head.S | |||
@@ -0,0 +1,111 @@ | |||
1 | /* | ||
2 | * ARC CPU startup Code | ||
3 | * | ||
4 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | * | ||
10 | * Vineetg: Dec 2007 | ||
11 | * -Check if we are running on Simulator or on real hardware | ||
12 | * to skip certain things during boot on simulator | ||
13 | */ | ||
14 | |||
15 | #include <asm/asm-offsets.h> | ||
16 | #include <asm/entry.h> | ||
17 | #include <linux/linkage.h> | ||
18 | #include <asm/arcregs.h> | ||
19 | |||
20 | .cpu A7 | ||
21 | |||
22 | .section .init.text, "ax",@progbits | ||
23 | .type stext, @function | ||
24 | .globl stext | ||
25 | stext: | ||
26 | ;------------------------------------------------------------------- | ||
27 | ; Don't clobber r0-r4 yet. It might have bootloader provided info | ||
28 | ;------------------------------------------------------------------- | ||
29 | |||
30 | #ifdef CONFIG_SMP | ||
31 | ; Only Boot (Master) proceeds. Others wait in platform dependent way | ||
32 | ; IDENTITY Reg [ 3 2 1 0 ] | ||
33 | ; (cpu-id) ^^^ => Zero for UP ARC700 | ||
34 | ; => #Core-ID if SMP (Master 0) | ||
35 | GET_CPU_ID r5 | ||
36 | cmp r5, 0 | ||
37 | jnz arc_platform_smp_wait_to_boot | ||
38 | #endif | ||
39 | ; Clear BSS before updating any globals | ||
40 | ; XXX: use ZOL here | ||
41 | mov r5, __bss_start | ||
42 | mov r6, __bss_stop | ||
43 | 1: | ||
44 | st.ab 0, [r5,4] | ||
45 | brlt r5, r6, 1b | ||
46 | |||
47 | #ifdef CONFIG_CMDLINE_UBOOT | ||
48 | ; support for bootloader provided cmdline | ||
49 | ; If cmdline passed by u-boot, then | ||
50 | ; r0 = 1 (because ATAGS parsing, now retired, used to use 0) | ||
51 | ; r1 = magic number (board identity) | ||
52 | ; r2 = addr of cmdline string (somewhere in memory/flash) | ||
53 | |||
54 | brne r0, 1, .Lother_bootup_chores ; u-boot didn't pass cmdline | ||
55 | breq r2, 0, .Lother_bootup_chores ; or cmdline is NULL | ||
56 | |||
57 | mov r5, @command_line | ||
58 | 1: | ||
59 | ldb.ab r6, [r2, 1] | ||
60 | breq r6, 0, .Lother_bootup_chores | ||
61 | b.d 1b | ||
62 | stb.ab r6, [r5, 1] | ||
63 | #endif | ||
64 | |||
65 | .Lother_bootup_chores: | ||
66 | |||
67 | ; Identify if running on ISS vs Silicon | ||
68 | ; IDENTITY Reg [ 3 2 1 0 ] | ||
69 | ; (chip-id) ^^^^^ ==> 0xffff for ISS | ||
70 | lr r0, [identity] | ||
71 | lsr r3, r0, 16 | ||
72 | cmp r3, 0xffff | ||
73 | mov.z r4, 0 | ||
74 | mov.nz r4, 1 | ||
75 | st r4, [@running_on_hw] | ||
76 | |||
77 | ; setup "current" tsk and optionally cache it in dedicated r25 | ||
78 | mov r9, @init_task | ||
79 | SET_CURR_TASK_ON_CPU r9, r0 ; r9 = tsk, r0 = scratch | ||
80 | |||
81 | ; setup stack (fp, sp) | ||
82 | mov fp, 0 | ||
83 | |||
84 | ; tsk->thread_info is really a PAGE, whose bottom hoists stack | ||
85 | GET_TSK_STACK_BASE r9, sp ; r9 = tsk, sp = stack base(output) | ||
86 | |||
87 | j start_kernel ; "C" entry point | ||
88 | |||
89 | #ifdef CONFIG_SMP | ||
90 | ;---------------------------------------------------------------- | ||
91 | ; First lines of code run by secondary before jumping to 'C' | ||
92 | ;---------------------------------------------------------------- | ||
93 | .section .init.text, "ax",@progbits | ||
94 | .type first_lines_of_secondary, @function | ||
95 | .globl first_lines_of_secondary | ||
96 | |||
97 | first_lines_of_secondary: | ||
98 | |||
99 | ; setup per-cpu idle task as "current" on this CPU | ||
100 | ld r0, [@secondary_idle_tsk] | ||
101 | SET_CURR_TASK_ON_CPU r0, r1 | ||
102 | |||
103 | ; setup stack (fp, sp) | ||
104 | mov fp, 0 | ||
105 | |||
106 | ; set it's stack base to tsk->thread_info bottom | ||
107 | GET_TSK_STACK_BASE r0, sp | ||
108 | |||
109 | j start_kernel_secondary | ||
110 | |||
111 | #endif | ||
diff --git a/arch/arc/kernel/irq.c b/arch/arc/kernel/irq.c new file mode 100644 index 000000000000..551c10dff481 --- /dev/null +++ b/arch/arc/kernel/irq.c | |||
@@ -0,0 +1,273 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2011-12 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | */ | ||
9 | |||
10 | #include <linux/interrupt.h> | ||
11 | #include <linux/module.h> | ||
12 | #include <linux/of.h> | ||
13 | #include <linux/irqdomain.h> | ||
14 | #include <asm/sections.h> | ||
15 | #include <asm/irq.h> | ||
16 | #include <asm/mach_desc.h> | ||
17 | |||
18 | /* | ||
19 | * Early Hardware specific Interrupt setup | ||
20 | * -Called very early (start_kernel -> setup_arch -> setup_processor) | ||
21 | * -Platform Independent (must for any ARC700) | ||
22 | * -Needed for each CPU (hence not foldable into init_IRQ) | ||
23 | * | ||
24 | * what it does ? | ||
25 | * -setup Vector Table Base Reg - in case Linux not linked at 0x8000_0000 | ||
26 | * -Disable all IRQs (on CPU side) | ||
27 | * -Optionally, setup the High priority Interrupts as Level 2 IRQs | ||
28 | */ | ||
29 | void __init arc_init_IRQ(void) | ||
30 | { | ||
31 | int level_mask = 0; | ||
32 | |||
33 | write_aux_reg(AUX_INTR_VEC_BASE, _int_vec_base_lds); | ||
34 | |||
35 | /* Disable all IRQs: enable them as devices request */ | ||
36 | write_aux_reg(AUX_IENABLE, 0); | ||
37 | |||
38 | /* setup any high priority Interrupts (Level2 in ARCompact jargon) */ | ||
39 | #ifdef CONFIG_ARC_IRQ3_LV2 | ||
40 | level_mask |= (1 << 3); | ||
41 | #endif | ||
42 | #ifdef CONFIG_ARC_IRQ5_LV2 | ||
43 | level_mask |= (1 << 5); | ||
44 | #endif | ||
45 | #ifdef CONFIG_ARC_IRQ6_LV2 | ||
46 | level_mask |= (1 << 6); | ||
47 | #endif | ||
48 | |||
49 | if (level_mask) { | ||
50 | pr_info("Level-2 interrupts bitset %x\n", level_mask); | ||
51 | write_aux_reg(AUX_IRQ_LEV, level_mask); | ||
52 | } | ||
53 | } | ||
54 | |||
55 | /* | ||
56 | * ARC700 core includes a simple on-chip intc supporting | ||
57 | * -per IRQ enable/disable | ||
58 | * -2 levels of interrupts (high/low) | ||
59 | * -all interrupts being level triggered | ||
60 | * | ||
61 | * To reduce platform code, we assume all IRQs directly hooked-up into intc. | ||
62 | * Platforms with external intc, hence cascaded IRQs, are free to over-ride | ||
63 | * below, per IRQ. | ||
64 | */ | ||
65 | |||
66 | static void arc_mask_irq(struct irq_data *data) | ||
67 | { | ||
68 | arch_mask_irq(data->irq); | ||
69 | } | ||
70 | |||
71 | static void arc_unmask_irq(struct irq_data *data) | ||
72 | { | ||
73 | arch_unmask_irq(data->irq); | ||
74 | } | ||
75 | |||
76 | static struct irq_chip onchip_intc = { | ||
77 | .name = "ARC In-core Intc", | ||
78 | .irq_mask = arc_mask_irq, | ||
79 | .irq_unmask = arc_unmask_irq, | ||
80 | }; | ||
81 | |||
82 | static int arc_intc_domain_map(struct irq_domain *d, unsigned int irq, | ||
83 | irq_hw_number_t hw) | ||
84 | { | ||
85 | if (irq == TIMER0_IRQ) | ||
86 | irq_set_chip_and_handler(irq, &onchip_intc, handle_percpu_irq); | ||
87 | else | ||
88 | irq_set_chip_and_handler(irq, &onchip_intc, handle_level_irq); | ||
89 | |||
90 | return 0; | ||
91 | } | ||
92 | |||
93 | static const struct irq_domain_ops arc_intc_domain_ops = { | ||
94 | .xlate = irq_domain_xlate_onecell, | ||
95 | .map = arc_intc_domain_map, | ||
96 | }; | ||
97 | |||
98 | static struct irq_domain *root_domain; | ||
99 | |||
100 | void __init init_onchip_IRQ(void) | ||
101 | { | ||
102 | struct device_node *intc = NULL; | ||
103 | |||
104 | intc = of_find_compatible_node(NULL, NULL, "snps,arc700-intc"); | ||
105 | if(!intc) | ||
106 | panic("DeviceTree Missing incore intc\n"); | ||
107 | |||
108 | root_domain = irq_domain_add_legacy(intc, NR_IRQS, 0, 0, | ||
109 | &arc_intc_domain_ops, NULL); | ||
110 | |||
111 | if (!root_domain) | ||
112 | panic("root irq domain not avail\n"); | ||
113 | |||
114 | /* with this we don't need to export root_domain */ | ||
115 | irq_set_default_host(root_domain); | ||
116 | } | ||
117 | |||
118 | /* | ||
119 | * Late Interrupt system init called from start_kernel for Boot CPU only | ||
120 | * | ||
121 | * Since slab must already be initialized, platforms can start doing any | ||
122 | * needed request_irq( )s | ||
123 | */ | ||
124 | void __init init_IRQ(void) | ||
125 | { | ||
126 | init_onchip_IRQ(); | ||
127 | |||
128 | /* Any external intc can be setup here */ | ||
129 | if (machine_desc->init_irq) | ||
130 | machine_desc->init_irq(); | ||
131 | |||
132 | #ifdef CONFIG_SMP | ||
133 | /* Master CPU can initialize it's side of IPI */ | ||
134 | if (machine_desc->init_smp) | ||
135 | machine_desc->init_smp(smp_processor_id()); | ||
136 | #endif | ||
137 | } | ||
138 | |||
139 | /* | ||
140 | * "C" Entry point for any ARC ISR, called from low level vector handler | ||
141 | * @irq is the vector number read from ICAUSE reg of on-chip intc | ||
142 | */ | ||
143 | void arch_do_IRQ(unsigned int irq, struct pt_regs *regs) | ||
144 | { | ||
145 | struct pt_regs *old_regs = set_irq_regs(regs); | ||
146 | |||
147 | irq_enter(); | ||
148 | generic_handle_irq(irq); | ||
149 | irq_exit(); | ||
150 | set_irq_regs(old_regs); | ||
151 | } | ||
152 | |||
153 | int __init get_hw_config_num_irq(void) | ||
154 | { | ||
155 | uint32_t val = read_aux_reg(ARC_REG_VECBASE_BCR); | ||
156 | |||
157 | switch (val & 0x03) { | ||
158 | case 0: | ||
159 | return 16; | ||
160 | case 1: | ||
161 | return 32; | ||
162 | case 2: | ||
163 | return 8; | ||
164 | default: | ||
165 | return 0; | ||
166 | } | ||
167 | |||
168 | return 0; | ||
169 | } | ||
170 | |||
171 | /* | ||
172 | * arch_local_irq_enable - Enable interrupts. | ||
173 | * | ||
174 | * 1. Explicitly called to re-enable interrupts | ||
175 | * 2. Implicitly called from spin_unlock_irq, write_unlock_irq etc | ||
176 | * which maybe in hard ISR itself | ||
177 | * | ||
178 | * Semantics of this function change depending on where it is called from: | ||
179 | * | ||
180 | * -If called from hard-ISR, it must not invert interrupt priorities | ||
181 | * e.g. suppose TIMER is high priority (Level 2) IRQ | ||
182 | * Time hard-ISR, timer_interrupt( ) calls spin_unlock_irq several times. | ||
183 | * Here local_irq_enable( ) shd not re-enable lower priority interrupts | ||
184 | * -If called from soft-ISR, it must re-enable all interrupts | ||
185 | * soft ISR are low prioity jobs which can be very slow, thus all IRQs | ||
186 | * must be enabled while they run. | ||
187 | * Now hardware context wise we may still be in L2 ISR (not done rtie) | ||
188 | * still we must re-enable both L1 and L2 IRQs | ||
189 | * Another twist is prev scenario with flow being | ||
190 | * L1 ISR ==> interrupted by L2 ISR ==> L2 soft ISR | ||
191 | * here we must not re-enable Ll as prev Ll Interrupt's h/w context will get | ||
192 | * over-written (this is deficiency in ARC700 Interrupt mechanism) | ||
193 | */ | ||
194 | |||
195 | #ifdef CONFIG_ARC_COMPACT_IRQ_LEVELS /* Complex version for 2 IRQ levels */ | ||
196 | |||
197 | void arch_local_irq_enable(void) | ||
198 | { | ||
199 | |||
200 | unsigned long flags; | ||
201 | flags = arch_local_save_flags(); | ||
202 | |||
203 | /* Allow both L1 and L2 at the onset */ | ||
204 | flags |= (STATUS_E1_MASK | STATUS_E2_MASK); | ||
205 | |||
206 | /* Called from hard ISR (between irq_enter and irq_exit) */ | ||
207 | if (in_irq()) { | ||
208 | |||
209 | /* If in L2 ISR, don't re-enable any further IRQs as this can | ||
210 | * cause IRQ priorities to get upside down. e.g. it could allow | ||
211 | * L1 be taken while in L2 hard ISR which is wrong not only in | ||
212 | * theory, it can also cause the dreaded L1-L2-L1 scenario | ||
213 | */ | ||
214 | if (flags & STATUS_A2_MASK) | ||
215 | flags &= ~(STATUS_E1_MASK | STATUS_E2_MASK); | ||
216 | |||
217 | /* Even if in L1 ISR, allowe Higher prio L2 IRQs */ | ||
218 | else if (flags & STATUS_A1_MASK) | ||
219 | flags &= ~(STATUS_E1_MASK); | ||
220 | } | ||
221 | |||
222 | /* called from soft IRQ, ideally we want to re-enable all levels */ | ||
223 | |||
224 | else if (in_softirq()) { | ||
225 | |||
226 | /* However if this is case of L1 interrupted by L2, | ||
227 | * re-enabling both may cause whaco L1-L2-L1 scenario | ||
228 | * because ARC700 allows level 1 to interrupt an active L2 ISR | ||
229 | * Thus we disable both | ||
230 | * However some code, executing in soft ISR wants some IRQs | ||
231 | * to be enabled so we re-enable L2 only | ||
232 | * | ||
233 | * How do we determine L1 intr by L2 | ||
234 | * -A2 is set (means in L2 ISR) | ||
235 | * -E1 is set in this ISR's pt_regs->status32 which is | ||
236 | * saved copy of status32_l2 when l2 ISR happened | ||
237 | */ | ||
238 | struct pt_regs *pt = get_irq_regs(); | ||
239 | if ((flags & STATUS_A2_MASK) && pt && | ||
240 | (pt->status32 & STATUS_A1_MASK)) { | ||
241 | /*flags &= ~(STATUS_E1_MASK | STATUS_E2_MASK); */ | ||
242 | flags &= ~(STATUS_E1_MASK); | ||
243 | } | ||
244 | } | ||
245 | |||
246 | arch_local_irq_restore(flags); | ||
247 | } | ||
248 | |||
249 | #else /* ! CONFIG_ARC_COMPACT_IRQ_LEVELS */ | ||
250 | |||
251 | /* | ||
252 | * Simpler version for only 1 level of interrupt | ||
253 | * Here we only Worry about Level 1 Bits | ||
254 | */ | ||
255 | void arch_local_irq_enable(void) | ||
256 | { | ||
257 | unsigned long flags; | ||
258 | |||
259 | /* | ||
260 | * ARC IDE Drivers tries to re-enable interrupts from hard-isr | ||
261 | * context which is simply wrong | ||
262 | */ | ||
263 | if (in_irq()) { | ||
264 | WARN_ONCE(1, "IRQ enabled from hard-isr"); | ||
265 | return; | ||
266 | } | ||
267 | |||
268 | flags = arch_local_save_flags(); | ||
269 | flags |= (STATUS_E1_MASK | STATUS_E2_MASK); | ||
270 | arch_local_irq_restore(flags); | ||
271 | } | ||
272 | #endif | ||
273 | EXPORT_SYMBOL(arch_local_irq_enable); | ||
diff --git a/arch/arc/kernel/kgdb.c b/arch/arc/kernel/kgdb.c new file mode 100644 index 000000000000..2888ba5be47e --- /dev/null +++ b/arch/arc/kernel/kgdb.c | |||
@@ -0,0 +1,205 @@ | |||
1 | /* | ||
2 | * kgdb support for ARC | ||
3 | * | ||
4 | * Copyright (C) 2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | */ | ||
10 | |||
11 | #include <linux/kgdb.h> | ||
12 | #include <asm/disasm.h> | ||
13 | #include <asm/cacheflush.h> | ||
14 | |||
15 | static void to_gdb_regs(unsigned long *gdb_regs, struct pt_regs *kernel_regs, | ||
16 | struct callee_regs *cregs) | ||
17 | { | ||
18 | int regno; | ||
19 | |||
20 | for (regno = 0; regno <= 26; regno++) | ||
21 | gdb_regs[_R0 + regno] = get_reg(regno, kernel_regs, cregs); | ||
22 | |||
23 | for (regno = 27; regno < GDB_MAX_REGS; regno++) | ||
24 | gdb_regs[regno] = 0; | ||
25 | |||
26 | gdb_regs[_FP] = kernel_regs->fp; | ||
27 | gdb_regs[__SP] = kernel_regs->sp; | ||
28 | gdb_regs[_BLINK] = kernel_regs->blink; | ||
29 | gdb_regs[_RET] = kernel_regs->ret; | ||
30 | gdb_regs[_STATUS32] = kernel_regs->status32; | ||
31 | gdb_regs[_LP_COUNT] = kernel_regs->lp_count; | ||
32 | gdb_regs[_LP_END] = kernel_regs->lp_end; | ||
33 | gdb_regs[_LP_START] = kernel_regs->lp_start; | ||
34 | gdb_regs[_BTA] = kernel_regs->bta; | ||
35 | gdb_regs[_STOP_PC] = kernel_regs->ret; | ||
36 | } | ||
37 | |||
38 | static void from_gdb_regs(unsigned long *gdb_regs, struct pt_regs *kernel_regs, | ||
39 | struct callee_regs *cregs) | ||
40 | { | ||
41 | int regno; | ||
42 | |||
43 | for (regno = 0; regno <= 26; regno++) | ||
44 | set_reg(regno, gdb_regs[regno + _R0], kernel_regs, cregs); | ||
45 | |||
46 | kernel_regs->fp = gdb_regs[_FP]; | ||
47 | kernel_regs->sp = gdb_regs[__SP]; | ||
48 | kernel_regs->blink = gdb_regs[_BLINK]; | ||
49 | kernel_regs->ret = gdb_regs[_RET]; | ||
50 | kernel_regs->status32 = gdb_regs[_STATUS32]; | ||
51 | kernel_regs->lp_count = gdb_regs[_LP_COUNT]; | ||
52 | kernel_regs->lp_end = gdb_regs[_LP_END]; | ||
53 | kernel_regs->lp_start = gdb_regs[_LP_START]; | ||
54 | kernel_regs->bta = gdb_regs[_BTA]; | ||
55 | } | ||
56 | |||
57 | |||
58 | void pt_regs_to_gdb_regs(unsigned long *gdb_regs, struct pt_regs *kernel_regs) | ||
59 | { | ||
60 | to_gdb_regs(gdb_regs, kernel_regs, (struct callee_regs *) | ||
61 | current->thread.callee_reg); | ||
62 | } | ||
63 | |||
64 | void gdb_regs_to_pt_regs(unsigned long *gdb_regs, struct pt_regs *kernel_regs) | ||
65 | { | ||
66 | from_gdb_regs(gdb_regs, kernel_regs, (struct callee_regs *) | ||
67 | current->thread.callee_reg); | ||
68 | } | ||
69 | |||
70 | void sleeping_thread_to_gdb_regs(unsigned long *gdb_regs, | ||
71 | struct task_struct *task) | ||
72 | { | ||
73 | if (task) | ||
74 | to_gdb_regs(gdb_regs, task_pt_regs(task), | ||
75 | (struct callee_regs *) task->thread.callee_reg); | ||
76 | } | ||
77 | |||
78 | struct single_step_data_t { | ||
79 | uint16_t opcode[2]; | ||
80 | unsigned long address[2]; | ||
81 | int is_branch; | ||
82 | int armed; | ||
83 | } single_step_data; | ||
84 | |||
85 | static void undo_single_step(struct pt_regs *regs) | ||
86 | { | ||
87 | if (single_step_data.armed) { | ||
88 | int i; | ||
89 | |||
90 | for (i = 0; i < (single_step_data.is_branch ? 2 : 1); i++) { | ||
91 | memcpy((void *) single_step_data.address[i], | ||
92 | &single_step_data.opcode[i], | ||
93 | BREAK_INSTR_SIZE); | ||
94 | |||
95 | flush_icache_range(single_step_data.address[i], | ||
96 | single_step_data.address[i] + | ||
97 | BREAK_INSTR_SIZE); | ||
98 | } | ||
99 | single_step_data.armed = 0; | ||
100 | } | ||
101 | } | ||
102 | |||
103 | static void place_trap(unsigned long address, void *save) | ||
104 | { | ||
105 | memcpy(save, (void *) address, BREAK_INSTR_SIZE); | ||
106 | memcpy((void *) address, &arch_kgdb_ops.gdb_bpt_instr, | ||
107 | BREAK_INSTR_SIZE); | ||
108 | flush_icache_range(address, address + BREAK_INSTR_SIZE); | ||
109 | } | ||
110 | |||
111 | static void do_single_step(struct pt_regs *regs) | ||
112 | { | ||
113 | single_step_data.is_branch = disasm_next_pc((unsigned long) | ||
114 | regs->ret, regs, (struct callee_regs *) | ||
115 | current->thread.callee_reg, | ||
116 | &single_step_data.address[0], | ||
117 | &single_step_data.address[1]); | ||
118 | |||
119 | place_trap(single_step_data.address[0], &single_step_data.opcode[0]); | ||
120 | |||
121 | if (single_step_data.is_branch) { | ||
122 | place_trap(single_step_data.address[1], | ||
123 | &single_step_data.opcode[1]); | ||
124 | } | ||
125 | |||
126 | single_step_data.armed++; | ||
127 | } | ||
128 | |||
129 | int kgdb_arch_handle_exception(int e_vector, int signo, int err_code, | ||
130 | char *remcomInBuffer, char *remcomOutBuffer, | ||
131 | struct pt_regs *regs) | ||
132 | { | ||
133 | unsigned long addr; | ||
134 | char *ptr; | ||
135 | |||
136 | undo_single_step(regs); | ||
137 | |||
138 | switch (remcomInBuffer[0]) { | ||
139 | case 's': | ||
140 | case 'c': | ||
141 | ptr = &remcomInBuffer[1]; | ||
142 | if (kgdb_hex2long(&ptr, &addr)) | ||
143 | regs->ret = addr; | ||
144 | |||
145 | case 'D': | ||
146 | case 'k': | ||
147 | atomic_set(&kgdb_cpu_doing_single_step, -1); | ||
148 | |||
149 | if (remcomInBuffer[0] == 's') { | ||
150 | do_single_step(regs); | ||
151 | atomic_set(&kgdb_cpu_doing_single_step, | ||
152 | smp_processor_id()); | ||
153 | } | ||
154 | |||
155 | return 0; | ||
156 | } | ||
157 | return -1; | ||
158 | } | ||
159 | |||
160 | unsigned long kgdb_arch_pc(int exception, struct pt_regs *regs) | ||
161 | { | ||
162 | return instruction_pointer(regs); | ||
163 | } | ||
164 | |||
165 | int kgdb_arch_init(void) | ||
166 | { | ||
167 | single_step_data.armed = 0; | ||
168 | return 0; | ||
169 | } | ||
170 | |||
171 | void kgdb_trap(struct pt_regs *regs, int param) | ||
172 | { | ||
173 | /* trap_s 3 is used for breakpoints that overwrite existing | ||
174 | * instructions, while trap_s 4 is used for compiled breakpoints. | ||
175 | * | ||
176 | * with trap_s 3 breakpoints the original instruction needs to be | ||
177 | * restored and continuation needs to start at the location of the | ||
178 | * breakpoint. | ||
179 | * | ||
180 | * with trap_s 4 (compiled) breakpoints, continuation needs to | ||
181 | * start after the breakpoint. | ||
182 | */ | ||
183 | if (param == 3) | ||
184 | instruction_pointer(regs) -= BREAK_INSTR_SIZE; | ||
185 | |||
186 | kgdb_handle_exception(1, SIGTRAP, 0, regs); | ||
187 | } | ||
188 | |||
189 | void kgdb_arch_exit(void) | ||
190 | { | ||
191 | } | ||
192 | |||
193 | void kgdb_arch_set_pc(struct pt_regs *regs, unsigned long ip) | ||
194 | { | ||
195 | instruction_pointer(regs) = ip; | ||
196 | } | ||
197 | |||
198 | struct kgdb_arch arch_kgdb_ops = { | ||
199 | /* breakpoint instruction: TRAP_S 0x3 */ | ||
200 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
201 | .gdb_bpt_instr = {0x78, 0x7e}, | ||
202 | #else | ||
203 | .gdb_bpt_instr = {0x7e, 0x78}, | ||
204 | #endif | ||
205 | }; | ||
diff --git a/arch/arc/kernel/kprobes.c b/arch/arc/kernel/kprobes.c new file mode 100644 index 000000000000..3bfeacb674de --- /dev/null +++ b/arch/arc/kernel/kprobes.c | |||
@@ -0,0 +1,525 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <linux/types.h> | ||
10 | #include <linux/kprobes.h> | ||
11 | #include <linux/slab.h> | ||
12 | #include <linux/module.h> | ||
13 | #include <linux/kprobes.h> | ||
14 | #include <linux/kdebug.h> | ||
15 | #include <linux/sched.h> | ||
16 | #include <linux/uaccess.h> | ||
17 | #include <asm/cacheflush.h> | ||
18 | #include <asm/current.h> | ||
19 | #include <asm/disasm.h> | ||
20 | |||
21 | #define MIN_STACK_SIZE(addr) min((unsigned long)MAX_STACK_SIZE, \ | ||
22 | (unsigned long)current_thread_info() + THREAD_SIZE - (addr)) | ||
23 | |||
24 | DEFINE_PER_CPU(struct kprobe *, current_kprobe) = NULL; | ||
25 | DEFINE_PER_CPU(struct kprobe_ctlblk, kprobe_ctlblk); | ||
26 | |||
27 | int __kprobes arch_prepare_kprobe(struct kprobe *p) | ||
28 | { | ||
29 | /* Attempt to probe at unaligned address */ | ||
30 | if ((unsigned long)p->addr & 0x01) | ||
31 | return -EINVAL; | ||
32 | |||
33 | /* Address should not be in exception handling code */ | ||
34 | |||
35 | p->ainsn.is_short = is_short_instr((unsigned long)p->addr); | ||
36 | p->opcode = *p->addr; | ||
37 | |||
38 | return 0; | ||
39 | } | ||
40 | |||
41 | void __kprobes arch_arm_kprobe(struct kprobe *p) | ||
42 | { | ||
43 | *p->addr = UNIMP_S_INSTRUCTION; | ||
44 | |||
45 | flush_icache_range((unsigned long)p->addr, | ||
46 | (unsigned long)p->addr + sizeof(kprobe_opcode_t)); | ||
47 | } | ||
48 | |||
49 | void __kprobes arch_disarm_kprobe(struct kprobe *p) | ||
50 | { | ||
51 | *p->addr = p->opcode; | ||
52 | |||
53 | flush_icache_range((unsigned long)p->addr, | ||
54 | (unsigned long)p->addr + sizeof(kprobe_opcode_t)); | ||
55 | } | ||
56 | |||
57 | void __kprobes arch_remove_kprobe(struct kprobe *p) | ||
58 | { | ||
59 | arch_disarm_kprobe(p); | ||
60 | |||
61 | /* Can we remove the kprobe in the middle of kprobe handling? */ | ||
62 | if (p->ainsn.t1_addr) { | ||
63 | *(p->ainsn.t1_addr) = p->ainsn.t1_opcode; | ||
64 | |||
65 | flush_icache_range((unsigned long)p->ainsn.t1_addr, | ||
66 | (unsigned long)p->ainsn.t1_addr + | ||
67 | sizeof(kprobe_opcode_t)); | ||
68 | |||
69 | p->ainsn.t1_addr = NULL; | ||
70 | } | ||
71 | |||
72 | if (p->ainsn.t2_addr) { | ||
73 | *(p->ainsn.t2_addr) = p->ainsn.t2_opcode; | ||
74 | |||
75 | flush_icache_range((unsigned long)p->ainsn.t2_addr, | ||
76 | (unsigned long)p->ainsn.t2_addr + | ||
77 | sizeof(kprobe_opcode_t)); | ||
78 | |||
79 | p->ainsn.t2_addr = NULL; | ||
80 | } | ||
81 | } | ||
82 | |||
83 | static void __kprobes save_previous_kprobe(struct kprobe_ctlblk *kcb) | ||
84 | { | ||
85 | kcb->prev_kprobe.kp = kprobe_running(); | ||
86 | kcb->prev_kprobe.status = kcb->kprobe_status; | ||
87 | } | ||
88 | |||
89 | static void __kprobes restore_previous_kprobe(struct kprobe_ctlblk *kcb) | ||
90 | { | ||
91 | __get_cpu_var(current_kprobe) = kcb->prev_kprobe.kp; | ||
92 | kcb->kprobe_status = kcb->prev_kprobe.status; | ||
93 | } | ||
94 | |||
95 | static inline void __kprobes set_current_kprobe(struct kprobe *p) | ||
96 | { | ||
97 | __get_cpu_var(current_kprobe) = p; | ||
98 | } | ||
99 | |||
100 | static void __kprobes resume_execution(struct kprobe *p, unsigned long addr, | ||
101 | struct pt_regs *regs) | ||
102 | { | ||
103 | /* Remove the trap instructions inserted for single step and | ||
104 | * restore the original instructions | ||
105 | */ | ||
106 | if (p->ainsn.t1_addr) { | ||
107 | *(p->ainsn.t1_addr) = p->ainsn.t1_opcode; | ||
108 | |||
109 | flush_icache_range((unsigned long)p->ainsn.t1_addr, | ||
110 | (unsigned long)p->ainsn.t1_addr + | ||
111 | sizeof(kprobe_opcode_t)); | ||
112 | |||
113 | p->ainsn.t1_addr = NULL; | ||
114 | } | ||
115 | |||
116 | if (p->ainsn.t2_addr) { | ||
117 | *(p->ainsn.t2_addr) = p->ainsn.t2_opcode; | ||
118 | |||
119 | flush_icache_range((unsigned long)p->ainsn.t2_addr, | ||
120 | (unsigned long)p->ainsn.t2_addr + | ||
121 | sizeof(kprobe_opcode_t)); | ||
122 | |||
123 | p->ainsn.t2_addr = NULL; | ||
124 | } | ||
125 | |||
126 | return; | ||
127 | } | ||
128 | |||
129 | static void __kprobes setup_singlestep(struct kprobe *p, struct pt_regs *regs) | ||
130 | { | ||
131 | unsigned long next_pc; | ||
132 | unsigned long tgt_if_br = 0; | ||
133 | int is_branch; | ||
134 | unsigned long bta; | ||
135 | |||
136 | /* Copy the opcode back to the kprobe location and execute the | ||
137 | * instruction. Because of this we will not be able to get into the | ||
138 | * same kprobe until this kprobe is done | ||
139 | */ | ||
140 | *(p->addr) = p->opcode; | ||
141 | |||
142 | flush_icache_range((unsigned long)p->addr, | ||
143 | (unsigned long)p->addr + sizeof(kprobe_opcode_t)); | ||
144 | |||
145 | /* Now we insert the trap at the next location after this instruction to | ||
146 | * single step. If it is a branch we insert the trap at possible branch | ||
147 | * targets | ||
148 | */ | ||
149 | |||
150 | bta = regs->bta; | ||
151 | |||
152 | if (regs->status32 & 0x40) { | ||
153 | /* We are in a delay slot with the branch taken */ | ||
154 | |||
155 | next_pc = bta & ~0x01; | ||
156 | |||
157 | if (!p->ainsn.is_short) { | ||
158 | if (bta & 0x01) | ||
159 | regs->blink += 2; | ||
160 | else { | ||
161 | /* Branch not taken */ | ||
162 | next_pc += 2; | ||
163 | |||
164 | /* next pc is taken from bta after executing the | ||
165 | * delay slot instruction | ||
166 | */ | ||
167 | regs->bta += 2; | ||
168 | } | ||
169 | } | ||
170 | |||
171 | is_branch = 0; | ||
172 | } else | ||
173 | is_branch = | ||
174 | disasm_next_pc((unsigned long)p->addr, regs, | ||
175 | (struct callee_regs *) current->thread.callee_reg, | ||
176 | &next_pc, &tgt_if_br); | ||
177 | |||
178 | p->ainsn.t1_addr = (kprobe_opcode_t *) next_pc; | ||
179 | p->ainsn.t1_opcode = *(p->ainsn.t1_addr); | ||
180 | *(p->ainsn.t1_addr) = TRAP_S_2_INSTRUCTION; | ||
181 | |||
182 | flush_icache_range((unsigned long)p->ainsn.t1_addr, | ||
183 | (unsigned long)p->ainsn.t1_addr + | ||
184 | sizeof(kprobe_opcode_t)); | ||
185 | |||
186 | if (is_branch) { | ||
187 | p->ainsn.t2_addr = (kprobe_opcode_t *) tgt_if_br; | ||
188 | p->ainsn.t2_opcode = *(p->ainsn.t2_addr); | ||
189 | *(p->ainsn.t2_addr) = TRAP_S_2_INSTRUCTION; | ||
190 | |||
191 | flush_icache_range((unsigned long)p->ainsn.t2_addr, | ||
192 | (unsigned long)p->ainsn.t2_addr + | ||
193 | sizeof(kprobe_opcode_t)); | ||
194 | } | ||
195 | } | ||
196 | |||
197 | int __kprobes arc_kprobe_handler(unsigned long addr, struct pt_regs *regs) | ||
198 | { | ||
199 | struct kprobe *p; | ||
200 | struct kprobe_ctlblk *kcb; | ||
201 | |||
202 | preempt_disable(); | ||
203 | |||
204 | kcb = get_kprobe_ctlblk(); | ||
205 | p = get_kprobe((unsigned long *)addr); | ||
206 | |||
207 | if (p) { | ||
208 | /* | ||
209 | * We have reentered the kprobe_handler, since another kprobe | ||
210 | * was hit while within the handler, we save the original | ||
211 | * kprobes and single step on the instruction of the new probe | ||
212 | * without calling any user handlers to avoid recursive | ||
213 | * kprobes. | ||
214 | */ | ||
215 | if (kprobe_running()) { | ||
216 | save_previous_kprobe(kcb); | ||
217 | set_current_kprobe(p); | ||
218 | kprobes_inc_nmissed_count(p); | ||
219 | setup_singlestep(p, regs); | ||
220 | kcb->kprobe_status = KPROBE_REENTER; | ||
221 | return 1; | ||
222 | } | ||
223 | |||
224 | set_current_kprobe(p); | ||
225 | kcb->kprobe_status = KPROBE_HIT_ACTIVE; | ||
226 | |||
227 | /* If we have no pre-handler or it returned 0, we continue with | ||
228 | * normal processing. If we have a pre-handler and it returned | ||
229 | * non-zero - which is expected from setjmp_pre_handler for | ||
230 | * jprobe, we return without single stepping and leave that to | ||
231 | * the break-handler which is invoked by a kprobe from | ||
232 | * jprobe_return | ||
233 | */ | ||
234 | if (!p->pre_handler || !p->pre_handler(p, regs)) { | ||
235 | setup_singlestep(p, regs); | ||
236 | kcb->kprobe_status = KPROBE_HIT_SS; | ||
237 | } | ||
238 | |||
239 | return 1; | ||
240 | } else if (kprobe_running()) { | ||
241 | p = __get_cpu_var(current_kprobe); | ||
242 | if (p->break_handler && p->break_handler(p, regs)) { | ||
243 | setup_singlestep(p, regs); | ||
244 | kcb->kprobe_status = KPROBE_HIT_SS; | ||
245 | return 1; | ||
246 | } | ||
247 | } | ||
248 | |||
249 | /* no_kprobe: */ | ||
250 | preempt_enable_no_resched(); | ||
251 | return 0; | ||
252 | } | ||
253 | |||
254 | static int __kprobes arc_post_kprobe_handler(unsigned long addr, | ||
255 | struct pt_regs *regs) | ||
256 | { | ||
257 | struct kprobe *cur = kprobe_running(); | ||
258 | struct kprobe_ctlblk *kcb = get_kprobe_ctlblk(); | ||
259 | |||
260 | if (!cur) | ||
261 | return 0; | ||
262 | |||
263 | resume_execution(cur, addr, regs); | ||
264 | |||
265 | /* Rearm the kprobe */ | ||
266 | arch_arm_kprobe(cur); | ||
267 | |||
268 | /* | ||
269 | * When we return from trap instruction we go to the next instruction | ||
270 | * We restored the actual instruction in resume_exectuiont and we to | ||
271 | * return to the same address and execute it | ||
272 | */ | ||
273 | regs->ret = addr; | ||
274 | |||
275 | if ((kcb->kprobe_status != KPROBE_REENTER) && cur->post_handler) { | ||
276 | kcb->kprobe_status = KPROBE_HIT_SSDONE; | ||
277 | cur->post_handler(cur, regs, 0); | ||
278 | } | ||
279 | |||
280 | if (kcb->kprobe_status == KPROBE_REENTER) { | ||
281 | restore_previous_kprobe(kcb); | ||
282 | goto out; | ||
283 | } | ||
284 | |||
285 | reset_current_kprobe(); | ||
286 | |||
287 | out: | ||
288 | preempt_enable_no_resched(); | ||
289 | return 1; | ||
290 | } | ||
291 | |||
292 | /* | ||
293 | * Fault can be for the instruction being single stepped or for the | ||
294 | * pre/post handlers in the module. | ||
295 | * This is applicable for applications like user probes, where we have the | ||
296 | * probe in user space and the handlers in the kernel | ||
297 | */ | ||
298 | |||
299 | int __kprobes kprobe_fault_handler(struct pt_regs *regs, unsigned long trapnr) | ||
300 | { | ||
301 | struct kprobe *cur = kprobe_running(); | ||
302 | struct kprobe_ctlblk *kcb = get_kprobe_ctlblk(); | ||
303 | |||
304 | switch (kcb->kprobe_status) { | ||
305 | case KPROBE_HIT_SS: | ||
306 | case KPROBE_REENTER: | ||
307 | /* | ||
308 | * We are here because the instruction being single stepped | ||
309 | * caused the fault. We reset the current kprobe and allow the | ||
310 | * exception handler as if it is regular exception. In our | ||
311 | * case it doesn't matter because the system will be halted | ||
312 | */ | ||
313 | resume_execution(cur, (unsigned long)cur->addr, regs); | ||
314 | |||
315 | if (kcb->kprobe_status == KPROBE_REENTER) | ||
316 | restore_previous_kprobe(kcb); | ||
317 | else | ||
318 | reset_current_kprobe(); | ||
319 | |||
320 | preempt_enable_no_resched(); | ||
321 | break; | ||
322 | |||
323 | case KPROBE_HIT_ACTIVE: | ||
324 | case KPROBE_HIT_SSDONE: | ||
325 | /* | ||
326 | * We are here because the instructions in the pre/post handler | ||
327 | * caused the fault. | ||
328 | */ | ||
329 | |||
330 | /* We increment the nmissed count for accounting, | ||
331 | * we can also use npre/npostfault count for accouting | ||
332 | * these specific fault cases. | ||
333 | */ | ||
334 | kprobes_inc_nmissed_count(cur); | ||
335 | |||
336 | /* | ||
337 | * We come here because instructions in the pre/post | ||
338 | * handler caused the page_fault, this could happen | ||
339 | * if handler tries to access user space by | ||
340 | * copy_from_user(), get_user() etc. Let the | ||
341 | * user-specified handler try to fix it first. | ||
342 | */ | ||
343 | if (cur->fault_handler && cur->fault_handler(cur, regs, trapnr)) | ||
344 | return 1; | ||
345 | |||
346 | /* | ||
347 | * In case the user-specified fault handler returned zero, | ||
348 | * try to fix up. | ||
349 | */ | ||
350 | if (fixup_exception(regs)) | ||
351 | return 1; | ||
352 | |||
353 | /* | ||
354 | * fixup_exception() could not handle it, | ||
355 | * Let do_page_fault() fix it. | ||
356 | */ | ||
357 | break; | ||
358 | |||
359 | default: | ||
360 | break; | ||
361 | } | ||
362 | return 0; | ||
363 | } | ||
364 | |||
365 | int __kprobes kprobe_exceptions_notify(struct notifier_block *self, | ||
366 | unsigned long val, void *data) | ||
367 | { | ||
368 | struct die_args *args = data; | ||
369 | unsigned long addr = args->err; | ||
370 | int ret = NOTIFY_DONE; | ||
371 | |||
372 | switch (val) { | ||
373 | case DIE_IERR: | ||
374 | if (arc_kprobe_handler(addr, args->regs)) | ||
375 | return NOTIFY_STOP; | ||
376 | break; | ||
377 | |||
378 | case DIE_TRAP: | ||
379 | if (arc_post_kprobe_handler(addr, args->regs)) | ||
380 | return NOTIFY_STOP; | ||
381 | break; | ||
382 | |||
383 | default: | ||
384 | break; | ||
385 | } | ||
386 | |||
387 | return ret; | ||
388 | } | ||
389 | |||
390 | int __kprobes setjmp_pre_handler(struct kprobe *p, struct pt_regs *regs) | ||
391 | { | ||
392 | struct jprobe *jp = container_of(p, struct jprobe, kp); | ||
393 | struct kprobe_ctlblk *kcb = get_kprobe_ctlblk(); | ||
394 | unsigned long sp_addr = regs->sp; | ||
395 | |||
396 | kcb->jprobe_saved_regs = *regs; | ||
397 | memcpy(kcb->jprobes_stack, (void *)sp_addr, MIN_STACK_SIZE(sp_addr)); | ||
398 | regs->ret = (unsigned long)(jp->entry); | ||
399 | |||
400 | return 1; | ||
401 | } | ||
402 | |||
403 | void __kprobes jprobe_return(void) | ||
404 | { | ||
405 | __asm__ __volatile__("unimp_s"); | ||
406 | return; | ||
407 | } | ||
408 | |||
409 | int __kprobes longjmp_break_handler(struct kprobe *p, struct pt_regs *regs) | ||
410 | { | ||
411 | struct kprobe_ctlblk *kcb = get_kprobe_ctlblk(); | ||
412 | unsigned long sp_addr; | ||
413 | |||
414 | *regs = kcb->jprobe_saved_regs; | ||
415 | sp_addr = regs->sp; | ||
416 | memcpy((void *)sp_addr, kcb->jprobes_stack, MIN_STACK_SIZE(sp_addr)); | ||
417 | preempt_enable_no_resched(); | ||
418 | |||
419 | return 1; | ||
420 | } | ||
421 | |||
422 | static void __used kretprobe_trampoline_holder(void) | ||
423 | { | ||
424 | __asm__ __volatile__(".global kretprobe_trampoline\n" | ||
425 | "kretprobe_trampoline:\n" "nop\n"); | ||
426 | } | ||
427 | |||
428 | void __kprobes arch_prepare_kretprobe(struct kretprobe_instance *ri, | ||
429 | struct pt_regs *regs) | ||
430 | { | ||
431 | |||
432 | ri->ret_addr = (kprobe_opcode_t *) regs->blink; | ||
433 | |||
434 | /* Replace the return addr with trampoline addr */ | ||
435 | regs->blink = (unsigned long)&kretprobe_trampoline; | ||
436 | } | ||
437 | |||
438 | static int __kprobes trampoline_probe_handler(struct kprobe *p, | ||
439 | struct pt_regs *regs) | ||
440 | { | ||
441 | struct kretprobe_instance *ri = NULL; | ||
442 | struct hlist_head *head, empty_rp; | ||
443 | struct hlist_node *tmp; | ||
444 | unsigned long flags, orig_ret_address = 0; | ||
445 | unsigned long trampoline_address = (unsigned long)&kretprobe_trampoline; | ||
446 | |||
447 | INIT_HLIST_HEAD(&empty_rp); | ||
448 | kretprobe_hash_lock(current, &head, &flags); | ||
449 | |||
450 | /* | ||
451 | * It is possible to have multiple instances associated with a given | ||
452 | * task either because an multiple functions in the call path | ||
453 | * have a return probe installed on them, and/or more than one return | ||
454 | * return probe was registered for a target function. | ||
455 | * | ||
456 | * We can handle this because: | ||
457 | * - instances are always inserted at the head of the list | ||
458 | * - when multiple return probes are registered for the same | ||
459 | * function, the first instance's ret_addr will point to the | ||
460 | * real return address, and all the rest will point to | ||
461 | * kretprobe_trampoline | ||
462 | */ | ||
463 | hlist_for_each_entry_safe(ri, tmp, head, hlist) { | ||
464 | if (ri->task != current) | ||
465 | /* another task is sharing our hash bucket */ | ||
466 | continue; | ||
467 | |||
468 | if (ri->rp && ri->rp->handler) | ||
469 | ri->rp->handler(ri, regs); | ||
470 | |||
471 | orig_ret_address = (unsigned long)ri->ret_addr; | ||
472 | recycle_rp_inst(ri, &empty_rp); | ||
473 | |||
474 | if (orig_ret_address != trampoline_address) { | ||
475 | /* | ||
476 | * This is the real return address. Any other | ||
477 | * instances associated with this task are for | ||
478 | * other calls deeper on the call stack | ||
479 | */ | ||
480 | break; | ||
481 | } | ||
482 | } | ||
483 | |||
484 | kretprobe_assert(ri, orig_ret_address, trampoline_address); | ||
485 | regs->ret = orig_ret_address; | ||
486 | |||
487 | reset_current_kprobe(); | ||
488 | kretprobe_hash_unlock(current, &flags); | ||
489 | preempt_enable_no_resched(); | ||
490 | |||
491 | hlist_for_each_entry_safe(ri, tmp, &empty_rp, hlist) { | ||
492 | hlist_del(&ri->hlist); | ||
493 | kfree(ri); | ||
494 | } | ||
495 | |||
496 | /* By returning a non zero value, we are telling the kprobe handler | ||
497 | * that we don't want the post_handler to run | ||
498 | */ | ||
499 | return 1; | ||
500 | } | ||
501 | |||
502 | static struct kprobe trampoline_p = { | ||
503 | .addr = (kprobe_opcode_t *) &kretprobe_trampoline, | ||
504 | .pre_handler = trampoline_probe_handler | ||
505 | }; | ||
506 | |||
507 | int __init arch_init_kprobes(void) | ||
508 | { | ||
509 | /* Registering the trampoline code for the kret probe */ | ||
510 | return register_kprobe(&trampoline_p); | ||
511 | } | ||
512 | |||
513 | int __kprobes arch_trampoline_kprobe(struct kprobe *p) | ||
514 | { | ||
515 | if (p->addr == (kprobe_opcode_t *) &kretprobe_trampoline) | ||
516 | return 1; | ||
517 | |||
518 | return 0; | ||
519 | } | ||
520 | |||
521 | void trap_is_kprobe(unsigned long cause, unsigned long address, | ||
522 | struct pt_regs *regs) | ||
523 | { | ||
524 | notify_die(DIE_TRAP, "kprobe_trap", regs, address, cause, SIGTRAP); | ||
525 | } | ||
diff --git a/arch/arc/kernel/module.c b/arch/arc/kernel/module.c new file mode 100644 index 000000000000..cdd359352c0a --- /dev/null +++ b/arch/arc/kernel/module.c | |||
@@ -0,0 +1,145 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <linux/module.h> | ||
10 | #include <linux/moduleloader.h> | ||
11 | #include <linux/kernel.h> | ||
12 | #include <linux/elf.h> | ||
13 | #include <linux/vmalloc.h> | ||
14 | #include <linux/slab.h> | ||
15 | #include <linux/fs.h> | ||
16 | #include <linux/string.h> | ||
17 | #include <asm/unwind.h> | ||
18 | |||
19 | static inline void arc_write_me(unsigned short *addr, unsigned long value) | ||
20 | { | ||
21 | *addr = (value & 0xffff0000) >> 16; | ||
22 | *(addr + 1) = (value & 0xffff); | ||
23 | } | ||
24 | |||
25 | /* ARC specific section quirks - before relocation loop in generic loader | ||
26 | * | ||
27 | * For dwarf unwinding out of modules, this needs to | ||
28 | * 1. Ensure the .debug_frame is allocatable (ARC Linker bug: despite | ||
29 | * -fasynchronous-unwind-tables it doesn't). | ||
30 | * 2. Since we are iterating thru sec hdr tbl anyways, make a note of | ||
31 | * the exact section index, for later use. | ||
32 | */ | ||
33 | int module_frob_arch_sections(Elf_Ehdr *hdr, Elf_Shdr *sechdrs, | ||
34 | char *secstr, struct module *mod) | ||
35 | { | ||
36 | #ifdef CONFIG_ARC_DW2_UNWIND | ||
37 | int i; | ||
38 | |||
39 | mod->arch.unw_sec_idx = 0; | ||
40 | mod->arch.unw_info = NULL; | ||
41 | |||
42 | for (i = 1; i < hdr->e_shnum; i++) { | ||
43 | if (strcmp(secstr+sechdrs[i].sh_name, ".debug_frame") == 0) { | ||
44 | sechdrs[i].sh_flags |= SHF_ALLOC; | ||
45 | mod->arch.unw_sec_idx = i; | ||
46 | break; | ||
47 | } | ||
48 | } | ||
49 | #endif | ||
50 | return 0; | ||
51 | } | ||
52 | |||
53 | void module_arch_cleanup(struct module *mod) | ||
54 | { | ||
55 | #ifdef CONFIG_ARC_DW2_UNWIND | ||
56 | if (mod->arch.unw_info) | ||
57 | unwind_remove_table(mod->arch.unw_info, 0); | ||
58 | #endif | ||
59 | } | ||
60 | |||
61 | int apply_relocate_add(Elf32_Shdr *sechdrs, | ||
62 | const char *strtab, | ||
63 | unsigned int symindex, /* sec index for sym tbl */ | ||
64 | unsigned int relsec, /* sec index for relo sec */ | ||
65 | struct module *module) | ||
66 | { | ||
67 | int i, n; | ||
68 | Elf32_Rela *rel_entry = (void *)sechdrs[relsec].sh_addr; | ||
69 | Elf32_Sym *sym_entry, *sym_sec; | ||
70 | Elf32_Addr relocation; | ||
71 | Elf32_Addr location; | ||
72 | Elf32_Addr sec_to_patch; | ||
73 | int relo_type; | ||
74 | |||
75 | sec_to_patch = sechdrs[sechdrs[relsec].sh_info].sh_addr; | ||
76 | sym_sec = (Elf32_Sym *) sechdrs[symindex].sh_addr; | ||
77 | n = sechdrs[relsec].sh_size / sizeof(*rel_entry); | ||
78 | |||
79 | pr_debug("\n========== Module Sym reloc ===========================\n"); | ||
80 | pr_debug("Section to fixup %x\n", sec_to_patch); | ||
81 | pr_debug("=========================================================\n"); | ||
82 | pr_debug("rela->r_off | rela->addend | sym->st_value | ADDR | VALUE\n"); | ||
83 | pr_debug("=========================================================\n"); | ||
84 | |||
85 | /* Loop thru entries in relocation section */ | ||
86 | for (i = 0; i < n; i++) { | ||
87 | |||
88 | /* This is where to make the change */ | ||
89 | location = sec_to_patch + rel_entry[i].r_offset; | ||
90 | |||
91 | /* This is the symbol it is referring to. Note that all | ||
92 | undefined symbols have been resolved. */ | ||
93 | sym_entry = sym_sec + ELF32_R_SYM(rel_entry[i].r_info); | ||
94 | |||
95 | relocation = sym_entry->st_value + rel_entry[i].r_addend; | ||
96 | |||
97 | pr_debug("\t%x\t\t%x\t\t%x %x %x [%s]\n", | ||
98 | rel_entry[i].r_offset, rel_entry[i].r_addend, | ||
99 | sym_entry->st_value, location, relocation, | ||
100 | strtab + sym_entry->st_name); | ||
101 | |||
102 | /* This assumes modules are built with -mlong-calls | ||
103 | * so any branches/jumps are absolute 32 bit jmps | ||
104 | * global data access again is abs 32 bit. | ||
105 | * Both of these are handled by same relocation type | ||
106 | */ | ||
107 | relo_type = ELF32_R_TYPE(rel_entry[i].r_info); | ||
108 | |||
109 | if (likely(R_ARC_32_ME == relo_type)) | ||
110 | arc_write_me((unsigned short *)location, relocation); | ||
111 | else if (R_ARC_32 == relo_type) | ||
112 | *((Elf32_Addr *) location) = relocation; | ||
113 | else | ||
114 | goto relo_err; | ||
115 | |||
116 | } | ||
117 | return 0; | ||
118 | |||
119 | relo_err: | ||
120 | pr_err("%s: unknown relocation: %u\n", | ||
121 | module->name, ELF32_R_TYPE(rel_entry[i].r_info)); | ||
122 | return -ENOEXEC; | ||
123 | |||
124 | } | ||
125 | |||
126 | /* Just before lift off: After sections have been relocated, we add the | ||
127 | * dwarf section to unwinder table pool | ||
128 | * This couldn't be done in module_frob_arch_sections() because | ||
129 | * relocations had not been applied by then | ||
130 | */ | ||
131 | int module_finalize(const Elf32_Ehdr *hdr, const Elf_Shdr *sechdrs, | ||
132 | struct module *mod) | ||
133 | { | ||
134 | #ifdef CONFIG_ARC_DW2_UNWIND | ||
135 | void *unw; | ||
136 | int unwsec = mod->arch.unw_sec_idx; | ||
137 | |||
138 | if (unwsec) { | ||
139 | unw = unwind_add_table(mod, (void *)sechdrs[unwsec].sh_addr, | ||
140 | sechdrs[unwsec].sh_size); | ||
141 | mod->arch.unw_info = unw; | ||
142 | } | ||
143 | #endif | ||
144 | return 0; | ||
145 | } | ||
diff --git a/arch/arc/kernel/process.c b/arch/arc/kernel/process.c new file mode 100644 index 000000000000..0a7531d99294 --- /dev/null +++ b/arch/arc/kernel/process.c | |||
@@ -0,0 +1,235 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Amit Bhor, Kanika Nema: Codito Technologies 2004 | ||
9 | */ | ||
10 | |||
11 | #include <linux/errno.h> | ||
12 | #include <linux/module.h> | ||
13 | #include <linux/sched.h> | ||
14 | #include <linux/mm.h> | ||
15 | #include <linux/fs.h> | ||
16 | #include <linux/unistd.h> | ||
17 | #include <linux/ptrace.h> | ||
18 | #include <linux/slab.h> | ||
19 | #include <linux/syscalls.h> | ||
20 | #include <linux/elf.h> | ||
21 | #include <linux/tick.h> | ||
22 | |||
23 | SYSCALL_DEFINE1(arc_settls, void *, user_tls_data_ptr) | ||
24 | { | ||
25 | task_thread_info(current)->thr_ptr = (unsigned int)user_tls_data_ptr; | ||
26 | return 0; | ||
27 | } | ||
28 | |||
29 | /* | ||
30 | * We return the user space TLS data ptr as sys-call return code | ||
31 | * Ideally it should be copy to user. | ||
32 | * However we can cheat by the fact that some sys-calls do return | ||
33 | * absurdly high values | ||
34 | * Since the tls dat aptr is not going to be in range of 0xFFFF_xxxx | ||
35 | * it won't be considered a sys-call error | ||
36 | * and it will be loads better than copy-to-user, which is a definite | ||
37 | * D-TLB Miss | ||
38 | */ | ||
39 | SYSCALL_DEFINE0(arc_gettls) | ||
40 | { | ||
41 | return task_thread_info(current)->thr_ptr; | ||
42 | } | ||
43 | |||
44 | static inline void arch_idle(void) | ||
45 | { | ||
46 | /* sleep, but enable all interrupts before committing */ | ||
47 | __asm__("sleep 0x3"); | ||
48 | } | ||
49 | |||
50 | void cpu_idle(void) | ||
51 | { | ||
52 | /* Since we SLEEP in idle loop, TIF_POLLING_NRFLAG can't be set */ | ||
53 | |||
54 | /* endless idle loop with no priority at all */ | ||
55 | while (1) { | ||
56 | tick_nohz_idle_enter(); | ||
57 | rcu_idle_enter(); | ||
58 | |||
59 | doze: | ||
60 | local_irq_disable(); | ||
61 | if (!need_resched()) { | ||
62 | arch_idle(); | ||
63 | goto doze; | ||
64 | } else { | ||
65 | local_irq_enable(); | ||
66 | } | ||
67 | |||
68 | rcu_idle_exit(); | ||
69 | tick_nohz_idle_exit(); | ||
70 | |||
71 | schedule_preempt_disabled(); | ||
72 | } | ||
73 | } | ||
74 | |||
75 | asmlinkage void ret_from_fork(void); | ||
76 | |||
77 | /* Layout of Child kernel mode stack as setup at the end of this function is | ||
78 | * | ||
79 | * | ... | | ||
80 | * | ... | | ||
81 | * | unused | | ||
82 | * | | | ||
83 | * ------------------ <==== top of Stack (thread.ksp) | ||
84 | * | UNUSED 1 word| | ||
85 | * ------------------ | ||
86 | * | r25 | | ||
87 | * ~ ~ | ||
88 | * | --to-- | (CALLEE Regs of user mode) | ||
89 | * | r13 | | ||
90 | * ------------------ | ||
91 | * | fp | | ||
92 | * | blink | @ret_from_fork | ||
93 | * ------------------ | ||
94 | * | | | ||
95 | * ~ ~ | ||
96 | * ~ ~ | ||
97 | * | | | ||
98 | * ------------------ | ||
99 | * | r12 | | ||
100 | * ~ ~ | ||
101 | * | --to-- | (scratch Regs of user mode) | ||
102 | * | r0 | | ||
103 | * ------------------ | ||
104 | * | UNUSED 1 word| | ||
105 | * ------------------ <===== END of PAGE | ||
106 | */ | ||
107 | int copy_thread(unsigned long clone_flags, | ||
108 | unsigned long usp, unsigned long arg, | ||
109 | struct task_struct *p) | ||
110 | { | ||
111 | struct pt_regs *c_regs; /* child's pt_regs */ | ||
112 | unsigned long *childksp; /* to unwind out of __switch_to() */ | ||
113 | struct callee_regs *c_callee; /* child's callee regs */ | ||
114 | struct callee_regs *parent_callee; /* paren't callee */ | ||
115 | struct pt_regs *regs = current_pt_regs(); | ||
116 | |||
117 | /* Mark the specific anchors to begin with (see pic above) */ | ||
118 | c_regs = task_pt_regs(p); | ||
119 | childksp = (unsigned long *)c_regs - 2; /* 2 words for FP/BLINK */ | ||
120 | c_callee = ((struct callee_regs *)childksp) - 1; | ||
121 | |||
122 | /* | ||
123 | * __switch_to() uses thread.ksp to start unwinding stack | ||
124 | * For kernel threads we don't need to create callee regs, the | ||
125 | * stack layout nevertheless needs to remain the same. | ||
126 | * Also, since __switch_to anyways unwinds callee regs, we use | ||
127 | * this to populate kernel thread entry-pt/args into callee regs, | ||
128 | * so that ret_from_kernel_thread() becomes simpler. | ||
129 | */ | ||
130 | p->thread.ksp = (unsigned long)c_callee; /* THREAD_KSP */ | ||
131 | |||
132 | /* __switch_to expects FP(0), BLINK(return addr) at top */ | ||
133 | childksp[0] = 0; /* fp */ | ||
134 | childksp[1] = (unsigned long)ret_from_fork; /* blink */ | ||
135 | |||
136 | if (unlikely(p->flags & PF_KTHREAD)) { | ||
137 | memset(c_regs, 0, sizeof(struct pt_regs)); | ||
138 | |||
139 | c_callee->r13 = arg; /* argument to kernel thread */ | ||
140 | c_callee->r14 = usp; /* function */ | ||
141 | |||
142 | return 0; | ||
143 | } | ||
144 | |||
145 | /*--------- User Task Only --------------*/ | ||
146 | |||
147 | /* __switch_to expects FP(0), BLINK(return addr) at top of stack */ | ||
148 | childksp[0] = 0; /* for POP fp */ | ||
149 | childksp[1] = (unsigned long)ret_from_fork; /* for POP blink */ | ||
150 | |||
151 | /* Copy parents pt regs on child's kernel mode stack */ | ||
152 | *c_regs = *regs; | ||
153 | |||
154 | if (usp) | ||
155 | c_regs->sp = usp; | ||
156 | |||
157 | c_regs->r0 = 0; /* fork returns 0 in child */ | ||
158 | |||
159 | parent_callee = ((struct callee_regs *)regs) - 1; | ||
160 | *c_callee = *parent_callee; | ||
161 | |||
162 | if (unlikely(clone_flags & CLONE_SETTLS)) { | ||
163 | /* | ||
164 | * set task's userland tls data ptr from 4th arg | ||
165 | * clone C-lib call is difft from clone sys-call | ||
166 | */ | ||
167 | task_thread_info(p)->thr_ptr = regs->r3; | ||
168 | } else { | ||
169 | /* Normal fork case: set parent's TLS ptr in child */ | ||
170 | task_thread_info(p)->thr_ptr = | ||
171 | task_thread_info(current)->thr_ptr; | ||
172 | } | ||
173 | |||
174 | return 0; | ||
175 | } | ||
176 | |||
177 | /* | ||
178 | * Some archs flush debug and FPU info here | ||
179 | */ | ||
180 | void flush_thread(void) | ||
181 | { | ||
182 | } | ||
183 | |||
184 | /* | ||
185 | * Free any architecture-specific thread data structures, etc. | ||
186 | */ | ||
187 | void exit_thread(void) | ||
188 | { | ||
189 | } | ||
190 | |||
191 | int dump_fpu(struct pt_regs *regs, elf_fpregset_t *fpu) | ||
192 | { | ||
193 | return 0; | ||
194 | } | ||
195 | |||
196 | /* | ||
197 | * API: expected by schedular Code: If thread is sleeping where is that. | ||
198 | * What is this good for? it will be always the scheduler or ret_from_fork. | ||
199 | * So we hard code that anyways. | ||
200 | */ | ||
201 | unsigned long thread_saved_pc(struct task_struct *t) | ||
202 | { | ||
203 | struct pt_regs *regs = task_pt_regs(t); | ||
204 | unsigned long blink = 0; | ||
205 | |||
206 | /* | ||
207 | * If the thread being queried for in not itself calling this, then it | ||
208 | * implies it is not executing, which in turn implies it is sleeping, | ||
209 | * which in turn implies it got switched OUT by the schedular. | ||
210 | * In that case, it's kernel mode blink can reliably retrieved as per | ||
211 | * the picture above (right above pt_regs). | ||
212 | */ | ||
213 | if (t != current && t->state != TASK_RUNNING) | ||
214 | blink = *((unsigned int *)regs - 1); | ||
215 | |||
216 | return blink; | ||
217 | } | ||
218 | |||
219 | int elf_check_arch(const struct elf32_hdr *x) | ||
220 | { | ||
221 | unsigned int eflags; | ||
222 | |||
223 | if (x->e_machine != EM_ARCOMPACT) | ||
224 | return 0; | ||
225 | |||
226 | eflags = x->e_flags; | ||
227 | if ((eflags & EF_ARC_OSABI_MSK) < EF_ARC_OSABI_CURRENT) { | ||
228 | pr_err("ABI mismatch - you need newer toolchain\n"); | ||
229 | force_sigsegv(SIGSEGV, current); | ||
230 | return 0; | ||
231 | } | ||
232 | |||
233 | return 1; | ||
234 | } | ||
235 | EXPORT_SYMBOL(elf_check_arch); | ||
diff --git a/arch/arc/kernel/ptrace.c b/arch/arc/kernel/ptrace.c new file mode 100644 index 000000000000..c6a81c58d0f3 --- /dev/null +++ b/arch/arc/kernel/ptrace.c | |||
@@ -0,0 +1,158 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <linux/ptrace.h> | ||
10 | #include <linux/tracehook.h> | ||
11 | #include <linux/regset.h> | ||
12 | #include <linux/unistd.h> | ||
13 | #include <linux/elf.h> | ||
14 | |||
15 | static struct callee_regs *task_callee_regs(struct task_struct *tsk) | ||
16 | { | ||
17 | struct callee_regs *tmp = (struct callee_regs *)tsk->thread.callee_reg; | ||
18 | return tmp; | ||
19 | } | ||
20 | |||
21 | static int genregs_get(struct task_struct *target, | ||
22 | const struct user_regset *regset, | ||
23 | unsigned int pos, unsigned int count, | ||
24 | void *kbuf, void __user *ubuf) | ||
25 | { | ||
26 | const struct pt_regs *ptregs = task_pt_regs(target); | ||
27 | const struct callee_regs *cregs = task_callee_regs(target); | ||
28 | int ret = 0; | ||
29 | unsigned int stop_pc_val; | ||
30 | |||
31 | #define REG_O_CHUNK(START, END, PTR) \ | ||
32 | if (!ret) \ | ||
33 | ret = user_regset_copyout(&pos, &count, &kbuf, &ubuf, PTR, \ | ||
34 | offsetof(struct user_regs_struct, START), \ | ||
35 | offsetof(struct user_regs_struct, END)); | ||
36 | |||
37 | #define REG_O_ONE(LOC, PTR) \ | ||
38 | if (!ret) \ | ||
39 | ret = user_regset_copyout(&pos, &count, &kbuf, &ubuf, PTR, \ | ||
40 | offsetof(struct user_regs_struct, LOC), \ | ||
41 | offsetof(struct user_regs_struct, LOC) + 4); | ||
42 | |||
43 | REG_O_CHUNK(scratch, callee, ptregs); | ||
44 | REG_O_CHUNK(callee, efa, cregs); | ||
45 | REG_O_CHUNK(efa, stop_pc, &target->thread.fault_address); | ||
46 | |||
47 | if (!ret) { | ||
48 | if (in_brkpt_trap(ptregs)) { | ||
49 | stop_pc_val = target->thread.fault_address; | ||
50 | pr_debug("\t\tstop_pc (brk-pt)\n"); | ||
51 | } else { | ||
52 | stop_pc_val = ptregs->ret; | ||
53 | pr_debug("\t\tstop_pc (others)\n"); | ||
54 | } | ||
55 | |||
56 | REG_O_ONE(stop_pc, &stop_pc_val); | ||
57 | } | ||
58 | |||
59 | return ret; | ||
60 | } | ||
61 | |||
62 | static int genregs_set(struct task_struct *target, | ||
63 | const struct user_regset *regset, | ||
64 | unsigned int pos, unsigned int count, | ||
65 | const void *kbuf, const void __user *ubuf) | ||
66 | { | ||
67 | const struct pt_regs *ptregs = task_pt_regs(target); | ||
68 | const struct callee_regs *cregs = task_callee_regs(target); | ||
69 | int ret = 0; | ||
70 | |||
71 | #define REG_IN_CHUNK(FIRST, NEXT, PTR) \ | ||
72 | if (!ret) \ | ||
73 | ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf, \ | ||
74 | (void *)(PTR), \ | ||
75 | offsetof(struct user_regs_struct, FIRST), \ | ||
76 | offsetof(struct user_regs_struct, NEXT)); | ||
77 | |||
78 | #define REG_IN_ONE(LOC, PTR) \ | ||
79 | if (!ret) \ | ||
80 | ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf, \ | ||
81 | (void *)(PTR), \ | ||
82 | offsetof(struct user_regs_struct, LOC), \ | ||
83 | offsetof(struct user_regs_struct, LOC) + 4); | ||
84 | |||
85 | #define REG_IGNORE_ONE(LOC) \ | ||
86 | if (!ret) \ | ||
87 | ret = user_regset_copyin_ignore(&pos, &count, &kbuf, &ubuf, \ | ||
88 | offsetof(struct user_regs_struct, LOC), \ | ||
89 | offsetof(struct user_regs_struct, LOC) + 4); | ||
90 | |||
91 | /* TBD: disallow updates to STATUS32, orig_r8 etc*/ | ||
92 | REG_IN_CHUNK(scratch, callee, ptregs); /* pt_regs[bta..orig_r8] */ | ||
93 | REG_IN_CHUNK(callee, efa, cregs); /* callee_regs[r25..r13] */ | ||
94 | REG_IGNORE_ONE(efa); /* efa update invalid */ | ||
95 | REG_IN_ONE(stop_pc, &ptregs->ret); /* stop_pc: PC update */ | ||
96 | |||
97 | return ret; | ||
98 | } | ||
99 | |||
100 | enum arc_getset { | ||
101 | REGSET_GENERAL, | ||
102 | }; | ||
103 | |||
104 | static const struct user_regset arc_regsets[] = { | ||
105 | [REGSET_GENERAL] = { | ||
106 | .core_note_type = NT_PRSTATUS, | ||
107 | .n = ELF_NGREG, | ||
108 | .size = sizeof(unsigned long), | ||
109 | .align = sizeof(unsigned long), | ||
110 | .get = genregs_get, | ||
111 | .set = genregs_set, | ||
112 | } | ||
113 | }; | ||
114 | |||
115 | static const struct user_regset_view user_arc_view = { | ||
116 | .name = UTS_MACHINE, | ||
117 | .e_machine = EM_ARCOMPACT, | ||
118 | .regsets = arc_regsets, | ||
119 | .n = ARRAY_SIZE(arc_regsets) | ||
120 | }; | ||
121 | |||
122 | const struct user_regset_view *task_user_regset_view(struct task_struct *task) | ||
123 | { | ||
124 | return &user_arc_view; | ||
125 | } | ||
126 | |||
127 | void ptrace_disable(struct task_struct *child) | ||
128 | { | ||
129 | } | ||
130 | |||
131 | long arch_ptrace(struct task_struct *child, long request, | ||
132 | unsigned long addr, unsigned long data) | ||
133 | { | ||
134 | int ret = -EIO; | ||
135 | |||
136 | pr_debug("REQ=%ld: ADDR =0x%lx, DATA=0x%lx)\n", request, addr, data); | ||
137 | |||
138 | switch (request) { | ||
139 | default: | ||
140 | ret = ptrace_request(child, request, addr, data); | ||
141 | break; | ||
142 | } | ||
143 | |||
144 | return ret; | ||
145 | } | ||
146 | |||
147 | asmlinkage int syscall_trace_entry(struct pt_regs *regs) | ||
148 | { | ||
149 | if (tracehook_report_syscall_entry(regs)) | ||
150 | return ULONG_MAX; | ||
151 | |||
152 | return regs->r8; | ||
153 | } | ||
154 | |||
155 | asmlinkage void syscall_trace_exit(struct pt_regs *regs) | ||
156 | { | ||
157 | tracehook_report_syscall_exit(regs, 0); | ||
158 | } | ||
diff --git a/arch/arc/kernel/reset.c b/arch/arc/kernel/reset.c new file mode 100644 index 000000000000..e227a2b1c943 --- /dev/null +++ b/arch/arc/kernel/reset.c | |||
@@ -0,0 +1,33 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <linux/kernel.h> | ||
10 | #include <linux/printk.h> | ||
11 | #include <linux/reboot.h> | ||
12 | #include <linux/pm.h> | ||
13 | |||
14 | void machine_halt(void) | ||
15 | { | ||
16 | /* Halt the processor */ | ||
17 | __asm__ __volatile__("flag 1\n"); | ||
18 | } | ||
19 | |||
20 | void machine_restart(char *__unused) | ||
21 | { | ||
22 | /* Soft reset : jump to reset vector */ | ||
23 | pr_info("Put your restart handler here\n"); | ||
24 | machine_halt(); | ||
25 | } | ||
26 | |||
27 | void machine_power_off(void) | ||
28 | { | ||
29 | /* FIXME :: power off ??? */ | ||
30 | machine_halt(); | ||
31 | } | ||
32 | |||
33 | void (*pm_power_off) (void) = NULL; | ||
diff --git a/arch/arc/kernel/setup.c b/arch/arc/kernel/setup.c new file mode 100644 index 000000000000..dc0f968dae0a --- /dev/null +++ b/arch/arc/kernel/setup.c | |||
@@ -0,0 +1,473 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <linux/seq_file.h> | ||
10 | #include <linux/fs.h> | ||
11 | #include <linux/delay.h> | ||
12 | #include <linux/root_dev.h> | ||
13 | #include <linux/console.h> | ||
14 | #include <linux/module.h> | ||
15 | #include <linux/cpu.h> | ||
16 | #include <linux/of_fdt.h> | ||
17 | #include <asm/sections.h> | ||
18 | #include <asm/arcregs.h> | ||
19 | #include <asm/tlb.h> | ||
20 | #include <asm/cache.h> | ||
21 | #include <asm/setup.h> | ||
22 | #include <asm/page.h> | ||
23 | #include <asm/irq.h> | ||
24 | #include <asm/arcregs.h> | ||
25 | #include <asm/prom.h> | ||
26 | #include <asm/unwind.h> | ||
27 | #include <asm/clk.h> | ||
28 | #include <asm/mach_desc.h> | ||
29 | |||
30 | #define FIX_PTR(x) __asm__ __volatile__(";" : "+r"(x)) | ||
31 | |||
32 | int running_on_hw = 1; /* vs. on ISS */ | ||
33 | |||
34 | char __initdata command_line[COMMAND_LINE_SIZE]; | ||
35 | struct machine_desc *machine_desc __initdata; | ||
36 | |||
37 | struct task_struct *_current_task[NR_CPUS]; /* For stack switching */ | ||
38 | |||
39 | struct cpuinfo_arc cpuinfo_arc700[NR_CPUS]; | ||
40 | |||
41 | |||
42 | void __init read_arc_build_cfg_regs(void) | ||
43 | { | ||
44 | struct bcr_perip uncached_space; | ||
45 | struct cpuinfo_arc *cpu = &cpuinfo_arc700[smp_processor_id()]; | ||
46 | FIX_PTR(cpu); | ||
47 | |||
48 | READ_BCR(AUX_IDENTITY, cpu->core); | ||
49 | |||
50 | cpu->timers = read_aux_reg(ARC_REG_TIMERS_BCR); | ||
51 | |||
52 | cpu->vec_base = read_aux_reg(AUX_INTR_VEC_BASE); | ||
53 | if (cpu->vec_base == 0) | ||
54 | cpu->vec_base = (unsigned int)_int_vec_base_lds; | ||
55 | |||
56 | READ_BCR(ARC_REG_D_UNCACH_BCR, uncached_space); | ||
57 | cpu->uncached_base = uncached_space.start << 24; | ||
58 | |||
59 | cpu->extn.mul = read_aux_reg(ARC_REG_MUL_BCR); | ||
60 | cpu->extn.swap = read_aux_reg(ARC_REG_SWAP_BCR); | ||
61 | cpu->extn.norm = read_aux_reg(ARC_REG_NORM_BCR); | ||
62 | cpu->extn.minmax = read_aux_reg(ARC_REG_MIXMAX_BCR); | ||
63 | cpu->extn.barrel = read_aux_reg(ARC_REG_BARREL_BCR); | ||
64 | READ_BCR(ARC_REG_MAC_BCR, cpu->extn_mac_mul); | ||
65 | |||
66 | cpu->extn.ext_arith = read_aux_reg(ARC_REG_EXTARITH_BCR); | ||
67 | cpu->extn.crc = read_aux_reg(ARC_REG_CRC_BCR); | ||
68 | |||
69 | /* Note that we read the CCM BCRs independent of kernel config | ||
70 | * This is to catch the cases where user doesn't know that | ||
71 | * CCMs are present in hardware build | ||
72 | */ | ||
73 | { | ||
74 | struct bcr_iccm iccm; | ||
75 | struct bcr_dccm dccm; | ||
76 | struct bcr_dccm_base dccm_base; | ||
77 | unsigned int bcr_32bit_val; | ||
78 | |||
79 | bcr_32bit_val = read_aux_reg(ARC_REG_ICCM_BCR); | ||
80 | if (bcr_32bit_val) { | ||
81 | iccm = *((struct bcr_iccm *)&bcr_32bit_val); | ||
82 | cpu->iccm.base_addr = iccm.base << 16; | ||
83 | cpu->iccm.sz = 0x2000 << (iccm.sz - 1); | ||
84 | } | ||
85 | |||
86 | bcr_32bit_val = read_aux_reg(ARC_REG_DCCM_BCR); | ||
87 | if (bcr_32bit_val) { | ||
88 | dccm = *((struct bcr_dccm *)&bcr_32bit_val); | ||
89 | cpu->dccm.sz = 0x800 << (dccm.sz); | ||
90 | |||
91 | READ_BCR(ARC_REG_DCCMBASE_BCR, dccm_base); | ||
92 | cpu->dccm.base_addr = dccm_base.addr << 8; | ||
93 | } | ||
94 | } | ||
95 | |||
96 | READ_BCR(ARC_REG_XY_MEM_BCR, cpu->extn_xymem); | ||
97 | |||
98 | read_decode_mmu_bcr(); | ||
99 | read_decode_cache_bcr(); | ||
100 | |||
101 | READ_BCR(ARC_REG_FP_BCR, cpu->fp); | ||
102 | READ_BCR(ARC_REG_DPFP_BCR, cpu->dpfp); | ||
103 | } | ||
104 | |||
105 | static const struct cpuinfo_data arc_cpu_tbl[] = { | ||
106 | { {0x10, "ARCTangent A5"}, 0x1F}, | ||
107 | { {0x20, "ARC 600" }, 0x2F}, | ||
108 | { {0x30, "ARC 700" }, 0x33}, | ||
109 | { {0x34, "ARC 700 R4.10"}, 0x34}, | ||
110 | { {0x00, NULL } } | ||
111 | }; | ||
112 | |||
113 | char *arc_cpu_mumbojumbo(int cpu_id, char *buf, int len) | ||
114 | { | ||
115 | int n = 0; | ||
116 | struct cpuinfo_arc *cpu = &cpuinfo_arc700[cpu_id]; | ||
117 | struct bcr_identity *core = &cpu->core; | ||
118 | const struct cpuinfo_data *tbl; | ||
119 | int be = 0; | ||
120 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
121 | be = 1; | ||
122 | #endif | ||
123 | FIX_PTR(cpu); | ||
124 | |||
125 | n += scnprintf(buf + n, len - n, | ||
126 | "\nARC IDENTITY\t: Family [%#02x]" | ||
127 | " Cpu-id [%#02x] Chip-id [%#4x]\n", | ||
128 | core->family, core->cpu_id, | ||
129 | core->chip_id); | ||
130 | |||
131 | for (tbl = &arc_cpu_tbl[0]; tbl->info.id != 0; tbl++) { | ||
132 | if ((core->family >= tbl->info.id) && | ||
133 | (core->family <= tbl->up_range)) { | ||
134 | n += scnprintf(buf + n, len - n, | ||
135 | "processor\t: %s %s\n", | ||
136 | tbl->info.str, | ||
137 | be ? "[Big Endian]" : ""); | ||
138 | break; | ||
139 | } | ||
140 | } | ||
141 | |||
142 | if (tbl->info.id == 0) | ||
143 | n += scnprintf(buf + n, len - n, "UNKNOWN ARC Processor\n"); | ||
144 | |||
145 | n += scnprintf(buf + n, len - n, "CPU speed\t: %u.%02u Mhz\n", | ||
146 | (unsigned int)(arc_get_core_freq() / 1000000), | ||
147 | (unsigned int)(arc_get_core_freq() / 10000) % 100); | ||
148 | |||
149 | n += scnprintf(buf + n, len - n, "Timers\t\t: %s %s\n", | ||
150 | (cpu->timers & 0x200) ? "TIMER1" : "", | ||
151 | (cpu->timers & 0x100) ? "TIMER0" : ""); | ||
152 | |||
153 | n += scnprintf(buf + n, len - n, "Vect Tbl Base\t: %#x\n", | ||
154 | cpu->vec_base); | ||
155 | |||
156 | n += scnprintf(buf + n, len - n, "UNCACHED Base\t: %#x\n", | ||
157 | cpu->uncached_base); | ||
158 | |||
159 | return buf; | ||
160 | } | ||
161 | |||
162 | static const struct id_to_str mul_type_nm[] = { | ||
163 | { 0x0, "N/A"}, | ||
164 | { 0x1, "32x32 (spl Result Reg)" }, | ||
165 | { 0x2, "32x32 (ANY Result Reg)" } | ||
166 | }; | ||
167 | |||
168 | static const struct id_to_str mac_mul_nm[] = { | ||
169 | {0x0, "N/A"}, | ||
170 | {0x1, "N/A"}, | ||
171 | {0x2, "Dual 16 x 16"}, | ||
172 | {0x3, "N/A"}, | ||
173 | {0x4, "32x16"}, | ||
174 | {0x5, "N/A"}, | ||
175 | {0x6, "Dual 16x16 and 32x16"} | ||
176 | }; | ||
177 | |||
178 | char *arc_extn_mumbojumbo(int cpu_id, char *buf, int len) | ||
179 | { | ||
180 | int n = 0; | ||
181 | struct cpuinfo_arc *cpu = &cpuinfo_arc700[cpu_id]; | ||
182 | |||
183 | FIX_PTR(cpu); | ||
184 | #define IS_AVAIL1(var, str) ((var) ? str : "") | ||
185 | #define IS_AVAIL2(var, str) ((var == 0x2) ? str : "") | ||
186 | #define IS_USED(var) ((var) ? "(in-use)" : "(not used)") | ||
187 | |||
188 | n += scnprintf(buf + n, len - n, | ||
189 | "Extn [700-Base]\t: %s %s %s %s %s %s\n", | ||
190 | IS_AVAIL2(cpu->extn.norm, "norm,"), | ||
191 | IS_AVAIL2(cpu->extn.barrel, "barrel-shift,"), | ||
192 | IS_AVAIL1(cpu->extn.swap, "swap,"), | ||
193 | IS_AVAIL2(cpu->extn.minmax, "minmax,"), | ||
194 | IS_AVAIL1(cpu->extn.crc, "crc,"), | ||
195 | IS_AVAIL2(cpu->extn.ext_arith, "ext-arith")); | ||
196 | |||
197 | n += scnprintf(buf + n, len - n, "Extn [700-MPY]\t: %s", | ||
198 | mul_type_nm[cpu->extn.mul].str); | ||
199 | |||
200 | n += scnprintf(buf + n, len - n, " MAC MPY: %s\n", | ||
201 | mac_mul_nm[cpu->extn_mac_mul.type].str); | ||
202 | |||
203 | if (cpu->core.family == 0x34) { | ||
204 | n += scnprintf(buf + n, len - n, | ||
205 | "Extn [700-4.10]\t: LLOCK/SCOND %s, SWAPE %s, RTSC %s\n", | ||
206 | IS_USED(__CONFIG_ARC_HAS_LLSC_VAL), | ||
207 | IS_USED(__CONFIG_ARC_HAS_SWAPE_VAL), | ||
208 | IS_USED(__CONFIG_ARC_HAS_RTSC_VAL)); | ||
209 | } | ||
210 | |||
211 | n += scnprintf(buf + n, len - n, "Extn [CCM]\t: %s", | ||
212 | !(cpu->dccm.sz || cpu->iccm.sz) ? "N/A" : ""); | ||
213 | |||
214 | if (cpu->dccm.sz) | ||
215 | n += scnprintf(buf + n, len - n, "DCCM: @ %x, %d KB ", | ||
216 | cpu->dccm.base_addr, TO_KB(cpu->dccm.sz)); | ||
217 | |||
218 | if (cpu->iccm.sz) | ||
219 | n += scnprintf(buf + n, len - n, "ICCM: @ %x, %d KB", | ||
220 | cpu->iccm.base_addr, TO_KB(cpu->iccm.sz)); | ||
221 | |||
222 | n += scnprintf(buf + n, len - n, "\nExtn [FPU]\t: %s", | ||
223 | !(cpu->fp.ver || cpu->dpfp.ver) ? "N/A" : ""); | ||
224 | |||
225 | if (cpu->fp.ver) | ||
226 | n += scnprintf(buf + n, len - n, "SP [v%d] %s", | ||
227 | cpu->fp.ver, cpu->fp.fast ? "(fast)" : ""); | ||
228 | |||
229 | if (cpu->dpfp.ver) | ||
230 | n += scnprintf(buf + n, len - n, "DP [v%d] %s", | ||
231 | cpu->dpfp.ver, cpu->dpfp.fast ? "(fast)" : ""); | ||
232 | |||
233 | n += scnprintf(buf + n, len - n, "\n"); | ||
234 | |||
235 | #ifdef _ASM_GENERIC_UNISTD_H | ||
236 | n += scnprintf(buf + n, len - n, | ||
237 | "OS ABI [v2]\t: asm-generic/{unistd,stat,fcntl}\n"); | ||
238 | #endif | ||
239 | |||
240 | return buf; | ||
241 | } | ||
242 | |||
243 | void __init arc_chk_ccms(void) | ||
244 | { | ||
245 | #if defined(CONFIG_ARC_HAS_DCCM) || defined(CONFIG_ARC_HAS_ICCM) | ||
246 | struct cpuinfo_arc *cpu = &cpuinfo_arc700[smp_processor_id()]; | ||
247 | |||
248 | #ifdef CONFIG_ARC_HAS_DCCM | ||
249 | /* | ||
250 | * DCCM can be arbit placed in hardware. | ||
251 | * Make sure it's placement/sz matches what Linux is built with | ||
252 | */ | ||
253 | if ((unsigned int)__arc_dccm_base != cpu->dccm.base_addr) | ||
254 | panic("Linux built with incorrect DCCM Base address\n"); | ||
255 | |||
256 | if (CONFIG_ARC_DCCM_SZ != cpu->dccm.sz) | ||
257 | panic("Linux built with incorrect DCCM Size\n"); | ||
258 | #endif | ||
259 | |||
260 | #ifdef CONFIG_ARC_HAS_ICCM | ||
261 | if (CONFIG_ARC_ICCM_SZ != cpu->iccm.sz) | ||
262 | panic("Linux built with incorrect ICCM Size\n"); | ||
263 | #endif | ||
264 | #endif | ||
265 | } | ||
266 | |||
267 | /* | ||
268 | * Ensure that FP hardware and kernel config match | ||
269 | * -If hardware contains DPFP, kernel needs to save/restore FPU state | ||
270 | * across context switches | ||
271 | * -If hardware lacks DPFP, but kernel configured to save FPU state then | ||
272 | * kernel trying to access non-existant DPFP regs will crash | ||
273 | * | ||
274 | * We only check for Dbl precision Floating Point, because only DPFP | ||
275 | * hardware has dedicated regs which need to be saved/restored on ctx-sw | ||
276 | * (Single Precision uses core regs), thus kernel is kind of oblivious to it | ||
277 | */ | ||
278 | void __init arc_chk_fpu(void) | ||
279 | { | ||
280 | struct cpuinfo_arc *cpu = &cpuinfo_arc700[smp_processor_id()]; | ||
281 | |||
282 | if (cpu->dpfp.ver) { | ||
283 | #ifndef CONFIG_ARC_FPU_SAVE_RESTORE | ||
284 | pr_warn("DPFP support broken in this kernel...\n"); | ||
285 | #endif | ||
286 | } else { | ||
287 | #ifdef CONFIG_ARC_FPU_SAVE_RESTORE | ||
288 | panic("H/w lacks DPFP support, apps won't work\n"); | ||
289 | #endif | ||
290 | } | ||
291 | } | ||
292 | |||
293 | /* | ||
294 | * Initialize and setup the processor core | ||
295 | * This is called by all the CPUs thus should not do special case stuff | ||
296 | * such as only for boot CPU etc | ||
297 | */ | ||
298 | |||
299 | void __init setup_processor(void) | ||
300 | { | ||
301 | char str[512]; | ||
302 | int cpu_id = smp_processor_id(); | ||
303 | |||
304 | read_arc_build_cfg_regs(); | ||
305 | arc_init_IRQ(); | ||
306 | |||
307 | printk(arc_cpu_mumbojumbo(cpu_id, str, sizeof(str))); | ||
308 | |||
309 | arc_mmu_init(); | ||
310 | arc_cache_init(); | ||
311 | arc_chk_ccms(); | ||
312 | |||
313 | printk(arc_extn_mumbojumbo(cpu_id, str, sizeof(str))); | ||
314 | |||
315 | #ifdef CONFIG_SMP | ||
316 | printk(arc_platform_smp_cpuinfo()); | ||
317 | #endif | ||
318 | |||
319 | arc_chk_fpu(); | ||
320 | } | ||
321 | |||
322 | void __init setup_arch(char **cmdline_p) | ||
323 | { | ||
324 | #ifdef CONFIG_CMDLINE_UBOOT | ||
325 | /* Make sure that a whitespace is inserted before */ | ||
326 | strlcat(command_line, " ", sizeof(command_line)); | ||
327 | #endif | ||
328 | /* | ||
329 | * Append .config cmdline to base command line, which might already | ||
330 | * contain u-boot "bootargs" (handled by head.S, if so configured) | ||
331 | */ | ||
332 | strlcat(command_line, CONFIG_CMDLINE, sizeof(command_line)); | ||
333 | |||
334 | /* Save unparsed command line copy for /proc/cmdline */ | ||
335 | strlcpy(boot_command_line, command_line, COMMAND_LINE_SIZE); | ||
336 | *cmdline_p = command_line; | ||
337 | |||
338 | machine_desc = setup_machine_fdt(__dtb_start); | ||
339 | if (!machine_desc) | ||
340 | panic("Embedded DT invalid\n"); | ||
341 | |||
342 | /* To force early parsing of things like mem=xxx */ | ||
343 | parse_early_param(); | ||
344 | |||
345 | /* Platform/board specific: e.g. early console registration */ | ||
346 | if (machine_desc->init_early) | ||
347 | machine_desc->init_early(); | ||
348 | |||
349 | setup_processor(); | ||
350 | |||
351 | #ifdef CONFIG_SMP | ||
352 | smp_init_cpus(); | ||
353 | #endif | ||
354 | |||
355 | setup_arch_memory(); | ||
356 | |||
357 | /* copy flat DT out of .init and then unflatten it */ | ||
358 | copy_devtree(); | ||
359 | unflatten_device_tree(); | ||
360 | |||
361 | /* Can be issue if someone passes cmd line arg "ro" | ||
362 | * But that is unlikely so keeping it as it is | ||
363 | */ | ||
364 | root_mountflags &= ~MS_RDONLY; | ||
365 | |||
366 | console_verbose(); | ||
367 | |||
368 | #if defined(CONFIG_VT) && defined(CONFIG_DUMMY_CONSOLE) | ||
369 | conswitchp = &dummy_con; | ||
370 | #endif | ||
371 | |||
372 | arc_unwind_init(); | ||
373 | arc_unwind_setup(); | ||
374 | } | ||
375 | |||
376 | static int __init customize_machine(void) | ||
377 | { | ||
378 | /* Add platform devices */ | ||
379 | if (machine_desc->init_machine) | ||
380 | machine_desc->init_machine(); | ||
381 | |||
382 | return 0; | ||
383 | } | ||
384 | arch_initcall(customize_machine); | ||
385 | |||
386 | static int __init init_late_machine(void) | ||
387 | { | ||
388 | if (machine_desc->init_late) | ||
389 | machine_desc->init_late(); | ||
390 | |||
391 | return 0; | ||
392 | } | ||
393 | late_initcall(init_late_machine); | ||
394 | /* | ||
395 | * Get CPU information for use by the procfs. | ||
396 | */ | ||
397 | |||
398 | #define cpu_to_ptr(c) ((void *)(0xFFFF0000 | (unsigned int)(c))) | ||
399 | #define ptr_to_cpu(p) (~0xFFFF0000UL & (unsigned int)(p)) | ||
400 | |||
401 | static int show_cpuinfo(struct seq_file *m, void *v) | ||
402 | { | ||
403 | char *str; | ||
404 | int cpu_id = ptr_to_cpu(v); | ||
405 | |||
406 | str = (char *)__get_free_page(GFP_TEMPORARY); | ||
407 | if (!str) | ||
408 | goto done; | ||
409 | |||
410 | seq_printf(m, arc_cpu_mumbojumbo(cpu_id, str, PAGE_SIZE)); | ||
411 | |||
412 | seq_printf(m, "Bogo MIPS : \t%lu.%02lu\n", | ||
413 | loops_per_jiffy / (500000 / HZ), | ||
414 | (loops_per_jiffy / (5000 / HZ)) % 100); | ||
415 | |||
416 | seq_printf(m, arc_mmu_mumbojumbo(cpu_id, str, PAGE_SIZE)); | ||
417 | |||
418 | seq_printf(m, arc_cache_mumbojumbo(cpu_id, str, PAGE_SIZE)); | ||
419 | |||
420 | seq_printf(m, arc_extn_mumbojumbo(cpu_id, str, PAGE_SIZE)); | ||
421 | |||
422 | #ifdef CONFIG_SMP | ||
423 | seq_printf(m, arc_platform_smp_cpuinfo()); | ||
424 | #endif | ||
425 | |||
426 | free_page((unsigned long)str); | ||
427 | done: | ||
428 | seq_printf(m, "\n\n"); | ||
429 | |||
430 | return 0; | ||
431 | } | ||
432 | |||
433 | static void *c_start(struct seq_file *m, loff_t *pos) | ||
434 | { | ||
435 | /* | ||
436 | * Callback returns cpu-id to iterator for show routine, NULL to stop. | ||
437 | * However since NULL is also a valid cpu-id (0), we use a round-about | ||
438 | * way to pass it w/o having to kmalloc/free a 2 byte string. | ||
439 | * Encode cpu-id as 0xFFcccc, which is decoded by show routine. | ||
440 | */ | ||
441 | return *pos < num_possible_cpus() ? cpu_to_ptr(*pos) : NULL; | ||
442 | } | ||
443 | |||
444 | static void *c_next(struct seq_file *m, void *v, loff_t *pos) | ||
445 | { | ||
446 | ++*pos; | ||
447 | return c_start(m, pos); | ||
448 | } | ||
449 | |||
450 | static void c_stop(struct seq_file *m, void *v) | ||
451 | { | ||
452 | } | ||
453 | |||
454 | const struct seq_operations cpuinfo_op = { | ||
455 | .start = c_start, | ||
456 | .next = c_next, | ||
457 | .stop = c_stop, | ||
458 | .show = show_cpuinfo | ||
459 | }; | ||
460 | |||
461 | static DEFINE_PER_CPU(struct cpu, cpu_topology); | ||
462 | |||
463 | static int __init topology_init(void) | ||
464 | { | ||
465 | int cpu; | ||
466 | |||
467 | for_each_present_cpu(cpu) | ||
468 | register_cpu(&per_cpu(cpu_topology, cpu), cpu); | ||
469 | |||
470 | return 0; | ||
471 | } | ||
472 | |||
473 | subsys_initcall(topology_init); | ||
diff --git a/arch/arc/kernel/signal.c b/arch/arc/kernel/signal.c new file mode 100644 index 000000000000..ee6ef2f60a28 --- /dev/null +++ b/arch/arc/kernel/signal.c | |||
@@ -0,0 +1,360 @@ | |||
1 | /* | ||
2 | * Signal Handling for ARC | ||
3 | * | ||
4 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | * | ||
10 | * vineetg: Jan 2010 (Restarting of timer related syscalls) | ||
11 | * | ||
12 | * vineetg: Nov 2009 (Everything needed for TIF_RESTORE_SIGMASK) | ||
13 | * -do_signal() supports TIF_RESTORE_SIGMASK | ||
14 | * -do_signal() no loner needs oldset, required by OLD sys_sigsuspend | ||
15 | * -sys_rt_sigsuspend() now comes from generic code, so discard arch implemen | ||
16 | * -sys_sigsuspend() no longer needs to fudge ptregs, hence that arg removed | ||
17 | * -sys_sigsuspend() no longer loops for do_signal(), sets TIF_xxx and leaves | ||
18 | * the job to do_signal() | ||
19 | * | ||
20 | * vineetg: July 2009 | ||
21 | * -Modified Code to support the uClibc provided userland sigreturn stub | ||
22 | * to avoid kernel synthesing it on user stack at runtime, costing TLB | ||
23 | * probes and Cache line flushes. | ||
24 | * | ||
25 | * vineetg: July 2009 | ||
26 | * -In stash_usr_regs( ) and restore_usr_regs( ), save/restore of user regs | ||
27 | * in done in block copy rather than one word at a time. | ||
28 | * This saves around 2K of code and improves LMBench lat_sig <catch> | ||
29 | * | ||
30 | * rajeshwarr: Feb 2009 | ||
31 | * - Support for Realtime Signals | ||
32 | * | ||
33 | * vineetg: Aug 11th 2008: Bug #94183 | ||
34 | * -ViXS were still seeing crashes when using insmod to load drivers. | ||
35 | * It turned out that the code to change Execute permssions for TLB entries | ||
36 | * of user was not guarded for interrupts (mod_tlb_permission) | ||
37 | * This was cauing TLB entries to be overwritten on unrelated indexes | ||
38 | * | ||
39 | * Vineetg: July 15th 2008: Bug #94183 | ||
40 | * -Exception happens in Delay slot of a JMP, and before user space resumes, | ||
41 | * Signal is delivered (Ctrl + C) = >SIGINT. | ||
42 | * setup_frame( ) sets up PC,SP,BLINK to enable user space signal handler | ||
43 | * to run, but doesn't clear the Delay slot bit from status32. As a result, | ||
44 | * on resuming user mode, signal handler branches off to BTA of orig JMP | ||
45 | * -FIX: clear the DE bit from status32 in setup_frame( ) | ||
46 | * | ||
47 | * Rahul Trivedi, Kanika Nema: Codito Technologies 2004 | ||
48 | */ | ||
49 | |||
50 | #include <linux/signal.h> | ||
51 | #include <linux/ptrace.h> | ||
52 | #include <linux/personality.h> | ||
53 | #include <linux/uaccess.h> | ||
54 | #include <linux/syscalls.h> | ||
55 | #include <linux/tracehook.h> | ||
56 | #include <asm/ucontext.h> | ||
57 | |||
58 | struct rt_sigframe { | ||
59 | struct siginfo info; | ||
60 | struct ucontext uc; | ||
61 | #define MAGIC_SIGALTSTK 0x07302004 | ||
62 | unsigned int sigret_magic; | ||
63 | }; | ||
64 | |||
65 | static int | ||
66 | stash_usr_regs(struct rt_sigframe __user *sf, struct pt_regs *regs, | ||
67 | sigset_t *set) | ||
68 | { | ||
69 | int err; | ||
70 | err = __copy_to_user(&(sf->uc.uc_mcontext.regs), regs, | ||
71 | sizeof(sf->uc.uc_mcontext.regs.scratch)); | ||
72 | err |= __copy_to_user(&sf->uc.uc_sigmask, set, sizeof(sigset_t)); | ||
73 | |||
74 | return err; | ||
75 | } | ||
76 | |||
77 | static int restore_usr_regs(struct pt_regs *regs, struct rt_sigframe __user *sf) | ||
78 | { | ||
79 | sigset_t set; | ||
80 | int err; | ||
81 | |||
82 | err = __copy_from_user(&set, &sf->uc.uc_sigmask, sizeof(set)); | ||
83 | if (!err) | ||
84 | set_current_blocked(&set); | ||
85 | |||
86 | err |= __copy_from_user(regs, &(sf->uc.uc_mcontext.regs), | ||
87 | sizeof(sf->uc.uc_mcontext.regs.scratch)); | ||
88 | |||
89 | return err; | ||
90 | } | ||
91 | |||
92 | static inline int is_do_ss_needed(unsigned int magic) | ||
93 | { | ||
94 | if (MAGIC_SIGALTSTK == magic) | ||
95 | return 1; | ||
96 | else | ||
97 | return 0; | ||
98 | } | ||
99 | |||
100 | SYSCALL_DEFINE0(rt_sigreturn) | ||
101 | { | ||
102 | struct rt_sigframe __user *sf; | ||
103 | unsigned int magic; | ||
104 | int err; | ||
105 | struct pt_regs *regs = current_pt_regs(); | ||
106 | |||
107 | /* Always make any pending restarted system calls return -EINTR */ | ||
108 | current_thread_info()->restart_block.fn = do_no_restart_syscall; | ||
109 | |||
110 | /* Since we stacked the signal on a word boundary, | ||
111 | * then 'sp' should be word aligned here. If it's | ||
112 | * not, then the user is trying to mess with us. | ||
113 | */ | ||
114 | if (regs->sp & 3) | ||
115 | goto badframe; | ||
116 | |||
117 | sf = (struct rt_sigframe __force __user *)(regs->sp); | ||
118 | |||
119 | if (!access_ok(VERIFY_READ, sf, sizeof(*sf))) | ||
120 | goto badframe; | ||
121 | |||
122 | err = restore_usr_regs(regs, sf); | ||
123 | err |= __get_user(magic, &sf->sigret_magic); | ||
124 | if (err) | ||
125 | goto badframe; | ||
126 | |||
127 | if (unlikely(is_do_ss_needed(magic))) | ||
128 | if (restore_altstack(&sf->uc.uc_stack)) | ||
129 | goto badframe; | ||
130 | |||
131 | /* Don't restart from sigreturn */ | ||
132 | syscall_wont_restart(regs); | ||
133 | |||
134 | return regs->r0; | ||
135 | |||
136 | badframe: | ||
137 | force_sig(SIGSEGV, current); | ||
138 | return 0; | ||
139 | } | ||
140 | |||
141 | /* | ||
142 | * Determine which stack to use.. | ||
143 | */ | ||
144 | static inline void __user *get_sigframe(struct k_sigaction *ka, | ||
145 | struct pt_regs *regs, | ||
146 | unsigned long framesize) | ||
147 | { | ||
148 | unsigned long sp = regs->sp; | ||
149 | void __user *frame; | ||
150 | |||
151 | /* This is the X/Open sanctioned signal stack switching */ | ||
152 | if ((ka->sa.sa_flags & SA_ONSTACK) && !sas_ss_flags(sp)) | ||
153 | sp = current->sas_ss_sp + current->sas_ss_size; | ||
154 | |||
155 | /* No matter what happens, 'sp' must be word | ||
156 | * aligned otherwise nasty things could happen | ||
157 | */ | ||
158 | |||
159 | /* ATPCS B01 mandates 8-byte alignment */ | ||
160 | frame = (void __user *)((sp - framesize) & ~7); | ||
161 | |||
162 | /* Check that we can actually write to the signal frame */ | ||
163 | if (!access_ok(VERIFY_WRITE, frame, framesize)) | ||
164 | frame = NULL; | ||
165 | |||
166 | return frame; | ||
167 | } | ||
168 | |||
169 | /* | ||
170 | * translate the signal | ||
171 | */ | ||
172 | static inline int map_sig(int sig) | ||
173 | { | ||
174 | struct thread_info *thread = current_thread_info(); | ||
175 | if (thread->exec_domain && thread->exec_domain->signal_invmap | ||
176 | && sig < 32) | ||
177 | sig = thread->exec_domain->signal_invmap[sig]; | ||
178 | return sig; | ||
179 | } | ||
180 | |||
181 | static int | ||
182 | setup_rt_frame(int signo, struct k_sigaction *ka, siginfo_t *info, | ||
183 | sigset_t *set, struct pt_regs *regs) | ||
184 | { | ||
185 | struct rt_sigframe __user *sf; | ||
186 | unsigned int magic = 0; | ||
187 | int err = 0; | ||
188 | |||
189 | sf = get_sigframe(ka, regs, sizeof(struct rt_sigframe)); | ||
190 | if (!sf) | ||
191 | return 1; | ||
192 | |||
193 | /* | ||
194 | * SA_SIGINFO requires 3 args to signal handler: | ||
195 | * #1: sig-no (common to any handler) | ||
196 | * #2: struct siginfo | ||
197 | * #3: struct ucontext (completely populated) | ||
198 | */ | ||
199 | if (unlikely(ka->sa.sa_flags & SA_SIGINFO)) { | ||
200 | err |= copy_siginfo_to_user(&sf->info, info); | ||
201 | err |= __put_user(0, &sf->uc.uc_flags); | ||
202 | err |= __put_user(NULL, &sf->uc.uc_link); | ||
203 | err |= __save_altstack(&sf->uc.uc_stack, regs->sp); | ||
204 | |||
205 | /* setup args 2 and 3 for user mode handler */ | ||
206 | regs->r1 = (unsigned long)&sf->info; | ||
207 | regs->r2 = (unsigned long)&sf->uc; | ||
208 | |||
209 | /* | ||
210 | * small optim to avoid unconditonally calling do_sigaltstack | ||
211 | * in sigreturn path, now that we only have rt_sigreturn | ||
212 | */ | ||
213 | magic = MAGIC_SIGALTSTK; | ||
214 | } | ||
215 | |||
216 | /* | ||
217 | * w/o SA_SIGINFO, struct ucontext is partially populated (only | ||
218 | * uc_mcontext/uc_sigmask) for kernel's normal user state preservation | ||
219 | * during signal handler execution. This works for SA_SIGINFO as well | ||
220 | * although the semantics are now overloaded (the same reg state can be | ||
221 | * inspected by userland: but are they allowed to fiddle with it ? | ||
222 | */ | ||
223 | err |= stash_usr_regs(sf, regs, set); | ||
224 | err |= __put_user(magic, &sf->sigret_magic); | ||
225 | if (err) | ||
226 | return err; | ||
227 | |||
228 | /* #1 arg to the user Signal handler */ | ||
229 | regs->r0 = map_sig(signo); | ||
230 | |||
231 | /* setup PC of user space signal handler */ | ||
232 | regs->ret = (unsigned long)ka->sa.sa_handler; | ||
233 | |||
234 | /* | ||
235 | * handler returns using sigreturn stub provided already by userpsace | ||
236 | */ | ||
237 | BUG_ON(!(ka->sa.sa_flags & SA_RESTORER)); | ||
238 | regs->blink = (unsigned long)ka->sa.sa_restorer; | ||
239 | |||
240 | /* User Stack for signal handler will be above the frame just carved */ | ||
241 | regs->sp = (unsigned long)sf; | ||
242 | |||
243 | /* | ||
244 | * Bug 94183, Clear the DE bit, so that when signal handler | ||
245 | * starts to run, it doesn't use BTA | ||
246 | */ | ||
247 | regs->status32 &= ~STATUS_DE_MASK; | ||
248 | regs->status32 |= STATUS_L_MASK; | ||
249 | |||
250 | return err; | ||
251 | } | ||
252 | |||
253 | static void arc_restart_syscall(struct k_sigaction *ka, struct pt_regs *regs) | ||
254 | { | ||
255 | switch (regs->r0) { | ||
256 | case -ERESTART_RESTARTBLOCK: | ||
257 | case -ERESTARTNOHAND: | ||
258 | /* | ||
259 | * ERESTARTNOHAND means that the syscall should | ||
260 | * only be restarted if there was no handler for | ||
261 | * the signal, and since we only get here if there | ||
262 | * is a handler, we don't restart | ||
263 | */ | ||
264 | regs->r0 = -EINTR; /* ERESTART_xxx is internal */ | ||
265 | break; | ||
266 | |||
267 | case -ERESTARTSYS: | ||
268 | /* | ||
269 | * ERESTARTSYS means to restart the syscall if | ||
270 | * there is no handler or the handler was | ||
271 | * registered with SA_RESTART | ||
272 | */ | ||
273 | if (!(ka->sa.sa_flags & SA_RESTART)) { | ||
274 | regs->r0 = -EINTR; | ||
275 | break; | ||
276 | } | ||
277 | /* fallthrough */ | ||
278 | |||
279 | case -ERESTARTNOINTR: | ||
280 | /* | ||
281 | * ERESTARTNOINTR means that the syscall should | ||
282 | * be called again after the signal handler returns. | ||
283 | * Setup reg state just as it was before doing the trap | ||
284 | * r0 has been clobbered with sys call ret code thus it | ||
285 | * needs to be reloaded with orig first arg to syscall | ||
286 | * in orig_r0. Rest of relevant reg-file: | ||
287 | * r8 (syscall num) and (r1 - r7) will be reset to | ||
288 | * their orig user space value when we ret from kernel | ||
289 | */ | ||
290 | regs->r0 = regs->orig_r0; | ||
291 | regs->ret -= 4; | ||
292 | break; | ||
293 | } | ||
294 | } | ||
295 | |||
296 | /* | ||
297 | * OK, we're invoking a handler | ||
298 | */ | ||
299 | static void | ||
300 | handle_signal(unsigned long sig, struct k_sigaction *ka, siginfo_t *info, | ||
301 | struct pt_regs *regs) | ||
302 | { | ||
303 | sigset_t *oldset = sigmask_to_save(); | ||
304 | int ret; | ||
305 | |||
306 | /* Set up the stack frame */ | ||
307 | ret = setup_rt_frame(sig, ka, info, oldset, regs); | ||
308 | |||
309 | if (ret) | ||
310 | force_sigsegv(sig, current); | ||
311 | else | ||
312 | signal_delivered(sig, info, ka, regs, 0); | ||
313 | } | ||
314 | |||
315 | void do_signal(struct pt_regs *regs) | ||
316 | { | ||
317 | struct k_sigaction ka; | ||
318 | siginfo_t info; | ||
319 | int signr; | ||
320 | int restart_scall; | ||
321 | |||
322 | signr = get_signal_to_deliver(&info, &ka, regs, NULL); | ||
323 | |||
324 | restart_scall = in_syscall(regs) && syscall_restartable(regs); | ||
325 | |||
326 | if (signr > 0) { | ||
327 | if (restart_scall) { | ||
328 | arc_restart_syscall(&ka, regs); | ||
329 | syscall_wont_restart(regs); /* No more restarts */ | ||
330 | } | ||
331 | handle_signal(signr, &ka, &info, regs); | ||
332 | return; | ||
333 | } | ||
334 | |||
335 | if (restart_scall) { | ||
336 | /* No handler for syscall: restart it */ | ||
337 | if (regs->r0 == -ERESTARTNOHAND || | ||
338 | regs->r0 == -ERESTARTSYS || regs->r0 == -ERESTARTNOINTR) { | ||
339 | regs->r0 = regs->orig_r0; | ||
340 | regs->ret -= 4; | ||
341 | } else if (regs->r0 == -ERESTART_RESTARTBLOCK) { | ||
342 | regs->r8 = __NR_restart_syscall; | ||
343 | regs->ret -= 4; | ||
344 | } | ||
345 | syscall_wont_restart(regs); /* No more restarts */ | ||
346 | } | ||
347 | |||
348 | /* If there's no signal to deliver, restore the saved sigmask back */ | ||
349 | restore_saved_sigmask(); | ||
350 | } | ||
351 | |||
352 | void do_notify_resume(struct pt_regs *regs) | ||
353 | { | ||
354 | /* | ||
355 | * ASM glue gaurantees that this is only called when returning to | ||
356 | * user mode | ||
357 | */ | ||
358 | if (test_and_clear_thread_flag(TIF_NOTIFY_RESUME)) | ||
359 | tracehook_notify_resume(regs); | ||
360 | } | ||
diff --git a/arch/arc/kernel/smp.c b/arch/arc/kernel/smp.c new file mode 100644 index 000000000000..3af3e06dcf02 --- /dev/null +++ b/arch/arc/kernel/smp.c | |||
@@ -0,0 +1,332 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * RajeshwarR: Dec 11, 2007 | ||
9 | * -- Added support for Inter Processor Interrupts | ||
10 | * | ||
11 | * Vineetg: Nov 1st, 2007 | ||
12 | * -- Initial Write (Borrowed heavily from ARM) | ||
13 | */ | ||
14 | |||
15 | #include <linux/module.h> | ||
16 | #include <linux/init.h> | ||
17 | #include <linux/spinlock.h> | ||
18 | #include <linux/sched.h> | ||
19 | #include <linux/interrupt.h> | ||
20 | #include <linux/profile.h> | ||
21 | #include <linux/errno.h> | ||
22 | #include <linux/err.h> | ||
23 | #include <linux/mm.h> | ||
24 | #include <linux/cpu.h> | ||
25 | #include <linux/smp.h> | ||
26 | #include <linux/irq.h> | ||
27 | #include <linux/delay.h> | ||
28 | #include <linux/atomic.h> | ||
29 | #include <linux/percpu.h> | ||
30 | #include <linux/cpumask.h> | ||
31 | #include <linux/spinlock_types.h> | ||
32 | #include <linux/reboot.h> | ||
33 | #include <asm/processor.h> | ||
34 | #include <asm/setup.h> | ||
35 | #include <asm/mach_desc.h> | ||
36 | |||
37 | arch_spinlock_t smp_atomic_ops_lock = __ARCH_SPIN_LOCK_UNLOCKED; | ||
38 | arch_spinlock_t smp_bitops_lock = __ARCH_SPIN_LOCK_UNLOCKED; | ||
39 | |||
40 | struct plat_smp_ops plat_smp_ops; | ||
41 | |||
42 | /* XXX: per cpu ? Only needed once in early seconday boot */ | ||
43 | struct task_struct *secondary_idle_tsk; | ||
44 | |||
45 | /* Called from start_kernel */ | ||
46 | void __init smp_prepare_boot_cpu(void) | ||
47 | { | ||
48 | } | ||
49 | |||
50 | /* | ||
51 | * Initialise the CPU possible map early - this describes the CPUs | ||
52 | * which may be present or become present in the system. | ||
53 | */ | ||
54 | void __init smp_init_cpus(void) | ||
55 | { | ||
56 | unsigned int i; | ||
57 | |||
58 | for (i = 0; i < NR_CPUS; i++) | ||
59 | set_cpu_possible(i, true); | ||
60 | } | ||
61 | |||
62 | /* called from init ( ) => process 1 */ | ||
63 | void __init smp_prepare_cpus(unsigned int max_cpus) | ||
64 | { | ||
65 | int i; | ||
66 | |||
67 | /* | ||
68 | * Initialise the present map, which describes the set of CPUs | ||
69 | * actually populated at the present time. | ||
70 | */ | ||
71 | for (i = 0; i < max_cpus; i++) | ||
72 | set_cpu_present(i, true); | ||
73 | } | ||
74 | |||
75 | void __init smp_cpus_done(unsigned int max_cpus) | ||
76 | { | ||
77 | |||
78 | } | ||
79 | |||
80 | /* | ||
81 | * After power-up, a non Master CPU needs to wait for Master to kick start it | ||
82 | * | ||
83 | * The default implementation halts | ||
84 | * | ||
85 | * This relies on platform specific support allowing Master to directly set | ||
86 | * this CPU's PC (to be @first_lines_of_secondary() and kick start it. | ||
87 | * | ||
88 | * In lack of such h/w assist, platforms can override this function | ||
89 | * - make this function busy-spin on a token, eventually set by Master | ||
90 | * (from arc_platform_smp_wakeup_cpu()) | ||
91 | * - Once token is available, jump to @first_lines_of_secondary | ||
92 | * (using inline asm). | ||
93 | * | ||
94 | * Alert: can NOT use stack here as it has not been determined/setup for CPU. | ||
95 | * If it turns out to be elaborate, it's better to code it in assembly | ||
96 | * | ||
97 | */ | ||
98 | void __attribute__((weak)) arc_platform_smp_wait_to_boot(int cpu) | ||
99 | { | ||
100 | /* | ||
101 | * As a hack for debugging - since debugger will single-step over the | ||
102 | * FLAG insn - wrap the halt itself it in a self loop | ||
103 | */ | ||
104 | __asm__ __volatile__( | ||
105 | "1: \n" | ||
106 | " flag 1 \n" | ||
107 | " b 1b \n"); | ||
108 | } | ||
109 | |||
110 | const char *arc_platform_smp_cpuinfo(void) | ||
111 | { | ||
112 | return plat_smp_ops.info; | ||
113 | } | ||
114 | |||
115 | /* | ||
116 | * The very first "C" code executed by secondary | ||
117 | * Called from asm stub in head.S | ||
118 | * "current"/R25 already setup by low level boot code | ||
119 | */ | ||
120 | void __cpuinit start_kernel_secondary(void) | ||
121 | { | ||
122 | struct mm_struct *mm = &init_mm; | ||
123 | unsigned int cpu = smp_processor_id(); | ||
124 | |||
125 | /* MMU, Caches, Vector Table, Interrupts etc */ | ||
126 | setup_processor(); | ||
127 | |||
128 | atomic_inc(&mm->mm_users); | ||
129 | atomic_inc(&mm->mm_count); | ||
130 | current->active_mm = mm; | ||
131 | |||
132 | notify_cpu_starting(cpu); | ||
133 | set_cpu_online(cpu, true); | ||
134 | |||
135 | pr_info("## CPU%u LIVE ##: Executing Code...\n", cpu); | ||
136 | |||
137 | if (machine_desc->init_smp) | ||
138 | machine_desc->init_smp(smp_processor_id()); | ||
139 | |||
140 | arc_local_timer_setup(cpu); | ||
141 | |||
142 | local_irq_enable(); | ||
143 | preempt_disable(); | ||
144 | cpu_idle(); | ||
145 | } | ||
146 | |||
147 | /* | ||
148 | * Called from kernel_init( ) -> smp_init( ) - for each CPU | ||
149 | * | ||
150 | * At this point, Secondary Processor is "HALT"ed: | ||
151 | * -It booted, but was halted in head.S | ||
152 | * -It was configured to halt-on-reset | ||
153 | * So need to wake it up. | ||
154 | * | ||
155 | * Essential requirements being where to run from (PC) and stack (SP) | ||
156 | */ | ||
157 | int __cpuinit __cpu_up(unsigned int cpu, struct task_struct *idle) | ||
158 | { | ||
159 | unsigned long wait_till; | ||
160 | |||
161 | secondary_idle_tsk = idle; | ||
162 | |||
163 | pr_info("Idle Task [%d] %p", cpu, idle); | ||
164 | pr_info("Trying to bring up CPU%u ...\n", cpu); | ||
165 | |||
166 | if (plat_smp_ops.cpu_kick) | ||
167 | plat_smp_ops.cpu_kick(cpu, | ||
168 | (unsigned long)first_lines_of_secondary); | ||
169 | |||
170 | /* wait for 1 sec after kicking the secondary */ | ||
171 | wait_till = jiffies + HZ; | ||
172 | while (time_before(jiffies, wait_till)) { | ||
173 | if (cpu_online(cpu)) | ||
174 | break; | ||
175 | } | ||
176 | |||
177 | if (!cpu_online(cpu)) { | ||
178 | pr_info("Timeout: CPU%u FAILED to comeup !!!\n", cpu); | ||
179 | return -1; | ||
180 | } | ||
181 | |||
182 | secondary_idle_tsk = NULL; | ||
183 | |||
184 | return 0; | ||
185 | } | ||
186 | |||
187 | /* | ||
188 | * not supported here | ||
189 | */ | ||
190 | int __init setup_profiling_timer(unsigned int multiplier) | ||
191 | { | ||
192 | return -EINVAL; | ||
193 | } | ||
194 | |||
195 | /*****************************************************************************/ | ||
196 | /* Inter Processor Interrupt Handling */ | ||
197 | /*****************************************************************************/ | ||
198 | |||
199 | /* | ||
200 | * structures for inter-processor calls | ||
201 | * A Collection of single bit ipi messages | ||
202 | * | ||
203 | */ | ||
204 | |||
205 | /* | ||
206 | * TODO_rajesh investigate tlb message types. | ||
207 | * IPI Timer not needed because each ARC has an individual Interrupting Timer | ||
208 | */ | ||
209 | enum ipi_msg_type { | ||
210 | IPI_NOP = 0, | ||
211 | IPI_RESCHEDULE = 1, | ||
212 | IPI_CALL_FUNC, | ||
213 | IPI_CALL_FUNC_SINGLE, | ||
214 | IPI_CPU_STOP | ||
215 | }; | ||
216 | |||
217 | struct ipi_data { | ||
218 | unsigned long bits; | ||
219 | }; | ||
220 | |||
221 | static DEFINE_PER_CPU(struct ipi_data, ipi_data); | ||
222 | |||
223 | static void ipi_send_msg(const struct cpumask *callmap, enum ipi_msg_type msg) | ||
224 | { | ||
225 | unsigned long flags; | ||
226 | unsigned int cpu; | ||
227 | |||
228 | local_irq_save(flags); | ||
229 | |||
230 | for_each_cpu(cpu, callmap) { | ||
231 | struct ipi_data *ipi = &per_cpu(ipi_data, cpu); | ||
232 | set_bit(msg, &ipi->bits); | ||
233 | } | ||
234 | |||
235 | /* Call the platform specific cross-CPU call function */ | ||
236 | if (plat_smp_ops.ipi_send) | ||
237 | plat_smp_ops.ipi_send((void *)callmap); | ||
238 | |||
239 | local_irq_restore(flags); | ||
240 | } | ||
241 | |||
242 | void smp_send_reschedule(int cpu) | ||
243 | { | ||
244 | ipi_send_msg(cpumask_of(cpu), IPI_RESCHEDULE); | ||
245 | } | ||
246 | |||
247 | void smp_send_stop(void) | ||
248 | { | ||
249 | struct cpumask targets; | ||
250 | cpumask_copy(&targets, cpu_online_mask); | ||
251 | cpumask_clear_cpu(smp_processor_id(), &targets); | ||
252 | ipi_send_msg(&targets, IPI_CPU_STOP); | ||
253 | } | ||
254 | |||
255 | void arch_send_call_function_single_ipi(int cpu) | ||
256 | { | ||
257 | ipi_send_msg(cpumask_of(cpu), IPI_CALL_FUNC_SINGLE); | ||
258 | } | ||
259 | |||
260 | void arch_send_call_function_ipi_mask(const struct cpumask *mask) | ||
261 | { | ||
262 | ipi_send_msg(mask, IPI_CALL_FUNC); | ||
263 | } | ||
264 | |||
265 | /* | ||
266 | * ipi_cpu_stop - handle IPI from smp_send_stop() | ||
267 | */ | ||
268 | static void ipi_cpu_stop(unsigned int cpu) | ||
269 | { | ||
270 | machine_halt(); | ||
271 | } | ||
272 | |||
273 | static inline void __do_IPI(unsigned long *ops, struct ipi_data *ipi, int cpu) | ||
274 | { | ||
275 | unsigned long msg = 0; | ||
276 | |||
277 | do { | ||
278 | msg = find_next_bit(ops, BITS_PER_LONG, msg+1); | ||
279 | |||
280 | switch (msg) { | ||
281 | case IPI_RESCHEDULE: | ||
282 | scheduler_ipi(); | ||
283 | break; | ||
284 | |||
285 | case IPI_CALL_FUNC: | ||
286 | generic_smp_call_function_interrupt(); | ||
287 | break; | ||
288 | |||
289 | case IPI_CALL_FUNC_SINGLE: | ||
290 | generic_smp_call_function_single_interrupt(); | ||
291 | break; | ||
292 | |||
293 | case IPI_CPU_STOP: | ||
294 | ipi_cpu_stop(cpu); | ||
295 | break; | ||
296 | } | ||
297 | } while (msg < BITS_PER_LONG); | ||
298 | |||
299 | } | ||
300 | |||
301 | /* | ||
302 | * arch-common ISR to handle for inter-processor interrupts | ||
303 | * Has hooks for platform specific IPI | ||
304 | */ | ||
305 | irqreturn_t do_IPI(int irq, void *dev_id) | ||
306 | { | ||
307 | int cpu = smp_processor_id(); | ||
308 | struct ipi_data *ipi = &per_cpu(ipi_data, cpu); | ||
309 | unsigned long ops; | ||
310 | |||
311 | if (plat_smp_ops.ipi_clear) | ||
312 | plat_smp_ops.ipi_clear(cpu, irq); | ||
313 | |||
314 | /* | ||
315 | * XXX: is this loop really needed | ||
316 | * And do we need to move ipi_clean inside | ||
317 | */ | ||
318 | while ((ops = xchg(&ipi->bits, 0)) != 0) | ||
319 | __do_IPI(&ops, ipi, cpu); | ||
320 | |||
321 | return IRQ_HANDLED; | ||
322 | } | ||
323 | |||
324 | /* | ||
325 | * API called by platform code to hookup arch-common ISR to their IPI IRQ | ||
326 | */ | ||
327 | static DEFINE_PER_CPU(int, ipi_dev); | ||
328 | int smp_ipi_irq_setup(int cpu, int irq) | ||
329 | { | ||
330 | int *dev_id = &per_cpu(ipi_dev, smp_processor_id()); | ||
331 | return request_percpu_irq(irq, do_IPI, "IPI Interrupt", dev_id); | ||
332 | } | ||
diff --git a/arch/arc/kernel/stacktrace.c b/arch/arc/kernel/stacktrace.c new file mode 100644 index 000000000000..a63ff842564b --- /dev/null +++ b/arch/arc/kernel/stacktrace.c | |||
@@ -0,0 +1,254 @@ | |||
1 | /* | ||
2 | * stacktrace.c : stacktracing APIs needed by rest of kernel | ||
3 | * (wrappers over ARC dwarf based unwinder) | ||
4 | * | ||
5 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
6 | * | ||
7 | * This program is free software; you can redistribute it and/or modify | ||
8 | * it under the terms of the GNU General Public License version 2 as | ||
9 | * published by the Free Software Foundation. | ||
10 | * | ||
11 | * vineetg: aug 2009 | ||
12 | * -Implemented CONFIG_STACKTRACE APIs, primarily save_stack_trace_tsk( ) | ||
13 | * for displaying task's kernel mode call stack in /proc/<pid>/stack | ||
14 | * -Iterator based approach to have single copy of unwinding core and APIs | ||
15 | * needing unwinding, implement the logic in iterator regarding: | ||
16 | * = which frame onwards to start capture | ||
17 | * = which frame to stop capturing (wchan) | ||
18 | * = specifics of data structs where trace is saved(CONFIG_STACKTRACE etc) | ||
19 | * | ||
20 | * vineetg: March 2009 | ||
21 | * -Implemented correct versions of thread_saved_pc() and get_wchan() | ||
22 | * | ||
23 | * rajeshwarr: 2008 | ||
24 | * -Initial implementation | ||
25 | */ | ||
26 | |||
27 | #include <linux/ptrace.h> | ||
28 | #include <linux/export.h> | ||
29 | #include <linux/stacktrace.h> | ||
30 | #include <linux/kallsyms.h> | ||
31 | #include <asm/arcregs.h> | ||
32 | #include <asm/unwind.h> | ||
33 | #include <asm/switch_to.h> | ||
34 | |||
35 | /*------------------------------------------------------------------------- | ||
36 | * Unwinder Iterator | ||
37 | *------------------------------------------------------------------------- | ||
38 | */ | ||
39 | |||
40 | #ifdef CONFIG_ARC_DW2_UNWIND | ||
41 | |||
42 | static void seed_unwind_frame_info(struct task_struct *tsk, | ||
43 | struct pt_regs *regs, | ||
44 | struct unwind_frame_info *frame_info) | ||
45 | { | ||
46 | if (tsk == NULL && regs == NULL) { | ||
47 | unsigned long fp, sp, blink, ret; | ||
48 | frame_info->task = current; | ||
49 | |||
50 | __asm__ __volatile__( | ||
51 | "mov %0,r27\n\t" | ||
52 | "mov %1,r28\n\t" | ||
53 | "mov %2,r31\n\t" | ||
54 | "mov %3,r63\n\t" | ||
55 | : "=r"(fp), "=r"(sp), "=r"(blink), "=r"(ret) | ||
56 | ); | ||
57 | |||
58 | frame_info->regs.r27 = fp; | ||
59 | frame_info->regs.r28 = sp; | ||
60 | frame_info->regs.r31 = blink; | ||
61 | frame_info->regs.r63 = ret; | ||
62 | frame_info->call_frame = 0; | ||
63 | } else if (regs == NULL) { | ||
64 | |||
65 | frame_info->task = tsk; | ||
66 | |||
67 | frame_info->regs.r27 = KSTK_FP(tsk); | ||
68 | frame_info->regs.r28 = KSTK_ESP(tsk); | ||
69 | frame_info->regs.r31 = KSTK_BLINK(tsk); | ||
70 | frame_info->regs.r63 = (unsigned int)__switch_to; | ||
71 | |||
72 | /* In the prologue of __switch_to, first FP is saved on stack | ||
73 | * and then SP is copied to FP. Dwarf assumes cfa as FP based | ||
74 | * but we didn't save FP. The value retrieved above is FP's | ||
75 | * state in previous frame. | ||
76 | * As a work around for this, we unwind from __switch_to start | ||
77 | * and adjust SP accordingly. The other limitation is that | ||
78 | * __switch_to macro is dwarf rules are not generated for inline | ||
79 | * assembly code | ||
80 | */ | ||
81 | frame_info->regs.r27 = 0; | ||
82 | frame_info->regs.r28 += 64; | ||
83 | frame_info->call_frame = 0; | ||
84 | |||
85 | } else { | ||
86 | frame_info->task = tsk; | ||
87 | |||
88 | frame_info->regs.r27 = regs->fp; | ||
89 | frame_info->regs.r28 = regs->sp; | ||
90 | frame_info->regs.r31 = regs->blink; | ||
91 | frame_info->regs.r63 = regs->ret; | ||
92 | frame_info->call_frame = 0; | ||
93 | } | ||
94 | } | ||
95 | |||
96 | #endif | ||
97 | |||
98 | static noinline unsigned int | ||
99 | arc_unwind_core(struct task_struct *tsk, struct pt_regs *regs, | ||
100 | int (*consumer_fn) (unsigned int, void *), void *arg) | ||
101 | { | ||
102 | #ifdef CONFIG_ARC_DW2_UNWIND | ||
103 | int ret = 0; | ||
104 | unsigned int address; | ||
105 | struct unwind_frame_info frame_info; | ||
106 | |||
107 | seed_unwind_frame_info(tsk, regs, &frame_info); | ||
108 | |||
109 | while (1) { | ||
110 | address = UNW_PC(&frame_info); | ||
111 | |||
112 | if (address && __kernel_text_address(address)) { | ||
113 | if (consumer_fn(address, arg) == -1) | ||
114 | break; | ||
115 | } | ||
116 | |||
117 | ret = arc_unwind(&frame_info); | ||
118 | |||
119 | if (ret == 0) { | ||
120 | frame_info.regs.r63 = frame_info.regs.r31; | ||
121 | continue; | ||
122 | } else { | ||
123 | break; | ||
124 | } | ||
125 | } | ||
126 | |||
127 | return address; /* return the last address it saw */ | ||
128 | #else | ||
129 | /* On ARC, only Dward based unwinder works. fp based backtracing is | ||
130 | * not possible (-fno-omit-frame-pointer) because of the way function | ||
131 | * prelogue is setup (callee regs saved and then fp set and not other | ||
132 | * way around | ||
133 | */ | ||
134 | pr_warn("CONFIG_ARC_DW2_UNWIND needs to be enabled\n"); | ||
135 | return 0; | ||
136 | |||
137 | #endif | ||
138 | } | ||
139 | |||
140 | /*------------------------------------------------------------------------- | ||
141 | * callbacks called by unwinder iterator to implement kernel APIs | ||
142 | * | ||
143 | * The callback can return -1 to force the iterator to stop, which by default | ||
144 | * keeps going till the bottom-most frame. | ||
145 | *------------------------------------------------------------------------- | ||
146 | */ | ||
147 | |||
148 | /* Call-back which plugs into unwinding core to dump the stack in | ||
149 | * case of panic/OOPs/BUG etc | ||
150 | */ | ||
151 | static int __print_sym(unsigned int address, void *unused) | ||
152 | { | ||
153 | __print_symbol(" %s\n", address); | ||
154 | return 0; | ||
155 | } | ||
156 | |||
157 | #ifdef CONFIG_STACKTRACE | ||
158 | |||
159 | /* Call-back which plugs into unwinding core to capture the | ||
160 | * traces needed by kernel on /proc/<pid>/stack | ||
161 | */ | ||
162 | static int __collect_all(unsigned int address, void *arg) | ||
163 | { | ||
164 | struct stack_trace *trace = arg; | ||
165 | |||
166 | if (trace->skip > 0) | ||
167 | trace->skip--; | ||
168 | else | ||
169 | trace->entries[trace->nr_entries++] = address; | ||
170 | |||
171 | if (trace->nr_entries >= trace->max_entries) | ||
172 | return -1; | ||
173 | |||
174 | return 0; | ||
175 | } | ||
176 | |||
177 | static int __collect_all_but_sched(unsigned int address, void *arg) | ||
178 | { | ||
179 | struct stack_trace *trace = arg; | ||
180 | |||
181 | if (in_sched_functions(address)) | ||
182 | return 0; | ||
183 | |||
184 | if (trace->skip > 0) | ||
185 | trace->skip--; | ||
186 | else | ||
187 | trace->entries[trace->nr_entries++] = address; | ||
188 | |||
189 | if (trace->nr_entries >= trace->max_entries) | ||
190 | return -1; | ||
191 | |||
192 | return 0; | ||
193 | } | ||
194 | |||
195 | #endif | ||
196 | |||
197 | static int __get_first_nonsched(unsigned int address, void *unused) | ||
198 | { | ||
199 | if (in_sched_functions(address)) | ||
200 | return 0; | ||
201 | |||
202 | return -1; | ||
203 | } | ||
204 | |||
205 | /*------------------------------------------------------------------------- | ||
206 | * APIs expected by various kernel sub-systems | ||
207 | *------------------------------------------------------------------------- | ||
208 | */ | ||
209 | |||
210 | noinline void show_stacktrace(struct task_struct *tsk, struct pt_regs *regs) | ||
211 | { | ||
212 | pr_info("\nStack Trace:\n"); | ||
213 | arc_unwind_core(tsk, regs, __print_sym, NULL); | ||
214 | } | ||
215 | EXPORT_SYMBOL(show_stacktrace); | ||
216 | |||
217 | /* Expected by sched Code */ | ||
218 | void show_stack(struct task_struct *tsk, unsigned long *sp) | ||
219 | { | ||
220 | show_stacktrace(tsk, NULL); | ||
221 | } | ||
222 | |||
223 | /* Expected by Rest of kernel code */ | ||
224 | void dump_stack(void) | ||
225 | { | ||
226 | show_stacktrace(NULL, NULL); | ||
227 | } | ||
228 | EXPORT_SYMBOL(dump_stack); | ||
229 | |||
230 | /* Another API expected by schedular, shows up in "ps" as Wait Channel | ||
231 | * Ofcourse just returning schedule( ) would be pointless so unwind until | ||
232 | * the function is not in schedular code | ||
233 | */ | ||
234 | unsigned int get_wchan(struct task_struct *tsk) | ||
235 | { | ||
236 | return arc_unwind_core(tsk, NULL, __get_first_nonsched, NULL); | ||
237 | } | ||
238 | |||
239 | #ifdef CONFIG_STACKTRACE | ||
240 | |||
241 | /* | ||
242 | * API required by CONFIG_STACKTRACE, CONFIG_LATENCYTOP. | ||
243 | * A typical use is when /proc/<pid>/stack is queried by userland | ||
244 | */ | ||
245 | void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace) | ||
246 | { | ||
247 | arc_unwind_core(tsk, NULL, __collect_all_but_sched, trace); | ||
248 | } | ||
249 | |||
250 | void save_stack_trace(struct stack_trace *trace) | ||
251 | { | ||
252 | arc_unwind_core(current, NULL, __collect_all, trace); | ||
253 | } | ||
254 | #endif | ||
diff --git a/arch/arc/kernel/sys.c b/arch/arc/kernel/sys.c new file mode 100644 index 000000000000..f6bdd07583f3 --- /dev/null +++ b/arch/arc/kernel/sys.c | |||
@@ -0,0 +1,18 @@ | |||
1 | |||
2 | #include <linux/syscalls.h> | ||
3 | #include <linux/signal.h> | ||
4 | #include <linux/unistd.h> | ||
5 | |||
6 | #include <asm/syscalls.h> | ||
7 | |||
8 | #define sys_clone sys_clone_wrapper | ||
9 | #define sys_fork sys_fork_wrapper | ||
10 | #define sys_vfork sys_vfork_wrapper | ||
11 | |||
12 | #undef __SYSCALL | ||
13 | #define __SYSCALL(nr, call) [nr] = (call), | ||
14 | |||
15 | void *sys_call_table[NR_syscalls] = { | ||
16 | [0 ... NR_syscalls-1] = sys_ni_syscall, | ||
17 | #include <asm/unistd.h> | ||
18 | }; | ||
diff --git a/arch/arc/kernel/time.c b/arch/arc/kernel/time.c new file mode 100644 index 000000000000..f13f72807aa5 --- /dev/null +++ b/arch/arc/kernel/time.c | |||
@@ -0,0 +1,265 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * vineetg: Jan 1011 | ||
9 | * -sched_clock( ) no longer jiffies based. Uses the same clocksource | ||
10 | * as gtod | ||
11 | * | ||
12 | * Rajeshwarr/Vineetg: Mar 2008 | ||
13 | * -Implemented CONFIG_GENERIC_TIME (rather deleted arch specific code) | ||
14 | * for arch independent gettimeofday() | ||
15 | * -Implemented CONFIG_GENERIC_CLOCKEVENTS as base for hrtimers | ||
16 | * | ||
17 | * Vineetg: Mar 2008: Forked off from time.c which now is time-jiff.c | ||
18 | */ | ||
19 | |||
20 | /* ARC700 has two 32bit independent prog Timers: TIMER0 and TIMER1 | ||
21 | * Each can programmed to go from @count to @limit and optionally | ||
22 | * interrupt when that happens. | ||
23 | * A write to Control Register clears the Interrupt | ||
24 | * | ||
25 | * We've designated TIMER0 for events (clockevents) | ||
26 | * while TIMER1 for free running (clocksource) | ||
27 | * | ||
28 | * Newer ARC700 cores have 64bit clk fetching RTSC insn, preferred over TIMER1 | ||
29 | */ | ||
30 | |||
31 | #include <linux/spinlock.h> | ||
32 | #include <linux/interrupt.h> | ||
33 | #include <linux/module.h> | ||
34 | #include <linux/sched.h> | ||
35 | #include <linux/kernel.h> | ||
36 | #include <linux/interrupt.h> | ||
37 | #include <linux/time.h> | ||
38 | #include <linux/init.h> | ||
39 | #include <linux/timex.h> | ||
40 | #include <linux/profile.h> | ||
41 | #include <linux/clocksource.h> | ||
42 | #include <linux/clockchips.h> | ||
43 | #include <asm/irq.h> | ||
44 | #include <asm/arcregs.h> | ||
45 | #include <asm/clk.h> | ||
46 | #include <asm/mach_desc.h> | ||
47 | |||
48 | #define ARC_TIMER_MAX 0xFFFFFFFF | ||
49 | |||
50 | /********** Clock Source Device *********/ | ||
51 | |||
52 | #ifdef CONFIG_ARC_HAS_RTSC | ||
53 | |||
54 | int __cpuinit arc_counter_setup(void) | ||
55 | { | ||
56 | /* RTSC insn taps into cpu clk, needs no setup */ | ||
57 | |||
58 | /* For SMP, only allowed if cross-core-sync, hence usable as cs */ | ||
59 | return 1; | ||
60 | } | ||
61 | |||
62 | static cycle_t arc_counter_read(struct clocksource *cs) | ||
63 | { | ||
64 | unsigned long flags; | ||
65 | union { | ||
66 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
67 | struct { u32 high, low; }; | ||
68 | #else | ||
69 | struct { u32 low, high; }; | ||
70 | #endif | ||
71 | cycle_t full; | ||
72 | } stamp; | ||
73 | |||
74 | flags = arch_local_irq_save(); | ||
75 | |||
76 | __asm__ __volatile( | ||
77 | " .extCoreRegister tsch, 58, r, cannot_shortcut \n" | ||
78 | " rtsc %0, 0 \n" | ||
79 | " mov %1, 0 \n" | ||
80 | : "=r" (stamp.low), "=r" (stamp.high)); | ||
81 | |||
82 | arch_local_irq_restore(flags); | ||
83 | |||
84 | return stamp.full; | ||
85 | } | ||
86 | |||
87 | static struct clocksource arc_counter = { | ||
88 | .name = "ARC RTSC", | ||
89 | .rating = 300, | ||
90 | .read = arc_counter_read, | ||
91 | .mask = CLOCKSOURCE_MASK(32), | ||
92 | .flags = CLOCK_SOURCE_IS_CONTINUOUS, | ||
93 | }; | ||
94 | |||
95 | #else /* !CONFIG_ARC_HAS_RTSC */ | ||
96 | |||
97 | static bool is_usable_as_clocksource(void) | ||
98 | { | ||
99 | #ifdef CONFIG_SMP | ||
100 | return 0; | ||
101 | #else | ||
102 | return 1; | ||
103 | #endif | ||
104 | } | ||
105 | |||
106 | /* | ||
107 | * set 32bit TIMER1 to keep counting monotonically and wraparound | ||
108 | */ | ||
109 | int __cpuinit arc_counter_setup(void) | ||
110 | { | ||
111 | write_aux_reg(ARC_REG_TIMER1_LIMIT, ARC_TIMER_MAX); | ||
112 | write_aux_reg(ARC_REG_TIMER1_CNT, 0); | ||
113 | write_aux_reg(ARC_REG_TIMER1_CTRL, TIMER_CTRL_NH); | ||
114 | |||
115 | return is_usable_as_clocksource(); | ||
116 | } | ||
117 | |||
118 | static cycle_t arc_counter_read(struct clocksource *cs) | ||
119 | { | ||
120 | return (cycle_t) read_aux_reg(ARC_REG_TIMER1_CNT); | ||
121 | } | ||
122 | |||
123 | static struct clocksource arc_counter = { | ||
124 | .name = "ARC Timer1", | ||
125 | .rating = 300, | ||
126 | .read = arc_counter_read, | ||
127 | .mask = CLOCKSOURCE_MASK(32), | ||
128 | .flags = CLOCK_SOURCE_IS_CONTINUOUS, | ||
129 | }; | ||
130 | |||
131 | #endif | ||
132 | |||
133 | /********** Clock Event Device *********/ | ||
134 | |||
135 | /* | ||
136 | * Arm the timer to interrupt after @limit cycles | ||
137 | * The distinction for oneshot/periodic is done in arc_event_timer_ack() below | ||
138 | */ | ||
139 | static void arc_timer_event_setup(unsigned int limit) | ||
140 | { | ||
141 | write_aux_reg(ARC_REG_TIMER0_LIMIT, limit); | ||
142 | write_aux_reg(ARC_REG_TIMER0_CNT, 0); /* start from 0 */ | ||
143 | |||
144 | write_aux_reg(ARC_REG_TIMER0_CTRL, TIMER_CTRL_IE | TIMER_CTRL_NH); | ||
145 | } | ||
146 | |||
147 | /* | ||
148 | * Acknowledge the interrupt (oneshot) and optionally re-arm it (periodic) | ||
149 | * -Any write to CTRL Reg will ack the intr (NH bit: Count when not halted) | ||
150 | * -Rearming is done by setting the IE bit | ||
151 | * | ||
152 | * Small optimisation: Normal code would have been | ||
153 | * if (irq_reenable) | ||
154 | * CTRL_REG = (IE | NH); | ||
155 | * else | ||
156 | * CTRL_REG = NH; | ||
157 | * However since IE is BIT0 we can fold the branch | ||
158 | */ | ||
159 | static void arc_timer_event_ack(unsigned int irq_reenable) | ||
160 | { | ||
161 | write_aux_reg(ARC_REG_TIMER0_CTRL, irq_reenable | TIMER_CTRL_NH); | ||
162 | } | ||
163 | |||
164 | static int arc_clkevent_set_next_event(unsigned long delta, | ||
165 | struct clock_event_device *dev) | ||
166 | { | ||
167 | arc_timer_event_setup(delta); | ||
168 | return 0; | ||
169 | } | ||
170 | |||
171 | static void arc_clkevent_set_mode(enum clock_event_mode mode, | ||
172 | struct clock_event_device *dev) | ||
173 | { | ||
174 | switch (mode) { | ||
175 | case CLOCK_EVT_MODE_PERIODIC: | ||
176 | arc_timer_event_setup(arc_get_core_freq() / HZ); | ||
177 | break; | ||
178 | case CLOCK_EVT_MODE_ONESHOT: | ||
179 | break; | ||
180 | default: | ||
181 | break; | ||
182 | } | ||
183 | |||
184 | return; | ||
185 | } | ||
186 | |||
187 | static DEFINE_PER_CPU(struct clock_event_device, arc_clockevent_device) = { | ||
188 | .name = "ARC Timer0", | ||
189 | .features = CLOCK_EVT_FEAT_ONESHOT | CLOCK_EVT_FEAT_PERIODIC, | ||
190 | .mode = CLOCK_EVT_MODE_UNUSED, | ||
191 | .rating = 300, | ||
192 | .irq = TIMER0_IRQ, /* hardwired, no need for resources */ | ||
193 | .set_next_event = arc_clkevent_set_next_event, | ||
194 | .set_mode = arc_clkevent_set_mode, | ||
195 | }; | ||
196 | |||
197 | static irqreturn_t timer_irq_handler(int irq, void *dev_id) | ||
198 | { | ||
199 | struct clock_event_device *clk = &__get_cpu_var(arc_clockevent_device); | ||
200 | |||
201 | arc_timer_event_ack(clk->mode == CLOCK_EVT_MODE_PERIODIC); | ||
202 | clk->event_handler(clk); | ||
203 | return IRQ_HANDLED; | ||
204 | } | ||
205 | |||
206 | static struct irqaction arc_timer_irq = { | ||
207 | .name = "Timer0 (clock-evt-dev)", | ||
208 | .flags = IRQF_TIMER | IRQF_PERCPU, | ||
209 | .handler = timer_irq_handler, | ||
210 | }; | ||
211 | |||
212 | /* | ||
213 | * Setup the local event timer for @cpu | ||
214 | * N.B. weak so that some exotic ARC SoCs can completely override it | ||
215 | */ | ||
216 | void __attribute__((weak)) __cpuinit arc_local_timer_setup(unsigned int cpu) | ||
217 | { | ||
218 | struct clock_event_device *clk = &per_cpu(arc_clockevent_device, cpu); | ||
219 | |||
220 | clockevents_calc_mult_shift(clk, arc_get_core_freq(), 5); | ||
221 | |||
222 | clk->max_delta_ns = clockevent_delta2ns(ARC_TIMER_MAX, clk); | ||
223 | clk->cpumask = cpumask_of(cpu); | ||
224 | |||
225 | clockevents_register_device(clk); | ||
226 | |||
227 | /* | ||
228 | * setup the per-cpu timer IRQ handler - for all cpus | ||
229 | * For non boot CPU explicitly unmask at intc | ||
230 | * setup_irq() -> .. -> irq_startup() already does this on boot-cpu | ||
231 | */ | ||
232 | if (!cpu) | ||
233 | setup_irq(TIMER0_IRQ, &arc_timer_irq); | ||
234 | else | ||
235 | arch_unmask_irq(TIMER0_IRQ); | ||
236 | } | ||
237 | |||
238 | /* | ||
239 | * Called from start_kernel() - boot CPU only | ||
240 | * | ||
241 | * -Sets up h/w timers as applicable on boot cpu | ||
242 | * -Also sets up any global state needed for timer subsystem: | ||
243 | * - for "counting" timer, registers a clocksource, usable across CPUs | ||
244 | * (provided that underlying counter h/w is synchronized across cores) | ||
245 | * - for "event" timer, sets up TIMER0 IRQ (as that is platform agnostic) | ||
246 | */ | ||
247 | void __init time_init(void) | ||
248 | { | ||
249 | /* | ||
250 | * sets up the timekeeping free-flowing counter which also returns | ||
251 | * whether the counter is usable as clocksource | ||
252 | */ | ||
253 | if (arc_counter_setup()) | ||
254 | /* | ||
255 | * CLK upto 4.29 GHz can be safely represented in 32 bits | ||
256 | * because Max 32 bit number is 4,294,967,295 | ||
257 | */ | ||
258 | clocksource_register_hz(&arc_counter, arc_get_core_freq()); | ||
259 | |||
260 | /* sets up the periodic event timer */ | ||
261 | arc_local_timer_setup(smp_processor_id()); | ||
262 | |||
263 | if (machine_desc->init_time) | ||
264 | machine_desc->init_time(); | ||
265 | } | ||
diff --git a/arch/arc/kernel/traps.c b/arch/arc/kernel/traps.c new file mode 100644 index 000000000000..7496995371e8 --- /dev/null +++ b/arch/arc/kernel/traps.c | |||
@@ -0,0 +1,170 @@ | |||
1 | /* | ||
2 | * Traps/Non-MMU Exception handling for ARC | ||
3 | * | ||
4 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | * | ||
10 | * vineetg: May 2011 | ||
11 | * -user-space unaligned access emulation | ||
12 | * | ||
13 | * Rahul Trivedi: Codito Technologies 2004 | ||
14 | */ | ||
15 | |||
16 | #include <linux/sched.h> | ||
17 | #include <linux/kdebug.h> | ||
18 | #include <linux/uaccess.h> | ||
19 | #include <asm/ptrace.h> | ||
20 | #include <asm/setup.h> | ||
21 | #include <asm/kprobes.h> | ||
22 | #include <asm/unaligned.h> | ||
23 | #include <asm/kgdb.h> | ||
24 | |||
25 | void __init trap_init(void) | ||
26 | { | ||
27 | return; | ||
28 | } | ||
29 | |||
30 | void die(const char *str, struct pt_regs *regs, unsigned long address, | ||
31 | unsigned long cause_reg) | ||
32 | { | ||
33 | show_kernel_fault_diag(str, regs, address, cause_reg); | ||
34 | |||
35 | /* DEAD END */ | ||
36 | __asm__("flag 1"); | ||
37 | } | ||
38 | |||
39 | /* | ||
40 | * Helper called for bulk of exceptions NOT needing specific handling | ||
41 | * -for user faults enqueues requested signal | ||
42 | * -for kernel, chk if due to copy_(to|from)_user, otherwise die() | ||
43 | */ | ||
44 | static noinline int handle_exception(unsigned long cause, char *str, | ||
45 | struct pt_regs *regs, siginfo_t *info) | ||
46 | { | ||
47 | if (user_mode(regs)) { | ||
48 | struct task_struct *tsk = current; | ||
49 | |||
50 | tsk->thread.fault_address = (__force unsigned int)info->si_addr; | ||
51 | tsk->thread.cause_code = cause; | ||
52 | |||
53 | force_sig_info(info->si_signo, info, tsk); | ||
54 | |||
55 | } else { | ||
56 | /* If not due to copy_(to|from)_user, we are doomed */ | ||
57 | if (fixup_exception(regs)) | ||
58 | return 0; | ||
59 | |||
60 | die(str, regs, (unsigned long)info->si_addr, cause); | ||
61 | } | ||
62 | |||
63 | return 1; | ||
64 | } | ||
65 | |||
66 | #define DO_ERROR_INFO(signr, str, name, sicode) \ | ||
67 | int name(unsigned long cause, unsigned long address, struct pt_regs *regs) \ | ||
68 | { \ | ||
69 | siginfo_t info = { \ | ||
70 | .si_signo = signr, \ | ||
71 | .si_errno = 0, \ | ||
72 | .si_code = sicode, \ | ||
73 | .si_addr = (void __user *)address, \ | ||
74 | }; \ | ||
75 | return handle_exception(cause, str, regs, &info);\ | ||
76 | } | ||
77 | |||
78 | /* | ||
79 | * Entry points for exceptions NOT needing specific handling | ||
80 | */ | ||
81 | DO_ERROR_INFO(SIGILL, "Priv Op/Disabled Extn", do_privilege_fault, ILL_PRVOPC) | ||
82 | DO_ERROR_INFO(SIGILL, "Invalid Extn Insn", do_extension_fault, ILL_ILLOPC) | ||
83 | DO_ERROR_INFO(SIGILL, "Illegal Insn (or Seq)", insterror_is_error, ILL_ILLOPC) | ||
84 | DO_ERROR_INFO(SIGBUS, "Invalid Mem Access", do_memory_error, BUS_ADRERR) | ||
85 | DO_ERROR_INFO(SIGTRAP, "Breakpoint Set", trap_is_brkpt, TRAP_BRKPT) | ||
86 | |||
87 | #ifdef CONFIG_ARC_MISALIGN_ACCESS | ||
88 | /* | ||
89 | * Entry Point for Misaligned Data access Exception, for emulating in software | ||
90 | */ | ||
91 | int do_misaligned_access(unsigned long cause, unsigned long address, | ||
92 | struct pt_regs *regs, struct callee_regs *cregs) | ||
93 | { | ||
94 | if (misaligned_fixup(address, regs, cause, cregs) != 0) { | ||
95 | siginfo_t info; | ||
96 | |||
97 | info.si_signo = SIGBUS; | ||
98 | info.si_errno = 0; | ||
99 | info.si_code = BUS_ADRALN; | ||
100 | info.si_addr = (void __user *)address; | ||
101 | return handle_exception(cause, "Misaligned Access", regs, | ||
102 | &info); | ||
103 | } | ||
104 | return 0; | ||
105 | } | ||
106 | |||
107 | #else | ||
108 | DO_ERROR_INFO(SIGSEGV, "Misaligned Access", do_misaligned_access, SEGV_ACCERR) | ||
109 | #endif | ||
110 | |||
111 | /* | ||
112 | * Entry point for miscll errors such as Nested Exceptions | ||
113 | * -Duplicate TLB entry is handled seperately though | ||
114 | */ | ||
115 | void do_machine_check_fault(unsigned long cause, unsigned long address, | ||
116 | struct pt_regs *regs) | ||
117 | { | ||
118 | die("Machine Check Exception", regs, address, cause); | ||
119 | } | ||
120 | |||
121 | |||
122 | /* | ||
123 | * Entry point for traps induced by ARCompact TRAP_S <n> insn | ||
124 | * This is same family as TRAP0/SWI insn (use the same vector). | ||
125 | * The only difference being SWI insn take no operand, while TRAP_S does | ||
126 | * which reflects in ECR Reg as 8 bit param. | ||
127 | * Thus TRAP_S <n> can be used for specific purpose | ||
128 | * -1 used for software breakpointing (gdb) | ||
129 | * -2 used by kprobes | ||
130 | */ | ||
131 | void do_non_swi_trap(unsigned long cause, unsigned long address, | ||
132 | struct pt_regs *regs) | ||
133 | { | ||
134 | unsigned int param = cause & 0xff; | ||
135 | |||
136 | switch (param) { | ||
137 | case 1: | ||
138 | trap_is_brkpt(cause, address, regs); | ||
139 | break; | ||
140 | |||
141 | case 2: | ||
142 | trap_is_kprobe(param, address, regs); | ||
143 | break; | ||
144 | |||
145 | case 3: | ||
146 | case 4: | ||
147 | kgdb_trap(regs, param); | ||
148 | break; | ||
149 | |||
150 | default: | ||
151 | break; | ||
152 | } | ||
153 | } | ||
154 | |||
155 | /* | ||
156 | * Entry point for Instruction Error Exception | ||
157 | * -For a corner case, ARC kprobes implementation resorts to using | ||
158 | * this exception, hence the check | ||
159 | */ | ||
160 | void do_insterror_or_kprobe(unsigned long cause, | ||
161 | unsigned long address, | ||
162 | struct pt_regs *regs) | ||
163 | { | ||
164 | /* Check if this exception is caused by kprobes */ | ||
165 | if (notify_die(DIE_IERR, "kprobe_ierr", regs, address, | ||
166 | cause, SIGILL) == NOTIFY_STOP) | ||
167 | return; | ||
168 | |||
169 | insterror_is_error(cause, address, regs); | ||
170 | } | ||
diff --git a/arch/arc/kernel/troubleshoot.c b/arch/arc/kernel/troubleshoot.c new file mode 100644 index 000000000000..7c10873c311f --- /dev/null +++ b/arch/arc/kernel/troubleshoot.c | |||
@@ -0,0 +1,322 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | */ | ||
7 | |||
8 | #include <linux/ptrace.h> | ||
9 | #include <linux/module.h> | ||
10 | #include <linux/mm.h> | ||
11 | #include <linux/fs.h> | ||
12 | #include <linux/kdev_t.h> | ||
13 | #include <linux/fs_struct.h> | ||
14 | #include <linux/proc_fs.h> | ||
15 | #include <linux/file.h> | ||
16 | #include <asm/arcregs.h> | ||
17 | |||
18 | /* | ||
19 | * Common routine to print scratch regs (r0-r12) or callee regs (r13-r25) | ||
20 | * -Prints 3 regs per line and a CR. | ||
21 | * -To continue, callee regs right after scratch, special handling of CR | ||
22 | */ | ||
23 | static noinline void print_reg_file(long *reg_rev, int start_num) | ||
24 | { | ||
25 | unsigned int i; | ||
26 | char buf[512]; | ||
27 | int n = 0, len = sizeof(buf); | ||
28 | |||
29 | /* weird loop because pt_regs regs rev r12..r0, r25..r13 */ | ||
30 | for (i = start_num; i < start_num + 13; i++) { | ||
31 | n += scnprintf(buf + n, len - n, "r%02u: 0x%08lx\t", | ||
32 | i, (unsigned long)*reg_rev); | ||
33 | |||
34 | if (((i + 1) % 3) == 0) | ||
35 | n += scnprintf(buf + n, len - n, "\n"); | ||
36 | |||
37 | reg_rev--; | ||
38 | } | ||
39 | |||
40 | if (start_num != 0) | ||
41 | n += scnprintf(buf + n, len - n, "\n\n"); | ||
42 | |||
43 | pr_info("%s", buf); | ||
44 | } | ||
45 | |||
46 | static void show_callee_regs(struct callee_regs *cregs) | ||
47 | { | ||
48 | print_reg_file(&(cregs->r13), 13); | ||
49 | } | ||
50 | |||
51 | void print_task_path_n_nm(struct task_struct *tsk, char *buf) | ||
52 | { | ||
53 | struct path path; | ||
54 | char *path_nm = NULL; | ||
55 | struct mm_struct *mm; | ||
56 | struct file *exe_file; | ||
57 | |||
58 | mm = get_task_mm(tsk); | ||
59 | if (!mm) | ||
60 | goto done; | ||
61 | |||
62 | exe_file = get_mm_exe_file(mm); | ||
63 | mmput(mm); | ||
64 | |||
65 | if (exe_file) { | ||
66 | path = exe_file->f_path; | ||
67 | path_get(&exe_file->f_path); | ||
68 | fput(exe_file); | ||
69 | path_nm = d_path(&path, buf, 255); | ||
70 | path_put(&path); | ||
71 | } | ||
72 | |||
73 | done: | ||
74 | pr_info("%s, TGID %u\n", path_nm, tsk->tgid); | ||
75 | } | ||
76 | EXPORT_SYMBOL(print_task_path_n_nm); | ||
77 | |||
78 | static void show_faulting_vma(unsigned long address, char *buf) | ||
79 | { | ||
80 | struct vm_area_struct *vma; | ||
81 | struct inode *inode; | ||
82 | unsigned long ino = 0; | ||
83 | dev_t dev = 0; | ||
84 | char *nm = buf; | ||
85 | |||
86 | vma = find_vma(current->active_mm, address); | ||
87 | |||
88 | /* check against the find_vma( ) behaviour which returns the next VMA | ||
89 | * if the container VMA is not found | ||
90 | */ | ||
91 | if (vma && (vma->vm_start <= address)) { | ||
92 | struct file *file = vma->vm_file; | ||
93 | if (file) { | ||
94 | struct path *path = &file->f_path; | ||
95 | nm = d_path(path, buf, PAGE_SIZE - 1); | ||
96 | inode = vma->vm_file->f_path.dentry->d_inode; | ||
97 | dev = inode->i_sb->s_dev; | ||
98 | ino = inode->i_ino; | ||
99 | } | ||
100 | pr_info(" @off 0x%lx in [%s]\n" | ||
101 | " VMA: 0x%08lx to 0x%08lx\n\n", | ||
102 | address - vma->vm_start, nm, vma->vm_start, vma->vm_end); | ||
103 | } else | ||
104 | pr_info(" @No matching VMA found\n"); | ||
105 | } | ||
106 | |||
107 | static void show_ecr_verbose(struct pt_regs *regs) | ||
108 | { | ||
109 | unsigned int vec, cause_code, cause_reg; | ||
110 | unsigned long address; | ||
111 | |||
112 | cause_reg = current->thread.cause_code; | ||
113 | pr_info("\n[ECR]: 0x%08x => ", cause_reg); | ||
114 | |||
115 | /* For Data fault, this is data address not instruction addr */ | ||
116 | address = current->thread.fault_address; | ||
117 | |||
118 | vec = cause_reg >> 16; | ||
119 | cause_code = (cause_reg >> 8) & 0xFF; | ||
120 | |||
121 | /* For DTLB Miss or ProtV, display the memory involved too */ | ||
122 | if (vec == ECR_V_DTLB_MISS) { | ||
123 | pr_cont("Invalid (%s) @ 0x%08lx by insn @ 0x%08lx\n", | ||
124 | (cause_code == 0x01) ? "Read From" : | ||
125 | ((cause_code == 0x02) ? "Write to" : "EX"), | ||
126 | address, regs->ret); | ||
127 | } else if (vec == ECR_V_ITLB_MISS) { | ||
128 | pr_cont("Insn could not be fetched\n"); | ||
129 | } else if (vec == ECR_V_MACH_CHK) { | ||
130 | pr_cont("%s\n", (cause_code == 0x0) ? | ||
131 | "Double Fault" : "Other Fatal Err"); | ||
132 | |||
133 | } else if (vec == ECR_V_PROTV) { | ||
134 | if (cause_code == ECR_C_PROTV_INST_FETCH) | ||
135 | pr_cont("Execute from Non-exec Page\n"); | ||
136 | else if (cause_code == ECR_C_PROTV_LOAD) | ||
137 | pr_cont("Read from Non-readable Page\n"); | ||
138 | else if (cause_code == ECR_C_PROTV_STORE) | ||
139 | pr_cont("Write to Non-writable Page\n"); | ||
140 | else if (cause_code == ECR_C_PROTV_XCHG) | ||
141 | pr_cont("Data exchange protection violation\n"); | ||
142 | else if (cause_code == ECR_C_PROTV_MISALIG_DATA) | ||
143 | pr_cont("Misaligned r/w from 0x%08lx\n", address); | ||
144 | } else if (vec == ECR_V_INSN_ERR) { | ||
145 | pr_cont("Illegal Insn\n"); | ||
146 | } else { | ||
147 | pr_cont("Check Programmer's Manual\n"); | ||
148 | } | ||
149 | } | ||
150 | |||
151 | /************************************************************************ | ||
152 | * API called by rest of kernel | ||
153 | ***********************************************************************/ | ||
154 | |||
155 | void show_regs(struct pt_regs *regs) | ||
156 | { | ||
157 | struct task_struct *tsk = current; | ||
158 | struct callee_regs *cregs; | ||
159 | char *buf; | ||
160 | |||
161 | buf = (char *)__get_free_page(GFP_TEMPORARY); | ||
162 | if (!buf) | ||
163 | return; | ||
164 | |||
165 | print_task_path_n_nm(tsk, buf); | ||
166 | |||
167 | if (current->thread.cause_code) | ||
168 | show_ecr_verbose(regs); | ||
169 | |||
170 | pr_info("[EFA]: 0x%08lx\n", current->thread.fault_address); | ||
171 | pr_info("[ERET]: 0x%08lx (PC of Faulting Instr)\n", regs->ret); | ||
172 | |||
173 | show_faulting_vma(regs->ret, buf); /* faulting code, not data */ | ||
174 | |||
175 | /* can't use print_vma_addr() yet as it doesn't check for | ||
176 | * non-inclusive vma | ||
177 | */ | ||
178 | |||
179 | /* print special regs */ | ||
180 | pr_info("status32: 0x%08lx\n", regs->status32); | ||
181 | pr_info(" SP: 0x%08lx\tFP: 0x%08lx\n", regs->sp, regs->fp); | ||
182 | pr_info("BTA: 0x%08lx\tBLINK: 0x%08lx\n", | ||
183 | regs->bta, regs->blink); | ||
184 | pr_info("LPS: 0x%08lx\tLPE: 0x%08lx\tLPC: 0x%08lx\n", | ||
185 | regs->lp_start, regs->lp_end, regs->lp_count); | ||
186 | |||
187 | /* print regs->r0 thru regs->r12 | ||
188 | * Sequential printing was generating horrible code | ||
189 | */ | ||
190 | print_reg_file(&(regs->r0), 0); | ||
191 | |||
192 | /* If Callee regs were saved, display them too */ | ||
193 | cregs = (struct callee_regs *)current->thread.callee_reg; | ||
194 | if (cregs) | ||
195 | show_callee_regs(cregs); | ||
196 | |||
197 | free_page((unsigned long)buf); | ||
198 | } | ||
199 | |||
200 | void show_kernel_fault_diag(const char *str, struct pt_regs *regs, | ||
201 | unsigned long address, unsigned long cause_reg) | ||
202 | { | ||
203 | current->thread.fault_address = address; | ||
204 | current->thread.cause_code = cause_reg; | ||
205 | |||
206 | /* Caller and Callee regs */ | ||
207 | show_regs(regs); | ||
208 | |||
209 | /* Show stack trace if this Fatality happened in kernel mode */ | ||
210 | if (!user_mode(regs)) | ||
211 | show_stacktrace(current, regs); | ||
212 | } | ||
213 | |||
214 | #ifdef CONFIG_DEBUG_FS | ||
215 | |||
216 | #include <linux/module.h> | ||
217 | #include <linux/fs.h> | ||
218 | #include <linux/mount.h> | ||
219 | #include <linux/pagemap.h> | ||
220 | #include <linux/init.h> | ||
221 | #include <linux/namei.h> | ||
222 | #include <linux/debugfs.h> | ||
223 | |||
224 | static struct dentry *test_dentry; | ||
225 | static struct dentry *test_dir; | ||
226 | static struct dentry *test_u32_dentry; | ||
227 | |||
228 | static u32 clr_on_read = 1; | ||
229 | |||
230 | #ifdef CONFIG_ARC_DBG_TLB_MISS_COUNT | ||
231 | u32 numitlb, numdtlb, num_pte_not_present; | ||
232 | |||
233 | static int fill_display_data(char *kbuf) | ||
234 | { | ||
235 | size_t num = 0; | ||
236 | num += sprintf(kbuf + num, "I-TLB Miss %x\n", numitlb); | ||
237 | num += sprintf(kbuf + num, "D-TLB Miss %x\n", numdtlb); | ||
238 | num += sprintf(kbuf + num, "PTE not present %x\n", num_pte_not_present); | ||
239 | |||
240 | if (clr_on_read) | ||
241 | numitlb = numdtlb = num_pte_not_present = 0; | ||
242 | |||
243 | return num; | ||
244 | } | ||
245 | |||
246 | static int tlb_stats_open(struct inode *inode, struct file *file) | ||
247 | { | ||
248 | file->private_data = (void *)__get_free_page(GFP_KERNEL); | ||
249 | return 0; | ||
250 | } | ||
251 | |||
252 | /* called on user read(): display the couters */ | ||
253 | static ssize_t tlb_stats_output(struct file *file, /* file descriptor */ | ||
254 | char __user *user_buf, /* user buffer */ | ||
255 | size_t len, /* length of buffer */ | ||
256 | loff_t *offset) /* offset in the file */ | ||
257 | { | ||
258 | size_t num; | ||
259 | char *kbuf = (char *)file->private_data; | ||
260 | |||
261 | /* All of the data can he shoved in one iteration */ | ||
262 | if (*offset != 0) | ||
263 | return 0; | ||
264 | |||
265 | num = fill_display_data(kbuf); | ||
266 | |||
267 | /* simple_read_from_buffer() is helper for copy to user space | ||
268 | It copies up to @2 (num) bytes from kernel buffer @4 (kbuf) at offset | ||
269 | @3 (offset) into the user space address starting at @1 (user_buf). | ||
270 | @5 (len) is max size of user buffer | ||
271 | */ | ||
272 | return simple_read_from_buffer(user_buf, num, offset, kbuf, len); | ||
273 | } | ||
274 | |||
275 | /* called on user write : clears the counters */ | ||
276 | static ssize_t tlb_stats_clear(struct file *file, const char __user *user_buf, | ||
277 | size_t length, loff_t *offset) | ||
278 | { | ||
279 | numitlb = numdtlb = num_pte_not_present = 0; | ||
280 | return length; | ||
281 | } | ||
282 | |||
283 | static int tlb_stats_close(struct inode *inode, struct file *file) | ||
284 | { | ||
285 | free_page((unsigned long)(file->private_data)); | ||
286 | return 0; | ||
287 | } | ||
288 | |||
289 | static const struct file_operations tlb_stats_file_ops = { | ||
290 | .read = tlb_stats_output, | ||
291 | .write = tlb_stats_clear, | ||
292 | .open = tlb_stats_open, | ||
293 | .release = tlb_stats_close | ||
294 | }; | ||
295 | #endif | ||
296 | |||
297 | static int __init arc_debugfs_init(void) | ||
298 | { | ||
299 | test_dir = debugfs_create_dir("arc", NULL); | ||
300 | |||
301 | #ifdef CONFIG_ARC_DBG_TLB_MISS_COUNT | ||
302 | test_dentry = debugfs_create_file("tlb_stats", 0444, test_dir, NULL, | ||
303 | &tlb_stats_file_ops); | ||
304 | #endif | ||
305 | |||
306 | test_u32_dentry = | ||
307 | debugfs_create_u32("clr_on_read", 0444, test_dir, &clr_on_read); | ||
308 | |||
309 | return 0; | ||
310 | } | ||
311 | |||
312 | module_init(arc_debugfs_init); | ||
313 | |||
314 | static void __exit arc_debugfs_exit(void) | ||
315 | { | ||
316 | debugfs_remove(test_u32_dentry); | ||
317 | debugfs_remove(test_dentry); | ||
318 | debugfs_remove(test_dir); | ||
319 | } | ||
320 | module_exit(arc_debugfs_exit); | ||
321 | |||
322 | #endif | ||
diff --git a/arch/arc/kernel/unaligned.c b/arch/arc/kernel/unaligned.c new file mode 100644 index 000000000000..4cd81633febd --- /dev/null +++ b/arch/arc/kernel/unaligned.c | |||
@@ -0,0 +1,245 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2011-2012 Synopsys (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * vineetg : May 2011 | ||
9 | * -Adapted (from .26 to .35) | ||
10 | * -original contribution by Tim.yao@amlogic.com | ||
11 | * | ||
12 | */ | ||
13 | |||
14 | #include <linux/types.h> | ||
15 | #include <linux/ptrace.h> | ||
16 | #include <linux/uaccess.h> | ||
17 | #include <asm/disasm.h> | ||
18 | |||
19 | #define __get8_unaligned_check(val, addr, err) \ | ||
20 | __asm__( \ | ||
21 | "1: ldb.ab %1, [%2, 1]\n" \ | ||
22 | "2:\n" \ | ||
23 | " .section .fixup,\"ax\"\n" \ | ||
24 | " .align 4\n" \ | ||
25 | "3: mov %0, 1\n" \ | ||
26 | " b 2b\n" \ | ||
27 | " .previous\n" \ | ||
28 | " .section __ex_table,\"a\"\n" \ | ||
29 | " .align 4\n" \ | ||
30 | " .long 1b, 3b\n" \ | ||
31 | " .previous\n" \ | ||
32 | : "=r" (err), "=&r" (val), "=r" (addr) \ | ||
33 | : "0" (err), "2" (addr)) | ||
34 | |||
35 | #define get16_unaligned_check(val, addr) \ | ||
36 | do { \ | ||
37 | unsigned int err = 0, v, a = addr; \ | ||
38 | __get8_unaligned_check(v, a, err); \ | ||
39 | val = v ; \ | ||
40 | __get8_unaligned_check(v, a, err); \ | ||
41 | val |= v << 8; \ | ||
42 | if (err) \ | ||
43 | goto fault; \ | ||
44 | } while (0) | ||
45 | |||
46 | #define get32_unaligned_check(val, addr) \ | ||
47 | do { \ | ||
48 | unsigned int err = 0, v, a = addr; \ | ||
49 | __get8_unaligned_check(v, a, err); \ | ||
50 | val = v << 0; \ | ||
51 | __get8_unaligned_check(v, a, err); \ | ||
52 | val |= v << 8; \ | ||
53 | __get8_unaligned_check(v, a, err); \ | ||
54 | val |= v << 16; \ | ||
55 | __get8_unaligned_check(v, a, err); \ | ||
56 | val |= v << 24; \ | ||
57 | if (err) \ | ||
58 | goto fault; \ | ||
59 | } while (0) | ||
60 | |||
61 | #define put16_unaligned_check(val, addr) \ | ||
62 | do { \ | ||
63 | unsigned int err = 0, v = val, a = addr;\ | ||
64 | \ | ||
65 | __asm__( \ | ||
66 | "1: stb.ab %1, [%2, 1]\n" \ | ||
67 | " lsr %1, %1, 8\n" \ | ||
68 | "2: stb %1, [%2]\n" \ | ||
69 | "3:\n" \ | ||
70 | " .section .fixup,\"ax\"\n" \ | ||
71 | " .align 4\n" \ | ||
72 | "4: mov %0, 1\n" \ | ||
73 | " b 3b\n" \ | ||
74 | " .previous\n" \ | ||
75 | " .section __ex_table,\"a\"\n" \ | ||
76 | " .align 4\n" \ | ||
77 | " .long 1b, 4b\n" \ | ||
78 | " .long 2b, 4b\n" \ | ||
79 | " .previous\n" \ | ||
80 | : "=r" (err), "=&r" (v), "=&r" (a) \ | ||
81 | : "0" (err), "1" (v), "2" (a)); \ | ||
82 | \ | ||
83 | if (err) \ | ||
84 | goto fault; \ | ||
85 | } while (0) | ||
86 | |||
87 | #define put32_unaligned_check(val, addr) \ | ||
88 | do { \ | ||
89 | unsigned int err = 0, v = val, a = addr;\ | ||
90 | __asm__( \ | ||
91 | \ | ||
92 | "1: stb.ab %1, [%2, 1]\n" \ | ||
93 | " lsr %1, %1, 8\n" \ | ||
94 | "2: stb.ab %1, [%2, 1]\n" \ | ||
95 | " lsr %1, %1, 8\n" \ | ||
96 | "3: stb.ab %1, [%2, 1]\n" \ | ||
97 | " lsr %1, %1, 8\n" \ | ||
98 | "4: stb %1, [%2]\n" \ | ||
99 | "5:\n" \ | ||
100 | " .section .fixup,\"ax\"\n" \ | ||
101 | " .align 4\n" \ | ||
102 | "6: mov %0, 1\n" \ | ||
103 | " b 5b\n" \ | ||
104 | " .previous\n" \ | ||
105 | " .section __ex_table,\"a\"\n" \ | ||
106 | " .align 4\n" \ | ||
107 | " .long 1b, 6b\n" \ | ||
108 | " .long 2b, 6b\n" \ | ||
109 | " .long 3b, 6b\n" \ | ||
110 | " .long 4b, 6b\n" \ | ||
111 | " .previous\n" \ | ||
112 | : "=r" (err), "=&r" (v), "=&r" (a) \ | ||
113 | : "0" (err), "1" (v), "2" (a)); \ | ||
114 | \ | ||
115 | if (err) \ | ||
116 | goto fault; \ | ||
117 | } while (0) | ||
118 | |||
119 | /* sysctl hooks */ | ||
120 | int unaligned_enabled __read_mostly = 1; /* Enabled by default */ | ||
121 | int no_unaligned_warning __read_mostly = 1; /* Only 1 warning by default */ | ||
122 | |||
123 | static void fixup_load(struct disasm_state *state, struct pt_regs *regs, | ||
124 | struct callee_regs *cregs) | ||
125 | { | ||
126 | int val; | ||
127 | |||
128 | /* register write back */ | ||
129 | if ((state->aa == 1) || (state->aa == 2)) { | ||
130 | set_reg(state->wb_reg, state->src1 + state->src2, regs, cregs); | ||
131 | |||
132 | if (state->aa == 2) | ||
133 | state->src2 = 0; | ||
134 | } | ||
135 | |||
136 | if (state->zz == 0) { | ||
137 | get32_unaligned_check(val, state->src1 + state->src2); | ||
138 | } else { | ||
139 | get16_unaligned_check(val, state->src1 + state->src2); | ||
140 | |||
141 | if (state->x) | ||
142 | val = (val << 16) >> 16; | ||
143 | } | ||
144 | |||
145 | if (state->pref == 0) | ||
146 | set_reg(state->dest, val, regs, cregs); | ||
147 | |||
148 | return; | ||
149 | |||
150 | fault: state->fault = 1; | ||
151 | } | ||
152 | |||
153 | static void fixup_store(struct disasm_state *state, struct pt_regs *regs, | ||
154 | struct callee_regs *cregs) | ||
155 | { | ||
156 | /* register write back */ | ||
157 | if ((state->aa == 1) || (state->aa == 2)) { | ||
158 | set_reg(state->wb_reg, state->src2 + state->src3, regs, cregs); | ||
159 | |||
160 | if (state->aa == 3) | ||
161 | state->src3 = 0; | ||
162 | } else if (state->aa == 3) { | ||
163 | if (state->zz == 2) { | ||
164 | set_reg(state->wb_reg, state->src2 + (state->src3 << 1), | ||
165 | regs, cregs); | ||
166 | } else if (!state->zz) { | ||
167 | set_reg(state->wb_reg, state->src2 + (state->src3 << 2), | ||
168 | regs, cregs); | ||
169 | } else { | ||
170 | goto fault; | ||
171 | } | ||
172 | } | ||
173 | |||
174 | /* write fix-up */ | ||
175 | if (!state->zz) | ||
176 | put32_unaligned_check(state->src1, state->src2 + state->src3); | ||
177 | else | ||
178 | put16_unaligned_check(state->src1, state->src2 + state->src3); | ||
179 | |||
180 | return; | ||
181 | |||
182 | fault: state->fault = 1; | ||
183 | } | ||
184 | |||
185 | /* | ||
186 | * Handle an unaligned access | ||
187 | * Returns 0 if successfully handled, 1 if some error happened | ||
188 | */ | ||
189 | int misaligned_fixup(unsigned long address, struct pt_regs *regs, | ||
190 | unsigned long cause, struct callee_regs *cregs) | ||
191 | { | ||
192 | struct disasm_state state; | ||
193 | char buf[TASK_COMM_LEN]; | ||
194 | |||
195 | /* handle user mode only and only if enabled by sysadmin */ | ||
196 | if (!user_mode(regs) || !unaligned_enabled) | ||
197 | return 1; | ||
198 | |||
199 | if (no_unaligned_warning) { | ||
200 | pr_warn_once("%s(%d) made unaligned access which was emulated" | ||
201 | " by kernel assist\n. This can degrade application" | ||
202 | " performance significantly\n. To enable further" | ||
203 | " logging of such instances, please \n" | ||
204 | " echo 0 > /proc/sys/kernel/ignore-unaligned-usertrap\n", | ||
205 | get_task_comm(buf, current), task_pid_nr(current)); | ||
206 | } else { | ||
207 | /* Add rate limiting if it gets down to it */ | ||
208 | pr_warn("%s(%d): unaligned access to/from 0x%lx by PC: 0x%lx\n", | ||
209 | get_task_comm(buf, current), task_pid_nr(current), | ||
210 | address, regs->ret); | ||
211 | |||
212 | } | ||
213 | |||
214 | disasm_instr(regs->ret, &state, 1, regs, cregs); | ||
215 | |||
216 | if (state.fault) | ||
217 | goto fault; | ||
218 | |||
219 | /* ldb/stb should not have unaligned exception */ | ||
220 | if ((state.zz == 1) || (state.di)) | ||
221 | goto fault; | ||
222 | |||
223 | if (!state.write) | ||
224 | fixup_load(&state, regs, cregs); | ||
225 | else | ||
226 | fixup_store(&state, regs, cregs); | ||
227 | |||
228 | if (state.fault) | ||
229 | goto fault; | ||
230 | |||
231 | if (delay_mode(regs)) { | ||
232 | regs->ret = regs->bta; | ||
233 | regs->status32 &= ~STATUS_DE_MASK; | ||
234 | } else { | ||
235 | regs->ret += state.instr_len; | ||
236 | } | ||
237 | |||
238 | return 0; | ||
239 | |||
240 | fault: | ||
241 | pr_err("Alignment trap: fault in fix-up %08lx at [<%08lx>]\n", | ||
242 | state.words[0], address); | ||
243 | |||
244 | return 1; | ||
245 | } | ||
diff --git a/arch/arc/kernel/unwind.c b/arch/arc/kernel/unwind.c new file mode 100644 index 000000000000..a8d02223da44 --- /dev/null +++ b/arch/arc/kernel/unwind.c | |||
@@ -0,0 +1,1329 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * Copyright (C) 2002-2006 Novell, Inc. | ||
4 | * Jan Beulich <jbeulich@novell.com> | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | * | ||
10 | * A simple API for unwinding kernel stacks. This is used for | ||
11 | * debugging and error reporting purposes. The kernel doesn't need | ||
12 | * full-blown stack unwinding with all the bells and whistles, so there | ||
13 | * is not much point in implementing the full Dwarf2 unwind API. | ||
14 | */ | ||
15 | |||
16 | #include <linux/sched.h> | ||
17 | #include <linux/module.h> | ||
18 | #include <linux/bootmem.h> | ||
19 | #include <linux/sort.h> | ||
20 | #include <linux/slab.h> | ||
21 | #include <linux/stop_machine.h> | ||
22 | #include <linux/uaccess.h> | ||
23 | #include <linux/ptrace.h> | ||
24 | #include <asm/sections.h> | ||
25 | #include <asm/unaligned.h> | ||
26 | #include <asm/unwind.h> | ||
27 | |||
28 | extern char __start_unwind[], __end_unwind[]; | ||
29 | /* extern const u8 __start_unwind_hdr[], __end_unwind_hdr[];*/ | ||
30 | |||
31 | /* #define UNWIND_DEBUG */ | ||
32 | |||
33 | #ifdef UNWIND_DEBUG | ||
34 | int dbg_unw; | ||
35 | #define unw_debug(fmt, ...) \ | ||
36 | do { \ | ||
37 | if (dbg_unw) \ | ||
38 | pr_info(fmt, ##__VA_ARGS__); \ | ||
39 | } while (0); | ||
40 | #else | ||
41 | #define unw_debug(fmt, ...) | ||
42 | #endif | ||
43 | |||
44 | #define MAX_STACK_DEPTH 8 | ||
45 | |||
46 | #define EXTRA_INFO(f) { \ | ||
47 | BUILD_BUG_ON_ZERO(offsetof(struct unwind_frame_info, f) \ | ||
48 | % FIELD_SIZEOF(struct unwind_frame_info, f)) \ | ||
49 | + offsetof(struct unwind_frame_info, f) \ | ||
50 | / FIELD_SIZEOF(struct unwind_frame_info, f), \ | ||
51 | FIELD_SIZEOF(struct unwind_frame_info, f) \ | ||
52 | } | ||
53 | #define PTREGS_INFO(f) EXTRA_INFO(regs.f) | ||
54 | |||
55 | static const struct { | ||
56 | unsigned offs:BITS_PER_LONG / 2; | ||
57 | unsigned width:BITS_PER_LONG / 2; | ||
58 | } reg_info[] = { | ||
59 | UNW_REGISTER_INFO}; | ||
60 | |||
61 | #undef PTREGS_INFO | ||
62 | #undef EXTRA_INFO | ||
63 | |||
64 | #ifndef REG_INVALID | ||
65 | #define REG_INVALID(r) (reg_info[r].width == 0) | ||
66 | #endif | ||
67 | |||
68 | #define DW_CFA_nop 0x00 | ||
69 | #define DW_CFA_set_loc 0x01 | ||
70 | #define DW_CFA_advance_loc1 0x02 | ||
71 | #define DW_CFA_advance_loc2 0x03 | ||
72 | #define DW_CFA_advance_loc4 0x04 | ||
73 | #define DW_CFA_offset_extended 0x05 | ||
74 | #define DW_CFA_restore_extended 0x06 | ||
75 | #define DW_CFA_undefined 0x07 | ||
76 | #define DW_CFA_same_value 0x08 | ||
77 | #define DW_CFA_register 0x09 | ||
78 | #define DW_CFA_remember_state 0x0a | ||
79 | #define DW_CFA_restore_state 0x0b | ||
80 | #define DW_CFA_def_cfa 0x0c | ||
81 | #define DW_CFA_def_cfa_register 0x0d | ||
82 | #define DW_CFA_def_cfa_offset 0x0e | ||
83 | #define DW_CFA_def_cfa_expression 0x0f | ||
84 | #define DW_CFA_expression 0x10 | ||
85 | #define DW_CFA_offset_extended_sf 0x11 | ||
86 | #define DW_CFA_def_cfa_sf 0x12 | ||
87 | #define DW_CFA_def_cfa_offset_sf 0x13 | ||
88 | #define DW_CFA_val_offset 0x14 | ||
89 | #define DW_CFA_val_offset_sf 0x15 | ||
90 | #define DW_CFA_val_expression 0x16 | ||
91 | #define DW_CFA_lo_user 0x1c | ||
92 | #define DW_CFA_GNU_window_save 0x2d | ||
93 | #define DW_CFA_GNU_args_size 0x2e | ||
94 | #define DW_CFA_GNU_negative_offset_extended 0x2f | ||
95 | #define DW_CFA_hi_user 0x3f | ||
96 | |||
97 | #define DW_EH_PE_FORM 0x07 | ||
98 | #define DW_EH_PE_native 0x00 | ||
99 | #define DW_EH_PE_leb128 0x01 | ||
100 | #define DW_EH_PE_data2 0x02 | ||
101 | #define DW_EH_PE_data4 0x03 | ||
102 | #define DW_EH_PE_data8 0x04 | ||
103 | #define DW_EH_PE_signed 0x08 | ||
104 | #define DW_EH_PE_ADJUST 0x70 | ||
105 | #define DW_EH_PE_abs 0x00 | ||
106 | #define DW_EH_PE_pcrel 0x10 | ||
107 | #define DW_EH_PE_textrel 0x20 | ||
108 | #define DW_EH_PE_datarel 0x30 | ||
109 | #define DW_EH_PE_funcrel 0x40 | ||
110 | #define DW_EH_PE_aligned 0x50 | ||
111 | #define DW_EH_PE_indirect 0x80 | ||
112 | #define DW_EH_PE_omit 0xff | ||
113 | |||
114 | typedef unsigned long uleb128_t; | ||
115 | typedef signed long sleb128_t; | ||
116 | |||
117 | static struct unwind_table { | ||
118 | struct { | ||
119 | unsigned long pc; | ||
120 | unsigned long range; | ||
121 | } core, init; | ||
122 | const void *address; | ||
123 | unsigned long size; | ||
124 | const unsigned char *header; | ||
125 | unsigned long hdrsz; | ||
126 | struct unwind_table *link; | ||
127 | const char *name; | ||
128 | } root_table; | ||
129 | |||
130 | struct unwind_item { | ||
131 | enum item_location { | ||
132 | Nowhere, | ||
133 | Memory, | ||
134 | Register, | ||
135 | Value | ||
136 | } where; | ||
137 | uleb128_t value; | ||
138 | }; | ||
139 | |||
140 | struct unwind_state { | ||
141 | uleb128_t loc, org; | ||
142 | const u8 *cieStart, *cieEnd; | ||
143 | uleb128_t codeAlign; | ||
144 | sleb128_t dataAlign; | ||
145 | struct cfa { | ||
146 | uleb128_t reg, offs; | ||
147 | } cfa; | ||
148 | struct unwind_item regs[ARRAY_SIZE(reg_info)]; | ||
149 | unsigned stackDepth:8; | ||
150 | unsigned version:8; | ||
151 | const u8 *label; | ||
152 | const u8 *stack[MAX_STACK_DEPTH]; | ||
153 | }; | ||
154 | |||
155 | static const struct cfa badCFA = { ARRAY_SIZE(reg_info), 1 }; | ||
156 | |||
157 | static struct unwind_table *find_table(unsigned long pc) | ||
158 | { | ||
159 | struct unwind_table *table; | ||
160 | |||
161 | for (table = &root_table; table; table = table->link) | ||
162 | if ((pc >= table->core.pc | ||
163 | && pc < table->core.pc + table->core.range) | ||
164 | || (pc >= table->init.pc | ||
165 | && pc < table->init.pc + table->init.range)) | ||
166 | break; | ||
167 | |||
168 | return table; | ||
169 | } | ||
170 | |||
171 | static unsigned long read_pointer(const u8 **pLoc, | ||
172 | const void *end, signed ptrType); | ||
173 | |||
174 | static void init_unwind_table(struct unwind_table *table, const char *name, | ||
175 | const void *core_start, unsigned long core_size, | ||
176 | const void *init_start, unsigned long init_size, | ||
177 | const void *table_start, unsigned long table_size, | ||
178 | const u8 *header_start, unsigned long header_size) | ||
179 | { | ||
180 | const u8 *ptr = header_start + 4; | ||
181 | const u8 *end = header_start + header_size; | ||
182 | |||
183 | table->core.pc = (unsigned long)core_start; | ||
184 | table->core.range = core_size; | ||
185 | table->init.pc = (unsigned long)init_start; | ||
186 | table->init.range = init_size; | ||
187 | table->address = table_start; | ||
188 | table->size = table_size; | ||
189 | |||
190 | /* See if the linker provided table looks valid. */ | ||
191 | if (header_size <= 4 | ||
192 | || header_start[0] != 1 | ||
193 | || (void *)read_pointer(&ptr, end, header_start[1]) != table_start | ||
194 | || header_start[2] == DW_EH_PE_omit | ||
195 | || read_pointer(&ptr, end, header_start[2]) <= 0 | ||
196 | || header_start[3] == DW_EH_PE_omit) | ||
197 | header_start = NULL; | ||
198 | |||
199 | table->hdrsz = header_size; | ||
200 | smp_wmb(); | ||
201 | table->header = header_start; | ||
202 | table->link = NULL; | ||
203 | table->name = name; | ||
204 | } | ||
205 | |||
206 | void __init arc_unwind_init(void) | ||
207 | { | ||
208 | init_unwind_table(&root_table, "kernel", _text, _end - _text, NULL, 0, | ||
209 | __start_unwind, __end_unwind - __start_unwind, | ||
210 | NULL, 0); | ||
211 | /*__start_unwind_hdr, __end_unwind_hdr - __start_unwind_hdr);*/ | ||
212 | } | ||
213 | |||
214 | static const u32 bad_cie, not_fde; | ||
215 | static const u32 *cie_for_fde(const u32 *fde, const struct unwind_table *); | ||
216 | static signed fde_pointer_type(const u32 *cie); | ||
217 | |||
218 | struct eh_frame_hdr_table_entry { | ||
219 | unsigned long start, fde; | ||
220 | }; | ||
221 | |||
222 | static int cmp_eh_frame_hdr_table_entries(const void *p1, const void *p2) | ||
223 | { | ||
224 | const struct eh_frame_hdr_table_entry *e1 = p1; | ||
225 | const struct eh_frame_hdr_table_entry *e2 = p2; | ||
226 | |||
227 | return (e1->start > e2->start) - (e1->start < e2->start); | ||
228 | } | ||
229 | |||
230 | static void swap_eh_frame_hdr_table_entries(void *p1, void *p2, int size) | ||
231 | { | ||
232 | struct eh_frame_hdr_table_entry *e1 = p1; | ||
233 | struct eh_frame_hdr_table_entry *e2 = p2; | ||
234 | unsigned long v; | ||
235 | |||
236 | v = e1->start; | ||
237 | e1->start = e2->start; | ||
238 | e2->start = v; | ||
239 | v = e1->fde; | ||
240 | e1->fde = e2->fde; | ||
241 | e2->fde = v; | ||
242 | } | ||
243 | |||
244 | static void __init setup_unwind_table(struct unwind_table *table, | ||
245 | void *(*alloc) (unsigned long)) | ||
246 | { | ||
247 | const u8 *ptr; | ||
248 | unsigned long tableSize = table->size, hdrSize; | ||
249 | unsigned n; | ||
250 | const u32 *fde; | ||
251 | struct { | ||
252 | u8 version; | ||
253 | u8 eh_frame_ptr_enc; | ||
254 | u8 fde_count_enc; | ||
255 | u8 table_enc; | ||
256 | unsigned long eh_frame_ptr; | ||
257 | unsigned int fde_count; | ||
258 | struct eh_frame_hdr_table_entry table[]; | ||
259 | } __attribute__ ((__packed__)) *header; | ||
260 | |||
261 | if (table->header) | ||
262 | return; | ||
263 | |||
264 | if (table->hdrsz) | ||
265 | pr_warn(".eh_frame_hdr for '%s' present but unusable\n", | ||
266 | table->name); | ||
267 | |||
268 | if (tableSize & (sizeof(*fde) - 1)) | ||
269 | return; | ||
270 | |||
271 | for (fde = table->address, n = 0; | ||
272 | tableSize > sizeof(*fde) && tableSize - sizeof(*fde) >= *fde; | ||
273 | tableSize -= sizeof(*fde) + *fde, fde += 1 + *fde / sizeof(*fde)) { | ||
274 | const u32 *cie = cie_for_fde(fde, table); | ||
275 | signed ptrType; | ||
276 | |||
277 | if (cie == ¬_fde) | ||
278 | continue; | ||
279 | if (cie == NULL || cie == &bad_cie) | ||
280 | return; | ||
281 | ptrType = fde_pointer_type(cie); | ||
282 | if (ptrType < 0) | ||
283 | return; | ||
284 | |||
285 | ptr = (const u8 *)(fde + 2); | ||
286 | if (!read_pointer(&ptr, (const u8 *)(fde + 1) + *fde, | ||
287 | ptrType)) { | ||
288 | /* FIXME_Rajesh We have 4 instances of null addresses | ||
289 | * instead of the initial loc addr | ||
290 | * return; | ||
291 | */ | ||
292 | } | ||
293 | ++n; | ||
294 | } | ||
295 | |||
296 | if (tableSize || !n) | ||
297 | return; | ||
298 | |||
299 | hdrSize = 4 + sizeof(unsigned long) + sizeof(unsigned int) | ||
300 | + 2 * n * sizeof(unsigned long); | ||
301 | header = alloc(hdrSize); | ||
302 | if (!header) | ||
303 | return; | ||
304 | header->version = 1; | ||
305 | header->eh_frame_ptr_enc = DW_EH_PE_abs | DW_EH_PE_native; | ||
306 | header->fde_count_enc = DW_EH_PE_abs | DW_EH_PE_data4; | ||
307 | header->table_enc = DW_EH_PE_abs | DW_EH_PE_native; | ||
308 | put_unaligned((unsigned long)table->address, &header->eh_frame_ptr); | ||
309 | BUILD_BUG_ON(offsetof(typeof(*header), fde_count) | ||
310 | % __alignof(typeof(header->fde_count))); | ||
311 | header->fde_count = n; | ||
312 | |||
313 | BUILD_BUG_ON(offsetof(typeof(*header), table) | ||
314 | % __alignof(typeof(*header->table))); | ||
315 | for (fde = table->address, tableSize = table->size, n = 0; | ||
316 | tableSize; | ||
317 | tableSize -= sizeof(*fde) + *fde, fde += 1 + *fde / sizeof(*fde)) { | ||
318 | /* const u32 *cie = fde + 1 - fde[1] / sizeof(*fde); */ | ||
319 | const u32 *cie = (const u32 *)(fde[1]); | ||
320 | |||
321 | if (fde[1] == 0xffffffff) | ||
322 | continue; /* this is a CIE */ | ||
323 | ptr = (const u8 *)(fde + 2); | ||
324 | header->table[n].start = read_pointer(&ptr, | ||
325 | (const u8 *)(fde + 1) + | ||
326 | *fde, | ||
327 | fde_pointer_type(cie)); | ||
328 | header->table[n].fde = (unsigned long)fde; | ||
329 | ++n; | ||
330 | } | ||
331 | WARN_ON(n != header->fde_count); | ||
332 | |||
333 | sort(header->table, | ||
334 | n, | ||
335 | sizeof(*header->table), | ||
336 | cmp_eh_frame_hdr_table_entries, swap_eh_frame_hdr_table_entries); | ||
337 | |||
338 | table->hdrsz = hdrSize; | ||
339 | smp_wmb(); | ||
340 | table->header = (const void *)header; | ||
341 | } | ||
342 | |||
343 | static void *__init balloc(unsigned long sz) | ||
344 | { | ||
345 | return __alloc_bootmem_nopanic(sz, | ||
346 | sizeof(unsigned int), | ||
347 | __pa(MAX_DMA_ADDRESS)); | ||
348 | } | ||
349 | |||
350 | void __init arc_unwind_setup(void) | ||
351 | { | ||
352 | setup_unwind_table(&root_table, balloc); | ||
353 | } | ||
354 | |||
355 | #ifdef CONFIG_MODULES | ||
356 | |||
357 | static struct unwind_table *last_table; | ||
358 | |||
359 | /* Must be called with module_mutex held. */ | ||
360 | void *unwind_add_table(struct module *module, const void *table_start, | ||
361 | unsigned long table_size) | ||
362 | { | ||
363 | struct unwind_table *table; | ||
364 | |||
365 | if (table_size <= 0) | ||
366 | return NULL; | ||
367 | |||
368 | table = kmalloc(sizeof(*table), GFP_KERNEL); | ||
369 | if (!table) | ||
370 | return NULL; | ||
371 | |||
372 | init_unwind_table(table, module->name, | ||
373 | module->module_core, module->core_size, | ||
374 | module->module_init, module->init_size, | ||
375 | table_start, table_size, | ||
376 | NULL, 0); | ||
377 | |||
378 | #ifdef UNWIND_DEBUG | ||
379 | unw_debug("Table added for [%s] %lx %lx\n", | ||
380 | module->name, table->core.pc, table->core.range); | ||
381 | #endif | ||
382 | if (last_table) | ||
383 | last_table->link = table; | ||
384 | else | ||
385 | root_table.link = table; | ||
386 | last_table = table; | ||
387 | |||
388 | return table; | ||
389 | } | ||
390 | |||
391 | struct unlink_table_info { | ||
392 | struct unwind_table *table; | ||
393 | int init_only; | ||
394 | }; | ||
395 | |||
396 | static int unlink_table(void *arg) | ||
397 | { | ||
398 | struct unlink_table_info *info = arg; | ||
399 | struct unwind_table *table = info->table, *prev; | ||
400 | |||
401 | for (prev = &root_table; prev->link && prev->link != table; | ||
402 | prev = prev->link) | ||
403 | ; | ||
404 | |||
405 | if (prev->link) { | ||
406 | if (info->init_only) { | ||
407 | table->init.pc = 0; | ||
408 | table->init.range = 0; | ||
409 | info->table = NULL; | ||
410 | } else { | ||
411 | prev->link = table->link; | ||
412 | if (!prev->link) | ||
413 | last_table = prev; | ||
414 | } | ||
415 | } else | ||
416 | info->table = NULL; | ||
417 | |||
418 | return 0; | ||
419 | } | ||
420 | |||
421 | /* Must be called with module_mutex held. */ | ||
422 | void unwind_remove_table(void *handle, int init_only) | ||
423 | { | ||
424 | struct unwind_table *table = handle; | ||
425 | struct unlink_table_info info; | ||
426 | |||
427 | if (!table || table == &root_table) | ||
428 | return; | ||
429 | |||
430 | if (init_only && table == last_table) { | ||
431 | table->init.pc = 0; | ||
432 | table->init.range = 0; | ||
433 | return; | ||
434 | } | ||
435 | |||
436 | info.table = table; | ||
437 | info.init_only = init_only; | ||
438 | |||
439 | unlink_table(&info); /* XXX: SMP */ | ||
440 | kfree(table); | ||
441 | } | ||
442 | |||
443 | #endif /* CONFIG_MODULES */ | ||
444 | |||
445 | static uleb128_t get_uleb128(const u8 **pcur, const u8 *end) | ||
446 | { | ||
447 | const u8 *cur = *pcur; | ||
448 | uleb128_t value; | ||
449 | unsigned shift; | ||
450 | |||
451 | for (shift = 0, value = 0; cur < end; shift += 7) { | ||
452 | if (shift + 7 > 8 * sizeof(value) | ||
453 | && (*cur & 0x7fU) >= (1U << (8 * sizeof(value) - shift))) { | ||
454 | cur = end + 1; | ||
455 | break; | ||
456 | } | ||
457 | value |= (uleb128_t) (*cur & 0x7f) << shift; | ||
458 | if (!(*cur++ & 0x80)) | ||
459 | break; | ||
460 | } | ||
461 | *pcur = cur; | ||
462 | |||
463 | return value; | ||
464 | } | ||
465 | |||
466 | static sleb128_t get_sleb128(const u8 **pcur, const u8 *end) | ||
467 | { | ||
468 | const u8 *cur = *pcur; | ||
469 | sleb128_t value; | ||
470 | unsigned shift; | ||
471 | |||
472 | for (shift = 0, value = 0; cur < end; shift += 7) { | ||
473 | if (shift + 7 > 8 * sizeof(value) | ||
474 | && (*cur & 0x7fU) >= (1U << (8 * sizeof(value) - shift))) { | ||
475 | cur = end + 1; | ||
476 | break; | ||
477 | } | ||
478 | value |= (sleb128_t) (*cur & 0x7f) << shift; | ||
479 | if (!(*cur & 0x80)) { | ||
480 | value |= -(*cur++ & 0x40) << shift; | ||
481 | break; | ||
482 | } | ||
483 | } | ||
484 | *pcur = cur; | ||
485 | |||
486 | return value; | ||
487 | } | ||
488 | |||
489 | static const u32 *cie_for_fde(const u32 *fde, const struct unwind_table *table) | ||
490 | { | ||
491 | const u32 *cie; | ||
492 | |||
493 | if (!*fde || (*fde & (sizeof(*fde) - 1))) | ||
494 | return &bad_cie; | ||
495 | |||
496 | if (fde[1] == 0xffffffff) | ||
497 | return ¬_fde; /* this is a CIE */ | ||
498 | |||
499 | if ((fde[1] & (sizeof(*fde) - 1))) | ||
500 | /* || fde[1] > (unsigned long)(fde + 1) - (unsigned long)table->address) */ | ||
501 | return NULL; /* this is not a valid FDE */ | ||
502 | |||
503 | /* cie = fde + 1 - fde[1] / sizeof(*fde); */ | ||
504 | cie = (u32 *) fde[1]; | ||
505 | |||
506 | if (*cie <= sizeof(*cie) + 4 || *cie >= fde[1] - sizeof(*fde) | ||
507 | || (*cie & (sizeof(*cie) - 1)) | ||
508 | || (cie[1] != 0xffffffff)) | ||
509 | return NULL; /* this is not a (valid) CIE */ | ||
510 | return cie; | ||
511 | } | ||
512 | |||
513 | static unsigned long read_pointer(const u8 **pLoc, const void *end, | ||
514 | signed ptrType) | ||
515 | { | ||
516 | unsigned long value = 0; | ||
517 | union { | ||
518 | const u8 *p8; | ||
519 | const u16 *p16u; | ||
520 | const s16 *p16s; | ||
521 | const u32 *p32u; | ||
522 | const s32 *p32s; | ||
523 | const unsigned long *pul; | ||
524 | } ptr; | ||
525 | |||
526 | if (ptrType < 0 || ptrType == DW_EH_PE_omit) | ||
527 | return 0; | ||
528 | ptr.p8 = *pLoc; | ||
529 | switch (ptrType & DW_EH_PE_FORM) { | ||
530 | case DW_EH_PE_data2: | ||
531 | if (end < (const void *)(ptr.p16u + 1)) | ||
532 | return 0; | ||
533 | if (ptrType & DW_EH_PE_signed) | ||
534 | value = get_unaligned((u16 *) ptr.p16s++); | ||
535 | else | ||
536 | value = get_unaligned((u16 *) ptr.p16u++); | ||
537 | break; | ||
538 | case DW_EH_PE_data4: | ||
539 | #ifdef CONFIG_64BIT | ||
540 | if (end < (const void *)(ptr.p32u + 1)) | ||
541 | return 0; | ||
542 | if (ptrType & DW_EH_PE_signed) | ||
543 | value = get_unaligned(ptr.p32s++); | ||
544 | else | ||
545 | value = get_unaligned(ptr.p32u++); | ||
546 | break; | ||
547 | case DW_EH_PE_data8: | ||
548 | BUILD_BUG_ON(sizeof(u64) != sizeof(value)); | ||
549 | #else | ||
550 | BUILD_BUG_ON(sizeof(u32) != sizeof(value)); | ||
551 | #endif | ||
552 | case DW_EH_PE_native: | ||
553 | if (end < (const void *)(ptr.pul + 1)) | ||
554 | return 0; | ||
555 | value = get_unaligned((unsigned long *)ptr.pul++); | ||
556 | break; | ||
557 | case DW_EH_PE_leb128: | ||
558 | BUILD_BUG_ON(sizeof(uleb128_t) > sizeof(value)); | ||
559 | value = ptrType & DW_EH_PE_signed ? get_sleb128(&ptr.p8, end) | ||
560 | : get_uleb128(&ptr.p8, end); | ||
561 | if ((const void *)ptr.p8 > end) | ||
562 | return 0; | ||
563 | break; | ||
564 | default: | ||
565 | return 0; | ||
566 | } | ||
567 | switch (ptrType & DW_EH_PE_ADJUST) { | ||
568 | case DW_EH_PE_abs: | ||
569 | break; | ||
570 | case DW_EH_PE_pcrel: | ||
571 | value += (unsigned long)*pLoc; | ||
572 | break; | ||
573 | default: | ||
574 | return 0; | ||
575 | } | ||
576 | if ((ptrType & DW_EH_PE_indirect) | ||
577 | && __get_user(value, (unsigned long __user *)value)) | ||
578 | return 0; | ||
579 | *pLoc = ptr.p8; | ||
580 | |||
581 | return value; | ||
582 | } | ||
583 | |||
584 | static signed fde_pointer_type(const u32 *cie) | ||
585 | { | ||
586 | const u8 *ptr = (const u8 *)(cie + 2); | ||
587 | unsigned version = *ptr; | ||
588 | |||
589 | if (version != 1) | ||
590 | return -1; /* unsupported */ | ||
591 | |||
592 | if (*++ptr) { | ||
593 | const char *aug; | ||
594 | const u8 *end = (const u8 *)(cie + 1) + *cie; | ||
595 | uleb128_t len; | ||
596 | |||
597 | /* check if augmentation size is first (and thus present) */ | ||
598 | if (*ptr != 'z') | ||
599 | return -1; | ||
600 | |||
601 | /* check if augmentation string is nul-terminated */ | ||
602 | aug = (const void *)ptr; | ||
603 | ptr = memchr(aug, 0, end - ptr); | ||
604 | if (ptr == NULL) | ||
605 | return -1; | ||
606 | |||
607 | ++ptr; /* skip terminator */ | ||
608 | get_uleb128(&ptr, end); /* skip code alignment */ | ||
609 | get_sleb128(&ptr, end); /* skip data alignment */ | ||
610 | /* skip return address column */ | ||
611 | version <= 1 ? (void) ++ptr : (void)get_uleb128(&ptr, end); | ||
612 | len = get_uleb128(&ptr, end); /* augmentation length */ | ||
613 | |||
614 | if (ptr + len < ptr || ptr + len > end) | ||
615 | return -1; | ||
616 | |||
617 | end = ptr + len; | ||
618 | while (*++aug) { | ||
619 | if (ptr >= end) | ||
620 | return -1; | ||
621 | switch (*aug) { | ||
622 | case 'L': | ||
623 | ++ptr; | ||
624 | break; | ||
625 | case 'P':{ | ||
626 | signed ptrType = *ptr++; | ||
627 | |||
628 | if (!read_pointer(&ptr, end, ptrType) | ||
629 | || ptr > end) | ||
630 | return -1; | ||
631 | } | ||
632 | break; | ||
633 | case 'R': | ||
634 | return *ptr; | ||
635 | default: | ||
636 | return -1; | ||
637 | } | ||
638 | } | ||
639 | } | ||
640 | return DW_EH_PE_native | DW_EH_PE_abs; | ||
641 | } | ||
642 | |||
643 | static int advance_loc(unsigned long delta, struct unwind_state *state) | ||
644 | { | ||
645 | state->loc += delta * state->codeAlign; | ||
646 | |||
647 | /* FIXME_Rajesh: Probably we are defining for the initial range as well; | ||
648 | return delta > 0; | ||
649 | */ | ||
650 | unw_debug("delta %3lu => loc 0x%lx: ", delta, state->loc); | ||
651 | return 1; | ||
652 | } | ||
653 | |||
654 | static void set_rule(uleb128_t reg, enum item_location where, uleb128_t value, | ||
655 | struct unwind_state *state) | ||
656 | { | ||
657 | if (reg < ARRAY_SIZE(state->regs)) { | ||
658 | state->regs[reg].where = where; | ||
659 | state->regs[reg].value = value; | ||
660 | |||
661 | #ifdef UNWIND_DEBUG | ||
662 | unw_debug("r%lu: ", reg); | ||
663 | switch (where) { | ||
664 | case Nowhere: | ||
665 | unw_debug("s "); | ||
666 | break; | ||
667 | case Memory: | ||
668 | unw_debug("c(%lu) ", value); | ||
669 | break; | ||
670 | case Register: | ||
671 | unw_debug("r(%lu) ", value); | ||
672 | break; | ||
673 | case Value: | ||
674 | unw_debug("v(%lu) ", value); | ||
675 | break; | ||
676 | default: | ||
677 | break; | ||
678 | } | ||
679 | #endif | ||
680 | } | ||
681 | } | ||
682 | |||
683 | static int processCFI(const u8 *start, const u8 *end, unsigned long targetLoc, | ||
684 | signed ptrType, struct unwind_state *state) | ||
685 | { | ||
686 | union { | ||
687 | const u8 *p8; | ||
688 | const u16 *p16; | ||
689 | const u32 *p32; | ||
690 | } ptr; | ||
691 | int result = 1; | ||
692 | u8 opcode; | ||
693 | |||
694 | if (start != state->cieStart) { | ||
695 | state->loc = state->org; | ||
696 | result = | ||
697 | processCFI(state->cieStart, state->cieEnd, 0, ptrType, | ||
698 | state); | ||
699 | if (targetLoc == 0 && state->label == NULL) | ||
700 | return result; | ||
701 | } | ||
702 | for (ptr.p8 = start; result && ptr.p8 < end;) { | ||
703 | switch (*ptr.p8 >> 6) { | ||
704 | uleb128_t value; | ||
705 | |||
706 | case 0: | ||
707 | opcode = *ptr.p8++; | ||
708 | |||
709 | switch (opcode) { | ||
710 | case DW_CFA_nop: | ||
711 | unw_debug("cfa nop "); | ||
712 | break; | ||
713 | case DW_CFA_set_loc: | ||
714 | state->loc = read_pointer(&ptr.p8, end, | ||
715 | ptrType); | ||
716 | if (state->loc == 0) | ||
717 | result = 0; | ||
718 | unw_debug("cfa_set_loc: 0x%lx ", state->loc); | ||
719 | break; | ||
720 | case DW_CFA_advance_loc1: | ||
721 | unw_debug("\ncfa advance loc1:"); | ||
722 | result = ptr.p8 < end | ||
723 | && advance_loc(*ptr.p8++, state); | ||
724 | break; | ||
725 | case DW_CFA_advance_loc2: | ||
726 | value = *ptr.p8++; | ||
727 | value += *ptr.p8++ << 8; | ||
728 | unw_debug("\ncfa advance loc2:"); | ||
729 | result = ptr.p8 <= end + 2 | ||
730 | /* && advance_loc(*ptr.p16++, state); */ | ||
731 | && advance_loc(value, state); | ||
732 | break; | ||
733 | case DW_CFA_advance_loc4: | ||
734 | unw_debug("\ncfa advance loc4:"); | ||
735 | result = ptr.p8 <= end + 4 | ||
736 | && advance_loc(*ptr.p32++, state); | ||
737 | break; | ||
738 | case DW_CFA_offset_extended: | ||
739 | value = get_uleb128(&ptr.p8, end); | ||
740 | unw_debug("cfa_offset_extended: "); | ||
741 | set_rule(value, Memory, | ||
742 | get_uleb128(&ptr.p8, end), state); | ||
743 | break; | ||
744 | case DW_CFA_val_offset: | ||
745 | value = get_uleb128(&ptr.p8, end); | ||
746 | set_rule(value, Value, | ||
747 | get_uleb128(&ptr.p8, end), state); | ||
748 | break; | ||
749 | case DW_CFA_offset_extended_sf: | ||
750 | value = get_uleb128(&ptr.p8, end); | ||
751 | set_rule(value, Memory, | ||
752 | get_sleb128(&ptr.p8, end), state); | ||
753 | break; | ||
754 | case DW_CFA_val_offset_sf: | ||
755 | value = get_uleb128(&ptr.p8, end); | ||
756 | set_rule(value, Value, | ||
757 | get_sleb128(&ptr.p8, end), state); | ||
758 | break; | ||
759 | case DW_CFA_restore_extended: | ||
760 | unw_debug("cfa_restore_extended: "); | ||
761 | case DW_CFA_undefined: | ||
762 | unw_debug("cfa_undefined: "); | ||
763 | case DW_CFA_same_value: | ||
764 | unw_debug("cfa_same_value: "); | ||
765 | set_rule(get_uleb128(&ptr.p8, end), Nowhere, 0, | ||
766 | state); | ||
767 | break; | ||
768 | case DW_CFA_register: | ||
769 | unw_debug("cfa_register: "); | ||
770 | value = get_uleb128(&ptr.p8, end); | ||
771 | set_rule(value, | ||
772 | Register, | ||
773 | get_uleb128(&ptr.p8, end), state); | ||
774 | break; | ||
775 | case DW_CFA_remember_state: | ||
776 | unw_debug("cfa_remember_state: "); | ||
777 | if (ptr.p8 == state->label) { | ||
778 | state->label = NULL; | ||
779 | return 1; | ||
780 | } | ||
781 | if (state->stackDepth >= MAX_STACK_DEPTH) | ||
782 | return 0; | ||
783 | state->stack[state->stackDepth++] = ptr.p8; | ||
784 | break; | ||
785 | case DW_CFA_restore_state: | ||
786 | unw_debug("cfa_restore_state: "); | ||
787 | if (state->stackDepth) { | ||
788 | const uleb128_t loc = state->loc; | ||
789 | const u8 *label = state->label; | ||
790 | |||
791 | state->label = | ||
792 | state->stack[state->stackDepth - 1]; | ||
793 | memcpy(&state->cfa, &badCFA, | ||
794 | sizeof(state->cfa)); | ||
795 | memset(state->regs, 0, | ||
796 | sizeof(state->regs)); | ||
797 | state->stackDepth = 0; | ||
798 | result = | ||
799 | processCFI(start, end, 0, ptrType, | ||
800 | state); | ||
801 | state->loc = loc; | ||
802 | state->label = label; | ||
803 | } else | ||
804 | return 0; | ||
805 | break; | ||
806 | case DW_CFA_def_cfa: | ||
807 | state->cfa.reg = get_uleb128(&ptr.p8, end); | ||
808 | unw_debug("cfa_def_cfa: r%lu ", state->cfa.reg); | ||
809 | /*nobreak*/ | ||
810 | case DW_CFA_def_cfa_offset: | ||
811 | state->cfa.offs = get_uleb128(&ptr.p8, end); | ||
812 | unw_debug("cfa_def_cfa_offset: 0x%lx ", | ||
813 | state->cfa.offs); | ||
814 | break; | ||
815 | case DW_CFA_def_cfa_sf: | ||
816 | state->cfa.reg = get_uleb128(&ptr.p8, end); | ||
817 | /*nobreak */ | ||
818 | case DW_CFA_def_cfa_offset_sf: | ||
819 | state->cfa.offs = get_sleb128(&ptr.p8, end) | ||
820 | * state->dataAlign; | ||
821 | break; | ||
822 | case DW_CFA_def_cfa_register: | ||
823 | unw_debug("cfa_def_cfa_regsiter: "); | ||
824 | state->cfa.reg = get_uleb128(&ptr.p8, end); | ||
825 | break; | ||
826 | /*todo case DW_CFA_def_cfa_expression: */ | ||
827 | /*todo case DW_CFA_expression: */ | ||
828 | /*todo case DW_CFA_val_expression: */ | ||
829 | case DW_CFA_GNU_args_size: | ||
830 | get_uleb128(&ptr.p8, end); | ||
831 | break; | ||
832 | case DW_CFA_GNU_negative_offset_extended: | ||
833 | value = get_uleb128(&ptr.p8, end); | ||
834 | set_rule(value, | ||
835 | Memory, | ||
836 | (uleb128_t) 0 - get_uleb128(&ptr.p8, | ||
837 | end), | ||
838 | state); | ||
839 | break; | ||
840 | case DW_CFA_GNU_window_save: | ||
841 | default: | ||
842 | unw_debug("UNKNOW OPCODE 0x%x\n", opcode); | ||
843 | result = 0; | ||
844 | break; | ||
845 | } | ||
846 | break; | ||
847 | case 1: | ||
848 | unw_debug("\ncfa_adv_loc: "); | ||
849 | result = advance_loc(*ptr.p8++ & 0x3f, state); | ||
850 | break; | ||
851 | case 2: | ||
852 | unw_debug("cfa_offset: "); | ||
853 | value = *ptr.p8++ & 0x3f; | ||
854 | set_rule(value, Memory, get_uleb128(&ptr.p8, end), | ||
855 | state); | ||
856 | break; | ||
857 | case 3: | ||
858 | unw_debug("cfa_restore: "); | ||
859 | set_rule(*ptr.p8++ & 0x3f, Nowhere, 0, state); | ||
860 | break; | ||
861 | } | ||
862 | |||
863 | if (ptr.p8 > end) | ||
864 | result = 0; | ||
865 | if (result && targetLoc != 0 && targetLoc < state->loc) | ||
866 | return 1; | ||
867 | } | ||
868 | |||
869 | return result && ptr.p8 == end && (targetLoc == 0 || ( | ||
870 | /*todo While in theory this should apply, gcc in practice omits | ||
871 | everything past the function prolog, and hence the location | ||
872 | never reaches the end of the function. | ||
873 | targetLoc < state->loc && */ state->label == NULL)); | ||
874 | } | ||
875 | |||
876 | /* Unwind to previous to frame. Returns 0 if successful, negative | ||
877 | * number in case of an error. */ | ||
878 | int arc_unwind(struct unwind_frame_info *frame) | ||
879 | { | ||
880 | #define FRAME_REG(r, t) (((t *)frame)[reg_info[r].offs]) | ||
881 | const u32 *fde = NULL, *cie = NULL; | ||
882 | const u8 *ptr = NULL, *end = NULL; | ||
883 | unsigned long pc = UNW_PC(frame) - frame->call_frame; | ||
884 | unsigned long startLoc = 0, endLoc = 0, cfa; | ||
885 | unsigned i; | ||
886 | signed ptrType = -1; | ||
887 | uleb128_t retAddrReg = 0; | ||
888 | const struct unwind_table *table; | ||
889 | struct unwind_state state; | ||
890 | unsigned long *fptr; | ||
891 | unsigned long addr; | ||
892 | |||
893 | unw_debug("\n\nUNWIND FRAME:\n"); | ||
894 | unw_debug("PC: 0x%lx BLINK: 0x%lx, SP: 0x%lx, FP: 0x%x\n", | ||
895 | UNW_PC(frame), UNW_BLINK(frame), UNW_SP(frame), | ||
896 | UNW_FP(frame)); | ||
897 | |||
898 | if (UNW_PC(frame) == 0) | ||
899 | return -EINVAL; | ||
900 | |||
901 | #ifdef UNWIND_DEBUG | ||
902 | { | ||
903 | unsigned long *sptr = (unsigned long *)UNW_SP(frame); | ||
904 | unw_debug("\nStack Dump:\n"); | ||
905 | for (i = 0; i < 20; i++, sptr++) | ||
906 | unw_debug("0x%p: 0x%lx\n", sptr, *sptr); | ||
907 | unw_debug("\n"); | ||
908 | } | ||
909 | #endif | ||
910 | |||
911 | table = find_table(pc); | ||
912 | if (table != NULL | ||
913 | && !(table->size & (sizeof(*fde) - 1))) { | ||
914 | const u8 *hdr = table->header; | ||
915 | unsigned long tableSize; | ||
916 | |||
917 | smp_rmb(); | ||
918 | if (hdr && hdr[0] == 1) { | ||
919 | switch (hdr[3] & DW_EH_PE_FORM) { | ||
920 | case DW_EH_PE_native: | ||
921 | tableSize = sizeof(unsigned long); | ||
922 | break; | ||
923 | case DW_EH_PE_data2: | ||
924 | tableSize = 2; | ||
925 | break; | ||
926 | case DW_EH_PE_data4: | ||
927 | tableSize = 4; | ||
928 | break; | ||
929 | case DW_EH_PE_data8: | ||
930 | tableSize = 8; | ||
931 | break; | ||
932 | default: | ||
933 | tableSize = 0; | ||
934 | break; | ||
935 | } | ||
936 | ptr = hdr + 4; | ||
937 | end = hdr + table->hdrsz; | ||
938 | if (tableSize && read_pointer(&ptr, end, hdr[1]) | ||
939 | == (unsigned long)table->address | ||
940 | && (i = read_pointer(&ptr, end, hdr[2])) > 0 | ||
941 | && i == (end - ptr) / (2 * tableSize) | ||
942 | && !((end - ptr) % (2 * tableSize))) { | ||
943 | do { | ||
944 | const u8 *cur = | ||
945 | ptr + (i / 2) * (2 * tableSize); | ||
946 | |||
947 | startLoc = read_pointer(&cur, | ||
948 | cur + tableSize, | ||
949 | hdr[3]); | ||
950 | if (pc < startLoc) | ||
951 | i /= 2; | ||
952 | else { | ||
953 | ptr = cur - tableSize; | ||
954 | i = (i + 1) / 2; | ||
955 | } | ||
956 | } while (startLoc && i > 1); | ||
957 | if (i == 1 | ||
958 | && (startLoc = read_pointer(&ptr, | ||
959 | ptr + tableSize, | ||
960 | hdr[3])) != 0 | ||
961 | && pc >= startLoc) | ||
962 | fde = (void *)read_pointer(&ptr, | ||
963 | ptr + | ||
964 | tableSize, | ||
965 | hdr[3]); | ||
966 | } | ||
967 | } | ||
968 | |||
969 | if (fde != NULL) { | ||
970 | cie = cie_for_fde(fde, table); | ||
971 | ptr = (const u8 *)(fde + 2); | ||
972 | if (cie != NULL | ||
973 | && cie != &bad_cie | ||
974 | && cie != ¬_fde | ||
975 | && (ptrType = fde_pointer_type(cie)) >= 0 | ||
976 | && read_pointer(&ptr, | ||
977 | (const u8 *)(fde + 1) + *fde, | ||
978 | ptrType) == startLoc) { | ||
979 | if (!(ptrType & DW_EH_PE_indirect)) | ||
980 | ptrType &= | ||
981 | DW_EH_PE_FORM | DW_EH_PE_signed; | ||
982 | endLoc = | ||
983 | startLoc + read_pointer(&ptr, | ||
984 | (const u8 *)(fde + | ||
985 | 1) + | ||
986 | *fde, ptrType); | ||
987 | if (pc >= endLoc) | ||
988 | fde = NULL; | ||
989 | } else | ||
990 | fde = NULL; | ||
991 | } | ||
992 | if (fde == NULL) { | ||
993 | for (fde = table->address, tableSize = table->size; | ||
994 | cie = NULL, tableSize > sizeof(*fde) | ||
995 | && tableSize - sizeof(*fde) >= *fde; | ||
996 | tableSize -= sizeof(*fde) + *fde, | ||
997 | fde += 1 + *fde / sizeof(*fde)) { | ||
998 | cie = cie_for_fde(fde, table); | ||
999 | if (cie == &bad_cie) { | ||
1000 | cie = NULL; | ||
1001 | break; | ||
1002 | } | ||
1003 | if (cie == NULL | ||
1004 | || cie == ¬_fde | ||
1005 | || (ptrType = fde_pointer_type(cie)) < 0) | ||
1006 | continue; | ||
1007 | ptr = (const u8 *)(fde + 2); | ||
1008 | startLoc = read_pointer(&ptr, | ||
1009 | (const u8 *)(fde + 1) + | ||
1010 | *fde, ptrType); | ||
1011 | if (!startLoc) | ||
1012 | continue; | ||
1013 | if (!(ptrType & DW_EH_PE_indirect)) | ||
1014 | ptrType &= | ||
1015 | DW_EH_PE_FORM | DW_EH_PE_signed; | ||
1016 | endLoc = | ||
1017 | startLoc + read_pointer(&ptr, | ||
1018 | (const u8 *)(fde + | ||
1019 | 1) + | ||
1020 | *fde, ptrType); | ||
1021 | if (pc >= startLoc && pc < endLoc) | ||
1022 | break; | ||
1023 | } | ||
1024 | } | ||
1025 | } | ||
1026 | if (cie != NULL) { | ||
1027 | memset(&state, 0, sizeof(state)); | ||
1028 | state.cieEnd = ptr; /* keep here temporarily */ | ||
1029 | ptr = (const u8 *)(cie + 2); | ||
1030 | end = (const u8 *)(cie + 1) + *cie; | ||
1031 | frame->call_frame = 1; | ||
1032 | if ((state.version = *ptr) != 1) | ||
1033 | cie = NULL; /* unsupported version */ | ||
1034 | else if (*++ptr) { | ||
1035 | /* check if augmentation size is first (thus present) */ | ||
1036 | if (*ptr == 'z') { | ||
1037 | while (++ptr < end && *ptr) { | ||
1038 | switch (*ptr) { | ||
1039 | /* chk for ignorable or already handled | ||
1040 | * nul-terminated augmentation string */ | ||
1041 | case 'L': | ||
1042 | case 'P': | ||
1043 | case 'R': | ||
1044 | continue; | ||
1045 | case 'S': | ||
1046 | frame->call_frame = 0; | ||
1047 | continue; | ||
1048 | default: | ||
1049 | break; | ||
1050 | } | ||
1051 | break; | ||
1052 | } | ||
1053 | } | ||
1054 | if (ptr >= end || *ptr) | ||
1055 | cie = NULL; | ||
1056 | } | ||
1057 | ++ptr; | ||
1058 | } | ||
1059 | if (cie != NULL) { | ||
1060 | /* get code aligment factor */ | ||
1061 | state.codeAlign = get_uleb128(&ptr, end); | ||
1062 | /* get data aligment factor */ | ||
1063 | state.dataAlign = get_sleb128(&ptr, end); | ||
1064 | if (state.codeAlign == 0 || state.dataAlign == 0 || ptr >= end) | ||
1065 | cie = NULL; | ||
1066 | else { | ||
1067 | retAddrReg = | ||
1068 | state.version <= 1 ? *ptr++ : get_uleb128(&ptr, | ||
1069 | end); | ||
1070 | unw_debug("CIE Frame Info:\n"); | ||
1071 | unw_debug("return Address register 0x%lx\n", | ||
1072 | retAddrReg); | ||
1073 | unw_debug("data Align: %ld\n", state.dataAlign); | ||
1074 | unw_debug("code Align: %lu\n", state.codeAlign); | ||
1075 | /* skip augmentation */ | ||
1076 | if (((const char *)(cie + 2))[1] == 'z') { | ||
1077 | uleb128_t augSize = get_uleb128(&ptr, end); | ||
1078 | |||
1079 | ptr += augSize; | ||
1080 | } | ||
1081 | if (ptr > end || retAddrReg >= ARRAY_SIZE(reg_info) | ||
1082 | || REG_INVALID(retAddrReg) | ||
1083 | || reg_info[retAddrReg].width != | ||
1084 | sizeof(unsigned long)) | ||
1085 | cie = NULL; | ||
1086 | } | ||
1087 | } | ||
1088 | if (cie != NULL) { | ||
1089 | state.cieStart = ptr; | ||
1090 | ptr = state.cieEnd; | ||
1091 | state.cieEnd = end; | ||
1092 | end = (const u8 *)(fde + 1) + *fde; | ||
1093 | /* skip augmentation */ | ||
1094 | if (((const char *)(cie + 2))[1] == 'z') { | ||
1095 | uleb128_t augSize = get_uleb128(&ptr, end); | ||
1096 | |||
1097 | if ((ptr += augSize) > end) | ||
1098 | fde = NULL; | ||
1099 | } | ||
1100 | } | ||
1101 | if (cie == NULL || fde == NULL) { | ||
1102 | #ifdef CONFIG_FRAME_POINTER | ||
1103 | unsigned long top, bottom; | ||
1104 | |||
1105 | top = STACK_TOP_UNW(frame->task); | ||
1106 | bottom = STACK_BOTTOM_UNW(frame->task); | ||
1107 | #if FRAME_RETADDR_OFFSET < 0 | ||
1108 | if (UNW_SP(frame) < top && UNW_FP(frame) <= UNW_SP(frame) | ||
1109 | && bottom < UNW_FP(frame) | ||
1110 | #else | ||
1111 | if (UNW_SP(frame) > top && UNW_FP(frame) >= UNW_SP(frame) | ||
1112 | && bottom > UNW_FP(frame) | ||
1113 | #endif | ||
1114 | && !((UNW_SP(frame) | UNW_FP(frame)) | ||
1115 | & (sizeof(unsigned long) - 1))) { | ||
1116 | unsigned long link; | ||
1117 | |||
1118 | if (!__get_user(link, (unsigned long *) | ||
1119 | (UNW_FP(frame) + FRAME_LINK_OFFSET)) | ||
1120 | #if FRAME_RETADDR_OFFSET < 0 | ||
1121 | && link > bottom && link < UNW_FP(frame) | ||
1122 | #else | ||
1123 | && link > UNW_FP(frame) && link < bottom | ||
1124 | #endif | ||
1125 | && !(link & (sizeof(link) - 1)) | ||
1126 | && !__get_user(UNW_PC(frame), | ||
1127 | (unsigned long *)(UNW_FP(frame) | ||
1128 | + FRAME_RETADDR_OFFSET))) | ||
1129 | { | ||
1130 | UNW_SP(frame) = | ||
1131 | UNW_FP(frame) + FRAME_RETADDR_OFFSET | ||
1132 | #if FRAME_RETADDR_OFFSET < 0 | ||
1133 | - | ||
1134 | #else | ||
1135 | + | ||
1136 | #endif | ||
1137 | sizeof(UNW_PC(frame)); | ||
1138 | UNW_FP(frame) = link; | ||
1139 | return 0; | ||
1140 | } | ||
1141 | } | ||
1142 | #endif | ||
1143 | return -ENXIO; | ||
1144 | } | ||
1145 | state.org = startLoc; | ||
1146 | memcpy(&state.cfa, &badCFA, sizeof(state.cfa)); | ||
1147 | |||
1148 | unw_debug("\nProcess instructions\n"); | ||
1149 | |||
1150 | /* process instructions | ||
1151 | * For ARC, we optimize by having blink(retAddrReg) with | ||
1152 | * the sameValue in the leaf function, so we should not check | ||
1153 | * state.regs[retAddrReg].where == Nowhere | ||
1154 | */ | ||
1155 | if (!processCFI(ptr, end, pc, ptrType, &state) | ||
1156 | || state.loc > endLoc | ||
1157 | /* || state.regs[retAddrReg].where == Nowhere */ | ||
1158 | || state.cfa.reg >= ARRAY_SIZE(reg_info) | ||
1159 | || reg_info[state.cfa.reg].width != sizeof(unsigned long) | ||
1160 | || state.cfa.offs % sizeof(unsigned long)) | ||
1161 | return -EIO; | ||
1162 | |||
1163 | #ifdef UNWIND_DEBUG | ||
1164 | unw_debug("\n"); | ||
1165 | |||
1166 | unw_debug("\nRegister State Based on the rules parsed from FDE:\n"); | ||
1167 | for (i = 0; i < ARRAY_SIZE(state.regs); ++i) { | ||
1168 | |||
1169 | if (REG_INVALID(i)) | ||
1170 | continue; | ||
1171 | |||
1172 | switch (state.regs[i].where) { | ||
1173 | case Nowhere: | ||
1174 | break; | ||
1175 | case Memory: | ||
1176 | unw_debug(" r%d: c(%lu),", i, state.regs[i].value); | ||
1177 | break; | ||
1178 | case Register: | ||
1179 | unw_debug(" r%d: r(%lu),", i, state.regs[i].value); | ||
1180 | break; | ||
1181 | case Value: | ||
1182 | unw_debug(" r%d: v(%lu),", i, state.regs[i].value); | ||
1183 | break; | ||
1184 | } | ||
1185 | } | ||
1186 | |||
1187 | unw_debug("\n"); | ||
1188 | #endif | ||
1189 | |||
1190 | /* update frame */ | ||
1191 | #ifndef CONFIG_AS_CFI_SIGNAL_FRAME | ||
1192 | if (frame->call_frame | ||
1193 | && !UNW_DEFAULT_RA(state.regs[retAddrReg], state.dataAlign)) | ||
1194 | frame->call_frame = 0; | ||
1195 | #endif | ||
1196 | cfa = FRAME_REG(state.cfa.reg, unsigned long) + state.cfa.offs; | ||
1197 | startLoc = min_t(unsigned long, UNW_SP(frame), cfa); | ||
1198 | endLoc = max_t(unsigned long, UNW_SP(frame), cfa); | ||
1199 | if (STACK_LIMIT(startLoc) != STACK_LIMIT(endLoc)) { | ||
1200 | startLoc = min(STACK_LIMIT(cfa), cfa); | ||
1201 | endLoc = max(STACK_LIMIT(cfa), cfa); | ||
1202 | } | ||
1203 | |||
1204 | unw_debug("\nCFA reg: 0x%lx, offset: 0x%lx => 0x%lx\n", | ||
1205 | state.cfa.reg, state.cfa.offs, cfa); | ||
1206 | |||
1207 | for (i = 0; i < ARRAY_SIZE(state.regs); ++i) { | ||
1208 | if (REG_INVALID(i)) { | ||
1209 | if (state.regs[i].where == Nowhere) | ||
1210 | continue; | ||
1211 | return -EIO; | ||
1212 | } | ||
1213 | switch (state.regs[i].where) { | ||
1214 | default: | ||
1215 | break; | ||
1216 | case Register: | ||
1217 | if (state.regs[i].value >= ARRAY_SIZE(reg_info) | ||
1218 | || REG_INVALID(state.regs[i].value) | ||
1219 | || reg_info[i].width > | ||
1220 | reg_info[state.regs[i].value].width) | ||
1221 | return -EIO; | ||
1222 | switch (reg_info[state.regs[i].value].width) { | ||
1223 | case sizeof(u8): | ||
1224 | state.regs[i].value = | ||
1225 | FRAME_REG(state.regs[i].value, const u8); | ||
1226 | break; | ||
1227 | case sizeof(u16): | ||
1228 | state.regs[i].value = | ||
1229 | FRAME_REG(state.regs[i].value, const u16); | ||
1230 | break; | ||
1231 | case sizeof(u32): | ||
1232 | state.regs[i].value = | ||
1233 | FRAME_REG(state.regs[i].value, const u32); | ||
1234 | break; | ||
1235 | #ifdef CONFIG_64BIT | ||
1236 | case sizeof(u64): | ||
1237 | state.regs[i].value = | ||
1238 | FRAME_REG(state.regs[i].value, const u64); | ||
1239 | break; | ||
1240 | #endif | ||
1241 | default: | ||
1242 | return -EIO; | ||
1243 | } | ||
1244 | break; | ||
1245 | } | ||
1246 | } | ||
1247 | |||
1248 | unw_debug("\nRegister state after evaluation with realtime Stack:\n"); | ||
1249 | fptr = (unsigned long *)(&frame->regs); | ||
1250 | for (i = 0; i < ARRAY_SIZE(state.regs); ++i, fptr++) { | ||
1251 | |||
1252 | if (REG_INVALID(i)) | ||
1253 | continue; | ||
1254 | switch (state.regs[i].where) { | ||
1255 | case Nowhere: | ||
1256 | if (reg_info[i].width != sizeof(UNW_SP(frame)) | ||
1257 | || &FRAME_REG(i, __typeof__(UNW_SP(frame))) | ||
1258 | != &UNW_SP(frame)) | ||
1259 | continue; | ||
1260 | UNW_SP(frame) = cfa; | ||
1261 | break; | ||
1262 | case Register: | ||
1263 | switch (reg_info[i].width) { | ||
1264 | case sizeof(u8): | ||
1265 | FRAME_REG(i, u8) = state.regs[i].value; | ||
1266 | break; | ||
1267 | case sizeof(u16): | ||
1268 | FRAME_REG(i, u16) = state.regs[i].value; | ||
1269 | break; | ||
1270 | case sizeof(u32): | ||
1271 | FRAME_REG(i, u32) = state.regs[i].value; | ||
1272 | break; | ||
1273 | #ifdef CONFIG_64BIT | ||
1274 | case sizeof(u64): | ||
1275 | FRAME_REG(i, u64) = state.regs[i].value; | ||
1276 | break; | ||
1277 | #endif | ||
1278 | default: | ||
1279 | return -EIO; | ||
1280 | } | ||
1281 | break; | ||
1282 | case Value: | ||
1283 | if (reg_info[i].width != sizeof(unsigned long)) | ||
1284 | return -EIO; | ||
1285 | FRAME_REG(i, unsigned long) = cfa + state.regs[i].value | ||
1286 | * state.dataAlign; | ||
1287 | break; | ||
1288 | case Memory: | ||
1289 | addr = cfa + state.regs[i].value * state.dataAlign; | ||
1290 | |||
1291 | if ((state.regs[i].value * state.dataAlign) | ||
1292 | % sizeof(unsigned long) | ||
1293 | || addr < startLoc | ||
1294 | || addr + sizeof(unsigned long) < addr | ||
1295 | || addr + sizeof(unsigned long) > endLoc) | ||
1296 | return -EIO; | ||
1297 | |||
1298 | switch (reg_info[i].width) { | ||
1299 | case sizeof(u8): | ||
1300 | __get_user(FRAME_REG(i, u8), | ||
1301 | (u8 __user *)addr); | ||
1302 | break; | ||
1303 | case sizeof(u16): | ||
1304 | __get_user(FRAME_REG(i, u16), | ||
1305 | (u16 __user *)addr); | ||
1306 | break; | ||
1307 | case sizeof(u32): | ||
1308 | __get_user(FRAME_REG(i, u32), | ||
1309 | (u32 __user *)addr); | ||
1310 | break; | ||
1311 | #ifdef CONFIG_64BIT | ||
1312 | case sizeof(u64): | ||
1313 | __get_user(FRAME_REG(i, u64), | ||
1314 | (u64 __user *)addr); | ||
1315 | break; | ||
1316 | #endif | ||
1317 | default: | ||
1318 | return -EIO; | ||
1319 | } | ||
1320 | |||
1321 | break; | ||
1322 | } | ||
1323 | unw_debug("r%d: 0x%lx ", i, *fptr); | ||
1324 | } | ||
1325 | |||
1326 | return 0; | ||
1327 | #undef FRAME_REG | ||
1328 | } | ||
1329 | EXPORT_SYMBOL(arc_unwind); | ||
diff --git a/arch/arc/kernel/vmlinux.lds.S b/arch/arc/kernel/vmlinux.lds.S new file mode 100644 index 000000000000..d3c92f52d444 --- /dev/null +++ b/arch/arc/kernel/vmlinux.lds.S | |||
@@ -0,0 +1,163 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <asm-generic/vmlinux.lds.h> | ||
10 | #include <asm/cache.h> | ||
11 | #include <asm/page.h> | ||
12 | #include <asm/thread_info.h> | ||
13 | |||
14 | OUTPUT_ARCH(arc) | ||
15 | ENTRY(_stext) | ||
16 | |||
17 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
18 | jiffies = jiffies_64 + 4; | ||
19 | #else | ||
20 | jiffies = jiffies_64; | ||
21 | #endif | ||
22 | |||
23 | SECTIONS | ||
24 | { | ||
25 | /* | ||
26 | * ICCM starts at 0x8000_0000. So if kernel is relocated to some other | ||
27 | * address, make sure peripheral at 0x8z doesn't clash with ICCM | ||
28 | * Essentially vector is also in ICCM. | ||
29 | */ | ||
30 | |||
31 | . = CONFIG_LINUX_LINK_BASE; | ||
32 | |||
33 | _int_vec_base_lds = .; | ||
34 | .vector : { | ||
35 | *(.vector) | ||
36 | . = ALIGN(PAGE_SIZE); | ||
37 | } | ||
38 | |||
39 | #ifdef CONFIG_ARC_HAS_ICCM | ||
40 | .text.arcfp : { | ||
41 | *(.text.arcfp) | ||
42 | . = ALIGN(CONFIG_ARC_ICCM_SZ * 1024); | ||
43 | } | ||
44 | #endif | ||
45 | |||
46 | /* | ||
47 | * The reason for having a seperate subsection .init.ramfs is to | ||
48 | * prevent objump from including it in kernel dumps | ||
49 | * | ||
50 | * Reason for having .init.ramfs above .init is to make sure that the | ||
51 | * binary blob is tucked away to one side, reducing the displacement | ||
52 | * between .init.text and .text, avoiding any possible relocation | ||
53 | * errors because of calls from .init.text to .text | ||
54 | * Yes such calls do exist. e.g. | ||
55 | * decompress_inflate.c:gunzip( ) -> zlib_inflate_workspace( ) | ||
56 | */ | ||
57 | |||
58 | __init_begin = .; | ||
59 | |||
60 | .init.ramfs : { INIT_RAM_FS } | ||
61 | |||
62 | . = ALIGN(PAGE_SIZE); | ||
63 | _stext = .; | ||
64 | |||
65 | HEAD_TEXT_SECTION | ||
66 | INIT_TEXT_SECTION(L1_CACHE_BYTES) | ||
67 | |||
68 | /* INIT_DATA_SECTION open-coded: special INIT_RAM_FS handling */ | ||
69 | .init.data : { | ||
70 | INIT_DATA | ||
71 | INIT_SETUP(L1_CACHE_BYTES) | ||
72 | INIT_CALLS | ||
73 | CON_INITCALL | ||
74 | SECURITY_INITCALL | ||
75 | } | ||
76 | |||
77 | .init.arch.info : { | ||
78 | __arch_info_begin = .; | ||
79 | *(.arch.info.init) | ||
80 | __arch_info_end = .; | ||
81 | } | ||
82 | |||
83 | PERCPU_SECTION(L1_CACHE_BYTES) | ||
84 | |||
85 | /* | ||
86 | * .exit.text is discard at runtime, not link time, to deal with | ||
87 | * references from .debug_frame | ||
88 | * It will be init freed, being inside [__init_start : __init_end] | ||
89 | */ | ||
90 | .exit.text : { EXIT_TEXT } | ||
91 | .exit.data : { EXIT_DATA } | ||
92 | |||
93 | . = ALIGN(PAGE_SIZE); | ||
94 | __init_end = .; | ||
95 | |||
96 | .text : { | ||
97 | _text = .; | ||
98 | TEXT_TEXT | ||
99 | SCHED_TEXT | ||
100 | LOCK_TEXT | ||
101 | KPROBES_TEXT | ||
102 | *(.fixup) | ||
103 | *(.gnu.warning) | ||
104 | } | ||
105 | EXCEPTION_TABLE(L1_CACHE_BYTES) | ||
106 | _etext = .; | ||
107 | |||
108 | _sdata = .; | ||
109 | RO_DATA_SECTION(PAGE_SIZE) | ||
110 | |||
111 | /* | ||
112 | * 1. this is .data essentially | ||
113 | * 2. THREAD_SIZE for init.task, must be kernel-stk sz aligned | ||
114 | */ | ||
115 | RW_DATA_SECTION(L1_CACHE_BYTES, PAGE_SIZE, THREAD_SIZE) | ||
116 | |||
117 | _edata = .; | ||
118 | |||
119 | BSS_SECTION(0, 0, 0) | ||
120 | |||
121 | #ifdef CONFIG_ARC_DW2_UNWIND | ||
122 | . = ALIGN(PAGE_SIZE); | ||
123 | .debug_frame : { | ||
124 | __start_unwind = .; | ||
125 | *(.debug_frame) | ||
126 | __end_unwind = .; | ||
127 | } | ||
128 | #else | ||
129 | /DISCARD/ : { *(.debug_frame) } | ||
130 | #endif | ||
131 | |||
132 | NOTES | ||
133 | |||
134 | . = ALIGN(PAGE_SIZE); | ||
135 | _end = . ; | ||
136 | |||
137 | STABS_DEBUG | ||
138 | DISCARDS | ||
139 | |||
140 | .arcextmap 0 : { | ||
141 | *(.gnu.linkonce.arcextmap.*) | ||
142 | *(.arcextmap.*) | ||
143 | } | ||
144 | |||
145 | /* open-coded because we need .debug_frame seperately for unwinding */ | ||
146 | .debug_aranges 0 : { *(.debug_aranges) } | ||
147 | .debug_pubnames 0 : { *(.debug_pubnames) } | ||
148 | .debug_info 0 : { *(.debug_info) } | ||
149 | .debug_abbrev 0 : { *(.debug_abbrev) } | ||
150 | .debug_line 0 : { *(.debug_line) } | ||
151 | .debug_str 0 : { *(.debug_str) } | ||
152 | .debug_loc 0 : { *(.debug_loc) } | ||
153 | .debug_macinfo 0 : { *(.debug_macinfo) } | ||
154 | |||
155 | #ifdef CONFIG_ARC_HAS_DCCM | ||
156 | . = CONFIG_ARC_DCCM_BASE; | ||
157 | __arc_dccm_base = .; | ||
158 | .data.arcfp : { | ||
159 | *(.data.arcfp) | ||
160 | } | ||
161 | . = ALIGN(CONFIG_ARC_DCCM_SZ * 1024); | ||
162 | #endif | ||
163 | } | ||
diff --git a/arch/arc/lib/Makefile b/arch/arc/lib/Makefile new file mode 100644 index 000000000000..db46e200baba --- /dev/null +++ b/arch/arc/lib/Makefile | |||
@@ -0,0 +1,9 @@ | |||
1 | # | ||
2 | # Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | # | ||
4 | # This program is free software; you can redistribute it and/or modify | ||
5 | # it under the terms of the GNU General Public License version 2 as | ||
6 | # published by the Free Software Foundation. | ||
7 | |||
8 | lib-y := strchr-700.o strcmp.o strcpy-700.o strlen.o | ||
9 | lib-y += memcmp.o memcpy-700.o memset.o | ||
diff --git a/arch/arc/lib/memcmp.S b/arch/arc/lib/memcmp.S new file mode 100644 index 000000000000..bc813d55b6c3 --- /dev/null +++ b/arch/arc/lib/memcmp.S | |||
@@ -0,0 +1,124 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <asm/linkage.h> | ||
10 | |||
11 | #ifdef __LITTLE_ENDIAN__ | ||
12 | #define WORD2 r2 | ||
13 | #define SHIFT r3 | ||
14 | #else /* BIG ENDIAN */ | ||
15 | #define WORD2 r3 | ||
16 | #define SHIFT r2 | ||
17 | #endif | ||
18 | |||
19 | ARC_ENTRY memcmp | ||
20 | or r12,r0,r1 | ||
21 | asl_s r12,r12,30 | ||
22 | sub r3,r2,1 | ||
23 | brls r2,r12,.Lbytewise | ||
24 | ld r4,[r0,0] | ||
25 | ld r5,[r1,0] | ||
26 | lsr.f lp_count,r3,3 | ||
27 | lpne .Loop_end | ||
28 | ld_s WORD2,[r0,4] | ||
29 | ld_s r12,[r1,4] | ||
30 | brne r4,r5,.Leven | ||
31 | ld.a r4,[r0,8] | ||
32 | ld.a r5,[r1,8] | ||
33 | brne WORD2,r12,.Lodd | ||
34 | .Loop_end: | ||
35 | asl_s SHIFT,SHIFT,3 | ||
36 | bhs_s .Last_cmp | ||
37 | brne r4,r5,.Leven | ||
38 | ld r4,[r0,4] | ||
39 | ld r5,[r1,4] | ||
40 | #ifdef __LITTLE_ENDIAN__ | ||
41 | nop_s | ||
42 | ; one more load latency cycle | ||
43 | .Last_cmp: | ||
44 | xor r0,r4,r5 | ||
45 | bset r0,r0,SHIFT | ||
46 | sub_s r1,r0,1 | ||
47 | bic_s r1,r1,r0 | ||
48 | norm r1,r1 | ||
49 | b.d .Leven_cmp | ||
50 | and r1,r1,24 | ||
51 | .Leven: | ||
52 | xor r0,r4,r5 | ||
53 | sub_s r1,r0,1 | ||
54 | bic_s r1,r1,r0 | ||
55 | norm r1,r1 | ||
56 | ; slow track insn | ||
57 | and r1,r1,24 | ||
58 | .Leven_cmp: | ||
59 | asl r2,r4,r1 | ||
60 | asl r12,r5,r1 | ||
61 | lsr_s r2,r2,1 | ||
62 | lsr_s r12,r12,1 | ||
63 | j_s.d [blink] | ||
64 | sub r0,r2,r12 | ||
65 | .balign 4 | ||
66 | .Lodd: | ||
67 | xor r0,WORD2,r12 | ||
68 | sub_s r1,r0,1 | ||
69 | bic_s r1,r1,r0 | ||
70 | norm r1,r1 | ||
71 | ; slow track insn | ||
72 | and r1,r1,24 | ||
73 | asl_s r2,r2,r1 | ||
74 | asl_s r12,r12,r1 | ||
75 | lsr_s r2,r2,1 | ||
76 | lsr_s r12,r12,1 | ||
77 | j_s.d [blink] | ||
78 | sub r0,r2,r12 | ||
79 | #else /* BIG ENDIAN */ | ||
80 | .Last_cmp: | ||
81 | neg_s SHIFT,SHIFT | ||
82 | lsr r4,r4,SHIFT | ||
83 | lsr r5,r5,SHIFT | ||
84 | ; slow track insn | ||
85 | .Leven: | ||
86 | sub.f r0,r4,r5 | ||
87 | mov.ne r0,1 | ||
88 | j_s.d [blink] | ||
89 | bset.cs r0,r0,31 | ||
90 | .Lodd: | ||
91 | cmp_s WORD2,r12 | ||
92 | |||
93 | mov_s r0,1 | ||
94 | j_s.d [blink] | ||
95 | bset.cs r0,r0,31 | ||
96 | #endif /* ENDIAN */ | ||
97 | .balign 4 | ||
98 | .Lbytewise: | ||
99 | breq r2,0,.Lnil | ||
100 | ldb r4,[r0,0] | ||
101 | ldb r5,[r1,0] | ||
102 | lsr.f lp_count,r3 | ||
103 | lpne .Lbyte_end | ||
104 | ldb_s r3,[r0,1] | ||
105 | ldb r12,[r1,1] | ||
106 | brne r4,r5,.Lbyte_even | ||
107 | ldb.a r4,[r0,2] | ||
108 | ldb.a r5,[r1,2] | ||
109 | brne r3,r12,.Lbyte_odd | ||
110 | .Lbyte_end: | ||
111 | bcc .Lbyte_even | ||
112 | brne r4,r5,.Lbyte_even | ||
113 | ldb_s r3,[r0,1] | ||
114 | ldb_s r12,[r1,1] | ||
115 | .Lbyte_odd: | ||
116 | j_s.d [blink] | ||
117 | sub r0,r3,r12 | ||
118 | .Lbyte_even: | ||
119 | j_s.d [blink] | ||
120 | sub r0,r4,r5 | ||
121 | .Lnil: | ||
122 | j_s.d [blink] | ||
123 | mov r0,0 | ||
124 | ARC_EXIT memcmp | ||
diff --git a/arch/arc/lib/memcpy-700.S b/arch/arc/lib/memcpy-700.S new file mode 100644 index 000000000000..b64cc10ac918 --- /dev/null +++ b/arch/arc/lib/memcpy-700.S | |||
@@ -0,0 +1,66 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <asm/linkage.h> | ||
10 | |||
11 | ARC_ENTRY memcpy | ||
12 | or r3,r0,r1 | ||
13 | asl_s r3,r3,30 | ||
14 | mov_s r5,r0 | ||
15 | brls.d r2,r3,.Lcopy_bytewise | ||
16 | sub.f r3,r2,1 | ||
17 | ld_s r12,[r1,0] | ||
18 | asr.f lp_count,r3,3 | ||
19 | bbit0.d r3,2,.Lnox4 | ||
20 | bmsk_s r2,r2,1 | ||
21 | st.ab r12,[r5,4] | ||
22 | ld.a r12,[r1,4] | ||
23 | .Lnox4: | ||
24 | lppnz .Lendloop | ||
25 | ld_s r3,[r1,4] | ||
26 | st.ab r12,[r5,4] | ||
27 | ld.a r12,[r1,8] | ||
28 | st.ab r3,[r5,4] | ||
29 | .Lendloop: | ||
30 | breq r2,0,.Last_store | ||
31 | ld r3,[r5,0] | ||
32 | #ifdef __LITTLE_ENDIAN__ | ||
33 | add3 r2,-1,r2 | ||
34 | ; uses long immediate | ||
35 | xor_s r12,r12,r3 | ||
36 | bmsk r12,r12,r2 | ||
37 | xor_s r12,r12,r3 | ||
38 | #else /* BIG ENDIAN */ | ||
39 | sub3 r2,31,r2 | ||
40 | ; uses long immediate | ||
41 | xor_s r3,r3,r12 | ||
42 | bmsk r3,r3,r2 | ||
43 | xor_s r12,r12,r3 | ||
44 | #endif /* ENDIAN */ | ||
45 | .Last_store: | ||
46 | j_s.d [blink] | ||
47 | st r12,[r5,0] | ||
48 | |||
49 | .balign 4 | ||
50 | .Lcopy_bytewise: | ||
51 | jcs [blink] | ||
52 | ldb_s r12,[r1,0] | ||
53 | lsr.f lp_count,r3 | ||
54 | bhs_s .Lnox1 | ||
55 | stb.ab r12,[r5,1] | ||
56 | ldb.a r12,[r1,1] | ||
57 | .Lnox1: | ||
58 | lppnz .Lendbloop | ||
59 | ldb_s r3,[r1,1] | ||
60 | stb.ab r12,[r5,1] | ||
61 | ldb.a r12,[r1,2] | ||
62 | stb.ab r3,[r5,1] | ||
63 | .Lendbloop: | ||
64 | j_s.d [blink] | ||
65 | stb r12,[r5,0] | ||
66 | ARC_EXIT memcpy | ||
diff --git a/arch/arc/lib/memset.S b/arch/arc/lib/memset.S new file mode 100644 index 000000000000..9b2d88d2e141 --- /dev/null +++ b/arch/arc/lib/memset.S | |||
@@ -0,0 +1,59 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <asm/linkage.h> | ||
10 | |||
11 | #define SMALL 7 /* Must be at least 6 to deal with alignment/loop issues. */ | ||
12 | |||
13 | ARC_ENTRY memset | ||
14 | mov_s r4,r0 | ||
15 | or r12,r0,r2 | ||
16 | bmsk.f r12,r12,1 | ||
17 | extb_s r1,r1 | ||
18 | asl r3,r1,8 | ||
19 | beq.d .Laligned | ||
20 | or_s r1,r1,r3 | ||
21 | brls r2,SMALL,.Ltiny | ||
22 | add r3,r2,r0 | ||
23 | stb r1,[r3,-1] | ||
24 | bclr_s r3,r3,0 | ||
25 | stw r1,[r3,-2] | ||
26 | bmsk.f r12,r0,1 | ||
27 | add_s r2,r2,r12 | ||
28 | sub.ne r2,r2,4 | ||
29 | stb.ab r1,[r4,1] | ||
30 | and r4,r4,-2 | ||
31 | stw.ab r1,[r4,2] | ||
32 | and r4,r4,-4 | ||
33 | .Laligned: ; This code address should be aligned for speed. | ||
34 | asl r3,r1,16 | ||
35 | lsr.f lp_count,r2,2 | ||
36 | or_s r1,r1,r3 | ||
37 | lpne .Loop_end | ||
38 | st.ab r1,[r4,4] | ||
39 | .Loop_end: | ||
40 | j_s [blink] | ||
41 | |||
42 | .balign 4 | ||
43 | .Ltiny: | ||
44 | mov.f lp_count,r2 | ||
45 | lpne .Ltiny_end | ||
46 | stb.ab r1,[r4,1] | ||
47 | .Ltiny_end: | ||
48 | j_s [blink] | ||
49 | ARC_EXIT memset | ||
50 | |||
51 | ; memzero: @r0 = mem, @r1 = size_t | ||
52 | ; memset: @r0 = mem, @r1 = char, @r2 = size_t | ||
53 | |||
54 | ARC_ENTRY memzero | ||
55 | ; adjust bzero args to memset args | ||
56 | mov r2, r1 | ||
57 | mov r1, 0 | ||
58 | b memset ;tail call so need to tinker with blink | ||
59 | ARC_EXIT memzero | ||
diff --git a/arch/arc/lib/strchr-700.S b/arch/arc/lib/strchr-700.S new file mode 100644 index 000000000000..99c10475d477 --- /dev/null +++ b/arch/arc/lib/strchr-700.S | |||
@@ -0,0 +1,123 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | /* ARC700 has a relatively long pipeline and branch prediction, so we want | ||
10 | to avoid branches that are hard to predict. On the other hand, the | ||
11 | presence of the norm instruction makes it easier to operate on whole | ||
12 | words branch-free. */ | ||
13 | |||
14 | #include <asm/linkage.h> | ||
15 | |||
16 | ARC_ENTRY strchr | ||
17 | extb_s r1,r1 | ||
18 | asl r5,r1,8 | ||
19 | bmsk r2,r0,1 | ||
20 | or r5,r5,r1 | ||
21 | mov_s r3,0x01010101 | ||
22 | breq.d r2,r0,.Laligned | ||
23 | asl r4,r5,16 | ||
24 | sub_s r0,r0,r2 | ||
25 | asl r7,r2,3 | ||
26 | ld_s r2,[r0] | ||
27 | #ifdef __LITTLE_ENDIAN__ | ||
28 | asl r7,r3,r7 | ||
29 | #else | ||
30 | lsr r7,r3,r7 | ||
31 | #endif | ||
32 | or r5,r5,r4 | ||
33 | ror r4,r3 | ||
34 | sub r12,r2,r7 | ||
35 | bic_s r12,r12,r2 | ||
36 | and r12,r12,r4 | ||
37 | brne.d r12,0,.Lfound0_ua | ||
38 | xor r6,r2,r5 | ||
39 | ld.a r2,[r0,4] | ||
40 | sub r12,r6,r7 | ||
41 | bic r12,r12,r6 | ||
42 | and r7,r12,r4 | ||
43 | breq r7,0,.Loop ; For speed, we want this branch to be unaligned. | ||
44 | b .Lfound_char ; Likewise this one. | ||
45 | ; /* We require this code address to be unaligned for speed... */ | ||
46 | .Laligned: | ||
47 | ld_s r2,[r0] | ||
48 | or r5,r5,r4 | ||
49 | ror r4,r3 | ||
50 | ; /* ... so that this code address is aligned, for itself and ... */ | ||
51 | .Loop: | ||
52 | sub r12,r2,r3 | ||
53 | bic_s r12,r12,r2 | ||
54 | and r12,r12,r4 | ||
55 | brne.d r12,0,.Lfound0 | ||
56 | xor r6,r2,r5 | ||
57 | ld.a r2,[r0,4] | ||
58 | sub r12,r6,r3 | ||
59 | bic r12,r12,r6 | ||
60 | and r7,r12,r4 | ||
61 | breq r7,0,.Loop /* ... so that this branch is unaligned. */ | ||
62 | ; Found searched-for character. r0 has already advanced to next word. | ||
63 | #ifdef __LITTLE_ENDIAN__ | ||
64 | /* We only need the information about the first matching byte | ||
65 | (i.e. the least significant matching byte) to be exact, | ||
66 | hence there is no problem with carry effects. */ | ||
67 | .Lfound_char: | ||
68 | sub r3,r7,1 | ||
69 | bic r3,r3,r7 | ||
70 | norm r2,r3 | ||
71 | sub_s r0,r0,1 | ||
72 | asr_s r2,r2,3 | ||
73 | j.d [blink] | ||
74 | sub_s r0,r0,r2 | ||
75 | |||
76 | .balign 4 | ||
77 | .Lfound0_ua: | ||
78 | mov r3,r7 | ||
79 | .Lfound0: | ||
80 | sub r3,r6,r3 | ||
81 | bic r3,r3,r6 | ||
82 | and r2,r3,r4 | ||
83 | or_s r12,r12,r2 | ||
84 | sub_s r3,r12,1 | ||
85 | bic_s r3,r3,r12 | ||
86 | norm r3,r3 | ||
87 | add_s r0,r0,3 | ||
88 | asr_s r12,r3,3 | ||
89 | asl.f 0,r2,r3 | ||
90 | sub_s r0,r0,r12 | ||
91 | j_s.d [blink] | ||
92 | mov.pl r0,0 | ||
93 | #else /* BIG ENDIAN */ | ||
94 | .Lfound_char: | ||
95 | lsr r7,r7,7 | ||
96 | |||
97 | bic r2,r7,r6 | ||
98 | norm r2,r2 | ||
99 | sub_s r0,r0,4 | ||
100 | asr_s r2,r2,3 | ||
101 | j.d [blink] | ||
102 | add_s r0,r0,r2 | ||
103 | |||
104 | .Lfound0_ua: | ||
105 | mov_s r3,r7 | ||
106 | .Lfound0: | ||
107 | asl_s r2,r2,7 | ||
108 | or r7,r6,r4 | ||
109 | bic_s r12,r12,r2 | ||
110 | sub r2,r7,r3 | ||
111 | or r2,r2,r6 | ||
112 | bic r12,r2,r12 | ||
113 | bic.f r3,r4,r12 | ||
114 | norm r3,r3 | ||
115 | |||
116 | add.pl r3,r3,1 | ||
117 | asr_s r12,r3,3 | ||
118 | asl.f 0,r2,r3 | ||
119 | add_s r0,r0,r12 | ||
120 | j_s.d [blink] | ||
121 | mov.mi r0,0 | ||
122 | #endif /* ENDIAN */ | ||
123 | ARC_EXIT strchr | ||
diff --git a/arch/arc/lib/strcmp.S b/arch/arc/lib/strcmp.S new file mode 100644 index 000000000000..5dc802b45cf3 --- /dev/null +++ b/arch/arc/lib/strcmp.S | |||
@@ -0,0 +1,96 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | /* This is optimized primarily for the ARC700. | ||
10 | It would be possible to speed up the loops by one cycle / word | ||
11 | respective one cycle / byte by forcing double source 1 alignment, unrolling | ||
12 | by a factor of two, and speculatively loading the second word / byte of | ||
13 | source 1; however, that would increase the overhead for loop setup / finish, | ||
14 | and strcmp might often terminate early. */ | ||
15 | |||
16 | #include <asm/linkage.h> | ||
17 | |||
18 | ARC_ENTRY strcmp | ||
19 | or r2,r0,r1 | ||
20 | bmsk_s r2,r2,1 | ||
21 | brne r2,0,.Lcharloop | ||
22 | mov_s r12,0x01010101 | ||
23 | ror r5,r12 | ||
24 | .Lwordloop: | ||
25 | ld.ab r2,[r0,4] | ||
26 | ld.ab r3,[r1,4] | ||
27 | nop_s | ||
28 | sub r4,r2,r12 | ||
29 | bic r4,r4,r2 | ||
30 | and r4,r4,r5 | ||
31 | brne r4,0,.Lfound0 | ||
32 | breq r2,r3,.Lwordloop | ||
33 | #ifdef __LITTLE_ENDIAN__ | ||
34 | xor r0,r2,r3 ; mask for difference | ||
35 | sub_s r1,r0,1 | ||
36 | bic_s r0,r0,r1 ; mask for least significant difference bit | ||
37 | sub r1,r5,r0 | ||
38 | xor r0,r5,r1 ; mask for least significant difference byte | ||
39 | and_s r2,r2,r0 | ||
40 | and_s r3,r3,r0 | ||
41 | #endif /* LITTLE ENDIAN */ | ||
42 | cmp_s r2,r3 | ||
43 | mov_s r0,1 | ||
44 | j_s.d [blink] | ||
45 | bset.lo r0,r0,31 | ||
46 | |||
47 | .balign 4 | ||
48 | #ifdef __LITTLE_ENDIAN__ | ||
49 | .Lfound0: | ||
50 | xor r0,r2,r3 ; mask for difference | ||
51 | or r0,r0,r4 ; or in zero indicator | ||
52 | sub_s r1,r0,1 | ||
53 | bic_s r0,r0,r1 ; mask for least significant difference bit | ||
54 | sub r1,r5,r0 | ||
55 | xor r0,r5,r1 ; mask for least significant difference byte | ||
56 | and_s r2,r2,r0 | ||
57 | and_s r3,r3,r0 | ||
58 | sub.f r0,r2,r3 | ||
59 | mov.hi r0,1 | ||
60 | j_s.d [blink] | ||
61 | bset.lo r0,r0,31 | ||
62 | #else /* BIG ENDIAN */ | ||
63 | /* The zero-detection above can mis-detect 0x01 bytes as zeroes | ||
64 | because of carry-propagateion from a lower significant zero byte. | ||
65 | We can compensate for this by checking that bit0 is zero. | ||
66 | This compensation is not necessary in the step where we | ||
67 | get a low estimate for r2, because in any affected bytes | ||
68 | we already have 0x00 or 0x01, which will remain unchanged | ||
69 | when bit 7 is cleared. */ | ||
70 | .balign 4 | ||
71 | .Lfound0: | ||
72 | lsr r0,r4,8 | ||
73 | lsr_s r1,r2 | ||
74 | bic_s r2,r2,r0 ; get low estimate for r2 and get ... | ||
75 | bic_s r0,r0,r1 ; <this is the adjusted mask for zeros> | ||
76 | or_s r3,r3,r0 ; ... high estimate r3 so that r2 > r3 will ... | ||
77 | cmp_s r3,r2 ; ... be independent of trailing garbage | ||
78 | or_s r2,r2,r0 ; likewise for r3 > r2 | ||
79 | bic_s r3,r3,r0 | ||
80 | rlc r0,0 ; r0 := r2 > r3 ? 1 : 0 | ||
81 | cmp_s r2,r3 | ||
82 | j_s.d [blink] | ||
83 | bset.lo r0,r0,31 | ||
84 | #endif /* ENDIAN */ | ||
85 | |||
86 | .balign 4 | ||
87 | .Lcharloop: | ||
88 | ldb.ab r2,[r0,1] | ||
89 | ldb.ab r3,[r1,1] | ||
90 | nop_s | ||
91 | breq r2,0,.Lcmpend | ||
92 | breq r2,r3,.Lcharloop | ||
93 | .Lcmpend: | ||
94 | j_s.d [blink] | ||
95 | sub r0,r2,r3 | ||
96 | ARC_EXIT strcmp | ||
diff --git a/arch/arc/lib/strcpy-700.S b/arch/arc/lib/strcpy-700.S new file mode 100644 index 000000000000..b7ca4ae81d88 --- /dev/null +++ b/arch/arc/lib/strcpy-700.S | |||
@@ -0,0 +1,70 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | /* If dst and src are 4 byte aligned, copy 8 bytes at a time. | ||
10 | If the src is 4, but not 8 byte aligned, we first read 4 bytes to get | ||
11 | it 8 byte aligned. Thus, we can do a little read-ahead, without | ||
12 | dereferencing a cache line that we should not touch. | ||
13 | Note that short and long instructions have been scheduled to avoid | ||
14 | branch stalls. | ||
15 | The beq_s to r3z could be made unaligned & long to avoid a stall | ||
16 | there, but the it is not likely to be taken often, and it | ||
17 | would also be likey to cost an unaligned mispredict at the next call. */ | ||
18 | |||
19 | #include <asm/linkage.h> | ||
20 | |||
21 | ARC_ENTRY strcpy | ||
22 | or r2,r0,r1 | ||
23 | bmsk_s r2,r2,1 | ||
24 | brne.d r2,0,charloop | ||
25 | mov_s r10,r0 | ||
26 | ld_s r3,[r1,0] | ||
27 | mov r8,0x01010101 | ||
28 | bbit0.d r1,2,loop_start | ||
29 | ror r12,r8 | ||
30 | sub r2,r3,r8 | ||
31 | bic_s r2,r2,r3 | ||
32 | tst_s r2,r12 | ||
33 | bne r3z | ||
34 | mov_s r4,r3 | ||
35 | .balign 4 | ||
36 | loop: | ||
37 | ld.a r3,[r1,4] | ||
38 | st.ab r4,[r10,4] | ||
39 | loop_start: | ||
40 | ld.a r4,[r1,4] | ||
41 | sub r2,r3,r8 | ||
42 | bic_s r2,r2,r3 | ||
43 | tst_s r2,r12 | ||
44 | bne_s r3z | ||
45 | st.ab r3,[r10,4] | ||
46 | sub r2,r4,r8 | ||
47 | bic r2,r2,r4 | ||
48 | tst r2,r12 | ||
49 | beq loop | ||
50 | mov_s r3,r4 | ||
51 | #ifdef __LITTLE_ENDIAN__ | ||
52 | r3z: bmsk.f r1,r3,7 | ||
53 | lsr_s r3,r3,8 | ||
54 | #else | ||
55 | r3z: lsr.f r1,r3,24 | ||
56 | asl_s r3,r3,8 | ||
57 | #endif | ||
58 | bne.d r3z | ||
59 | stb.ab r1,[r10,1] | ||
60 | j_s [blink] | ||
61 | |||
62 | .balign 4 | ||
63 | charloop: | ||
64 | ldb.ab r3,[r1,1] | ||
65 | |||
66 | |||
67 | brne.d r3,0,charloop | ||
68 | stb.ab r3,[r10,1] | ||
69 | j [blink] | ||
70 | ARC_EXIT strcpy | ||
diff --git a/arch/arc/lib/strlen.S b/arch/arc/lib/strlen.S new file mode 100644 index 000000000000..39759e099696 --- /dev/null +++ b/arch/arc/lib/strlen.S | |||
@@ -0,0 +1,83 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <asm/linkage.h> | ||
10 | |||
11 | ARC_ENTRY strlen | ||
12 | or r3,r0,7 | ||
13 | ld r2,[r3,-7] | ||
14 | ld.a r6,[r3,-3] | ||
15 | mov r4,0x01010101 | ||
16 | ; uses long immediate | ||
17 | #ifdef __LITTLE_ENDIAN__ | ||
18 | asl_s r1,r0,3 | ||
19 | btst_s r0,2 | ||
20 | asl r7,r4,r1 | ||
21 | ror r5,r4 | ||
22 | sub r1,r2,r7 | ||
23 | bic_s r1,r1,r2 | ||
24 | mov.eq r7,r4 | ||
25 | sub r12,r6,r7 | ||
26 | bic r12,r12,r6 | ||
27 | or.eq r12,r12,r1 | ||
28 | and r12,r12,r5 | ||
29 | brne r12,0,.Learly_end | ||
30 | #else /* BIG ENDIAN */ | ||
31 | ror r5,r4 | ||
32 | btst_s r0,2 | ||
33 | mov_s r1,31 | ||
34 | sub3 r7,r1,r0 | ||
35 | sub r1,r2,r4 | ||
36 | bic_s r1,r1,r2 | ||
37 | bmsk r1,r1,r7 | ||
38 | sub r12,r6,r4 | ||
39 | bic r12,r12,r6 | ||
40 | bmsk.ne r12,r12,r7 | ||
41 | or.eq r12,r12,r1 | ||
42 | and r12,r12,r5 | ||
43 | brne r12,0,.Learly_end | ||
44 | #endif /* ENDIAN */ | ||
45 | |||
46 | .Loop: | ||
47 | ld_s r2,[r3,4] | ||
48 | ld.a r6,[r3,8] | ||
49 | ; stall for load result | ||
50 | sub r1,r2,r4 | ||
51 | bic_s r1,r1,r2 | ||
52 | sub r12,r6,r4 | ||
53 | bic r12,r12,r6 | ||
54 | or r12,r12,r1 | ||
55 | and r12,r12,r5 | ||
56 | breq r12,0,.Loop | ||
57 | .Lend: | ||
58 | and.f r1,r1,r5 | ||
59 | sub.ne r3,r3,4 | ||
60 | mov.eq r1,r12 | ||
61 | #ifdef __LITTLE_ENDIAN__ | ||
62 | sub_s r2,r1,1 | ||
63 | bic_s r2,r2,r1 | ||
64 | norm r1,r2 | ||
65 | sub_s r0,r0,3 | ||
66 | lsr_s r1,r1,3 | ||
67 | sub r0,r3,r0 | ||
68 | j_s.d [blink] | ||
69 | sub r0,r0,r1 | ||
70 | #else /* BIG ENDIAN */ | ||
71 | lsr_s r1,r1,7 | ||
72 | mov.eq r2,r6 | ||
73 | bic_s r1,r1,r2 | ||
74 | norm r1,r1 | ||
75 | sub r0,r3,r0 | ||
76 | lsr_s r1,r1,3 | ||
77 | j_s.d [blink] | ||
78 | add r0,r0,r1 | ||
79 | #endif /* ENDIAN */ | ||
80 | .Learly_end: | ||
81 | b.d .Lend | ||
82 | sub_s.ne r1,r1,r1 | ||
83 | ARC_EXIT strlen | ||
diff --git a/arch/arc/mm/Makefile b/arch/arc/mm/Makefile new file mode 100644 index 000000000000..168dc146a8f6 --- /dev/null +++ b/arch/arc/mm/Makefile | |||
@@ -0,0 +1,10 @@ | |||
1 | # | ||
2 | # Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | # | ||
4 | # This program is free software; you can redistribute it and/or modify | ||
5 | # it under the terms of the GNU General Public License version 2 as | ||
6 | # published by the Free Software Foundation. | ||
7 | # | ||
8 | |||
9 | obj-y := extable.o ioremap.o dma.o fault.o init.o | ||
10 | obj-y += tlb.o tlbex.o cache_arc700.o | ||
diff --git a/arch/arc/mm/cache_arc700.c b/arch/arc/mm/cache_arc700.c new file mode 100644 index 000000000000..88d617d84234 --- /dev/null +++ b/arch/arc/mm/cache_arc700.c | |||
@@ -0,0 +1,768 @@ | |||
1 | /* | ||
2 | * ARC700 VIPT Cache Management | ||
3 | * | ||
4 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | * | ||
10 | * vineetg: May 2011: for Non-aliasing VIPT D-cache following can be NOPs | ||
11 | * -flush_cache_dup_mm (fork) | ||
12 | * -likewise for flush_cache_mm (exit/execve) | ||
13 | * -likewise for flush_cache_range,flush_cache_page (munmap, exit, COW-break) | ||
14 | * | ||
15 | * vineetg: Apr 2011 | ||
16 | * -Now that MMU can support larger pg sz (16K), the determiniation of | ||
17 | * aliasing shd not be based on assumption of 8k pg | ||
18 | * | ||
19 | * vineetg: Mar 2011 | ||
20 | * -optimised version of flush_icache_range( ) for making I/D coherent | ||
21 | * when vaddr is available (agnostic of num of aliases) | ||
22 | * | ||
23 | * vineetg: Mar 2011 | ||
24 | * -Added documentation about I-cache aliasing on ARC700 and the way it | ||
25 | * was handled up until MMU V2. | ||
26 | * -Spotted a three year old bug when killing the 4 aliases, which needs | ||
27 | * bottom 2 bits, so we need to do paddr | {0x00, 0x01, 0x02, 0x03} | ||
28 | * instead of paddr | {0x00, 0x01, 0x10, 0x11} | ||
29 | * (Rajesh you owe me one now) | ||
30 | * | ||
31 | * vineetg: Dec 2010 | ||
32 | * -Off-by-one error when computing num_of_lines to flush | ||
33 | * This broke signal handling with bionic which uses synthetic sigret stub | ||
34 | * | ||
35 | * vineetg: Mar 2010 | ||
36 | * -GCC can't generate ZOL for core cache flush loops. | ||
37 | * Conv them into iterations based as opposed to while (start < end) types | ||
38 | * | ||
39 | * Vineetg: July 2009 | ||
40 | * -In I-cache flush routine we used to chk for aliasing for every line INV. | ||
41 | * Instead now we setup routines per cache geometry and invoke them | ||
42 | * via function pointers. | ||
43 | * | ||
44 | * Vineetg: Jan 2009 | ||
45 | * -Cache Line flush routines used to flush an extra line beyond end addr | ||
46 | * because check was while (end >= start) instead of (end > start) | ||
47 | * =Some call sites had to work around by doing -1, -4 etc to end param | ||
48 | * =Some callers didnt care. This was spec bad in case of INV routines | ||
49 | * which would discard valid data (cause of the horrible ext2 bug | ||
50 | * in ARC IDE driver) | ||
51 | * | ||
52 | * vineetg: June 11th 2008: Fixed flush_icache_range( ) | ||
53 | * -Since ARC700 caches are not coherent (I$ doesnt snoop D$) both need | ||
54 | * to be flushed, which it was not doing. | ||
55 | * -load_module( ) passes vmalloc addr (Kernel Virtual Addr) to the API, | ||
56 | * however ARC cache maintenance OPs require PHY addr. Thus need to do | ||
57 | * vmalloc_to_phy. | ||
58 | * -Also added optimisation there, that for range > PAGE SIZE we flush the | ||
59 | * entire cache in one shot rather than line by line. For e.g. a module | ||
60 | * with Code sz 600k, old code flushed 600k worth of cache (line-by-line), | ||
61 | * while cache is only 16 or 32k. | ||
62 | */ | ||
63 | |||
64 | #include <linux/module.h> | ||
65 | #include <linux/mm.h> | ||
66 | #include <linux/sched.h> | ||
67 | #include <linux/cache.h> | ||
68 | #include <linux/mmu_context.h> | ||
69 | #include <linux/syscalls.h> | ||
70 | #include <linux/uaccess.h> | ||
71 | #include <asm/cacheflush.h> | ||
72 | #include <asm/cachectl.h> | ||
73 | #include <asm/setup.h> | ||
74 | |||
75 | |||
76 | #ifdef CONFIG_ARC_HAS_ICACHE | ||
77 | static void __ic_line_inv_no_alias(unsigned long, int); | ||
78 | static void __ic_line_inv_2_alias(unsigned long, int); | ||
79 | static void __ic_line_inv_4_alias(unsigned long, int); | ||
80 | |||
81 | /* Holds the ptr to flush routine, dependign on size due to aliasing issues */ | ||
82 | static void (*___flush_icache_rtn) (unsigned long, int); | ||
83 | #endif | ||
84 | |||
85 | char *arc_cache_mumbojumbo(int cpu_id, char *buf, int len) | ||
86 | { | ||
87 | int n = 0; | ||
88 | unsigned int c = smp_processor_id(); | ||
89 | |||
90 | #define PR_CACHE(p, enb, str) \ | ||
91 | { \ | ||
92 | if (!(p)->ver) \ | ||
93 | n += scnprintf(buf + n, len - n, str"\t\t: N/A\n"); \ | ||
94 | else \ | ||
95 | n += scnprintf(buf + n, len - n, \ | ||
96 | str"\t\t: (%uK) VIPT, %dway set-asc, %ub Line %s\n", \ | ||
97 | TO_KB((p)->sz), (p)->assoc, (p)->line_len, \ | ||
98 | enb ? "" : "DISABLED (kernel-build)"); \ | ||
99 | } | ||
100 | |||
101 | PR_CACHE(&cpuinfo_arc700[c].icache, __CONFIG_ARC_HAS_ICACHE, "I-Cache"); | ||
102 | PR_CACHE(&cpuinfo_arc700[c].dcache, __CONFIG_ARC_HAS_DCACHE, "D-Cache"); | ||
103 | |||
104 | return buf; | ||
105 | } | ||
106 | |||
107 | /* | ||
108 | * Read the Cache Build Confuration Registers, Decode them and save into | ||
109 | * the cpuinfo structure for later use. | ||
110 | * No Validation done here, simply read/convert the BCRs | ||
111 | */ | ||
112 | void __init read_decode_cache_bcr(void) | ||
113 | { | ||
114 | struct bcr_cache ibcr, dbcr; | ||
115 | struct cpuinfo_arc_cache *p_ic, *p_dc; | ||
116 | unsigned int cpu = smp_processor_id(); | ||
117 | |||
118 | p_ic = &cpuinfo_arc700[cpu].icache; | ||
119 | READ_BCR(ARC_REG_IC_BCR, ibcr); | ||
120 | |||
121 | if (ibcr.config == 0x3) | ||
122 | p_ic->assoc = 2; | ||
123 | p_ic->line_len = 8 << ibcr.line_len; | ||
124 | p_ic->sz = 0x200 << ibcr.sz; | ||
125 | p_ic->ver = ibcr.ver; | ||
126 | |||
127 | p_dc = &cpuinfo_arc700[cpu].dcache; | ||
128 | READ_BCR(ARC_REG_DC_BCR, dbcr); | ||
129 | |||
130 | if (dbcr.config == 0x2) | ||
131 | p_dc->assoc = 4; | ||
132 | p_dc->line_len = 16 << dbcr.line_len; | ||
133 | p_dc->sz = 0x200 << dbcr.sz; | ||
134 | p_dc->ver = dbcr.ver; | ||
135 | } | ||
136 | |||
137 | /* | ||
138 | * 1. Validate the Cache Geomtery (compile time config matches hardware) | ||
139 | * 2. If I-cache suffers from aliasing, setup work arounds (difft flush rtn) | ||
140 | * (aliasing D-cache configurations are not supported YET) | ||
141 | * 3. Enable the Caches, setup default flush mode for D-Cache | ||
142 | * 3. Calculate the SHMLBA used by user space | ||
143 | */ | ||
144 | void __init arc_cache_init(void) | ||
145 | { | ||
146 | unsigned int temp; | ||
147 | unsigned int cpu = smp_processor_id(); | ||
148 | struct cpuinfo_arc_cache *ic = &cpuinfo_arc700[cpu].icache; | ||
149 | struct cpuinfo_arc_cache *dc = &cpuinfo_arc700[cpu].dcache; | ||
150 | int way_pg_ratio = way_pg_ratio; | ||
151 | char str[256]; | ||
152 | |||
153 | printk(arc_cache_mumbojumbo(0, str, sizeof(str))); | ||
154 | |||
155 | if (!ic->ver) | ||
156 | goto chk_dc; | ||
157 | |||
158 | #ifdef CONFIG_ARC_HAS_ICACHE | ||
159 | /* 1. Confirm some of I-cache params which Linux assumes */ | ||
160 | if ((ic->assoc != ARC_ICACHE_WAYS) || | ||
161 | (ic->line_len != ARC_ICACHE_LINE_LEN)) { | ||
162 | panic("Cache H/W doesn't match kernel Config"); | ||
163 | } | ||
164 | #if (CONFIG_ARC_MMU_VER > 2) | ||
165 | if (ic->ver != 3) { | ||
166 | if (running_on_hw) | ||
167 | panic("Cache ver doesn't match MMU ver\n"); | ||
168 | |||
169 | /* For ISS - suggest the toggles to use */ | ||
170 | pr_err("Use -prop=icache_version=3,-prop=dcache_version=3\n"); | ||
171 | |||
172 | } | ||
173 | #endif | ||
174 | |||
175 | /* | ||
176 | * if Cache way size is <= page size then no aliasing exhibited | ||
177 | * otherwise ratio determines num of aliases. | ||
178 | * e.g. 32K I$, 2 way set assoc, 8k pg size | ||
179 | * way-sz = 32k/2 = 16k | ||
180 | * way-pg-ratio = 16k/8k = 2, so 2 aliases possible | ||
181 | * (meaning 1 line could be in 2 possible locations). | ||
182 | */ | ||
183 | way_pg_ratio = ic->sz / ARC_ICACHE_WAYS / PAGE_SIZE; | ||
184 | switch (way_pg_ratio) { | ||
185 | case 0: | ||
186 | case 1: | ||
187 | ___flush_icache_rtn = __ic_line_inv_no_alias; | ||
188 | break; | ||
189 | case 2: | ||
190 | ___flush_icache_rtn = __ic_line_inv_2_alias; | ||
191 | break; | ||
192 | case 4: | ||
193 | ___flush_icache_rtn = __ic_line_inv_4_alias; | ||
194 | break; | ||
195 | default: | ||
196 | panic("Unsupported I-Cache Sz\n"); | ||
197 | } | ||
198 | #endif | ||
199 | |||
200 | /* Enable/disable I-Cache */ | ||
201 | temp = read_aux_reg(ARC_REG_IC_CTRL); | ||
202 | |||
203 | #ifdef CONFIG_ARC_HAS_ICACHE | ||
204 | temp &= ~IC_CTRL_CACHE_DISABLE; | ||
205 | #else | ||
206 | temp |= IC_CTRL_CACHE_DISABLE; | ||
207 | #endif | ||
208 | |||
209 | write_aux_reg(ARC_REG_IC_CTRL, temp); | ||
210 | |||
211 | chk_dc: | ||
212 | if (!dc->ver) | ||
213 | return; | ||
214 | |||
215 | #ifdef CONFIG_ARC_HAS_DCACHE | ||
216 | if ((dc->assoc != ARC_DCACHE_WAYS) || | ||
217 | (dc->line_len != ARC_DCACHE_LINE_LEN)) { | ||
218 | panic("Cache H/W doesn't match kernel Config"); | ||
219 | } | ||
220 | |||
221 | /* check for D-Cache aliasing */ | ||
222 | if ((dc->sz / ARC_DCACHE_WAYS) > PAGE_SIZE) | ||
223 | panic("D$ aliasing not handled right now\n"); | ||
224 | #endif | ||
225 | |||
226 | /* Set the default Invalidate Mode to "simpy discard dirty lines" | ||
227 | * as this is more frequent then flush before invalidate | ||
228 | * Ofcourse we toggle this default behviour when desired | ||
229 | */ | ||
230 | temp = read_aux_reg(ARC_REG_DC_CTRL); | ||
231 | temp &= ~DC_CTRL_INV_MODE_FLUSH; | ||
232 | |||
233 | #ifdef CONFIG_ARC_HAS_DCACHE | ||
234 | /* Enable D-Cache: Clear Bit 0 */ | ||
235 | write_aux_reg(ARC_REG_DC_CTRL, temp & ~IC_CTRL_CACHE_DISABLE); | ||
236 | #else | ||
237 | /* Flush D cache */ | ||
238 | write_aux_reg(ARC_REG_DC_FLSH, 0x1); | ||
239 | /* Disable D cache */ | ||
240 | write_aux_reg(ARC_REG_DC_CTRL, temp | IC_CTRL_CACHE_DISABLE); | ||
241 | #endif | ||
242 | |||
243 | return; | ||
244 | } | ||
245 | |||
246 | #define OP_INV 0x1 | ||
247 | #define OP_FLUSH 0x2 | ||
248 | #define OP_FLUSH_N_INV 0x3 | ||
249 | |||
250 | #ifdef CONFIG_ARC_HAS_DCACHE | ||
251 | |||
252 | /*************************************************************** | ||
253 | * Machine specific helpers for Entire D-Cache or Per Line ops | ||
254 | */ | ||
255 | |||
256 | static inline void wait_for_flush(void) | ||
257 | { | ||
258 | while (read_aux_reg(ARC_REG_DC_CTRL) & DC_CTRL_FLUSH_STATUS) | ||
259 | ; | ||
260 | } | ||
261 | |||
262 | /* | ||
263 | * Operation on Entire D-Cache | ||
264 | * @cacheop = {OP_INV, OP_FLUSH, OP_FLUSH_N_INV} | ||
265 | * Note that constant propagation ensures all the checks are gone | ||
266 | * in generated code | ||
267 | */ | ||
268 | static inline void __dc_entire_op(const int cacheop) | ||
269 | { | ||
270 | unsigned long flags, tmp = tmp; | ||
271 | int aux; | ||
272 | |||
273 | local_irq_save(flags); | ||
274 | |||
275 | if (cacheop == OP_FLUSH_N_INV) { | ||
276 | /* Dcache provides 2 cmd: FLUSH or INV | ||
277 | * INV inturn has sub-modes: DISCARD or FLUSH-BEFORE | ||
278 | * flush-n-inv is achieved by INV cmd but with IM=1 | ||
279 | * Default INV sub-mode is DISCARD, which needs to be toggled | ||
280 | */ | ||
281 | tmp = read_aux_reg(ARC_REG_DC_CTRL); | ||
282 | write_aux_reg(ARC_REG_DC_CTRL, tmp | DC_CTRL_INV_MODE_FLUSH); | ||
283 | } | ||
284 | |||
285 | if (cacheop & OP_INV) /* Inv or flush-n-inv use same cmd reg */ | ||
286 | aux = ARC_REG_DC_IVDC; | ||
287 | else | ||
288 | aux = ARC_REG_DC_FLSH; | ||
289 | |||
290 | write_aux_reg(aux, 0x1); | ||
291 | |||
292 | if (cacheop & OP_FLUSH) /* flush / flush-n-inv both wait */ | ||
293 | wait_for_flush(); | ||
294 | |||
295 | /* Switch back the DISCARD ONLY Invalidate mode */ | ||
296 | if (cacheop == OP_FLUSH_N_INV) | ||
297 | write_aux_reg(ARC_REG_DC_CTRL, tmp & ~DC_CTRL_INV_MODE_FLUSH); | ||
298 | |||
299 | local_irq_restore(flags); | ||
300 | } | ||
301 | |||
302 | /* | ||
303 | * Per Line Operation on D-Cache | ||
304 | * Doesn't deal with type-of-op/IRQ-disabling/waiting-for-flush-to-complete | ||
305 | * It's sole purpose is to help gcc generate ZOL | ||
306 | */ | ||
307 | static inline void __dc_line_loop(unsigned long start, unsigned long sz, | ||
308 | int aux_reg) | ||
309 | { | ||
310 | int num_lines, slack; | ||
311 | |||
312 | /* Ensure we properly floor/ceil the non-line aligned/sized requests | ||
313 | * and have @start - aligned to cache line and integral @num_lines. | ||
314 | * This however can be avoided for page sized since: | ||
315 | * -@start will be cache-line aligned already (being page aligned) | ||
316 | * -@sz will be integral multiple of line size (being page sized). | ||
317 | */ | ||
318 | if (!(__builtin_constant_p(sz) && sz == PAGE_SIZE)) { | ||
319 | slack = start & ~DCACHE_LINE_MASK; | ||
320 | sz += slack; | ||
321 | start -= slack; | ||
322 | } | ||
323 | |||
324 | num_lines = DIV_ROUND_UP(sz, ARC_DCACHE_LINE_LEN); | ||
325 | |||
326 | while (num_lines-- > 0) { | ||
327 | #if (CONFIG_ARC_MMU_VER > 2) | ||
328 | /* | ||
329 | * Just as for I$, in MMU v3, D$ ops also require | ||
330 | * "tag" bits in DC_PTAG, "index" bits in FLDL,IVDL ops | ||
331 | * But we pass phy addr for both. This works since Linux | ||
332 | * doesn't support aliasing configs for D$, yet. | ||
333 | * Thus paddr is enough to provide both tag and index. | ||
334 | */ | ||
335 | write_aux_reg(ARC_REG_DC_PTAG, start); | ||
336 | #endif | ||
337 | write_aux_reg(aux_reg, start); | ||
338 | start += ARC_DCACHE_LINE_LEN; | ||
339 | } | ||
340 | } | ||
341 | |||
342 | /* | ||
343 | * D-Cache : Per Line INV (discard or wback+discard) or FLUSH (wback) | ||
344 | */ | ||
345 | static inline void __dc_line_op(unsigned long start, unsigned long sz, | ||
346 | const int cacheop) | ||
347 | { | ||
348 | unsigned long flags, tmp = tmp; | ||
349 | int aux; | ||
350 | |||
351 | local_irq_save(flags); | ||
352 | |||
353 | if (cacheop == OP_FLUSH_N_INV) { | ||
354 | /* | ||
355 | * Dcache provides 2 cmd: FLUSH or INV | ||
356 | * INV inturn has sub-modes: DISCARD or FLUSH-BEFORE | ||
357 | * flush-n-inv is achieved by INV cmd but with IM=1 | ||
358 | * Default INV sub-mode is DISCARD, which needs to be toggled | ||
359 | */ | ||
360 | tmp = read_aux_reg(ARC_REG_DC_CTRL); | ||
361 | write_aux_reg(ARC_REG_DC_CTRL, tmp | DC_CTRL_INV_MODE_FLUSH); | ||
362 | } | ||
363 | |||
364 | if (cacheop & OP_INV) /* Inv / flush-n-inv use same cmd reg */ | ||
365 | aux = ARC_REG_DC_IVDL; | ||
366 | else | ||
367 | aux = ARC_REG_DC_FLDL; | ||
368 | |||
369 | __dc_line_loop(start, sz, aux); | ||
370 | |||
371 | if (cacheop & OP_FLUSH) /* flush / flush-n-inv both wait */ | ||
372 | wait_for_flush(); | ||
373 | |||
374 | /* Switch back the DISCARD ONLY Invalidate mode */ | ||
375 | if (cacheop == OP_FLUSH_N_INV) | ||
376 | write_aux_reg(ARC_REG_DC_CTRL, tmp & ~DC_CTRL_INV_MODE_FLUSH); | ||
377 | |||
378 | local_irq_restore(flags); | ||
379 | } | ||
380 | |||
381 | #else | ||
382 | |||
383 | #define __dc_entire_op(cacheop) | ||
384 | #define __dc_line_op(start, sz, cacheop) | ||
385 | |||
386 | #endif /* CONFIG_ARC_HAS_DCACHE */ | ||
387 | |||
388 | |||
389 | #ifdef CONFIG_ARC_HAS_ICACHE | ||
390 | |||
391 | /* | ||
392 | * I-Cache Aliasing in ARC700 VIPT caches | ||
393 | * | ||
394 | * For fetching code from I$, ARC700 uses vaddr (embedded in program code) | ||
395 | * to "index" into SET of cache-line and paddr from MMU to match the TAG | ||
396 | * in the WAYS of SET. | ||
397 | * | ||
398 | * However the CDU iterface (to flush/inv) lines from software, only takes | ||
399 | * paddr (to have simpler hardware interface). For simpler cases, using paddr | ||
400 | * alone suffices. | ||
401 | * e.g. 2-way-set-assoc, 16K I$ (8k MMU pg sz, 32b cache line size): | ||
402 | * way_sz = cache_sz / num_ways = 16k/2 = 8k | ||
403 | * num_sets = way_sz / line_sz = 8k/32 = 256 => 8 bits | ||
404 | * Ignoring the bottom 5 bits corresp to the off within a 32b cacheline, | ||
405 | * bits req for calc set-index = bits 12:5 (0 based). Since this range fits | ||
406 | * inside the bottom 13 bits of paddr, which are same for vaddr and paddr | ||
407 | * (with 8k pg sz), paddr alone can be safely used by CDU to unambigously | ||
408 | * locate a cache-line. | ||
409 | * | ||
410 | * However for a difft sized cache, say 32k I$, above math yields need | ||
411 | * for 14 bits of vaddr to locate a cache line, which can't be provided by | ||
412 | * paddr, since the bit 13 (0 based) might differ between the two. | ||
413 | * | ||
414 | * This lack of extra bits needed for correct line addressing, defines the | ||
415 | * classical problem of Cache aliasing with VIPT architectures | ||
416 | * num_aliases = 1 << extra_bits | ||
417 | * e.g. 2-way-set-assoc, 32K I$ with 8k MMU pg sz => 2 aliases | ||
418 | * 2-way-set-assoc, 64K I$ with 8k MMU pg sz => 4 aliases | ||
419 | * 2-way-set-assoc, 16K I$ with 8k MMU pg sz => NO aliases | ||
420 | * | ||
421 | * ------------------ | ||
422 | * MMU v1/v2 (Fixed Page Size 8k) | ||
423 | * ------------------ | ||
424 | * The solution was to provide CDU with these additonal vaddr bits. These | ||
425 | * would be bits [x:13], x would depend on cache-geom. | ||
426 | * H/w folks chose [17:13] to be a future safe range, and moreso these 5 bits | ||
427 | * of vaddr could easily be "stuffed" in the paddr as bits [4:0] since the | ||
428 | * orig 5 bits of paddr were anyways ignored by CDU line ops, as they | ||
429 | * represent the offset within cache-line. The adv of using this "clumsy" | ||
430 | * interface for additional info was no new reg was needed in CDU. | ||
431 | * | ||
432 | * 17:13 represented the max num of bits passable, actual bits needed were | ||
433 | * fewer, based on the num-of-aliases possible. | ||
434 | * -for 2 alias possibility, only bit 13 needed (32K cache) | ||
435 | * -for 4 alias possibility, bits 14:13 needed (64K cache) | ||
436 | * | ||
437 | * Since vaddr was not available for all instances of I$ flush req by core | ||
438 | * kernel, the only safe way (non-optimal though) was to kill all possible | ||
439 | * lines which could represent an alias (even if they didnt represent one | ||
440 | * in execution). | ||
441 | * e.g. for 64K I$, 4 aliases possible, so we did | ||
442 | * flush start | ||
443 | * flush start | 0x01 | ||
444 | * flush start | 0x2 | ||
445 | * flush start | 0x3 | ||
446 | * | ||
447 | * The penalty was invoking the operation itself, since tag match is anyways | ||
448 | * paddr based, a line which didn't represent an alias would not match the | ||
449 | * paddr, hence wont be killed | ||
450 | * | ||
451 | * Note that aliasing concerns are independent of line-sz for a given cache | ||
452 | * geometry (size + set_assoc) because the extra bits required by line-sz are | ||
453 | * reduced from the set calc. | ||
454 | * e.g. 2-way-set-assoc, 32K I$ with 8k MMU pg sz and using math above | ||
455 | * 32b line-sz: 9 bits set-index-calc, 5 bits offset-in-line => 1 extra bit | ||
456 | * 64b line-sz: 8 bits set-index-calc, 6 bits offset-in-line => 1 extra bit | ||
457 | * | ||
458 | * ------------------ | ||
459 | * MMU v3 | ||
460 | * ------------------ | ||
461 | * This ver of MMU supports var page sizes (1k-16k) - Linux will support | ||
462 | * 8k (default), 16k and 4k. | ||
463 | * However from hardware perspective, smaller page sizes aggrevate aliasing | ||
464 | * meaning more vaddr bits needed to disambiguate the cache-line-op ; | ||
465 | * the existing scheme of piggybacking won't work for certain configurations. | ||
466 | * Two new registers IC_PTAG and DC_PTAG inttoduced. | ||
467 | * "tag" bits are provided in PTAG, index bits in existing IVIL/IVDL/FLDL regs | ||
468 | */ | ||
469 | |||
470 | /*********************************************************** | ||
471 | * Machine specific helpers for per line I-Cache invalidate. | ||
472 | * 3 routines to accpunt for 1, 2, 4 aliases possible | ||
473 | */ | ||
474 | |||
475 | static void __ic_line_inv_no_alias(unsigned long start, int num_lines) | ||
476 | { | ||
477 | while (num_lines-- > 0) { | ||
478 | #if (CONFIG_ARC_MMU_VER > 2) | ||
479 | write_aux_reg(ARC_REG_IC_PTAG, start); | ||
480 | #endif | ||
481 | write_aux_reg(ARC_REG_IC_IVIL, start); | ||
482 | start += ARC_ICACHE_LINE_LEN; | ||
483 | } | ||
484 | } | ||
485 | |||
486 | static void __ic_line_inv_2_alias(unsigned long start, int num_lines) | ||
487 | { | ||
488 | while (num_lines-- > 0) { | ||
489 | |||
490 | #if (CONFIG_ARC_MMU_VER > 2) | ||
491 | /* | ||
492 | * MMU v3, CDU prog model (for line ops) now uses a new IC_PTAG | ||
493 | * reg to pass the "tag" bits and existing IVIL reg only looks | ||
494 | * at bits relevant for "index" (details above) | ||
495 | * Programming Notes: | ||
496 | * -when writing tag to PTAG reg, bit chopping can be avoided, | ||
497 | * CDU ignores non-tag bits. | ||
498 | * -Ideally "index" must be computed from vaddr, but it is not | ||
499 | * avail in these rtns. So to be safe, we kill the lines in all | ||
500 | * possible indexes corresp to num of aliases possible for | ||
501 | * given cache config. | ||
502 | */ | ||
503 | write_aux_reg(ARC_REG_IC_PTAG, start); | ||
504 | write_aux_reg(ARC_REG_IC_IVIL, | ||
505 | start & ~(0x1 << PAGE_SHIFT)); | ||
506 | write_aux_reg(ARC_REG_IC_IVIL, start | (0x1 << PAGE_SHIFT)); | ||
507 | #else | ||
508 | write_aux_reg(ARC_REG_IC_IVIL, start); | ||
509 | write_aux_reg(ARC_REG_IC_IVIL, start | 0x01); | ||
510 | #endif | ||
511 | start += ARC_ICACHE_LINE_LEN; | ||
512 | } | ||
513 | } | ||
514 | |||
515 | static void __ic_line_inv_4_alias(unsigned long start, int num_lines) | ||
516 | { | ||
517 | while (num_lines-- > 0) { | ||
518 | |||
519 | #if (CONFIG_ARC_MMU_VER > 2) | ||
520 | write_aux_reg(ARC_REG_IC_PTAG, start); | ||
521 | |||
522 | write_aux_reg(ARC_REG_IC_IVIL, | ||
523 | start & ~(0x3 << PAGE_SHIFT)); | ||
524 | write_aux_reg(ARC_REG_IC_IVIL, | ||
525 | start & ~(0x2 << PAGE_SHIFT)); | ||
526 | write_aux_reg(ARC_REG_IC_IVIL, | ||
527 | start & ~(0x1 << PAGE_SHIFT)); | ||
528 | write_aux_reg(ARC_REG_IC_IVIL, start | (0x3 << PAGE_SHIFT)); | ||
529 | #else | ||
530 | write_aux_reg(ARC_REG_IC_IVIL, start); | ||
531 | write_aux_reg(ARC_REG_IC_IVIL, start | 0x01); | ||
532 | write_aux_reg(ARC_REG_IC_IVIL, start | 0x02); | ||
533 | write_aux_reg(ARC_REG_IC_IVIL, start | 0x03); | ||
534 | #endif | ||
535 | start += ARC_ICACHE_LINE_LEN; | ||
536 | } | ||
537 | } | ||
538 | |||
539 | static void __ic_line_inv(unsigned long start, unsigned long sz) | ||
540 | { | ||
541 | unsigned long flags; | ||
542 | int num_lines, slack; | ||
543 | |||
544 | /* | ||
545 | * Ensure we properly floor/ceil the non-line aligned/sized requests | ||
546 | * and have @start - aligned to cache line, and integral @num_lines | ||
547 | * However page sized flushes can be compile time optimised. | ||
548 | * -@start will be cache-line aligned already (being page aligned) | ||
549 | * -@sz will be integral multiple of line size (being page sized). | ||
550 | */ | ||
551 | if (!(__builtin_constant_p(sz) && sz == PAGE_SIZE)) { | ||
552 | slack = start & ~ICACHE_LINE_MASK; | ||
553 | sz += slack; | ||
554 | start -= slack; | ||
555 | } | ||
556 | |||
557 | num_lines = DIV_ROUND_UP(sz, ARC_ICACHE_LINE_LEN); | ||
558 | |||
559 | local_irq_save(flags); | ||
560 | (*___flush_icache_rtn) (start, num_lines); | ||
561 | local_irq_restore(flags); | ||
562 | } | ||
563 | |||
564 | /* Unlike routines above, having vaddr for flush op (along with paddr), | ||
565 | * prevents the need to speculatively kill the lines in multiple sets | ||
566 | * based on ratio of way_sz : pg_sz | ||
567 | */ | ||
568 | static void __ic_line_inv_vaddr(unsigned long phy_start, | ||
569 | unsigned long vaddr, unsigned long sz) | ||
570 | { | ||
571 | unsigned long flags; | ||
572 | int num_lines, slack; | ||
573 | unsigned int addr; | ||
574 | |||
575 | slack = phy_start & ~ICACHE_LINE_MASK; | ||
576 | sz += slack; | ||
577 | phy_start -= slack; | ||
578 | num_lines = DIV_ROUND_UP(sz, ARC_ICACHE_LINE_LEN); | ||
579 | |||
580 | #if (CONFIG_ARC_MMU_VER > 2) | ||
581 | vaddr &= ~ICACHE_LINE_MASK; | ||
582 | addr = phy_start; | ||
583 | #else | ||
584 | /* bits 17:13 of vaddr go as bits 4:0 of paddr */ | ||
585 | addr = phy_start | ((vaddr >> 13) & 0x1F); | ||
586 | #endif | ||
587 | |||
588 | local_irq_save(flags); | ||
589 | while (num_lines-- > 0) { | ||
590 | #if (CONFIG_ARC_MMU_VER > 2) | ||
591 | /* tag comes from phy addr */ | ||
592 | write_aux_reg(ARC_REG_IC_PTAG, addr); | ||
593 | |||
594 | /* index bits come from vaddr */ | ||
595 | write_aux_reg(ARC_REG_IC_IVIL, vaddr); | ||
596 | vaddr += ARC_ICACHE_LINE_LEN; | ||
597 | #else | ||
598 | /* this paddr contains vaddrs bits as needed */ | ||
599 | write_aux_reg(ARC_REG_IC_IVIL, addr); | ||
600 | #endif | ||
601 | addr += ARC_ICACHE_LINE_LEN; | ||
602 | } | ||
603 | local_irq_restore(flags); | ||
604 | } | ||
605 | |||
606 | #else | ||
607 | |||
608 | #define __ic_line_inv(start, sz) | ||
609 | #define __ic_line_inv_vaddr(pstart, vstart, sz) | ||
610 | |||
611 | #endif /* CONFIG_ARC_HAS_ICACHE */ | ||
612 | |||
613 | |||
614 | /*********************************************************** | ||
615 | * Exported APIs | ||
616 | */ | ||
617 | |||
618 | /* TBD: use pg_arch_1 to optimize this */ | ||
619 | void flush_dcache_page(struct page *page) | ||
620 | { | ||
621 | __dc_line_op((unsigned long)page_address(page), PAGE_SIZE, OP_FLUSH); | ||
622 | } | ||
623 | EXPORT_SYMBOL(flush_dcache_page); | ||
624 | |||
625 | |||
626 | void dma_cache_wback_inv(unsigned long start, unsigned long sz) | ||
627 | { | ||
628 | __dc_line_op(start, sz, OP_FLUSH_N_INV); | ||
629 | } | ||
630 | EXPORT_SYMBOL(dma_cache_wback_inv); | ||
631 | |||
632 | void dma_cache_inv(unsigned long start, unsigned long sz) | ||
633 | { | ||
634 | __dc_line_op(start, sz, OP_INV); | ||
635 | } | ||
636 | EXPORT_SYMBOL(dma_cache_inv); | ||
637 | |||
638 | void dma_cache_wback(unsigned long start, unsigned long sz) | ||
639 | { | ||
640 | __dc_line_op(start, sz, OP_FLUSH); | ||
641 | } | ||
642 | EXPORT_SYMBOL(dma_cache_wback); | ||
643 | |||
644 | /* | ||
645 | * This is API for making I/D Caches consistent when modifying code | ||
646 | * (loadable modules, kprobes, etc) | ||
647 | * This is called on insmod, with kernel virtual address for CODE of | ||
648 | * the module. ARC cache maintenance ops require PHY address thus we | ||
649 | * need to convert vmalloc addr to PHY addr | ||
650 | */ | ||
651 | void flush_icache_range(unsigned long kstart, unsigned long kend) | ||
652 | { | ||
653 | unsigned int tot_sz, off, sz; | ||
654 | unsigned long phy, pfn; | ||
655 | unsigned long flags; | ||
656 | |||
657 | /* printk("Kernel Cache Cohenercy: %lx to %lx\n",kstart, kend); */ | ||
658 | |||
659 | /* This is not the right API for user virtual address */ | ||
660 | if (kstart < TASK_SIZE) { | ||
661 | BUG_ON("Flush icache range for user virtual addr space"); | ||
662 | return; | ||
663 | } | ||
664 | |||
665 | /* Shortcut for bigger flush ranges. | ||
666 | * Here we don't care if this was kernel virtual or phy addr | ||
667 | */ | ||
668 | tot_sz = kend - kstart; | ||
669 | if (tot_sz > PAGE_SIZE) { | ||
670 | flush_cache_all(); | ||
671 | return; | ||
672 | } | ||
673 | |||
674 | /* Case: Kernel Phy addr (0x8000_0000 onwards) */ | ||
675 | if (likely(kstart > PAGE_OFFSET)) { | ||
676 | __ic_line_inv(kstart, kend - kstart); | ||
677 | __dc_line_op(kstart, kend - kstart, OP_FLUSH); | ||
678 | return; | ||
679 | } | ||
680 | |||
681 | /* | ||
682 | * Case: Kernel Vaddr (0x7000_0000 to 0x7fff_ffff) | ||
683 | * (1) ARC Cache Maintenance ops only take Phy addr, hence special | ||
684 | * handling of kernel vaddr. | ||
685 | * | ||
686 | * (2) Despite @tot_sz being < PAGE_SIZE (bigger cases handled already), | ||
687 | * it still needs to handle a 2 page scenario, where the range | ||
688 | * straddles across 2 virtual pages and hence need for loop | ||
689 | */ | ||
690 | while (tot_sz > 0) { | ||
691 | off = kstart % PAGE_SIZE; | ||
692 | pfn = vmalloc_to_pfn((void *)kstart); | ||
693 | phy = (pfn << PAGE_SHIFT) + off; | ||
694 | sz = min_t(unsigned int, tot_sz, PAGE_SIZE - off); | ||
695 | local_irq_save(flags); | ||
696 | __dc_line_op(phy, sz, OP_FLUSH); | ||
697 | __ic_line_inv(phy, sz); | ||
698 | local_irq_restore(flags); | ||
699 | kstart += sz; | ||
700 | tot_sz -= sz; | ||
701 | } | ||
702 | } | ||
703 | |||
704 | /* | ||
705 | * Optimised ver of flush_icache_range() with spec callers: ptrace/signals | ||
706 | * where vaddr is also available. This allows passing both vaddr and paddr | ||
707 | * bits to CDU for cache flush, short-circuting the current pessimistic algo | ||
708 | * which kills all possible aliases. | ||
709 | * An added adv of knowing that vaddr is user-vaddr avoids various checks | ||
710 | * and handling for k-vaddr, k-paddr as done in orig ver above | ||
711 | */ | ||
712 | void flush_icache_range_vaddr(unsigned long paddr, unsigned long u_vaddr, | ||
713 | int len) | ||
714 | { | ||
715 | __ic_line_inv_vaddr(paddr, u_vaddr, len); | ||
716 | __dc_line_op(paddr, len, OP_FLUSH); | ||
717 | } | ||
718 | |||
719 | /* | ||
720 | * XXX: This also needs to be optim using pg_arch_1 | ||
721 | * This is called when a page-cache page is about to be mapped into a | ||
722 | * user process' address space. It offers an opportunity for a | ||
723 | * port to ensure d-cache/i-cache coherency if necessary. | ||
724 | */ | ||
725 | void flush_icache_page(struct vm_area_struct *vma, struct page *page) | ||
726 | { | ||
727 | if (!(vma->vm_flags & VM_EXEC)) | ||
728 | return; | ||
729 | |||
730 | __ic_line_inv((unsigned long)page_address(page), PAGE_SIZE); | ||
731 | } | ||
732 | |||
733 | void flush_icache_all(void) | ||
734 | { | ||
735 | unsigned long flags; | ||
736 | |||
737 | local_irq_save(flags); | ||
738 | |||
739 | write_aux_reg(ARC_REG_IC_IVIC, 1); | ||
740 | |||
741 | /* lr will not complete till the icache inv operation is not over */ | ||
742 | read_aux_reg(ARC_REG_IC_CTRL); | ||
743 | local_irq_restore(flags); | ||
744 | } | ||
745 | |||
746 | noinline void flush_cache_all(void) | ||
747 | { | ||
748 | unsigned long flags; | ||
749 | |||
750 | local_irq_save(flags); | ||
751 | |||
752 | flush_icache_all(); | ||
753 | __dc_entire_op(OP_FLUSH_N_INV); | ||
754 | |||
755 | local_irq_restore(flags); | ||
756 | |||
757 | } | ||
758 | |||
759 | /********************************************************************** | ||
760 | * Explicit Cache flush request from user space via syscall | ||
761 | * Needed for JITs which generate code on the fly | ||
762 | */ | ||
763 | SYSCALL_DEFINE3(cacheflush, uint32_t, start, uint32_t, sz, uint32_t, flags) | ||
764 | { | ||
765 | /* TBD: optimize this */ | ||
766 | flush_cache_all(); | ||
767 | return 0; | ||
768 | } | ||
diff --git a/arch/arc/mm/dma.c b/arch/arc/mm/dma.c new file mode 100644 index 000000000000..12cc6485b218 --- /dev/null +++ b/arch/arc/mm/dma.c | |||
@@ -0,0 +1,94 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | /* | ||
10 | * DMA Coherent API Notes | ||
11 | * | ||
12 | * I/O is inherently non-coherent on ARC. So a coherent DMA buffer is | ||
13 | * implemented by accessintg it using a kernel virtual address, with | ||
14 | * Cache bit off in the TLB entry. | ||
15 | * | ||
16 | * The default DMA address == Phy address which is 0x8000_0000 based. | ||
17 | * A platform/device can make it zero based, by over-riding | ||
18 | * plat_{dma,kernel}_addr_to_{kernel,dma} | ||
19 | */ | ||
20 | |||
21 | #include <linux/dma-mapping.h> | ||
22 | #include <linux/dma-debug.h> | ||
23 | #include <linux/export.h> | ||
24 | #include <asm/cacheflush.h> | ||
25 | |||
26 | /* | ||
27 | * Helpers for Coherent DMA API. | ||
28 | */ | ||
29 | void *dma_alloc_noncoherent(struct device *dev, size_t size, | ||
30 | dma_addr_t *dma_handle, gfp_t gfp) | ||
31 | { | ||
32 | void *paddr; | ||
33 | |||
34 | /* This is linear addr (0x8000_0000 based) */ | ||
35 | paddr = alloc_pages_exact(size, gfp); | ||
36 | if (!paddr) | ||
37 | return NULL; | ||
38 | |||
39 | /* This is bus address, platform dependent */ | ||
40 | *dma_handle = plat_kernel_addr_to_dma(dev, paddr); | ||
41 | |||
42 | return paddr; | ||
43 | } | ||
44 | EXPORT_SYMBOL(dma_alloc_noncoherent); | ||
45 | |||
46 | void dma_free_noncoherent(struct device *dev, size_t size, void *vaddr, | ||
47 | dma_addr_t dma_handle) | ||
48 | { | ||
49 | free_pages_exact((void *)plat_dma_addr_to_kernel(dev, dma_handle), | ||
50 | size); | ||
51 | } | ||
52 | EXPORT_SYMBOL(dma_free_noncoherent); | ||
53 | |||
54 | void *dma_alloc_coherent(struct device *dev, size_t size, | ||
55 | dma_addr_t *dma_handle, gfp_t gfp) | ||
56 | { | ||
57 | void *paddr, *kvaddr; | ||
58 | |||
59 | /* This is linear addr (0x8000_0000 based) */ | ||
60 | paddr = alloc_pages_exact(size, gfp); | ||
61 | if (!paddr) | ||
62 | return NULL; | ||
63 | |||
64 | /* This is kernel Virtual address (0x7000_0000 based) */ | ||
65 | kvaddr = ioremap_nocache((unsigned long)paddr, size); | ||
66 | if (kvaddr != NULL) | ||
67 | memset(kvaddr, 0, size); | ||
68 | |||
69 | /* This is bus address, platform dependent */ | ||
70 | *dma_handle = plat_kernel_addr_to_dma(dev, paddr); | ||
71 | |||
72 | return kvaddr; | ||
73 | } | ||
74 | EXPORT_SYMBOL(dma_alloc_coherent); | ||
75 | |||
76 | void dma_free_coherent(struct device *dev, size_t size, void *kvaddr, | ||
77 | dma_addr_t dma_handle) | ||
78 | { | ||
79 | iounmap((void __force __iomem *)kvaddr); | ||
80 | |||
81 | free_pages_exact((void *)plat_dma_addr_to_kernel(dev, dma_handle), | ||
82 | size); | ||
83 | } | ||
84 | EXPORT_SYMBOL(dma_free_coherent); | ||
85 | |||
86 | /* | ||
87 | * Helper for streaming DMA... | ||
88 | */ | ||
89 | void __arc_dma_cache_sync(unsigned long paddr, size_t size, | ||
90 | enum dma_data_direction dir) | ||
91 | { | ||
92 | __inline_dma_cache_sync(paddr, size, dir); | ||
93 | } | ||
94 | EXPORT_SYMBOL(__arc_dma_cache_sync); | ||
diff --git a/arch/arc/mm/extable.c b/arch/arc/mm/extable.c new file mode 100644 index 000000000000..014172ba8432 --- /dev/null +++ b/arch/arc/mm/extable.c | |||
@@ -0,0 +1,63 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Borrowed heavily from MIPS | ||
9 | */ | ||
10 | |||
11 | #include <linux/module.h> | ||
12 | #include <linux/uaccess.h> | ||
13 | |||
14 | int fixup_exception(struct pt_regs *regs) | ||
15 | { | ||
16 | const struct exception_table_entry *fixup; | ||
17 | |||
18 | fixup = search_exception_tables(instruction_pointer(regs)); | ||
19 | if (fixup) { | ||
20 | regs->ret = fixup->fixup; | ||
21 | |||
22 | return 1; | ||
23 | } | ||
24 | |||
25 | return 0; | ||
26 | } | ||
27 | |||
28 | #ifdef CONFIG_CC_OPTIMIZE_FOR_SIZE | ||
29 | |||
30 | long arc_copy_from_user_noinline(void *to, const void __user * from, | ||
31 | unsigned long n) | ||
32 | { | ||
33 | return __arc_copy_from_user(to, from, n); | ||
34 | } | ||
35 | EXPORT_SYMBOL(arc_copy_from_user_noinline); | ||
36 | |||
37 | long arc_copy_to_user_noinline(void __user *to, const void *from, | ||
38 | unsigned long n) | ||
39 | { | ||
40 | return __arc_copy_to_user(to, from, n); | ||
41 | } | ||
42 | EXPORT_SYMBOL(arc_copy_to_user_noinline); | ||
43 | |||
44 | unsigned long arc_clear_user_noinline(void __user *to, | ||
45 | unsigned long n) | ||
46 | { | ||
47 | return __arc_clear_user(to, n); | ||
48 | } | ||
49 | EXPORT_SYMBOL(arc_clear_user_noinline); | ||
50 | |||
51 | long arc_strncpy_from_user_noinline (char *dst, const char __user *src, | ||
52 | long count) | ||
53 | { | ||
54 | return __arc_strncpy_from_user(dst, src, count); | ||
55 | } | ||
56 | EXPORT_SYMBOL(arc_strncpy_from_user_noinline); | ||
57 | |||
58 | long arc_strnlen_user_noinline(const char __user *src, long n) | ||
59 | { | ||
60 | return __arc_strnlen_user(src, n); | ||
61 | } | ||
62 | EXPORT_SYMBOL(arc_strnlen_user_noinline); | ||
63 | #endif | ||
diff --git a/arch/arc/mm/fault.c b/arch/arc/mm/fault.c new file mode 100644 index 000000000000..af55aab803d2 --- /dev/null +++ b/arch/arc/mm/fault.c | |||
@@ -0,0 +1,228 @@ | |||
1 | /* Page Fault Handling for ARC (TLB Miss / ProtV) | ||
2 | * | ||
3 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify | ||
6 | * it under the terms of the GNU General Public License version 2 as | ||
7 | * published by the Free Software Foundation. | ||
8 | */ | ||
9 | |||
10 | #include <linux/signal.h> | ||
11 | #include <linux/interrupt.h> | ||
12 | #include <linux/sched.h> | ||
13 | #include <linux/errno.h> | ||
14 | #include <linux/ptrace.h> | ||
15 | #include <linux/version.h> | ||
16 | #include <linux/uaccess.h> | ||
17 | #include <linux/kdebug.h> | ||
18 | #include <asm/pgalloc.h> | ||
19 | |||
20 | static int handle_vmalloc_fault(struct mm_struct *mm, unsigned long address) | ||
21 | { | ||
22 | /* | ||
23 | * Synchronize this task's top level page-table | ||
24 | * with the 'reference' page table. | ||
25 | */ | ||
26 | pgd_t *pgd, *pgd_k; | ||
27 | pud_t *pud, *pud_k; | ||
28 | pmd_t *pmd, *pmd_k; | ||
29 | |||
30 | pgd = pgd_offset_fast(mm, address); | ||
31 | pgd_k = pgd_offset_k(address); | ||
32 | |||
33 | if (!pgd_present(*pgd_k)) | ||
34 | goto bad_area; | ||
35 | |||
36 | pud = pud_offset(pgd, address); | ||
37 | pud_k = pud_offset(pgd_k, address); | ||
38 | if (!pud_present(*pud_k)) | ||
39 | goto bad_area; | ||
40 | |||
41 | pmd = pmd_offset(pud, address); | ||
42 | pmd_k = pmd_offset(pud_k, address); | ||
43 | if (!pmd_present(*pmd_k)) | ||
44 | goto bad_area; | ||
45 | |||
46 | set_pmd(pmd, *pmd_k); | ||
47 | |||
48 | /* XXX: create the TLB entry here */ | ||
49 | return 0; | ||
50 | |||
51 | bad_area: | ||
52 | return 1; | ||
53 | } | ||
54 | |||
55 | void do_page_fault(struct pt_regs *regs, int write, unsigned long address, | ||
56 | unsigned long cause_code) | ||
57 | { | ||
58 | struct vm_area_struct *vma = NULL; | ||
59 | struct task_struct *tsk = current; | ||
60 | struct mm_struct *mm = tsk->mm; | ||
61 | siginfo_t info; | ||
62 | int fault, ret; | ||
63 | unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE | | ||
64 | (write ? FAULT_FLAG_WRITE : 0); | ||
65 | |||
66 | /* | ||
67 | * We fault-in kernel-space virtual memory on-demand. The | ||
68 | * 'reference' page table is init_mm.pgd. | ||
69 | * | ||
70 | * NOTE! We MUST NOT take any locks for this case. We may | ||
71 | * be in an interrupt or a critical region, and should | ||
72 | * only copy the information from the master page table, | ||
73 | * nothing more. | ||
74 | */ | ||
75 | if (address >= VMALLOC_START && address <= VMALLOC_END) { | ||
76 | ret = handle_vmalloc_fault(mm, address); | ||
77 | if (unlikely(ret)) | ||
78 | goto bad_area_nosemaphore; | ||
79 | else | ||
80 | return; | ||
81 | } | ||
82 | |||
83 | info.si_code = SEGV_MAPERR; | ||
84 | |||
85 | /* | ||
86 | * If we're in an interrupt or have no user | ||
87 | * context, we must not take the fault.. | ||
88 | */ | ||
89 | if (in_atomic() || !mm) | ||
90 | goto no_context; | ||
91 | |||
92 | retry: | ||
93 | down_read(&mm->mmap_sem); | ||
94 | vma = find_vma(mm, address); | ||
95 | if (!vma) | ||
96 | goto bad_area; | ||
97 | if (vma->vm_start <= address) | ||
98 | goto good_area; | ||
99 | if (!(vma->vm_flags & VM_GROWSDOWN)) | ||
100 | goto bad_area; | ||
101 | if (expand_stack(vma, address)) | ||
102 | goto bad_area; | ||
103 | |||
104 | /* | ||
105 | * Ok, we have a good vm_area for this memory access, so | ||
106 | * we can handle it.. | ||
107 | */ | ||
108 | good_area: | ||
109 | info.si_code = SEGV_ACCERR; | ||
110 | |||
111 | /* Handle protection violation, execute on heap or stack */ | ||
112 | |||
113 | if (cause_code == ((ECR_V_PROTV << 16) | ECR_C_PROTV_INST_FETCH)) | ||
114 | goto bad_area; | ||
115 | |||
116 | if (write) { | ||
117 | if (!(vma->vm_flags & VM_WRITE)) | ||
118 | goto bad_area; | ||
119 | } else { | ||
120 | if (!(vma->vm_flags & (VM_READ | VM_EXEC))) | ||
121 | goto bad_area; | ||
122 | } | ||
123 | |||
124 | survive: | ||
125 | /* | ||
126 | * If for any reason at all we couldn't handle the fault, | ||
127 | * make sure we exit gracefully rather than endlessly redo | ||
128 | * the fault. | ||
129 | */ | ||
130 | fault = handle_mm_fault(mm, vma, address, flags); | ||
131 | |||
132 | /* If Pagefault was interrupted by SIGKILL, exit page fault "early" */ | ||
133 | if (unlikely(fatal_signal_pending(current))) { | ||
134 | if ((fault & VM_FAULT_ERROR) && !(fault & VM_FAULT_RETRY)) | ||
135 | up_read(&mm->mmap_sem); | ||
136 | if (user_mode(regs)) | ||
137 | return; | ||
138 | } | ||
139 | |||
140 | if (likely(!(fault & VM_FAULT_ERROR))) { | ||
141 | if (flags & FAULT_FLAG_ALLOW_RETRY) { | ||
142 | /* To avoid updating stats twice for retry case */ | ||
143 | if (fault & VM_FAULT_MAJOR) | ||
144 | tsk->maj_flt++; | ||
145 | else | ||
146 | tsk->min_flt++; | ||
147 | |||
148 | if (fault & VM_FAULT_RETRY) { | ||
149 | flags &= ~FAULT_FLAG_ALLOW_RETRY; | ||
150 | flags |= FAULT_FLAG_TRIED; | ||
151 | goto retry; | ||
152 | } | ||
153 | } | ||
154 | |||
155 | /* Fault Handled Gracefully */ | ||
156 | up_read(&mm->mmap_sem); | ||
157 | return; | ||
158 | } | ||
159 | |||
160 | /* TBD: switch to pagefault_out_of_memory() */ | ||
161 | if (fault & VM_FAULT_OOM) | ||
162 | goto out_of_memory; | ||
163 | else if (fault & VM_FAULT_SIGBUS) | ||
164 | goto do_sigbus; | ||
165 | |||
166 | /* no man's land */ | ||
167 | BUG(); | ||
168 | |||
169 | /* | ||
170 | * Something tried to access memory that isn't in our memory map.. | ||
171 | * Fix it, but check if it's kernel or user first.. | ||
172 | */ | ||
173 | bad_area: | ||
174 | up_read(&mm->mmap_sem); | ||
175 | |||
176 | bad_area_nosemaphore: | ||
177 | /* User mode accesses just cause a SIGSEGV */ | ||
178 | if (user_mode(regs)) { | ||
179 | tsk->thread.fault_address = address; | ||
180 | tsk->thread.cause_code = cause_code; | ||
181 | info.si_signo = SIGSEGV; | ||
182 | info.si_errno = 0; | ||
183 | /* info.si_code has been set above */ | ||
184 | info.si_addr = (void __user *)address; | ||
185 | force_sig_info(SIGSEGV, &info, tsk); | ||
186 | return; | ||
187 | } | ||
188 | |||
189 | no_context: | ||
190 | /* Are we prepared to handle this kernel fault? | ||
191 | * | ||
192 | * (The kernel has valid exception-points in the source | ||
193 | * when it acesses user-memory. When it fails in one | ||
194 | * of those points, we find it in a table and do a jump | ||
195 | * to some fixup code that loads an appropriate error | ||
196 | * code) | ||
197 | */ | ||
198 | if (fixup_exception(regs)) | ||
199 | return; | ||
200 | |||
201 | die("Oops", regs, address, cause_code); | ||
202 | |||
203 | out_of_memory: | ||
204 | if (is_global_init(tsk)) { | ||
205 | yield(); | ||
206 | goto survive; | ||
207 | } | ||
208 | up_read(&mm->mmap_sem); | ||
209 | |||
210 | if (user_mode(regs)) | ||
211 | do_group_exit(SIGKILL); /* This will never return */ | ||
212 | |||
213 | goto no_context; | ||
214 | |||
215 | do_sigbus: | ||
216 | up_read(&mm->mmap_sem); | ||
217 | |||
218 | if (!user_mode(regs)) | ||
219 | goto no_context; | ||
220 | |||
221 | tsk->thread.fault_address = address; | ||
222 | tsk->thread.cause_code = cause_code; | ||
223 | info.si_signo = SIGBUS; | ||
224 | info.si_errno = 0; | ||
225 | info.si_code = BUS_ADRERR; | ||
226 | info.si_addr = (void __user *)address; | ||
227 | force_sig_info(SIGBUS, &info, tsk); | ||
228 | } | ||
diff --git a/arch/arc/mm/init.c b/arch/arc/mm/init.c new file mode 100644 index 000000000000..caf797de23fc --- /dev/null +++ b/arch/arc/mm/init.c | |||
@@ -0,0 +1,187 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <linux/kernel.h> | ||
10 | #include <linux/mm.h> | ||
11 | #include <linux/bootmem.h> | ||
12 | #include <linux/memblock.h> | ||
13 | #ifdef CONFIG_BLOCK_DEV_RAM | ||
14 | #include <linux/blk.h> | ||
15 | #endif | ||
16 | #include <linux/swap.h> | ||
17 | #include <linux/module.h> | ||
18 | #include <asm/page.h> | ||
19 | #include <asm/pgalloc.h> | ||
20 | #include <asm/sections.h> | ||
21 | #include <asm/arcregs.h> | ||
22 | |||
23 | pgd_t swapper_pg_dir[PTRS_PER_PGD] __aligned(PAGE_SIZE); | ||
24 | char empty_zero_page[PAGE_SIZE] __aligned(PAGE_SIZE); | ||
25 | EXPORT_SYMBOL(empty_zero_page); | ||
26 | |||
27 | /* Default tot mem from .config */ | ||
28 | static unsigned long arc_mem_sz = 0x20000000; /* some default */ | ||
29 | |||
30 | /* User can over-ride above with "mem=nnn[KkMm]" in cmdline */ | ||
31 | static int __init setup_mem_sz(char *str) | ||
32 | { | ||
33 | arc_mem_sz = memparse(str, NULL) & PAGE_MASK; | ||
34 | |||
35 | /* early console might not be setup yet - it will show up later */ | ||
36 | pr_info("\"mem=%s\": mem sz set to %ldM\n", str, TO_MB(arc_mem_sz)); | ||
37 | |||
38 | return 0; | ||
39 | } | ||
40 | early_param("mem", setup_mem_sz); | ||
41 | |||
42 | void __init early_init_dt_add_memory_arch(u64 base, u64 size) | ||
43 | { | ||
44 | arc_mem_sz = size & PAGE_MASK; | ||
45 | pr_info("Memory size set via devicetree %ldM\n", TO_MB(arc_mem_sz)); | ||
46 | } | ||
47 | |||
48 | /* | ||
49 | * First memory setup routine called from setup_arch() | ||
50 | * 1. setup swapper's mm @init_mm | ||
51 | * 2. Count the pages we have and setup bootmem allocator | ||
52 | * 3. zone setup | ||
53 | */ | ||
54 | void __init setup_arch_memory(void) | ||
55 | { | ||
56 | unsigned long zones_size[MAX_NR_ZONES] = { 0, 0 }; | ||
57 | unsigned long end_mem = CONFIG_LINUX_LINK_BASE + arc_mem_sz; | ||
58 | |||
59 | init_mm.start_code = (unsigned long)_text; | ||
60 | init_mm.end_code = (unsigned long)_etext; | ||
61 | init_mm.end_data = (unsigned long)_edata; | ||
62 | init_mm.brk = (unsigned long)_end; | ||
63 | |||
64 | /* | ||
65 | * We do it here, so that memory is correctly instantiated | ||
66 | * even if "mem=xxx" cmline over-ride is given and/or | ||
67 | * DT has memory node. Each causes an update to @arc_mem_sz | ||
68 | * and we finally add memory one here | ||
69 | */ | ||
70 | memblock_add(CONFIG_LINUX_LINK_BASE, arc_mem_sz); | ||
71 | |||
72 | /*------------- externs in mm need setting up ---------------*/ | ||
73 | |||
74 | /* first page of system - kernel .vector starts here */ | ||
75 | min_low_pfn = PFN_DOWN(CONFIG_LINUX_LINK_BASE); | ||
76 | |||
77 | /* Last usable page of low mem (no HIGHMEM yet for ARC port) */ | ||
78 | max_low_pfn = max_pfn = PFN_DOWN(end_mem); | ||
79 | |||
80 | max_mapnr = num_physpages = max_low_pfn - min_low_pfn; | ||
81 | |||
82 | /*------------- reserve kernel image -----------------------*/ | ||
83 | memblock_reserve(CONFIG_LINUX_LINK_BASE, | ||
84 | __pa(_end) - CONFIG_LINUX_LINK_BASE); | ||
85 | |||
86 | memblock_dump_all(); | ||
87 | |||
88 | /*-------------- node setup --------------------------------*/ | ||
89 | memset(zones_size, 0, sizeof(zones_size)); | ||
90 | zones_size[ZONE_NORMAL] = num_physpages; | ||
91 | |||
92 | /* | ||
93 | * We can't use the helper free_area_init(zones[]) because it uses | ||
94 | * PAGE_OFFSET to compute the @min_low_pfn which would be wrong | ||
95 | * when our kernel doesn't start at PAGE_OFFSET, i.e. | ||
96 | * PAGE_OFFSET != CONFIG_LINUX_LINK_BASE | ||
97 | */ | ||
98 | free_area_init_node(0, /* node-id */ | ||
99 | zones_size, /* num pages per zone */ | ||
100 | min_low_pfn, /* first pfn of node */ | ||
101 | NULL); /* NO holes */ | ||
102 | } | ||
103 | |||
104 | /* | ||
105 | * mem_init - initializes memory | ||
106 | * | ||
107 | * Frees up bootmem | ||
108 | * Calculates and displays memory available/used | ||
109 | */ | ||
110 | void __init mem_init(void) | ||
111 | { | ||
112 | int codesize, datasize, initsize, reserved_pages, free_pages; | ||
113 | int tmp; | ||
114 | |||
115 | high_memory = (void *)(CONFIG_LINUX_LINK_BASE + arc_mem_sz); | ||
116 | |||
117 | totalram_pages = free_all_bootmem(); | ||
118 | |||
119 | /* count all reserved pages [kernel code/data/mem_map..] */ | ||
120 | reserved_pages = 0; | ||
121 | for (tmp = 0; tmp < max_mapnr; tmp++) | ||
122 | if (PageReserved(mem_map + tmp)) | ||
123 | reserved_pages++; | ||
124 | |||
125 | /* XXX: nr_free_pages() is equivalent */ | ||
126 | free_pages = max_mapnr - reserved_pages; | ||
127 | |||
128 | /* | ||
129 | * For the purpose of display below, split the "reserve mem" | ||
130 | * kernel code/data is already shown explicitly, | ||
131 | * Show any other reservations (mem_map[ ] et al) | ||
132 | */ | ||
133 | reserved_pages -= (((unsigned int)_end - CONFIG_LINUX_LINK_BASE) >> | ||
134 | PAGE_SHIFT); | ||
135 | |||
136 | codesize = _etext - _text; | ||
137 | datasize = _end - _etext; | ||
138 | initsize = __init_end - __init_begin; | ||
139 | |||
140 | pr_info("Memory Available: %dM / %ldM (%dK code, %dK data, %dK init, %dK reserv)\n", | ||
141 | PAGES_TO_MB(free_pages), | ||
142 | TO_MB(arc_mem_sz), | ||
143 | TO_KB(codesize), TO_KB(datasize), TO_KB(initsize), | ||
144 | PAGES_TO_KB(reserved_pages)); | ||
145 | } | ||
146 | |||
147 | static void __init free_init_pages(const char *what, unsigned long begin, | ||
148 | unsigned long end) | ||
149 | { | ||
150 | unsigned long addr; | ||
151 | |||
152 | pr_info("Freeing %s: %ldk [%lx] to [%lx]\n", | ||
153 | what, TO_KB(end - begin), begin, end); | ||
154 | |||
155 | /* need to check that the page we free is not a partial page */ | ||
156 | for (addr = begin; addr + PAGE_SIZE <= end; addr += PAGE_SIZE) { | ||
157 | ClearPageReserved(virt_to_page(addr)); | ||
158 | init_page_count(virt_to_page(addr)); | ||
159 | free_page(addr); | ||
160 | totalram_pages++; | ||
161 | } | ||
162 | } | ||
163 | |||
164 | /* | ||
165 | * free_initmem: Free all the __init memory. | ||
166 | */ | ||
167 | void __init_refok free_initmem(void) | ||
168 | { | ||
169 | free_init_pages("unused kernel memory", | ||
170 | (unsigned long)__init_begin, | ||
171 | (unsigned long)__init_end); | ||
172 | } | ||
173 | |||
174 | #ifdef CONFIG_BLK_DEV_INITRD | ||
175 | void __init free_initrd_mem(unsigned long start, unsigned long end) | ||
176 | { | ||
177 | free_init_pages("initrd memory", start, end); | ||
178 | } | ||
179 | #endif | ||
180 | |||
181 | #ifdef CONFIG_OF_FLATTREE | ||
182 | void __init early_init_dt_setup_initrd_arch(unsigned long start, | ||
183 | unsigned long end) | ||
184 | { | ||
185 | pr_err("%s(%lx, %lx)\n", __func__, start, end); | ||
186 | } | ||
187 | #endif /* CONFIG_OF_FLATTREE */ | ||
diff --git a/arch/arc/mm/ioremap.c b/arch/arc/mm/ioremap.c new file mode 100644 index 000000000000..3e5c92c79936 --- /dev/null +++ b/arch/arc/mm/ioremap.c | |||
@@ -0,0 +1,91 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <linux/vmalloc.h> | ||
10 | #include <linux/init.h> | ||
11 | #include <linux/module.h> | ||
12 | #include <linux/io.h> | ||
13 | #include <linux/mm.h> | ||
14 | #include <linux/slab.h> | ||
15 | #include <asm/cache.h> | ||
16 | |||
17 | void __iomem *ioremap(unsigned long paddr, unsigned long size) | ||
18 | { | ||
19 | unsigned long end; | ||
20 | |||
21 | /* Don't allow wraparound or zero size */ | ||
22 | end = paddr + size - 1; | ||
23 | if (!size || (end < paddr)) | ||
24 | return NULL; | ||
25 | |||
26 | /* If the region is h/w uncached, avoid MMU mappings */ | ||
27 | if (paddr >= ARC_UNCACHED_ADDR_SPACE) | ||
28 | return (void __iomem *)paddr; | ||
29 | |||
30 | return ioremap_prot(paddr, size, PAGE_KERNEL_NO_CACHE); | ||
31 | } | ||
32 | EXPORT_SYMBOL(ioremap); | ||
33 | |||
34 | /* | ||
35 | * ioremap with access flags | ||
36 | * Cache semantics wise it is same as ioremap - "forced" uncached. | ||
37 | * However unline vanilla ioremap which bypasses ARC MMU for addresses in | ||
38 | * ARC hardware uncached region, this one still goes thru the MMU as caller | ||
39 | * might need finer access control (R/W/X) | ||
40 | */ | ||
41 | void __iomem *ioremap_prot(phys_addr_t paddr, unsigned long size, | ||
42 | unsigned long flags) | ||
43 | { | ||
44 | void __iomem *vaddr; | ||
45 | struct vm_struct *area; | ||
46 | unsigned long off, end; | ||
47 | pgprot_t prot = __pgprot(flags); | ||
48 | |||
49 | /* Don't allow wraparound, zero size */ | ||
50 | end = paddr + size - 1; | ||
51 | if ((!size) || (end < paddr)) | ||
52 | return NULL; | ||
53 | |||
54 | /* An early platform driver might end up here */ | ||
55 | if (!slab_is_available()) | ||
56 | return NULL; | ||
57 | |||
58 | /* force uncached */ | ||
59 | prot = pgprot_noncached(prot); | ||
60 | |||
61 | /* Mappings have to be page-aligned */ | ||
62 | off = paddr & ~PAGE_MASK; | ||
63 | paddr &= PAGE_MASK; | ||
64 | size = PAGE_ALIGN(end + 1) - paddr; | ||
65 | |||
66 | /* | ||
67 | * Ok, go for it.. | ||
68 | */ | ||
69 | area = get_vm_area(size, VM_IOREMAP); | ||
70 | if (!area) | ||
71 | return NULL; | ||
72 | area->phys_addr = paddr; | ||
73 | vaddr = (void __iomem *)area->addr; | ||
74 | if (ioremap_page_range((unsigned long)vaddr, | ||
75 | (unsigned long)vaddr + size, paddr, prot)) { | ||
76 | vunmap((void __force *)vaddr); | ||
77 | return NULL; | ||
78 | } | ||
79 | return (void __iomem *)(off + (char __iomem *)vaddr); | ||
80 | } | ||
81 | EXPORT_SYMBOL(ioremap_prot); | ||
82 | |||
83 | |||
84 | void iounmap(const void __iomem *addr) | ||
85 | { | ||
86 | if (addr >= (void __force __iomem *)ARC_UNCACHED_ADDR_SPACE) | ||
87 | return; | ||
88 | |||
89 | vfree((void *)(PAGE_MASK & (unsigned long __force)addr)); | ||
90 | } | ||
91 | EXPORT_SYMBOL(iounmap); | ||
diff --git a/arch/arc/mm/tlb.c b/arch/arc/mm/tlb.c new file mode 100644 index 000000000000..9b9ce23f4ec3 --- /dev/null +++ b/arch/arc/mm/tlb.c | |||
@@ -0,0 +1,645 @@ | |||
1 | /* | ||
2 | * TLB Management (flush/create/diagnostics) for ARC700 | ||
3 | * | ||
4 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | * | ||
10 | * vineetg: Aug 2011 | ||
11 | * -Reintroduce duplicate PD fixup - some customer chips still have the issue | ||
12 | * | ||
13 | * vineetg: May 2011 | ||
14 | * -No need to flush_cache_page( ) for each call to update_mmu_cache() | ||
15 | * some of the LMBench tests improved amazingly | ||
16 | * = page-fault thrice as fast (75 usec to 28 usec) | ||
17 | * = mmap twice as fast (9.6 msec to 4.6 msec), | ||
18 | * = fork (5.3 msec to 3.7 msec) | ||
19 | * | ||
20 | * vineetg: April 2011 : | ||
21 | * -MMU v3: PD{0,1} bits layout changed: They don't overlap anymore, | ||
22 | * helps avoid a shift when preparing PD0 from PTE | ||
23 | * | ||
24 | * vineetg: April 2011 : Preparing for MMU V3 | ||
25 | * -MMU v2/v3 BCRs decoded differently | ||
26 | * -Remove TLB_SIZE hardcoding as it's variable now: 256 or 512 | ||
27 | * -tlb_entry_erase( ) can be void | ||
28 | * -local_flush_tlb_range( ): | ||
29 | * = need not "ceil" @end | ||
30 | * = walks MMU only if range spans < 32 entries, as opposed to 256 | ||
31 | * | ||
32 | * Vineetg: Sept 10th 2008 | ||
33 | * -Changes related to MMU v2 (Rel 4.8) | ||
34 | * | ||
35 | * Vineetg: Aug 29th 2008 | ||
36 | * -In TLB Flush operations (Metal Fix MMU) there is a explict command to | ||
37 | * flush Micro-TLBS. If TLB Index Reg is invalid prior to TLBIVUTLB cmd, | ||
38 | * it fails. Thus need to load it with ANY valid value before invoking | ||
39 | * TLBIVUTLB cmd | ||
40 | * | ||
41 | * Vineetg: Aug 21th 2008: | ||
42 | * -Reduced the duration of IRQ lockouts in TLB Flush routines | ||
43 | * -Multiple copies of TLB erase code seperated into a "single" function | ||
44 | * -In TLB Flush routines, interrupt disabling moved UP to retrieve ASID | ||
45 | * in interrupt-safe region. | ||
46 | * | ||
47 | * Vineetg: April 23rd Bug #93131 | ||
48 | * Problem: tlb_flush_kernel_range() doesnt do anything if the range to | ||
49 | * flush is more than the size of TLB itself. | ||
50 | * | ||
51 | * Rahul Trivedi : Codito Technologies 2004 | ||
52 | */ | ||
53 | |||
54 | #include <linux/module.h> | ||
55 | #include <asm/arcregs.h> | ||
56 | #include <asm/setup.h> | ||
57 | #include <asm/mmu_context.h> | ||
58 | #include <asm/tlb.h> | ||
59 | |||
60 | /* Need for ARC MMU v2 | ||
61 | * | ||
62 | * ARC700 MMU-v1 had a Joint-TLB for Code and Data and is 2 way set-assoc. | ||
63 | * For a memcpy operation with 3 players (src/dst/code) such that all 3 pages | ||
64 | * map into same set, there would be contention for the 2 ways causing severe | ||
65 | * Thrashing. | ||
66 | * | ||
67 | * Although J-TLB is 2 way set assoc, ARC700 caches J-TLB into uTLBS which has | ||
68 | * much higher associativity. u-D-TLB is 8 ways, u-I-TLB is 4 ways. | ||
69 | * Given this, the thrasing problem should never happen because once the 3 | ||
70 | * J-TLB entries are created (even though 3rd will knock out one of the prev | ||
71 | * two), the u-D-TLB and u-I-TLB will have what is required to accomplish memcpy | ||
72 | * | ||
73 | * Yet we still see the Thrashing because a J-TLB Write cause flush of u-TLBs. | ||
74 | * This is a simple design for keeping them in sync. So what do we do? | ||
75 | * The solution which James came up was pretty neat. It utilised the assoc | ||
76 | * of uTLBs by not invalidating always but only when absolutely necessary. | ||
77 | * | ||
78 | * - Existing TLB commands work as before | ||
79 | * - New command (TLBWriteNI) for TLB write without clearing uTLBs | ||
80 | * - New command (TLBIVUTLB) to invalidate uTLBs. | ||
81 | * | ||
82 | * The uTLBs need only be invalidated when pages are being removed from the | ||
83 | * OS page table. If a 'victim' TLB entry is being overwritten in the main TLB | ||
84 | * as a result of a miss, the removed entry is still allowed to exist in the | ||
85 | * uTLBs as it is still valid and present in the OS page table. This allows the | ||
86 | * full associativity of the uTLBs to hide the limited associativity of the main | ||
87 | * TLB. | ||
88 | * | ||
89 | * During a miss handler, the new "TLBWriteNI" command is used to load | ||
90 | * entries without clearing the uTLBs. | ||
91 | * | ||
92 | * When the OS page table is updated, TLB entries that may be associated with a | ||
93 | * removed page are removed (flushed) from the TLB using TLBWrite. In this | ||
94 | * circumstance, the uTLBs must also be cleared. This is done by using the | ||
95 | * existing TLBWrite command. An explicit IVUTLB is also required for those | ||
96 | * corner cases when TLBWrite was not executed at all because the corresp | ||
97 | * J-TLB entry got evicted/replaced. | ||
98 | */ | ||
99 | |||
100 | /* A copy of the ASID from the PID reg is kept in asid_cache */ | ||
101 | int asid_cache = FIRST_ASID; | ||
102 | |||
103 | /* ASID to mm struct mapping. We have one extra entry corresponding to | ||
104 | * NO_ASID to save us a compare when clearing the mm entry for old asid | ||
105 | * see get_new_mmu_context (asm-arc/mmu_context.h) | ||
106 | */ | ||
107 | struct mm_struct *asid_mm_map[NUM_ASID + 1]; | ||
108 | |||
109 | /* | ||
110 | * Utility Routine to erase a J-TLB entry | ||
111 | * The procedure is to look it up in the MMU. If found, ERASE it by | ||
112 | * issuing a TlbWrite CMD with PD0 = PD1 = 0 | ||
113 | */ | ||
114 | |||
115 | static void __tlb_entry_erase(void) | ||
116 | { | ||
117 | write_aux_reg(ARC_REG_TLBPD1, 0); | ||
118 | write_aux_reg(ARC_REG_TLBPD0, 0); | ||
119 | write_aux_reg(ARC_REG_TLBCOMMAND, TLBWrite); | ||
120 | } | ||
121 | |||
122 | static void tlb_entry_erase(unsigned int vaddr_n_asid) | ||
123 | { | ||
124 | unsigned int idx; | ||
125 | |||
126 | /* Locate the TLB entry for this vaddr + ASID */ | ||
127 | write_aux_reg(ARC_REG_TLBPD0, vaddr_n_asid); | ||
128 | write_aux_reg(ARC_REG_TLBCOMMAND, TLBProbe); | ||
129 | idx = read_aux_reg(ARC_REG_TLBINDEX); | ||
130 | |||
131 | /* No error means entry found, zero it out */ | ||
132 | if (likely(!(idx & TLB_LKUP_ERR))) { | ||
133 | __tlb_entry_erase(); | ||
134 | } else { /* Some sort of Error */ | ||
135 | |||
136 | /* Duplicate entry error */ | ||
137 | if (idx & 0x1) { | ||
138 | /* TODO we need to handle this case too */ | ||
139 | pr_emerg("unhandled Duplicate flush for %x\n", | ||
140 | vaddr_n_asid); | ||
141 | } | ||
142 | /* else entry not found so nothing to do */ | ||
143 | } | ||
144 | } | ||
145 | |||
146 | /**************************************************************************** | ||
147 | * ARC700 MMU caches recently used J-TLB entries (RAM) as uTLBs (FLOPs) | ||
148 | * | ||
149 | * New IVUTLB cmd in MMU v2 explictly invalidates the uTLB | ||
150 | * | ||
151 | * utlb_invalidate ( ) | ||
152 | * -For v2 MMU calls Flush uTLB Cmd | ||
153 | * -For v1 MMU does nothing (except for Metal Fix v1 MMU) | ||
154 | * This is because in v1 TLBWrite itself invalidate uTLBs | ||
155 | ***************************************************************************/ | ||
156 | |||
157 | static void utlb_invalidate(void) | ||
158 | { | ||
159 | #if (CONFIG_ARC_MMU_VER >= 2) | ||
160 | |||
161 | #if (CONFIG_ARC_MMU_VER < 3) | ||
162 | /* MMU v2 introduced the uTLB Flush command. | ||
163 | * There was however an obscure hardware bug, where uTLB flush would | ||
164 | * fail when a prior probe for J-TLB (both totally unrelated) would | ||
165 | * return lkup err - because the entry didnt exist in MMU. | ||
166 | * The Workround was to set Index reg with some valid value, prior to | ||
167 | * flush. This was fixed in MMU v3 hence not needed any more | ||
168 | */ | ||
169 | unsigned int idx; | ||
170 | |||
171 | /* make sure INDEX Reg is valid */ | ||
172 | idx = read_aux_reg(ARC_REG_TLBINDEX); | ||
173 | |||
174 | /* If not write some dummy val */ | ||
175 | if (unlikely(idx & TLB_LKUP_ERR)) | ||
176 | write_aux_reg(ARC_REG_TLBINDEX, 0xa); | ||
177 | #endif | ||
178 | |||
179 | write_aux_reg(ARC_REG_TLBCOMMAND, TLBIVUTLB); | ||
180 | #endif | ||
181 | |||
182 | } | ||
183 | |||
184 | /* | ||
185 | * Un-conditionally (without lookup) erase the entire MMU contents | ||
186 | */ | ||
187 | |||
188 | noinline void local_flush_tlb_all(void) | ||
189 | { | ||
190 | unsigned long flags; | ||
191 | unsigned int entry; | ||
192 | struct cpuinfo_arc_mmu *mmu = &cpuinfo_arc700[smp_processor_id()].mmu; | ||
193 | |||
194 | local_irq_save(flags); | ||
195 | |||
196 | /* Load PD0 and PD1 with template for a Blank Entry */ | ||
197 | write_aux_reg(ARC_REG_TLBPD1, 0); | ||
198 | write_aux_reg(ARC_REG_TLBPD0, 0); | ||
199 | |||
200 | for (entry = 0; entry < mmu->num_tlb; entry++) { | ||
201 | /* write this entry to the TLB */ | ||
202 | write_aux_reg(ARC_REG_TLBINDEX, entry); | ||
203 | write_aux_reg(ARC_REG_TLBCOMMAND, TLBWrite); | ||
204 | } | ||
205 | |||
206 | utlb_invalidate(); | ||
207 | |||
208 | local_irq_restore(flags); | ||
209 | } | ||
210 | |||
211 | /* | ||
212 | * Flush the entrie MM for userland. The fastest way is to move to Next ASID | ||
213 | */ | ||
214 | noinline void local_flush_tlb_mm(struct mm_struct *mm) | ||
215 | { | ||
216 | /* | ||
217 | * Small optimisation courtesy IA64 | ||
218 | * flush_mm called during fork,exit,munmap etc, multiple times as well. | ||
219 | * Only for fork( ) do we need to move parent to a new MMU ctxt, | ||
220 | * all other cases are NOPs, hence this check. | ||
221 | */ | ||
222 | if (atomic_read(&mm->mm_users) == 0) | ||
223 | return; | ||
224 | |||
225 | /* | ||
226 | * Workaround for Android weirdism: | ||
227 | * A binder VMA could end up in a task such that vma->mm != tsk->mm | ||
228 | * old code would cause h/w - s/w ASID to get out of sync | ||
229 | */ | ||
230 | if (current->mm != mm) | ||
231 | destroy_context(mm); | ||
232 | else | ||
233 | get_new_mmu_context(mm); | ||
234 | } | ||
235 | |||
236 | /* | ||
237 | * Flush a Range of TLB entries for userland. | ||
238 | * @start is inclusive, while @end is exclusive | ||
239 | * Difference between this and Kernel Range Flush is | ||
240 | * -Here the fastest way (if range is too large) is to move to next ASID | ||
241 | * without doing any explicit Shootdown | ||
242 | * -In case of kernel Flush, entry has to be shot down explictly | ||
243 | */ | ||
244 | void local_flush_tlb_range(struct vm_area_struct *vma, unsigned long start, | ||
245 | unsigned long end) | ||
246 | { | ||
247 | unsigned long flags; | ||
248 | unsigned int asid; | ||
249 | |||
250 | /* If range @start to @end is more than 32 TLB entries deep, | ||
251 | * its better to move to a new ASID rather than searching for | ||
252 | * individual entries and then shooting them down | ||
253 | * | ||
254 | * The calc above is rough, doesn't account for unaligned parts, | ||
255 | * since this is heuristics based anyways | ||
256 | */ | ||
257 | if (unlikely((end - start) >= PAGE_SIZE * 32)) { | ||
258 | local_flush_tlb_mm(vma->vm_mm); | ||
259 | return; | ||
260 | } | ||
261 | |||
262 | /* | ||
263 | * @start moved to page start: this alone suffices for checking | ||
264 | * loop end condition below, w/o need for aligning @end to end | ||
265 | * e.g. 2000 to 4001 will anyhow loop twice | ||
266 | */ | ||
267 | start &= PAGE_MASK; | ||
268 | |||
269 | local_irq_save(flags); | ||
270 | asid = vma->vm_mm->context.asid; | ||
271 | |||
272 | if (asid != NO_ASID) { | ||
273 | while (start < end) { | ||
274 | tlb_entry_erase(start | (asid & 0xff)); | ||
275 | start += PAGE_SIZE; | ||
276 | } | ||
277 | } | ||
278 | |||
279 | utlb_invalidate(); | ||
280 | |||
281 | local_irq_restore(flags); | ||
282 | } | ||
283 | |||
284 | /* Flush the kernel TLB entries - vmalloc/modules (Global from MMU perspective) | ||
285 | * @start, @end interpreted as kvaddr | ||
286 | * Interestingly, shared TLB entries can also be flushed using just | ||
287 | * @start,@end alone (interpreted as user vaddr), although technically SASID | ||
288 | * is also needed. However our smart TLbProbe lookup takes care of that. | ||
289 | */ | ||
290 | void local_flush_tlb_kernel_range(unsigned long start, unsigned long end) | ||
291 | { | ||
292 | unsigned long flags; | ||
293 | |||
294 | /* exactly same as above, except for TLB entry not taking ASID */ | ||
295 | |||
296 | if (unlikely((end - start) >= PAGE_SIZE * 32)) { | ||
297 | local_flush_tlb_all(); | ||
298 | return; | ||
299 | } | ||
300 | |||
301 | start &= PAGE_MASK; | ||
302 | |||
303 | local_irq_save(flags); | ||
304 | while (start < end) { | ||
305 | tlb_entry_erase(start); | ||
306 | start += PAGE_SIZE; | ||
307 | } | ||
308 | |||
309 | utlb_invalidate(); | ||
310 | |||
311 | local_irq_restore(flags); | ||
312 | } | ||
313 | |||
314 | /* | ||
315 | * Delete TLB entry in MMU for a given page (??? address) | ||
316 | * NOTE One TLB entry contains translation for single PAGE | ||
317 | */ | ||
318 | |||
319 | void local_flush_tlb_page(struct vm_area_struct *vma, unsigned long page) | ||
320 | { | ||
321 | unsigned long flags; | ||
322 | |||
323 | /* Note that it is critical that interrupts are DISABLED between | ||
324 | * checking the ASID and using it flush the TLB entry | ||
325 | */ | ||
326 | local_irq_save(flags); | ||
327 | |||
328 | if (vma->vm_mm->context.asid != NO_ASID) { | ||
329 | tlb_entry_erase((page & PAGE_MASK) | | ||
330 | (vma->vm_mm->context.asid & 0xff)); | ||
331 | utlb_invalidate(); | ||
332 | } | ||
333 | |||
334 | local_irq_restore(flags); | ||
335 | } | ||
336 | |||
337 | /* | ||
338 | * Routine to create a TLB entry | ||
339 | */ | ||
340 | void create_tlb(struct vm_area_struct *vma, unsigned long address, pte_t *ptep) | ||
341 | { | ||
342 | unsigned long flags; | ||
343 | unsigned int idx, asid_or_sasid; | ||
344 | unsigned long pd0_flags; | ||
345 | |||
346 | /* | ||
347 | * create_tlb() assumes that current->mm == vma->mm, since | ||
348 | * -it ASID for TLB entry is fetched from MMU ASID reg (valid for curr) | ||
349 | * -completes the lazy write to SASID reg (again valid for curr tsk) | ||
350 | * | ||
351 | * Removing the assumption involves | ||
352 | * -Using vma->mm->context{ASID,SASID}, as opposed to MMU reg. | ||
353 | * -Fix the TLB paranoid debug code to not trigger false negatives. | ||
354 | * -More importantly it makes this handler inconsistent with fast-path | ||
355 | * TLB Refill handler which always deals with "current" | ||
356 | * | ||
357 | * Lets see the use cases when current->mm != vma->mm and we land here | ||
358 | * 1. execve->copy_strings()->__get_user_pages->handle_mm_fault | ||
359 | * Here VM wants to pre-install a TLB entry for user stack while | ||
360 | * current->mm still points to pre-execve mm (hence the condition). | ||
361 | * However the stack vaddr is soon relocated (randomization) and | ||
362 | * move_page_tables() tries to undo that TLB entry. | ||
363 | * Thus not creating TLB entry is not any worse. | ||
364 | * | ||
365 | * 2. ptrace(POKETEXT) causes a CoW - debugger(current) inserting a | ||
366 | * breakpoint in debugged task. Not creating a TLB now is not | ||
367 | * performance critical. | ||
368 | * | ||
369 | * Both the cases above are not good enough for code churn. | ||
370 | */ | ||
371 | if (current->active_mm != vma->vm_mm) | ||
372 | return; | ||
373 | |||
374 | local_irq_save(flags); | ||
375 | |||
376 | tlb_paranoid_check(vma->vm_mm->context.asid, address); | ||
377 | |||
378 | address &= PAGE_MASK; | ||
379 | |||
380 | /* update this PTE credentials */ | ||
381 | pte_val(*ptep) |= (_PAGE_PRESENT | _PAGE_ACCESSED); | ||
382 | |||
383 | /* Create HW TLB entry Flags (in PD0) from PTE Flags */ | ||
384 | #if (CONFIG_ARC_MMU_VER <= 2) | ||
385 | pd0_flags = ((pte_val(*ptep) & PTE_BITS_IN_PD0) >> 1); | ||
386 | #else | ||
387 | pd0_flags = ((pte_val(*ptep) & PTE_BITS_IN_PD0)); | ||
388 | #endif | ||
389 | |||
390 | /* ASID for this task */ | ||
391 | asid_or_sasid = read_aux_reg(ARC_REG_PID) & 0xff; | ||
392 | |||
393 | write_aux_reg(ARC_REG_TLBPD0, address | pd0_flags | asid_or_sasid); | ||
394 | |||
395 | /* Load remaining info in PD1 (Page Frame Addr and Kx/Kw/Kr Flags) */ | ||
396 | write_aux_reg(ARC_REG_TLBPD1, (pte_val(*ptep) & PTE_BITS_IN_PD1)); | ||
397 | |||
398 | /* First verify if entry for this vaddr+ASID already exists */ | ||
399 | write_aux_reg(ARC_REG_TLBCOMMAND, TLBProbe); | ||
400 | idx = read_aux_reg(ARC_REG_TLBINDEX); | ||
401 | |||
402 | /* | ||
403 | * If Not already present get a free slot from MMU. | ||
404 | * Otherwise, Probe would have located the entry and set INDEX Reg | ||
405 | * with existing location. This will cause Write CMD to over-write | ||
406 | * existing entry with new PD0 and PD1 | ||
407 | */ | ||
408 | if (likely(idx & TLB_LKUP_ERR)) | ||
409 | write_aux_reg(ARC_REG_TLBCOMMAND, TLBGetIndex); | ||
410 | |||
411 | /* | ||
412 | * Commit the Entry to MMU | ||
413 | * It doesnt sound safe to use the TLBWriteNI cmd here | ||
414 | * which doesn't flush uTLBs. I'd rather be safe than sorry. | ||
415 | */ | ||
416 | write_aux_reg(ARC_REG_TLBCOMMAND, TLBWrite); | ||
417 | |||
418 | local_irq_restore(flags); | ||
419 | } | ||
420 | |||
421 | /* arch hook called by core VM at the end of handle_mm_fault( ), | ||
422 | * when a new PTE is entered in Page Tables or an existing one | ||
423 | * is modified. We aggresively pre-install a TLB entry | ||
424 | */ | ||
425 | |||
426 | void update_mmu_cache(struct vm_area_struct *vma, unsigned long vaddress, | ||
427 | pte_t *ptep) | ||
428 | { | ||
429 | |||
430 | create_tlb(vma, vaddress, ptep); | ||
431 | } | ||
432 | |||
433 | /* Read the Cache Build Confuration Registers, Decode them and save into | ||
434 | * the cpuinfo structure for later use. | ||
435 | * No Validation is done here, simply read/convert the BCRs | ||
436 | */ | ||
437 | void __init read_decode_mmu_bcr(void) | ||
438 | { | ||
439 | unsigned int tmp; | ||
440 | struct bcr_mmu_1_2 *mmu2; /* encoded MMU2 attr */ | ||
441 | struct bcr_mmu_3 *mmu3; /* encoded MMU3 attr */ | ||
442 | struct cpuinfo_arc_mmu *mmu = &cpuinfo_arc700[smp_processor_id()].mmu; | ||
443 | |||
444 | tmp = read_aux_reg(ARC_REG_MMU_BCR); | ||
445 | mmu->ver = (tmp >> 24); | ||
446 | |||
447 | if (mmu->ver <= 2) { | ||
448 | mmu2 = (struct bcr_mmu_1_2 *)&tmp; | ||
449 | mmu->pg_sz = PAGE_SIZE; | ||
450 | mmu->sets = 1 << mmu2->sets; | ||
451 | mmu->ways = 1 << mmu2->ways; | ||
452 | mmu->u_dtlb = mmu2->u_dtlb; | ||
453 | mmu->u_itlb = mmu2->u_itlb; | ||
454 | } else { | ||
455 | mmu3 = (struct bcr_mmu_3 *)&tmp; | ||
456 | mmu->pg_sz = 512 << mmu3->pg_sz; | ||
457 | mmu->sets = 1 << mmu3->sets; | ||
458 | mmu->ways = 1 << mmu3->ways; | ||
459 | mmu->u_dtlb = mmu3->u_dtlb; | ||
460 | mmu->u_itlb = mmu3->u_itlb; | ||
461 | } | ||
462 | |||
463 | mmu->num_tlb = mmu->sets * mmu->ways; | ||
464 | } | ||
465 | |||
466 | char *arc_mmu_mumbojumbo(int cpu_id, char *buf, int len) | ||
467 | { | ||
468 | int n = 0; | ||
469 | struct cpuinfo_arc_mmu *p_mmu = &cpuinfo_arc700[smp_processor_id()].mmu; | ||
470 | |||
471 | n += scnprintf(buf + n, len - n, "ARC700 MMU [v%x]\t: %dk PAGE, ", | ||
472 | p_mmu->ver, TO_KB(p_mmu->pg_sz)); | ||
473 | |||
474 | n += scnprintf(buf + n, len - n, | ||
475 | "J-TLB %d (%dx%d), uDTLB %d, uITLB %d, %s\n", | ||
476 | p_mmu->num_tlb, p_mmu->sets, p_mmu->ways, | ||
477 | p_mmu->u_dtlb, p_mmu->u_itlb, | ||
478 | __CONFIG_ARC_MMU_SASID_VAL ? "SASID" : ""); | ||
479 | |||
480 | return buf; | ||
481 | } | ||
482 | |||
483 | void __init arc_mmu_init(void) | ||
484 | { | ||
485 | char str[256]; | ||
486 | struct cpuinfo_arc_mmu *mmu = &cpuinfo_arc700[smp_processor_id()].mmu; | ||
487 | |||
488 | printk(arc_mmu_mumbojumbo(0, str, sizeof(str))); | ||
489 | |||
490 | /* For efficiency sake, kernel is compile time built for a MMU ver | ||
491 | * This must match the hardware it is running on. | ||
492 | * Linux built for MMU V2, if run on MMU V1 will break down because V1 | ||
493 | * hardware doesn't understand cmds such as WriteNI, or IVUTLB | ||
494 | * On the other hand, Linux built for V1 if run on MMU V2 will do | ||
495 | * un-needed workarounds to prevent memcpy thrashing. | ||
496 | * Similarly MMU V3 has new features which won't work on older MMU | ||
497 | */ | ||
498 | if (mmu->ver != CONFIG_ARC_MMU_VER) { | ||
499 | panic("MMU ver %d doesn't match kernel built for %d...\n", | ||
500 | mmu->ver, CONFIG_ARC_MMU_VER); | ||
501 | } | ||
502 | |||
503 | if (mmu->pg_sz != PAGE_SIZE) | ||
504 | panic("MMU pg size != PAGE_SIZE (%luk)\n", TO_KB(PAGE_SIZE)); | ||
505 | |||
506 | /* | ||
507 | * ASID mgmt data structures are compile time init | ||
508 | * asid_cache = FIRST_ASID and asid_mm_map[] all zeroes | ||
509 | */ | ||
510 | |||
511 | local_flush_tlb_all(); | ||
512 | |||
513 | /* Enable the MMU */ | ||
514 | write_aux_reg(ARC_REG_PID, MMU_ENABLE); | ||
515 | |||
516 | /* In smp we use this reg for interrupt 1 scratch */ | ||
517 | #ifndef CONFIG_SMP | ||
518 | /* swapper_pg_dir is the pgd for the kernel, used by vmalloc */ | ||
519 | write_aux_reg(ARC_REG_SCRATCH_DATA0, swapper_pg_dir); | ||
520 | #endif | ||
521 | } | ||
522 | |||
523 | /* | ||
524 | * TLB Programmer's Model uses Linear Indexes: 0 to {255, 511} for 128 x {2,4} | ||
525 | * The mapping is Column-first. | ||
526 | * --------------------- ----------- | ||
527 | * |way0|way1|way2|way3| |way0|way1| | ||
528 | * --------------------- ----------- | ||
529 | * [set0] | 0 | 1 | 2 | 3 | | 0 | 1 | | ||
530 | * [set1] | 4 | 5 | 6 | 7 | | 2 | 3 | | ||
531 | * ~ ~ ~ ~ | ||
532 | * [set127] | 508| 509| 510| 511| | 254| 255| | ||
533 | * --------------------- ----------- | ||
534 | * For normal operations we don't(must not) care how above works since | ||
535 | * MMU cmd getIndex(vaddr) abstracts that out. | ||
536 | * However for walking WAYS of a SET, we need to know this | ||
537 | */ | ||
538 | #define SET_WAY_TO_IDX(mmu, set, way) ((set) * mmu->ways + (way)) | ||
539 | |||
540 | /* Handling of Duplicate PD (TLB entry) in MMU. | ||
541 | * -Could be due to buggy customer tapeouts or obscure kernel bugs | ||
542 | * -MMU complaints not at the time of duplicate PD installation, but at the | ||
543 | * time of lookup matching multiple ways. | ||
544 | * -Ideally these should never happen - but if they do - workaround by deleting | ||
545 | * the duplicate one. | ||
546 | * -Knob to be verbose abt it.(TODO: hook them up to debugfs) | ||
547 | */ | ||
548 | volatile int dup_pd_verbose = 1;/* Be slient abt it or complain (default) */ | ||
549 | |||
550 | void do_tlb_overlap_fault(unsigned long cause, unsigned long address, | ||
551 | struct pt_regs *regs) | ||
552 | { | ||
553 | int set, way, n; | ||
554 | unsigned int pd0[4], pd1[4]; /* assume max 4 ways */ | ||
555 | unsigned long flags, is_valid; | ||
556 | struct cpuinfo_arc_mmu *mmu = &cpuinfo_arc700[smp_processor_id()].mmu; | ||
557 | |||
558 | local_irq_save(flags); | ||
559 | |||
560 | /* re-enable the MMU */ | ||
561 | write_aux_reg(ARC_REG_PID, MMU_ENABLE | read_aux_reg(ARC_REG_PID)); | ||
562 | |||
563 | /* loop thru all sets of TLB */ | ||
564 | for (set = 0; set < mmu->sets; set++) { | ||
565 | |||
566 | /* read out all the ways of current set */ | ||
567 | for (way = 0, is_valid = 0; way < mmu->ways; way++) { | ||
568 | write_aux_reg(ARC_REG_TLBINDEX, | ||
569 | SET_WAY_TO_IDX(mmu, set, way)); | ||
570 | write_aux_reg(ARC_REG_TLBCOMMAND, TLBRead); | ||
571 | pd0[way] = read_aux_reg(ARC_REG_TLBPD0); | ||
572 | pd1[way] = read_aux_reg(ARC_REG_TLBPD1); | ||
573 | is_valid |= pd0[way] & _PAGE_PRESENT; | ||
574 | } | ||
575 | |||
576 | /* If all the WAYS in SET are empty, skip to next SET */ | ||
577 | if (!is_valid) | ||
578 | continue; | ||
579 | |||
580 | /* Scan the set for duplicate ways: needs a nested loop */ | ||
581 | for (way = 0; way < mmu->ways; way++) { | ||
582 | if (!pd0[way]) | ||
583 | continue; | ||
584 | |||
585 | for (n = way + 1; n < mmu->ways; n++) { | ||
586 | if ((pd0[way] & PAGE_MASK) == | ||
587 | (pd0[n] & PAGE_MASK)) { | ||
588 | |||
589 | if (dup_pd_verbose) { | ||
590 | pr_info("Duplicate PD's @" | ||
591 | "[%d:%d]/[%d:%d]\n", | ||
592 | set, way, set, n); | ||
593 | pr_info("TLBPD0[%u]: %08x\n", | ||
594 | way, pd0[way]); | ||
595 | } | ||
596 | |||
597 | /* | ||
598 | * clear entry @way and not @n. This is | ||
599 | * critical to our optimised loop | ||
600 | */ | ||
601 | pd0[way] = pd1[way] = 0; | ||
602 | write_aux_reg(ARC_REG_TLBINDEX, | ||
603 | SET_WAY_TO_IDX(mmu, set, way)); | ||
604 | __tlb_entry_erase(); | ||
605 | } | ||
606 | } | ||
607 | } | ||
608 | } | ||
609 | |||
610 | local_irq_restore(flags); | ||
611 | } | ||
612 | |||
613 | /*********************************************************************** | ||
614 | * Diagnostic Routines | ||
615 | * -Called from Low Level TLB Hanlders if things don;t look good | ||
616 | **********************************************************************/ | ||
617 | |||
618 | #ifdef CONFIG_ARC_DBG_TLB_PARANOIA | ||
619 | |||
620 | /* | ||
621 | * Low Level ASM TLB handler calls this if it finds that HW and SW ASIDS | ||
622 | * don't match | ||
623 | */ | ||
624 | void print_asid_mismatch(int is_fast_path) | ||
625 | { | ||
626 | int pid_sw, pid_hw; | ||
627 | pid_sw = current->active_mm->context.asid; | ||
628 | pid_hw = read_aux_reg(ARC_REG_PID) & 0xff; | ||
629 | |||
630 | pr_emerg("ASID Mismatch in %s Path Handler: sw-pid=0x%x hw-pid=0x%x\n", | ||
631 | is_fast_path ? "Fast" : "Slow", pid_sw, pid_hw); | ||
632 | |||
633 | __asm__ __volatile__("flag 1"); | ||
634 | } | ||
635 | |||
636 | void tlb_paranoid_check(unsigned int pid_sw, unsigned long addr) | ||
637 | { | ||
638 | unsigned int pid_hw; | ||
639 | |||
640 | pid_hw = read_aux_reg(ARC_REG_PID) & 0xff; | ||
641 | |||
642 | if (addr < 0x70000000 && ((pid_hw != pid_sw) || (pid_sw == NO_ASID))) | ||
643 | print_asid_mismatch(0); | ||
644 | } | ||
645 | #endif | ||
diff --git a/arch/arc/mm/tlbex.S b/arch/arc/mm/tlbex.S new file mode 100644 index 000000000000..9df765dc7c3a --- /dev/null +++ b/arch/arc/mm/tlbex.S | |||
@@ -0,0 +1,408 @@ | |||
1 | /* | ||
2 | * TLB Exception Handling for ARC | ||
3 | * | ||
4 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | * | ||
10 | * Vineetg: April 2011 : | ||
11 | * -MMU v1: moved out legacy code into a seperate file | ||
12 | * -MMU v3: PD{0,1} bits layout changed: They don't overlap anymore, | ||
13 | * helps avoid a shift when preparing PD0 from PTE | ||
14 | * | ||
15 | * Vineetg: July 2009 | ||
16 | * -For MMU V2, we need not do heuristics at the time of commiting a D-TLB | ||
17 | * entry, so that it doesn't knock out it's I-TLB entry | ||
18 | * -Some more fine tuning: | ||
19 | * bmsk instead of add, asl.cc instead of branch, delay slot utilise etc | ||
20 | * | ||
21 | * Vineetg: July 2009 | ||
22 | * -Practically rewrote the I/D TLB Miss handlers | ||
23 | * Now 40 and 135 instructions a peice as compared to 131 and 449 resp. | ||
24 | * Hence Leaner by 1.5 K | ||
25 | * Used Conditional arithmetic to replace excessive branching | ||
26 | * Also used short instructions wherever possible | ||
27 | * | ||
28 | * Vineetg: Aug 13th 2008 | ||
29 | * -Passing ECR (Exception Cause REG) to do_page_fault( ) for printing | ||
30 | * more information in case of a Fatality | ||
31 | * | ||
32 | * Vineetg: March 25th Bug #92690 | ||
33 | * -Added Debug Code to check if sw-ASID == hw-ASID | ||
34 | |||
35 | * Rahul Trivedi, Amit Bhor: Codito Technologies 2004 | ||
36 | */ | ||
37 | |||
38 | .cpu A7 | ||
39 | |||
40 | #include <linux/linkage.h> | ||
41 | #include <asm/entry.h> | ||
42 | #include <asm/tlb.h> | ||
43 | #include <asm/pgtable.h> | ||
44 | #include <asm/arcregs.h> | ||
45 | #include <asm/cache.h> | ||
46 | #include <asm/processor.h> | ||
47 | #if (CONFIG_ARC_MMU_VER == 1) | ||
48 | #include <asm/tlb-mmu1.h> | ||
49 | #endif | ||
50 | |||
51 | ;-------------------------------------------------------------------------- | ||
52 | ; scratch memory to save the registers (r0-r3) used to code TLB refill Handler | ||
53 | ; For details refer to comments before TLBMISS_FREEUP_REGS below | ||
54 | ;-------------------------------------------------------------------------- | ||
55 | |||
56 | ARCFP_DATA ex_saved_reg1 | ||
57 | .align 1 << L1_CACHE_SHIFT ; IMP: Must be Cache Line aligned | ||
58 | .type ex_saved_reg1, @object | ||
59 | #ifdef CONFIG_SMP | ||
60 | .size ex_saved_reg1, (CONFIG_NR_CPUS << L1_CACHE_SHIFT) | ||
61 | ex_saved_reg1: | ||
62 | .zero (CONFIG_NR_CPUS << L1_CACHE_SHIFT) | ||
63 | #else | ||
64 | .size ex_saved_reg1, 16 | ||
65 | ex_saved_reg1: | ||
66 | .zero 16 | ||
67 | #endif | ||
68 | |||
69 | ;============================================================================ | ||
70 | ; Troubleshooting Stuff | ||
71 | ;============================================================================ | ||
72 | |||
73 | ; Linux keeps ASID (Address Space ID) in task->active_mm->context.asid | ||
74 | ; When Creating TLB Entries, instead of doing 3 dependent loads from memory, | ||
75 | ; we use the MMU PID Reg to get current ASID. | ||
76 | ; In bizzare scenrios SW and HW ASID can get out-of-sync which is trouble. | ||
77 | ; So we try to detect this in TLB Mis shandler | ||
78 | |||
79 | |||
80 | .macro DBG_ASID_MISMATCH | ||
81 | |||
82 | #ifdef CONFIG_ARC_DBG_TLB_PARANOIA | ||
83 | |||
84 | ; make sure h/w ASID is same as s/w ASID | ||
85 | |||
86 | GET_CURR_TASK_ON_CPU r3 | ||
87 | ld r0, [r3, TASK_ACT_MM] | ||
88 | ld r0, [r0, MM_CTXT+MM_CTXT_ASID] | ||
89 | |||
90 | lr r1, [ARC_REG_PID] | ||
91 | and r1, r1, 0xFF | ||
92 | breq r1, r0, 5f | ||
93 | |||
94 | ; Error if H/w and S/w ASID don't match, but NOT if in kernel mode | ||
95 | lr r0, [erstatus] | ||
96 | bbit0 r0, STATUS_U_BIT, 5f | ||
97 | |||
98 | ; We sure are in troubled waters, Flag the error, but to do so | ||
99 | ; need to switch to kernel mode stack to call error routine | ||
100 | GET_TSK_STACK_BASE r3, sp | ||
101 | |||
102 | ; Call printk to shoutout aloud | ||
103 | mov r0, 1 | ||
104 | j print_asid_mismatch | ||
105 | |||
106 | 5: ; ASIDs match so proceed normally | ||
107 | nop | ||
108 | |||
109 | #endif | ||
110 | |||
111 | .endm | ||
112 | |||
113 | ;============================================================================ | ||
114 | ;TLB Miss handling Code | ||
115 | ;============================================================================ | ||
116 | |||
117 | ;----------------------------------------------------------------------------- | ||
118 | ; This macro does the page-table lookup for the faulting address. | ||
119 | ; OUT: r0 = PTE faulted on, r1 = ptr to PTE, r2 = Faulting V-address | ||
120 | .macro LOAD_FAULT_PTE | ||
121 | |||
122 | lr r2, [efa] | ||
123 | |||
124 | #ifndef CONFIG_SMP | ||
125 | lr r1, [ARC_REG_SCRATCH_DATA0] ; current pgd | ||
126 | #else | ||
127 | GET_CURR_TASK_ON_CPU r1 | ||
128 | ld r1, [r1, TASK_ACT_MM] | ||
129 | ld r1, [r1, MM_PGD] | ||
130 | #endif | ||
131 | |||
132 | lsr r0, r2, PGDIR_SHIFT ; Bits for indexing into PGD | ||
133 | ld.as r1, [r1, r0] ; PGD entry corresp to faulting addr | ||
134 | and.f r1, r1, PAGE_MASK ; Ignoring protection and other flags | ||
135 | ; contains Ptr to Page Table | ||
136 | bz.d do_slow_path_pf ; if no Page Table, do page fault | ||
137 | |||
138 | ; Get the PTE entry: The idea is | ||
139 | ; (1) x = addr >> PAGE_SHIFT -> masks page-off bits from @fault-addr | ||
140 | ; (2) y = x & (PTRS_PER_PTE - 1) -> to get index | ||
141 | ; (3) z = pgtbl[y] | ||
142 | ; To avoid the multiply by in end, we do the -2, <<2 below | ||
143 | |||
144 | lsr r0, r2, (PAGE_SHIFT - 2) | ||
145 | and r0, r0, ( (PTRS_PER_PTE - 1) << 2) | ||
146 | ld.aw r0, [r1, r0] ; get PTE and PTE ptr for fault addr | ||
147 | #ifdef CONFIG_ARC_DBG_TLB_MISS_COUNT | ||
148 | and.f 0, r0, _PAGE_PRESENT | ||
149 | bz 1f | ||
150 | ld r2, [num_pte_not_present] | ||
151 | add r2, r2, 1 | ||
152 | st r2, [num_pte_not_present] | ||
153 | 1: | ||
154 | #endif | ||
155 | |||
156 | .endm | ||
157 | |||
158 | ;----------------------------------------------------------------- | ||
159 | ; Convert Linux PTE entry into TLB entry | ||
160 | ; A one-word PTE entry is programmed as two-word TLB Entry [PD0:PD1] in mmu | ||
161 | ; IN: r0 = PTE, r1 = ptr to PTE | ||
162 | |||
163 | .macro CONV_PTE_TO_TLB | ||
164 | and r3, r0, PTE_BITS_IN_PD1 ; Extract permission flags+PFN from PTE | ||
165 | sr r3, [ARC_REG_TLBPD1] ; these go in PD1 | ||
166 | |||
167 | and r2, r0, PTE_BITS_IN_PD0 ; Extract other PTE flags: (V)alid, (G)lb | ||
168 | #if (CONFIG_ARC_MMU_VER <= 2) /* Neednot be done with v3 onwards */ | ||
169 | lsr r2, r2 ; shift PTE flags to match layout in PD0 | ||
170 | #endif | ||
171 | |||
172 | lr r3,[ARC_REG_TLBPD0] ; MMU prepares PD0 with vaddr and asid | ||
173 | |||
174 | or r3, r3, r2 ; S | vaddr | {sasid|asid} | ||
175 | sr r3,[ARC_REG_TLBPD0] ; rewrite PD0 | ||
176 | .endm | ||
177 | |||
178 | ;----------------------------------------------------------------- | ||
179 | ; Commit the TLB entry into MMU | ||
180 | |||
181 | .macro COMMIT_ENTRY_TO_MMU | ||
182 | |||
183 | /* Get free TLB slot: Set = computed from vaddr, way = random */ | ||
184 | sr TLBGetIndex, [ARC_REG_TLBCOMMAND] | ||
185 | |||
186 | /* Commit the Write */ | ||
187 | #if (CONFIG_ARC_MMU_VER >= 2) /* introduced in v2 */ | ||
188 | sr TLBWriteNI, [ARC_REG_TLBCOMMAND] | ||
189 | #else | ||
190 | sr TLBWrite, [ARC_REG_TLBCOMMAND] | ||
191 | #endif | ||
192 | .endm | ||
193 | |||
194 | ;----------------------------------------------------------------- | ||
195 | ; ARC700 Exception Handling doesn't auto-switch stack and it only provides | ||
196 | ; ONE scratch AUX reg "ARC_REG_SCRATCH_DATA0" | ||
197 | ; | ||
198 | ; For Non-SMP, the scratch AUX reg is repurposed to cache task PGD, so a | ||
199 | ; "global" is used to free-up FIRST core reg to be able to code the rest of | ||
200 | ; exception prologue (IRQ auto-disabled on Exceptions, so it's IRQ-safe). | ||
201 | ; Since the Fast Path TLB Miss handler is coded with 4 regs, the remaining 3 | ||
202 | ; need to be saved as well by extending the "global" to be 4 words. Hence | ||
203 | ; ".size ex_saved_reg1, 16" | ||
204 | ; [All of this dance is to avoid stack switching for each TLB Miss, since we | ||
205 | ; only need to save only a handful of regs, as opposed to complete reg file] | ||
206 | ; | ||
207 | ; For ARC700 SMP, the "global" obviously can't be used for free up the FIRST | ||
208 | ; core reg as it will not be SMP safe. | ||
209 | ; Thus scratch AUX reg is used (and no longer used to cache task PGD). | ||
210 | ; To save the rest of 3 regs - per cpu, the global is made "per-cpu". | ||
211 | ; Epilogue thus has to locate the "per-cpu" storage for regs. | ||
212 | ; To avoid cache line bouncing the per-cpu global is aligned/sized per | ||
213 | ; L1_CACHE_SHIFT, despite fundamentally needing to be 12 bytes only. Hence | ||
214 | ; ".size ex_saved_reg1, (CONFIG_NR_CPUS << L1_CACHE_SHIFT)" | ||
215 | |||
216 | ; As simple as that.... | ||
217 | |||
218 | .macro TLBMISS_FREEUP_REGS | ||
219 | #ifdef CONFIG_SMP | ||
220 | sr r0, [ARC_REG_SCRATCH_DATA0] ; freeup r0 to code with | ||
221 | GET_CPU_ID r0 ; get to per cpu scratch mem, | ||
222 | lsl r0, r0, L1_CACHE_SHIFT ; cache line wide per cpu | ||
223 | add r0, @ex_saved_reg1, r0 | ||
224 | #else | ||
225 | st r0, [@ex_saved_reg1] | ||
226 | mov_s r0, @ex_saved_reg1 | ||
227 | #endif | ||
228 | st_s r1, [r0, 4] | ||
229 | st_s r2, [r0, 8] | ||
230 | st_s r3, [r0, 12] | ||
231 | |||
232 | ; VERIFY if the ASID in MMU-PID Reg is same as | ||
233 | ; one in Linux data structures | ||
234 | |||
235 | DBG_ASID_MISMATCH | ||
236 | .endm | ||
237 | |||
238 | ;----------------------------------------------------------------- | ||
239 | .macro TLBMISS_RESTORE_REGS | ||
240 | #ifdef CONFIG_SMP | ||
241 | GET_CPU_ID r0 ; get to per cpu scratch mem | ||
242 | lsl r0, r0, L1_CACHE_SHIFT ; each is cache line wide | ||
243 | add r0, @ex_saved_reg1, r0 | ||
244 | ld_s r3, [r0,12] | ||
245 | ld_s r2, [r0, 8] | ||
246 | ld_s r1, [r0, 4] | ||
247 | lr r0, [ARC_REG_SCRATCH_DATA0] | ||
248 | #else | ||
249 | mov_s r0, @ex_saved_reg1 | ||
250 | ld_s r3, [r0,12] | ||
251 | ld_s r2, [r0, 8] | ||
252 | ld_s r1, [r0, 4] | ||
253 | ld_s r0, [r0] | ||
254 | #endif | ||
255 | .endm | ||
256 | |||
257 | ARCFP_CODE ;Fast Path Code, candidate for ICCM | ||
258 | |||
259 | ;----------------------------------------------------------------------------- | ||
260 | ; I-TLB Miss Exception Handler | ||
261 | ;----------------------------------------------------------------------------- | ||
262 | |||
263 | ARC_ENTRY EV_TLBMissI | ||
264 | |||
265 | TLBMISS_FREEUP_REGS | ||
266 | |||
267 | #ifdef CONFIG_ARC_DBG_TLB_MISS_COUNT | ||
268 | ld r0, [@numitlb] | ||
269 | add r0, r0, 1 | ||
270 | st r0, [@numitlb] | ||
271 | #endif | ||
272 | |||
273 | ;---------------------------------------------------------------- | ||
274 | ; Get the PTE corresponding to V-addr accessed | ||
275 | LOAD_FAULT_PTE | ||
276 | |||
277 | ;---------------------------------------------------------------- | ||
278 | ; VERIFY_PTE: Check if PTE permissions approp for executing code | ||
279 | cmp_s r2, VMALLOC_START | ||
280 | mov.lo r2, (_PAGE_PRESENT | _PAGE_READ | _PAGE_EXECUTE) | ||
281 | mov.hs r2, (_PAGE_PRESENT | _PAGE_K_READ | _PAGE_K_EXECUTE) | ||
282 | |||
283 | and r3, r0, r2 ; Mask out NON Flag bits from PTE | ||
284 | xor.f r3, r3, r2 ; check ( ( pte & flags_test ) == flags_test ) | ||
285 | bnz do_slow_path_pf | ||
286 | |||
287 | ; Let Linux VM know that the page was accessed | ||
288 | or r0, r0, (_PAGE_PRESENT | _PAGE_ACCESSED) ; set Accessed Bit | ||
289 | st_s r0, [r1] ; Write back PTE | ||
290 | |||
291 | CONV_PTE_TO_TLB | ||
292 | COMMIT_ENTRY_TO_MMU | ||
293 | TLBMISS_RESTORE_REGS | ||
294 | rtie | ||
295 | |||
296 | ARC_EXIT EV_TLBMissI | ||
297 | |||
298 | ;----------------------------------------------------------------------------- | ||
299 | ; D-TLB Miss Exception Handler | ||
300 | ;----------------------------------------------------------------------------- | ||
301 | |||
302 | ARC_ENTRY EV_TLBMissD | ||
303 | |||
304 | TLBMISS_FREEUP_REGS | ||
305 | |||
306 | #ifdef CONFIG_ARC_DBG_TLB_MISS_COUNT | ||
307 | ld r0, [@numdtlb] | ||
308 | add r0, r0, 1 | ||
309 | st r0, [@numdtlb] | ||
310 | #endif | ||
311 | |||
312 | ;---------------------------------------------------------------- | ||
313 | ; Get the PTE corresponding to V-addr accessed | ||
314 | ; If PTE exists, it will setup, r0 = PTE, r1 = Ptr to PTE | ||
315 | LOAD_FAULT_PTE | ||
316 | |||
317 | ;---------------------------------------------------------------- | ||
318 | ; VERIFY_PTE: Chk if PTE permissions approp for data access (R/W/R+W) | ||
319 | |||
320 | mov_s r2, 0 | ||
321 | lr r3, [ecr] | ||
322 | btst_s r3, ECR_C_BIT_DTLB_LD_MISS ; Read Access | ||
323 | or.nz r2, r2, _PAGE_READ ; chk for Read flag in PTE | ||
324 | btst_s r3, ECR_C_BIT_DTLB_ST_MISS ; Write Access | ||
325 | or.nz r2, r2, _PAGE_WRITE ; chk for Write flag in PTE | ||
326 | ; Above laddering takes care of XCHG access | ||
327 | ; which is both Read and Write | ||
328 | |||
329 | ; If kernel mode access, ; make _PAGE_xx flags as _PAGE_K_xx | ||
330 | ; For copy_(to|from)_user, despite exception taken in kernel mode, | ||
331 | ; this code is not hit, because EFA would still be the user mode | ||
332 | ; address (EFA < 0x6000_0000). | ||
333 | ; This code is for legit kernel mode faults, vmalloc specifically | ||
334 | ; (EFA: 0x7000_0000 to 0x7FFF_FFFF) | ||
335 | |||
336 | lr r3, [efa] | ||
337 | cmp r3, VMALLOC_START - 1 ; If kernel mode access | ||
338 | asl.hi r2, r2, 3 ; make _PAGE_xx flags as _PAGE_K_xx | ||
339 | or r2, r2, _PAGE_PRESENT ; Common flag for K/U mode | ||
340 | |||
341 | ; By now, r2 setup with all the Flags we need to check in PTE | ||
342 | and r3, r0, r2 ; Mask out NON Flag bits from PTE | ||
343 | brne.d r3, r2, do_slow_path_pf ; is ((pte & flags_test) == flags_test) | ||
344 | |||
345 | ;---------------------------------------------------------------- | ||
346 | ; UPDATE_PTE: Let Linux VM know that page was accessed/dirty | ||
347 | lr r3, [ecr] | ||
348 | or r0, r0, (_PAGE_PRESENT | _PAGE_ACCESSED) ; Accessed bit always | ||
349 | btst_s r3, ECR_C_BIT_DTLB_ST_MISS ; See if it was a Write Access ? | ||
350 | or.nz r0, r0, _PAGE_MODIFIED ; if Write, set Dirty bit as well | ||
351 | st_s r0, [r1] ; Write back PTE | ||
352 | |||
353 | CONV_PTE_TO_TLB | ||
354 | |||
355 | #if (CONFIG_ARC_MMU_VER == 1) | ||
356 | ; MMU with 2 way set assoc J-TLB, needs some help in pathetic case of | ||
357 | ; memcpy where 3 parties contend for 2 ways, ensuing a livelock. | ||
358 | ; But only for old MMU or one with Metal Fix | ||
359 | TLB_WRITE_HEURISTICS | ||
360 | #endif | ||
361 | |||
362 | COMMIT_ENTRY_TO_MMU | ||
363 | TLBMISS_RESTORE_REGS | ||
364 | rtie | ||
365 | |||
366 | ;-------- Common routine to call Linux Page Fault Handler ----------- | ||
367 | do_slow_path_pf: | ||
368 | |||
369 | ; Restore the 4-scratch regs saved by fast path miss handler | ||
370 | TLBMISS_RESTORE_REGS | ||
371 | |||
372 | ; Slow path TLB Miss handled as a regular ARC Exception | ||
373 | ; (stack switching / save the complete reg-file). | ||
374 | ; That requires freeing up r9 | ||
375 | EXCPN_PROLOG_FREEUP_REG r9 | ||
376 | |||
377 | lr r9, [erstatus] | ||
378 | |||
379 | SWITCH_TO_KERNEL_STK | ||
380 | SAVE_ALL_SYS | ||
381 | |||
382 | ; ------- setup args for Linux Page fault Hanlder --------- | ||
383 | mov_s r0, sp | ||
384 | lr r2, [efa] | ||
385 | lr r3, [ecr] | ||
386 | |||
387 | ; Both st and ex imply WRITE access of some sort, hence do_page_fault( ) | ||
388 | ; invoked with write=1 for DTLB-st/ex Miss and write=0 for ITLB miss or | ||
389 | ; DTLB-ld Miss | ||
390 | ; DTLB Miss Cause code is ld = 0x01 , st = 0x02, ex = 0x03 | ||
391 | ; Following code uses that fact that st/ex have one bit in common | ||
392 | |||
393 | btst_s r3, ECR_C_BIT_DTLB_ST_MISS | ||
394 | mov.z r1, 0 | ||
395 | mov.nz r1, 1 | ||
396 | |||
397 | ; We don't want exceptions to be disabled while the fault is handled. | ||
398 | ; Now that we have saved the context we return from exception hence | ||
399 | ; exceptions get re-enable | ||
400 | |||
401 | FAKE_RET_FROM_EXCPN r9 | ||
402 | |||
403 | bl do_page_fault | ||
404 | b ret_from_exception | ||
405 | |||
406 | ARC_EXIT EV_TLBMissD | ||
407 | |||
408 | ARC_ENTRY EV_TLBMissB ; Bogus entry to measure sz of DTLBMiss hdlr | ||
diff --git a/arch/arc/oprofile/Makefile b/arch/arc/oprofile/Makefile new file mode 100644 index 000000000000..ce417a6e70b8 --- /dev/null +++ b/arch/arc/oprofile/Makefile | |||
@@ -0,0 +1,9 @@ | |||
1 | obj-$(CONFIG_OPROFILE) += oprofile.o | ||
2 | |||
3 | DRIVER_OBJS = $(addprefix ../../../drivers/oprofile/, \ | ||
4 | oprof.o cpu_buffer.o buffer_sync.o \ | ||
5 | event_buffer.o oprofile_files.o \ | ||
6 | oprofilefs.o oprofile_stats.o \ | ||
7 | timer_int.o ) | ||
8 | |||
9 | oprofile-y := $(DRIVER_OBJS) common.o | ||
diff --git a/arch/arc/oprofile/common.c b/arch/arc/oprofile/common.c new file mode 100644 index 000000000000..c80fcad4a5a7 --- /dev/null +++ b/arch/arc/oprofile/common.c | |||
@@ -0,0 +1,26 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Based on orig code from @author John Levon <levon@movementarian.org> | ||
9 | */ | ||
10 | |||
11 | #include <linux/oprofile.h> | ||
12 | #include <linux/perf_event.h> | ||
13 | |||
14 | int __init oprofile_arch_init(struct oprofile_operations *ops) | ||
15 | { | ||
16 | /* | ||
17 | * A failure here, forces oprofile core to switch to Timer based PC | ||
18 | * sampling, which will happen if say perf is not enabled/available | ||
19 | */ | ||
20 | return oprofile_perf_init(ops); | ||
21 | } | ||
22 | |||
23 | void oprofile_arch_exit(void) | ||
24 | { | ||
25 | oprofile_perf_exit(); | ||
26 | } | ||
diff --git a/arch/arc/plat-arcfpga/Kconfig b/arch/arc/plat-arcfpga/Kconfig new file mode 100644 index 000000000000..b41e786cdbc0 --- /dev/null +++ b/arch/arc/plat-arcfpga/Kconfig | |||
@@ -0,0 +1,84 @@ | |||
1 | # | ||
2 | # Copyright (C) 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | # | ||
4 | # This program is free software; you can redistribute it and/or modify | ||
5 | # it under the terms of the GNU General Public License version 2 as | ||
6 | # published by the Free Software Foundation. | ||
7 | # | ||
8 | |||
9 | menuconfig ARC_PLAT_FPGA_LEGACY | ||
10 | bool "\"Legacy\" ARC FPGA dev Boards" | ||
11 | select ISS_SMP_EXTN if SMP | ||
12 | help | ||
13 | Support for ARC development boards, provided by Synopsys. | ||
14 | These are based on FPGA or ISS. e.g. | ||
15 | - ARCAngel4 | ||
16 | - ML509 | ||
17 | - MetaWare ISS | ||
18 | |||
19 | if ARC_PLAT_FPGA_LEGACY | ||
20 | |||
21 | config ARC_BOARD_ANGEL4 | ||
22 | bool "ARC Angel4" | ||
23 | default y | ||
24 | help | ||
25 | ARC Angel4 FPGA Ref Platform (Xilinx Virtex Based) | ||
26 | |||
27 | config ARC_BOARD_ML509 | ||
28 | bool "ML509" | ||
29 | help | ||
30 | ARC ML509 FPGA Ref Platform (Xilinx Virtex-5 Based) | ||
31 | |||
32 | config ISS_SMP_EXTN | ||
33 | bool "ARC SMP Extensions (ISS Models only)" | ||
34 | default n | ||
35 | depends on SMP | ||
36 | select ARC_HAS_COH_RTSC | ||
37 | help | ||
38 | SMP Extensions to ARC700, in a "simulation only" Model, supported in | ||
39 | ARC ISS (Instruction Set Simulator). | ||
40 | The SMP extensions include: | ||
41 | -IDU (Interrupt Distribution Unit) | ||
42 | -XTL (To enable CPU start/stop/set-PC for another CPU) | ||
43 | It doesn't provide coherent Caches and/or Atomic Ops (LLOCK/SCOND) | ||
44 | |||
45 | config ARC_SERIAL_BAUD | ||
46 | int "UART Baud rate" | ||
47 | default "115200" | ||
48 | depends on SERIAL_ARC || SERIAL_ARC_CONSOLE | ||
49 | help | ||
50 | Baud rate for the ARC UART | ||
51 | |||
52 | menuconfig ARC_HAS_BVCI_LAT_UNIT | ||
53 | bool "BVCI Bus Latency Unit" | ||
54 | depends on ARC_BOARD_ML509 || ARC_BOARD_ANGEL4 | ||
55 | help | ||
56 | IP to add artifical latency to BVCI Bus Based FPGA builds. | ||
57 | The default latency (even worst case) for FPGA is non-realistic | ||
58 | (~10 SDRAM, ~5 SSRAM). | ||
59 | |||
60 | config BVCI_LAT_UNITS | ||
61 | hex "Latency Unit(s) Bitmap" | ||
62 | default "0x0" | ||
63 | depends on ARC_HAS_BVCI_LAT_UNIT | ||
64 | help | ||
65 | There are multiple Latency Units corresponding to the many | ||
66 | interfaces of the system bus arbiter (both CPU side as well as | ||
67 | the peripheral side). | ||
68 | To add latency to ALL memory transaction, choose Unit 0, otherwise | ||
69 | for finer grainer - interface wise latency, specify a bitmap (1 bit | ||
70 | per unit) of all units. e.g. 1,2,12 will be 0x1003 | ||
71 | |||
72 | Unit 0 - System Arb and Mem Controller | ||
73 | Unit 1 - I$ and System Bus | ||
74 | Unit 2 - D$ and System Bus | ||
75 | .. | ||
76 | Unit 12 - IDE Disk controller and System Bus | ||
77 | |||
78 | config BVCI_LAT_CYCLES | ||
79 | int "Latency Value in cycles" | ||
80 | range 0 63 | ||
81 | default "30" | ||
82 | depends on ARC_HAS_BVCI_LAT_UNIT | ||
83 | |||
84 | endif | ||
diff --git a/arch/arc/plat-arcfpga/Makefile b/arch/arc/plat-arcfpga/Makefile new file mode 100644 index 000000000000..a44e22ebc1b7 --- /dev/null +++ b/arch/arc/plat-arcfpga/Makefile | |||
@@ -0,0 +1,12 @@ | |||
1 | # | ||
2 | # Copyright (C) 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | # | ||
4 | # This program is free software; you can redistribute it and/or modify | ||
5 | # it under the terms of the GNU General Public License version 2 as | ||
6 | # published by the Free Software Foundation. | ||
7 | # | ||
8 | |||
9 | KBUILD_CFLAGS += -Iarch/arc/plat-arcfpga/include | ||
10 | |||
11 | obj-y := platform.o irq.o | ||
12 | obj-$(CONFIG_SMP) += smp.o | ||
diff --git a/arch/arc/plat-arcfpga/include/plat/irq.h b/arch/arc/plat-arcfpga/include/plat/irq.h new file mode 100644 index 000000000000..41e335670f60 --- /dev/null +++ b/arch/arc/plat-arcfpga/include/plat/irq.h | |||
@@ -0,0 +1,31 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * vineetg: Feb 2009 | ||
9 | * -For AA4 board, IRQ assignments to peripherals | ||
10 | */ | ||
11 | |||
12 | #ifndef __PLAT_IRQ_H | ||
13 | #define __PLAT_IRQ_H | ||
14 | |||
15 | #define UART0_IRQ 5 | ||
16 | #define UART1_IRQ 10 | ||
17 | #define UART2_IRQ 11 | ||
18 | |||
19 | #define VMAC_IRQ 6 | ||
20 | |||
21 | #define IDE_IRQ 13 | ||
22 | #define PCI_IRQ 14 | ||
23 | #define PS2_IRQ 15 | ||
24 | |||
25 | #ifdef CONFIG_SMP | ||
26 | #define IDU_INTERRUPT_0 16 | ||
27 | #endif | ||
28 | |||
29 | extern void __init plat_fpga_init_IRQ(void); | ||
30 | |||
31 | #endif | ||
diff --git a/arch/arc/plat-arcfpga/include/plat/memmap.h b/arch/arc/plat-arcfpga/include/plat/memmap.h new file mode 100644 index 000000000000..1663f3388085 --- /dev/null +++ b/arch/arc/plat-arcfpga/include/plat/memmap.h | |||
@@ -0,0 +1,31 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * vineetg: Feb 2009 | ||
9 | * -For AA4 board, System Memory Map for Peripherals etc | ||
10 | */ | ||
11 | |||
12 | #ifndef __PLAT_MEMMAP_H | ||
13 | #define __PLAT_MEMMAP_H | ||
14 | |||
15 | #define UART0_BASE 0xC0FC1000 | ||
16 | #define UART1_BASE 0xC0FC1100 | ||
17 | |||
18 | #define VMAC_REG_BASEADDR 0xC0FC2000 | ||
19 | |||
20 | #define IDE_CONTROLLER_BASE 0xC0FC9000 | ||
21 | |||
22 | #define AHB_PCI_HOST_BRG_BASE 0xC0FD0000 | ||
23 | |||
24 | #define PGU_BASEADDR 0xC0FC8000 | ||
25 | #define VLCK_ADDR 0xC0FCF028 | ||
26 | |||
27 | #define BVCI_LAT_UNIT_BASE 0xC0FED000 | ||
28 | |||
29 | #define PS2_BASE_ADDR 0xC0FCC000 | ||
30 | |||
31 | #endif | ||
diff --git a/arch/arc/plat-arcfpga/include/plat/smp.h b/arch/arc/plat-arcfpga/include/plat/smp.h new file mode 100644 index 000000000000..c09eb4cfc77c --- /dev/null +++ b/arch/arc/plat-arcfpga/include/plat/smp.h | |||
@@ -0,0 +1,118 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * Rajeshwar Ranga: Interrupt Distribution Unit API's | ||
9 | */ | ||
10 | |||
11 | #ifndef __PLAT_ARCFPGA_SMP_H | ||
12 | #define __PLAT_ARCFPGA_SMP_H | ||
13 | |||
14 | #ifdef CONFIG_SMP | ||
15 | |||
16 | #include <linux/types.h> | ||
17 | #include <asm/arcregs.h> | ||
18 | |||
19 | #define ARC_AUX_IDU_REG_CMD 0x2000 | ||
20 | #define ARC_AUX_IDU_REG_PARAM 0x2001 | ||
21 | |||
22 | #define ARC_AUX_XTL_REG_CMD 0x2002 | ||
23 | #define ARC_AUX_XTL_REG_PARAM 0x2003 | ||
24 | |||
25 | #define ARC_REG_MP_BCR 0x2021 | ||
26 | |||
27 | #define ARC_XTL_CMD_WRITE_PC 0x04 | ||
28 | #define ARC_XTL_CMD_CLEAR_HALT 0x02 | ||
29 | |||
30 | /* | ||
31 | * Build Configuration Register which identifies the sub-components | ||
32 | */ | ||
33 | struct bcr_mp { | ||
34 | #ifdef CONFIG_CPU_BIG_ENDIAN | ||
35 | unsigned int mp_arch:16, pad:5, sdu:1, idu:1, scu:1, ver:8; | ||
36 | #else | ||
37 | unsigned int ver:8, scu:1, idu:1, sdu:1, pad:5, mp_arch:16; | ||
38 | #endif | ||
39 | }; | ||
40 | |||
41 | /* IDU supports 256 common interrupts */ | ||
42 | #define NR_IDU_IRQS 256 | ||
43 | |||
44 | /* | ||
45 | * The Aux Regs layout is same bit-by-bit in both BE/LE modes. | ||
46 | * However when casted as a bitfield encoded "C" struct, gcc treats it as | ||
47 | * memory, generating different code for BE/LE, requiring strcture adj (see | ||
48 | * include/asm/arcregs.h) | ||
49 | * | ||
50 | * However when manually "carving" the value for a Aux, no special handling | ||
51 | * of BE is needed because of the property discribed above | ||
52 | */ | ||
53 | #define IDU_SET_COMMAND(irq, cmd) \ | ||
54 | do { \ | ||
55 | uint32_t __val; \ | ||
56 | __val = (((irq & 0xFF) << 8) | (cmd & 0xFF)); \ | ||
57 | write_aux_reg(ARC_AUX_IDU_REG_CMD, __val); \ | ||
58 | } while (0) | ||
59 | |||
60 | #define IDU_SET_PARAM(par) write_aux_reg(ARC_AUX_IDU_REG_PARAM, par) | ||
61 | #define IDU_GET_PARAM() read_aux_reg(ARC_AUX_IDU_REG_PARAM) | ||
62 | |||
63 | /* IDU Commands */ | ||
64 | #define IDU_DISABLE 0x00 | ||
65 | #define IDU_ENABLE 0x01 | ||
66 | #define IDU_IRQ_CLEAR 0x02 | ||
67 | #define IDU_IRQ_ASSERT 0x03 | ||
68 | #define IDU_IRQ_WMODE 0x04 | ||
69 | #define IDU_IRQ_STATUS 0x05 | ||
70 | #define IDU_IRQ_ACK 0x06 | ||
71 | #define IDU_IRQ_PEND 0x07 | ||
72 | #define IDU_IRQ_RMODE 0x08 | ||
73 | #define IDU_IRQ_WBITMASK 0x09 | ||
74 | #define IDU_IRQ_RBITMASK 0x0A | ||
75 | |||
76 | #define idu_enable() IDU_SET_COMMAND(0, IDU_ENABLE) | ||
77 | #define idu_disable() IDU_SET_COMMAND(0, IDU_DISABLE) | ||
78 | |||
79 | #define idu_irq_assert(irq) IDU_SET_COMMAND((irq), IDU_IRQ_ASSERT) | ||
80 | #define idu_irq_clear(irq) IDU_SET_COMMAND((irq), IDU_IRQ_CLEAR) | ||
81 | |||
82 | /* IDU Interrupt Mode - Destination Encoding */ | ||
83 | #define IDU_IRQ_MOD_DISABLE 0x00 | ||
84 | #define IDU_IRQ_MOD_ROUND_RECP 0x01 | ||
85 | #define IDU_IRQ_MOD_TCPU_FIRSTRECP 0x02 | ||
86 | #define IDU_IRQ_MOD_TCPU_ALLRECP 0x03 | ||
87 | |||
88 | /* IDU Interrupt Mode - Triggering Mode */ | ||
89 | #define IDU_IRQ_MODE_LEVEL_TRIG 0x00 | ||
90 | #define IDU_IRQ_MODE_PULSE_TRIG 0x01 | ||
91 | |||
92 | #define IDU_IRQ_MODE_PARAM(dest_mode, trig_mode) \ | ||
93 | (((trig_mode & 0x01) << 15) | (dest_mode & 0xFF)) | ||
94 | |||
95 | struct idu_irq_config { | ||
96 | uint8_t irq; | ||
97 | uint8_t dest_mode; | ||
98 | uint8_t trig_mode; | ||
99 | }; | ||
100 | |||
101 | struct idu_irq_status { | ||
102 | uint8_t irq; | ||
103 | bool enabled; | ||
104 | bool status; | ||
105 | bool ack; | ||
106 | bool pend; | ||
107 | uint8_t next_rr; | ||
108 | }; | ||
109 | |||
110 | extern void idu_irq_set_tgtcpu(uint8_t irq, uint32_t mask); | ||
111 | extern void idu_irq_set_mode(uint8_t irq, uint8_t dest_mode, uint8_t trig_mode); | ||
112 | |||
113 | extern void iss_model_init_smp(unsigned int cpu); | ||
114 | extern void iss_model_init_early_smp(void); | ||
115 | |||
116 | #endif /* CONFIG_SMP */ | ||
117 | |||
118 | #endif | ||
diff --git a/arch/arc/plat-arcfpga/irq.c b/arch/arc/plat-arcfpga/irq.c new file mode 100644 index 000000000000..d2215fd889c2 --- /dev/null +++ b/arch/arc/plat-arcfpga/irq.c | |||
@@ -0,0 +1,25 @@ | |||
1 | /* | ||
2 | * ARC FPGA Platform IRQ hookups | ||
3 | * | ||
4 | * Copyright (C) 2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | */ | ||
10 | |||
11 | #include <linux/interrupt.h> | ||
12 | #include <plat/irq.h> | ||
13 | |||
14 | void __init plat_fpga_init_IRQ(void) | ||
15 | { | ||
16 | /* | ||
17 | * SMP Hack because UART IRQ hardwired to cpu0 (boot-cpu) but if the | ||
18 | * request_irq() comes from any other CPU, the low level IRQ unamsking | ||
19 | * essential for getting Interrupts won't be enabled on cpu0, locking | ||
20 | * up the UART state machine. | ||
21 | */ | ||
22 | #ifdef CONFIG_SMP | ||
23 | arch_unmask_irq(UART0_IRQ); | ||
24 | #endif | ||
25 | } | ||
diff --git a/arch/arc/plat-arcfpga/platform.c b/arch/arc/plat-arcfpga/platform.c new file mode 100644 index 000000000000..4e20a1a5104d --- /dev/null +++ b/arch/arc/plat-arcfpga/platform.c | |||
@@ -0,0 +1,226 @@ | |||
1 | /* | ||
2 | * ARC FPGA Platform support code | ||
3 | * | ||
4 | * Copyright (C) 2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | */ | ||
10 | |||
11 | #include <linux/types.h> | ||
12 | #include <linux/init.h> | ||
13 | #include <linux/device.h> | ||
14 | #include <linux/platform_device.h> | ||
15 | #include <linux/io.h> | ||
16 | #include <linux/console.h> | ||
17 | #include <linux/of_platform.h> | ||
18 | #include <asm/setup.h> | ||
19 | #include <asm/clk.h> | ||
20 | #include <asm/mach_desc.h> | ||
21 | #include <plat/memmap.h> | ||
22 | #include <plat/smp.h> | ||
23 | #include <plat/irq.h> | ||
24 | |||
25 | /*-----------------------BVCI Latency Unit -----------------------------*/ | ||
26 | |||
27 | #ifdef CONFIG_ARC_HAS_BVCI_LAT_UNIT | ||
28 | |||
29 | int lat_cycles = CONFIG_BVCI_LAT_CYCLES; | ||
30 | |||
31 | /* BVCI Bus Profiler: Latency Unit */ | ||
32 | static void __init setup_bvci_lat_unit(void) | ||
33 | { | ||
34 | #define MAX_BVCI_UNITS 12 | ||
35 | |||
36 | unsigned int i; | ||
37 | unsigned int *base = (unsigned int *)BVCI_LAT_UNIT_BASE; | ||
38 | const unsigned long units_req = CONFIG_BVCI_LAT_UNITS; | ||
39 | const unsigned int REG_UNIT = 21; | ||
40 | const unsigned int REG_VAL = 22; | ||
41 | |||
42 | /* | ||
43 | * There are multiple Latency Units corresponding to the many | ||
44 | * interfaces of the system bus arbiter (both CPU side as well as | ||
45 | * the peripheral side). | ||
46 | * | ||
47 | * Unit 0 - System Arb and Mem Controller - adds latency to all | ||
48 | * memory trasactions | ||
49 | * Unit 1 - I$ and System Bus | ||
50 | * Unit 2 - D$ and System Bus | ||
51 | * .. | ||
52 | * Unit 12 - IDE Disk controller and System Bus | ||
53 | * | ||
54 | * The programmers model requires writing to lat_unit reg first | ||
55 | * and then the latency value (cycles) to lat_value reg | ||
56 | */ | ||
57 | |||
58 | if (CONFIG_BVCI_LAT_UNITS == 0) { | ||
59 | writel(0, base + REG_UNIT); | ||
60 | writel(lat_cycles, base + REG_VAL); | ||
61 | pr_info("BVCI Latency for all Memory Transactions %d cycles\n", | ||
62 | lat_cycles); | ||
63 | } else { | ||
64 | for_each_set_bit(i, &units_req, MAX_BVCI_UNITS) { | ||
65 | writel(i + 1, base + REG_UNIT); /* loop is 0 based */ | ||
66 | writel(lat_cycles, base + REG_VAL); | ||
67 | pr_info("BVCI Latency for Unit[%d] = %d cycles\n", | ||
68 | (i + 1), lat_cycles); | ||
69 | } | ||
70 | } | ||
71 | } | ||
72 | #else | ||
73 | static void __init setup_bvci_lat_unit(void) | ||
74 | { | ||
75 | } | ||
76 | #endif | ||
77 | |||
78 | /*----------------------- Platform Devices -----------------------------*/ | ||
79 | |||
80 | static unsigned long arc_uart_info[] = { | ||
81 | 0, /* uart->is_emulated (runtime @running_on_hw) */ | ||
82 | 0, /* uart->port.uartclk */ | ||
83 | 0, /* uart->baud */ | ||
84 | 0 | ||
85 | }; | ||
86 | |||
87 | #if defined(CONFIG_SERIAL_ARC_CONSOLE) | ||
88 | /* | ||
89 | * static platform data - but only for early serial | ||
90 | * TBD: derive this from a special DT node | ||
91 | */ | ||
92 | static struct resource arc_uart0_res[] = { | ||
93 | { | ||
94 | .start = UART0_BASE, | ||
95 | .end = UART0_BASE + 0xFF, | ||
96 | .flags = IORESOURCE_MEM, | ||
97 | }, | ||
98 | { | ||
99 | .start = UART0_IRQ, | ||
100 | .end = UART0_IRQ, | ||
101 | .flags = IORESOURCE_IRQ, | ||
102 | }, | ||
103 | }; | ||
104 | |||
105 | static struct platform_device arc_uart0_dev = { | ||
106 | .name = "arc-uart", | ||
107 | .id = 0, | ||
108 | .num_resources = ARRAY_SIZE(arc_uart0_res), | ||
109 | .resource = arc_uart0_res, | ||
110 | .dev = { | ||
111 | .platform_data = &arc_uart_info, | ||
112 | }, | ||
113 | }; | ||
114 | |||
115 | static struct platform_device *fpga_early_devs[] __initdata = { | ||
116 | &arc_uart0_dev, | ||
117 | }; | ||
118 | #endif | ||
119 | |||
120 | static void arc_fpga_serial_init(void) | ||
121 | { | ||
122 | /* To let driver workaround ISS bug: baudh Reg can't be set to 0 */ | ||
123 | arc_uart_info[0] = !running_on_hw; | ||
124 | |||
125 | arc_uart_info[1] = arc_get_core_freq(); | ||
126 | |||
127 | arc_uart_info[2] = CONFIG_ARC_SERIAL_BAUD; | ||
128 | |||
129 | #if defined(CONFIG_SERIAL_ARC_CONSOLE) | ||
130 | early_platform_add_devices(fpga_early_devs, | ||
131 | ARRAY_SIZE(fpga_early_devs)); | ||
132 | |||
133 | /* | ||
134 | * ARC console driver registers itself as an early platform driver | ||
135 | * of class "earlyprintk". | ||
136 | * Install it here, followed by probe of devices. | ||
137 | * The installation here doesn't require earlyprintk in command line | ||
138 | * To do so however, replace the lines below with | ||
139 | * parse_early_param(); | ||
140 | * early_platform_driver_probe("earlyprintk", 1, 1); | ||
141 | * ^^ | ||
142 | */ | ||
143 | early_platform_driver_register_all("earlyprintk"); | ||
144 | early_platform_driver_probe("earlyprintk", 1, 0); | ||
145 | |||
146 | /* | ||
147 | * This is to make sure that arc uart would be preferred console | ||
148 | * despite one/more of following: | ||
149 | * -command line lacked "console=ttyARC0" or | ||
150 | * -CONFIG_VT_CONSOLE was enabled (for no reason whatsoever) | ||
151 | * Note that this needs to be done after above early console is reg, | ||
152 | * otherwise the early console never gets a chance to run. | ||
153 | */ | ||
154 | add_preferred_console("ttyARC", 0, "115200"); | ||
155 | #endif | ||
156 | } | ||
157 | |||
158 | static void __init plat_fpga_early_init(void) | ||
159 | { | ||
160 | pr_info("[plat-arcfpga]: registering early dev resources\n"); | ||
161 | |||
162 | setup_bvci_lat_unit(); | ||
163 | |||
164 | arc_fpga_serial_init(); | ||
165 | |||
166 | #ifdef CONFIG_SMP | ||
167 | iss_model_init_early_smp(); | ||
168 | #endif | ||
169 | } | ||
170 | |||
171 | static struct of_dev_auxdata plat_auxdata_lookup[] __initdata = { | ||
172 | #if defined(CONFIG_SERIAL_ARC) || defined(CONFIG_SERIAL_ARC_MODULE) | ||
173 | OF_DEV_AUXDATA("snps,arc-uart", UART0_BASE, "arc-uart", arc_uart_info), | ||
174 | #endif | ||
175 | {} | ||
176 | }; | ||
177 | |||
178 | static void __init plat_fpga_populate_dev(void) | ||
179 | { | ||
180 | pr_info("[plat-arcfpga]: registering device resources\n"); | ||
181 | |||
182 | /* | ||
183 | * Traverses flattened DeviceTree - registering platform devices | ||
184 | * complete with their resources | ||
185 | */ | ||
186 | of_platform_populate(NULL, of_default_bus_match_table, | ||
187 | plat_auxdata_lookup, NULL); | ||
188 | } | ||
189 | |||
190 | /*----------------------- Machine Descriptions ------------------------------ | ||
191 | * | ||
192 | * Machine description is simply a set of platform/board specific callbacks | ||
193 | * This is not directly related to DeviceTree based dynamic device creation, | ||
194 | * however as part of early device tree scan, we also select the right | ||
195 | * callback set, by matching the DT compatible name. | ||
196 | */ | ||
197 | |||
198 | static const char *aa4_compat[] __initdata = { | ||
199 | "snps,arc-angel4", | ||
200 | NULL, | ||
201 | }; | ||
202 | |||
203 | MACHINE_START(ANGEL4, "angel4") | ||
204 | .dt_compat = aa4_compat, | ||
205 | .init_early = plat_fpga_early_init, | ||
206 | .init_machine = plat_fpga_populate_dev, | ||
207 | .init_irq = plat_fpga_init_IRQ, | ||
208 | #ifdef CONFIG_SMP | ||
209 | .init_smp = iss_model_init_smp, | ||
210 | #endif | ||
211 | MACHINE_END | ||
212 | |||
213 | static const char *ml509_compat[] __initdata = { | ||
214 | "snps,arc-ml509", | ||
215 | NULL, | ||
216 | }; | ||
217 | |||
218 | MACHINE_START(ML509, "ml509") | ||
219 | .dt_compat = ml509_compat, | ||
220 | .init_early = plat_fpga_early_init, | ||
221 | .init_machine = plat_fpga_populate_dev, | ||
222 | .init_irq = plat_fpga_init_IRQ, | ||
223 | #ifdef CONFIG_SMP | ||
224 | .init_smp = iss_model_init_smp, | ||
225 | #endif | ||
226 | MACHINE_END | ||
diff --git a/arch/arc/plat-arcfpga/smp.c b/arch/arc/plat-arcfpga/smp.c new file mode 100644 index 000000000000..91b55349a5f8 --- /dev/null +++ b/arch/arc/plat-arcfpga/smp.c | |||
@@ -0,0 +1,171 @@ | |||
1 | /* | ||
2 | * ARC700 Simulation-only Extensions for SMP | ||
3 | * | ||
4 | * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | * | ||
10 | * Vineet Gupta - 2012 : split off arch common and plat specific SMP | ||
11 | * Rajeshwar Ranga - 2007 : Interrupt Distribution Unit API's | ||
12 | */ | ||
13 | |||
14 | #include <linux/smp.h> | ||
15 | #include <linux/irq.h> | ||
16 | #include <plat/irq.h> | ||
17 | #include <plat/smp.h> | ||
18 | |||
19 | static char smp_cpuinfo_buf[128]; | ||
20 | |||
21 | /* | ||
22 | *------------------------------------------------------------------- | ||
23 | * Platform specific callbacks expected by arch SMP code | ||
24 | *------------------------------------------------------------------- | ||
25 | */ | ||
26 | |||
27 | /* | ||
28 | * Master kick starting another CPU | ||
29 | */ | ||
30 | static void iss_model_smp_wakeup_cpu(int cpu, unsigned long pc) | ||
31 | { | ||
32 | /* setup the start PC */ | ||
33 | write_aux_reg(ARC_AUX_XTL_REG_PARAM, pc); | ||
34 | |||
35 | /* Trigger WRITE_PC cmd for this cpu */ | ||
36 | write_aux_reg(ARC_AUX_XTL_REG_CMD, | ||
37 | (ARC_XTL_CMD_WRITE_PC | (cpu << 8))); | ||
38 | |||
39 | /* Take the cpu out of Halt */ | ||
40 | write_aux_reg(ARC_AUX_XTL_REG_CMD, | ||
41 | (ARC_XTL_CMD_CLEAR_HALT | (cpu << 8))); | ||
42 | |||
43 | } | ||
44 | |||
45 | /* | ||
46 | * Any SMP specific init any CPU does when it comes up. | ||
47 | * Here we setup the CPU to enable Inter-Processor-Interrupts | ||
48 | * Called for each CPU | ||
49 | * -Master : init_IRQ() | ||
50 | * -Other(s) : start_kernel_secondary() | ||
51 | */ | ||
52 | void iss_model_init_smp(unsigned int cpu) | ||
53 | { | ||
54 | /* Check if CPU is configured for more than 16 interrupts */ | ||
55 | if (NR_IRQS <= 16 || get_hw_config_num_irq() <= 16) | ||
56 | panic("[arcfpga] IRQ system can't support IDU IPI\n"); | ||
57 | |||
58 | idu_disable(); | ||
59 | |||
60 | /**************************************************************** | ||
61 | * IDU provides a set of Common IRQs, each of which can be dynamically | ||
62 | * attached to (1|many|all) CPUs. | ||
63 | * The Common IRQs [0-15] are mapped as CPU pvt [16-31] | ||
64 | * | ||
65 | * Here we use a simple 1:1 mapping: | ||
66 | * A CPU 'x' is wired to Common IRQ 'x'. | ||
67 | * So an IDU ASSERT on IRQ 'x' will trigger Interupt on CPU 'x', which | ||
68 | * makes up for our simple IPI plumbing. | ||
69 | * | ||
70 | * TBD: Have a dedicated multicast IRQ for sending IPIs to all CPUs | ||
71 | * w/o having to do one-at-a-time | ||
72 | ******************************************************************/ | ||
73 | |||
74 | /* | ||
75 | * Claim an IRQ which would trigger IPI on this CPU. | ||
76 | * In IDU parlance it involves setting up a cpu bitmask for the IRQ | ||
77 | * The bitmap here contains only 1 CPU (self). | ||
78 | */ | ||
79 | idu_irq_set_tgtcpu(cpu, 0x1 << cpu); | ||
80 | |||
81 | /* Set the IRQ destination to use the bitmask above */ | ||
82 | idu_irq_set_mode(cpu, 7, /* XXX: IDU_IRQ_MOD_TCPU_ALLRECP: ISS bug */ | ||
83 | IDU_IRQ_MODE_PULSE_TRIG); | ||
84 | |||
85 | idu_enable(); | ||
86 | |||
87 | /* Attach the arch-common IPI ISR to our IDU IRQ */ | ||
88 | smp_ipi_irq_setup(cpu, IDU_INTERRUPT_0 + cpu); | ||
89 | } | ||
90 | |||
91 | static void iss_model_ipi_send(void *arg) | ||
92 | { | ||
93 | struct cpumask *callmap = arg; | ||
94 | unsigned int cpu; | ||
95 | |||
96 | for_each_cpu(cpu, callmap) | ||
97 | idu_irq_assert(cpu); | ||
98 | } | ||
99 | |||
100 | static void iss_model_ipi_clear(int cpu, int irq) | ||
101 | { | ||
102 | idu_irq_clear(IDU_INTERRUPT_0 + cpu); | ||
103 | } | ||
104 | |||
105 | void iss_model_init_early_smp(void) | ||
106 | { | ||
107 | #define IS_AVAIL1(var, str) ((var) ? str : "") | ||
108 | |||
109 | struct bcr_mp mp; | ||
110 | |||
111 | READ_BCR(ARC_REG_MP_BCR, mp); | ||
112 | |||
113 | sprintf(smp_cpuinfo_buf, "Extn [ISS-SMP]: v%d, arch(%d) %s %s %s\n", | ||
114 | mp.ver, mp.mp_arch, IS_AVAIL1(mp.scu, "SCU"), | ||
115 | IS_AVAIL1(mp.idu, "IDU"), IS_AVAIL1(mp.sdu, "SDU")); | ||
116 | |||
117 | plat_smp_ops.info = smp_cpuinfo_buf; | ||
118 | |||
119 | plat_smp_ops.cpu_kick = iss_model_smp_wakeup_cpu; | ||
120 | plat_smp_ops.ipi_send = iss_model_ipi_send; | ||
121 | plat_smp_ops.ipi_clear = iss_model_ipi_clear; | ||
122 | } | ||
123 | |||
124 | /* | ||
125 | *------------------------------------------------------------------- | ||
126 | * Low level Platform IPI Providers | ||
127 | *------------------------------------------------------------------- | ||
128 | */ | ||
129 | |||
130 | /* Set the Mode for the Common IRQ */ | ||
131 | void idu_irq_set_mode(uint8_t irq, uint8_t dest_mode, uint8_t trig_mode) | ||
132 | { | ||
133 | uint32_t par = IDU_IRQ_MODE_PARAM(dest_mode, trig_mode); | ||
134 | |||
135 | IDU_SET_PARAM(par); | ||
136 | IDU_SET_COMMAND(irq, IDU_IRQ_WMODE); | ||
137 | } | ||
138 | |||
139 | /* Set the target cpu Bitmask for Common IRQ */ | ||
140 | void idu_irq_set_tgtcpu(uint8_t irq, uint32_t mask) | ||
141 | { | ||
142 | IDU_SET_PARAM(mask); | ||
143 | IDU_SET_COMMAND(irq, IDU_IRQ_WBITMASK); | ||
144 | } | ||
145 | |||
146 | /* Get the Interrupt Acknowledged status for IRQ (as CPU Bitmask) */ | ||
147 | bool idu_irq_get_ack(uint8_t irq) | ||
148 | { | ||
149 | uint32_t val; | ||
150 | |||
151 | IDU_SET_COMMAND(irq, IDU_IRQ_ACK); | ||
152 | val = IDU_GET_PARAM(); | ||
153 | |||
154 | return val & (1 << irq); | ||
155 | } | ||
156 | |||
157 | /* | ||
158 | * Get the Interrupt Pending status for IRQ (as CPU Bitmask) | ||
159 | * -Pending means CPU has not yet noticed the IRQ (e.g. disabled) | ||
160 | * -After Interrupt has been taken, the IPI expcitily needs to be | ||
161 | * cleared, to be acknowledged. | ||
162 | */ | ||
163 | bool idu_irq_get_pend(uint8_t irq) | ||
164 | { | ||
165 | uint32_t val; | ||
166 | |||
167 | IDU_SET_COMMAND(irq, IDU_IRQ_PEND); | ||
168 | val = IDU_GET_PARAM(); | ||
169 | |||
170 | return val & (1 << irq); | ||
171 | } | ||
diff --git a/arch/parisc/Kconfig b/arch/parisc/Kconfig index 4d5ea7648574..a2a47d9d6a22 100644 --- a/arch/parisc/Kconfig +++ b/arch/parisc/Kconfig | |||
@@ -18,6 +18,7 @@ config PARISC | |||
18 | select ARCH_HAVE_NMI_SAFE_CMPXCHG | 18 | select ARCH_HAVE_NMI_SAFE_CMPXCHG |
19 | select GENERIC_SMP_IDLE_THREAD | 19 | select GENERIC_SMP_IDLE_THREAD |
20 | select GENERIC_STRNCPY_FROM_USER | 20 | select GENERIC_STRNCPY_FROM_USER |
21 | select SYSCTL_ARCH_UNALIGN_ALLOW | ||
21 | select HAVE_MOD_ARCH_SPECIFIC | 22 | select HAVE_MOD_ARCH_SPECIFIC |
22 | select HAVE_VIRT_TO_BUS | 23 | select HAVE_VIRT_TO_BUS |
23 | select MODULES_USE_ELF_RELA | 24 | select MODULES_USE_ELF_RELA |
diff --git a/include/asm-generic/checksum.h b/include/asm-generic/checksum.h index c084767c88bc..59811df58c5b 100644 --- a/include/asm-generic/checksum.h +++ b/include/asm-generic/checksum.h | |||
@@ -38,12 +38,15 @@ extern __wsum csum_partial_copy_from_user(const void __user *src, void *dst, | |||
38 | csum_partial_copy((src), (dst), (len), (sum)) | 38 | csum_partial_copy((src), (dst), (len), (sum)) |
39 | #endif | 39 | #endif |
40 | 40 | ||
41 | #ifndef ip_fast_csum | ||
41 | /* | 42 | /* |
42 | * This is a version of ip_compute_csum() optimized for IP headers, | 43 | * This is a version of ip_compute_csum() optimized for IP headers, |
43 | * which always checksum on 4 octet boundaries. | 44 | * which always checksum on 4 octet boundaries. |
44 | */ | 45 | */ |
45 | extern __sum16 ip_fast_csum(const void *iph, unsigned int ihl); | 46 | extern __sum16 ip_fast_csum(const void *iph, unsigned int ihl); |
47 | #endif | ||
46 | 48 | ||
49 | #ifndef csum_fold | ||
47 | /* | 50 | /* |
48 | * Fold a partial checksum | 51 | * Fold a partial checksum |
49 | */ | 52 | */ |
@@ -54,6 +57,7 @@ static inline __sum16 csum_fold(__wsum csum) | |||
54 | sum = (sum & 0xffff) + (sum >> 16); | 57 | sum = (sum & 0xffff) + (sum >> 16); |
55 | return (__force __sum16)~sum; | 58 | return (__force __sum16)~sum; |
56 | } | 59 | } |
60 | #endif | ||
57 | 61 | ||
58 | #ifndef csum_tcpudp_nofold | 62 | #ifndef csum_tcpudp_nofold |
59 | /* | 63 | /* |
diff --git a/include/asm-generic/uaccess.h b/include/asm-generic/uaccess.h index 9788568f7978..c184aa8ec8cd 100644 --- a/include/asm-generic/uaccess.h +++ b/include/asm-generic/uaccess.h | |||
@@ -7,7 +7,6 @@ | |||
7 | * address space, e.g. all NOMMU machines. | 7 | * address space, e.g. all NOMMU machines. |
8 | */ | 8 | */ |
9 | #include <linux/sched.h> | 9 | #include <linux/sched.h> |
10 | #include <linux/mm.h> | ||
11 | #include <linux/string.h> | 10 | #include <linux/string.h> |
12 | 11 | ||
13 | #include <asm/segment.h> | 12 | #include <asm/segment.h> |
@@ -32,7 +31,9 @@ static inline void set_fs(mm_segment_t fs) | |||
32 | } | 31 | } |
33 | #endif | 32 | #endif |
34 | 33 | ||
34 | #ifndef segment_eq | ||
35 | #define segment_eq(a, b) ((a).seg == (b).seg) | 35 | #define segment_eq(a, b) ((a).seg == (b).seg) |
36 | #endif | ||
36 | 37 | ||
37 | #define VERIFY_READ 0 | 38 | #define VERIFY_READ 0 |
38 | #define VERIFY_WRITE 1 | 39 | #define VERIFY_WRITE 1 |
@@ -168,12 +169,18 @@ static inline __must_check long __copy_to_user(void __user *to, | |||
168 | -EFAULT; \ | 169 | -EFAULT; \ |
169 | }) | 170 | }) |
170 | 171 | ||
172 | #ifndef __put_user_fn | ||
173 | |||
171 | static inline int __put_user_fn(size_t size, void __user *ptr, void *x) | 174 | static inline int __put_user_fn(size_t size, void __user *ptr, void *x) |
172 | { | 175 | { |
173 | size = __copy_to_user(ptr, x, size); | 176 | size = __copy_to_user(ptr, x, size); |
174 | return size ? -EFAULT : size; | 177 | return size ? -EFAULT : size; |
175 | } | 178 | } |
176 | 179 | ||
180 | #define __put_user_fn(sz, u, k) __put_user_fn(sz, u, k) | ||
181 | |||
182 | #endif | ||
183 | |||
177 | extern int __put_user_bad(void) __attribute__((noreturn)); | 184 | extern int __put_user_bad(void) __attribute__((noreturn)); |
178 | 185 | ||
179 | #define __get_user(x, ptr) \ | 186 | #define __get_user(x, ptr) \ |
@@ -224,12 +231,17 @@ extern int __put_user_bad(void) __attribute__((noreturn)); | |||
224 | -EFAULT; \ | 231 | -EFAULT; \ |
225 | }) | 232 | }) |
226 | 233 | ||
234 | #ifndef __get_user_fn | ||
227 | static inline int __get_user_fn(size_t size, const void __user *ptr, void *x) | 235 | static inline int __get_user_fn(size_t size, const void __user *ptr, void *x) |
228 | { | 236 | { |
229 | size = __copy_from_user(x, ptr, size); | 237 | size = __copy_from_user(x, ptr, size); |
230 | return size ? -EFAULT : size; | 238 | return size ? -EFAULT : size; |
231 | } | 239 | } |
232 | 240 | ||
241 | #define __get_user_fn(sz, u, k) __get_user_fn(sz, u, k) | ||
242 | |||
243 | #endif | ||
244 | |||
233 | extern int __get_user_bad(void) __attribute__((noreturn)); | 245 | extern int __get_user_bad(void) __attribute__((noreturn)); |
234 | 246 | ||
235 | #ifndef __copy_from_user_inatomic | 247 | #ifndef __copy_from_user_inatomic |
diff --git a/init/Kconfig b/init/Kconfig index 0a5e80fb9ba2..22616cd434bc 100644 --- a/init/Kconfig +++ b/init/Kconfig | |||
@@ -1230,6 +1230,14 @@ config SYSCTL_ARCH_UNALIGN_NO_WARN | |||
1230 | Allows arch to define/use @no_unaligned_warning to possibly warn | 1230 | Allows arch to define/use @no_unaligned_warning to possibly warn |
1231 | about unaligned access emulation going on under the hood. | 1231 | about unaligned access emulation going on under the hood. |
1232 | 1232 | ||
1233 | config SYSCTL_ARCH_UNALIGN_ALLOW | ||
1234 | bool | ||
1235 | help | ||
1236 | Enable support for /proc/sys/kernel/unaligned-trap | ||
1237 | Allows arches to define/use @unaligned_enabled to runtime toggle | ||
1238 | the unaligned access emulation. | ||
1239 | see arch/parisc/kernel/unaligned.c for reference | ||
1240 | |||
1233 | config KALLSYMS | 1241 | config KALLSYMS |
1234 | bool "Load all symbols for debugging/ksymoops" if EXPERT | 1242 | bool "Load all symbols for debugging/ksymoops" if EXPERT |
1235 | default y | 1243 | default y |
diff --git a/kernel/sysctl.c b/kernel/sysctl.c index d1b4ee67d2df..afc1dc60f3f8 100644 --- a/kernel/sysctl.c +++ b/kernel/sysctl.c | |||
@@ -157,6 +157,9 @@ extern int sysctl_tsb_ratio; | |||
157 | 157 | ||
158 | #ifdef __hppa__ | 158 | #ifdef __hppa__ |
159 | extern int pwrsw_enabled; | 159 | extern int pwrsw_enabled; |
160 | #endif | ||
161 | |||
162 | #ifdef CONFIG_SYSCTL_ARCH_UNALIGN_ALLOW | ||
160 | extern int unaligned_enabled; | 163 | extern int unaligned_enabled; |
161 | #endif | 164 | #endif |
162 | 165 | ||
@@ -555,6 +558,8 @@ static struct ctl_table kern_table[] = { | |||
555 | .mode = 0644, | 558 | .mode = 0644, |
556 | .proc_handler = proc_dointvec, | 559 | .proc_handler = proc_dointvec, |
557 | }, | 560 | }, |
561 | #endif | ||
562 | #ifdef CONFIG_SYSCTL_ARCH_UNALIGN_ALLOW | ||
558 | { | 563 | { |
559 | .procname = "unaligned-trap", | 564 | .procname = "unaligned-trap", |
560 | .data = &unaligned_enabled, | 565 | .data = &unaligned_enabled, |
diff --git a/lib/checksum.c b/lib/checksum.c index 12dceb27ff20..129775eb6de6 100644 --- a/lib/checksum.c +++ b/lib/checksum.c | |||
@@ -102,6 +102,7 @@ out: | |||
102 | } | 102 | } |
103 | #endif | 103 | #endif |
104 | 104 | ||
105 | #ifndef ip_fast_csum | ||
105 | /* | 106 | /* |
106 | * This is a version of ip_compute_csum() optimized for IP headers, | 107 | * This is a version of ip_compute_csum() optimized for IP headers, |
107 | * which always checksum on 4 octet boundaries. | 108 | * which always checksum on 4 octet boundaries. |
@@ -111,6 +112,7 @@ __sum16 ip_fast_csum(const void *iph, unsigned int ihl) | |||
111 | return (__force __sum16)~do_csum(iph, ihl*4); | 112 | return (__force __sum16)~do_csum(iph, ihl*4); |
112 | } | 113 | } |
113 | EXPORT_SYMBOL(ip_fast_csum); | 114 | EXPORT_SYMBOL(ip_fast_csum); |
115 | #endif | ||
114 | 116 | ||
115 | /* | 117 | /* |
116 | * computes the checksum of a memory block at buff, length len, | 118 | * computes the checksum of a memory block at buff, length len, |
diff --git a/tools/perf/perf.h b/tools/perf/perf.h index c2206c87fc9f..d5818c98d051 100644 --- a/tools/perf/perf.h +++ b/tools/perf/perf.h | |||
@@ -94,6 +94,12 @@ | |||
94 | #define CPUINFO_PROC "cpu model" | 94 | #define CPUINFO_PROC "cpu model" |
95 | #endif | 95 | #endif |
96 | 96 | ||
97 | #ifdef __arc__ | ||
98 | #define rmb() asm volatile("" ::: "memory") | ||
99 | #define cpu_relax() rmb() | ||
100 | #define CPUINFO_PROC "Processor" | ||
101 | #endif | ||
102 | |||
97 | #include <time.h> | 103 | #include <time.h> |
98 | #include <unistd.h> | 104 | #include <unistd.h> |
99 | #include <sys/types.h> | 105 | #include <sys/types.h> |