Merge branch 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/s390/linux

Pull s390 updates from Martin Schwidefsky:
 "The most notable change for this pull request is the ftrace rework
  from Heiko.  It brings a small performance improvement and the ground
  work to support a new gcc option to replace the mcount blocks with a
  single nop.

  Two new s390 specific system calls are added to emulate user space
  mmio for PCI, an artifact of the how PCI memory is accessed.

  Two patches for the memory management with changes to common code.
  For KVM mm_forbids_zeropage is added which disables the empty zero
  page for an mm that is used by a KVM process.  And an optimization,
  pmdp_get_and_clear_full is added analog to ptep_get_and_clear_full.

  Some micro optimization for the cmpxchg and the spinlock code.

  And as usual bug fixes and cleanups"

* 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/s390/linux: (46 commits)
  s390/cputime: fix 31-bit compile
  s390/scm_block: make the number of reqs per HW req configurable
  s390/scm_block: handle multiple requests in one HW request
  s390/scm_block: allocate aidaw pages only when necessary
  s390/scm_block: use mempool to manage aidaw requests
  s390/eadm: change timeout value
  s390/mm: fix memory leak of ptlock in pmd_free_tlb
  s390: use local symbol names in entry[64].S
  s390/ptrace: always include vector registers in core files
  s390/simd: clear vector register pointer on fork/clone
  s390: translate cputime magic constants to macros
  s390/idle: convert open coded idle time seqcount
  s390/idle: add missing irq off lockdep annotation
  s390/debug: avoid function call for debug_sprintf_*
  s390/kprobes: fix instruction copy for out of line execution
  s390: remove diag 44 calls from cpu_relax()
  s390/dasd: retry partition detection
  s390/dasd: fix list corruption for sleep_on requests
  s390/dasd: fix infinite term I/O loop
  s390/dasd: remove unused code
  ...
This commit is contained in:
Linus Torvalds 2014-12-11 17:30:55 -08:00
commit 27afc5dbda
67 changed files with 1529 additions and 1508 deletions

View file

@ -26,11 +26,6 @@ The Linux for s/390 & z/Architecture Kernel Task Structure
Register Usage & Stackframes on Linux for s/390 & z/Architecture Register Usage & Stackframes on Linux for s/390 & z/Architecture
A sample program with comments A sample program with comments
Compiling programs for debugging on Linux for s/390 & z/Architecture Compiling programs for debugging on Linux for s/390 & z/Architecture
Figuring out gcc compile errors
Debugging Tools
objdump
strace
Performance Debugging
Debugging under VM Debugging under VM
s/390 & z/Architecture IO Overview s/390 & z/Architecture IO Overview
Debugging IO on s/390 & z/Architecture under VM Debugging IO on s/390 & z/Architecture under VM
@ -114,28 +109,25 @@ s/390 z/Architecture
16-17 16-17 Address Space Control 16-17 16-17 Address Space Control
00 Primary Space Mode when DAT on 00 Primary Space Mode:
The linux kernel currently runs in this mode, CR1 is affiliated with The register CR1 contains the primary address-space control ele-
this mode & points to the primary segment table origin etc. ment (PASCE), which points to the primary space region/segment
table origin.
01 Access register mode this mode is used in functions to 01 Access register mode
copy data between kernel & user space.
10 Secondary space mode not used in linux however CR7 the 10 Secondary Space Mode:
register affiliated with this mode is & this & normally The register CR7 contains the secondary address-space control
CR13=CR7 to allow us to copy data between kernel & user space. element (SASCE), which points to the secondary space region or
We do this as follows: segment table origin.
We set ar2 to 0 to designate its
affiliated gpr ( gpr2 )to point to primary=kernel space.
We set ar4 to 1 to designate its
affiliated gpr ( gpr4 ) to point to secondary=home=user space
& then essentially do a memcopy(gpr2,gpr4,size) to
copy data between the address spaces, the reason we use home space for the
kernel & don't keep secondary space free is that code will not run in
secondary space.
11 Home Space Mode all user programs run in this mode. 11 Home Space Mode:
it is affiliated with CR13. The register CR13 contains the home space address-space control
element (HASCE), which points to the home space region/segment
table origin.
See "Address Spaces on Linux for s/390 & z/Architecture" below
for more information about address space usage in Linux.
18-19 18-19 Condition codes (CC) 18-19 18-19 Condition codes (CC)
@ -249,9 +241,9 @@ currently 4TB of physical memory currently on z/Architecture.
Address Spaces on Linux for s/390 & z/Architecture Address Spaces on Linux for s/390 & z/Architecture
================================================== ==================================================
Our addressing scheme is as follows Our addressing scheme is basically as follows:
Primary Space Home Space
Himem 0x7fffffff 2GB on s/390 ***************** **************** Himem 0x7fffffff 2GB on s/390 ***************** ****************
currently 0x3ffffffffff (2^42)-1 * User Stack * * * currently 0x3ffffffffff (2^42)-1 * User Stack * * *
on z/Architecture. ***************** * * on z/Architecture. ***************** * *
@ -264,9 +256,46 @@ on z/Architecture. ***************** * *
* Sections * * * * Sections * * *
0x00000000 ***************** **************** 0x00000000 ***************** ****************
This also means that we need to look at the PSW problem state bit This also means that we need to look at the PSW problem state bit and the
or the addressing mode to decide whether we are looking at addressing mode to decide whether we are looking at user or kernel space.
user or kernel space.
User space runs in primary address mode (or access register mode within
the vdso code).
The kernel usually also runs in home space mode, however when accessing
user space the kernel switches to primary or secondary address mode if
the mvcos instruction is not available or if a compare-and-swap (futex)
instruction on a user space address is performed.
When also looking at the ASCE control registers, this means:
User space:
- runs in primary or access register mode
- cr1 contains the user asce
- cr7 contains the user asce
- cr13 contains the kernel asce
Kernel space:
- runs in home space mode
- cr1 contains the user or kernel asce
-> the kernel asce is loaded when a uaccess requires primary or
secondary address mode
- cr7 contains the user or kernel asce, (changed with set_fs())
- cr13 contains the kernel asce
In case of uaccess the kernel changes to:
- primary space mode in case of a uaccess (copy_to_user) and uses
e.g. the mvcp instruction to access user space. However the kernel
will stay in home space mode if the mvcos instruction is available
- secondary space mode in case of futex atomic operations, so that the
instructions come from primary address space and data from secondary
space
In case of KVM, the kernel runs in home space mode, but cr1 gets switched
to contain the gmap asce before the SIE instruction gets executed. When
the SIE instruction is finished, cr1 will be switched back to contain the
user asce.
Virtual Addresses on s/390 & z/Architecture Virtual Addresses on s/390 & z/Architecture
=========================================== ===========================================
@ -706,376 +735,7 @@ Debugging with optimisation has since much improved after fixing
some bugs, please make sure you are using gdb-5.0 or later developed some bugs, please make sure you are using gdb-5.0 or later developed
after Nov'2000. after Nov'2000.
Figuring out gcc compile errors
===============================
If you are getting a lot of syntax errors compiling a program & the problem
isn't blatantly obvious from the source.
It often helps to just preprocess the file, this is done with the -E
option in gcc.
What this does is that it runs through the very first phase of compilation
( compilation in gcc is done in several stages & gcc calls many programs to
achieve its end result ) with the -E option gcc just calls the gcc preprocessor (cpp).
The c preprocessor does the following, it joins all the files #included together
recursively ( #include files can #include other files ) & also the c file you wish to compile.
It puts a fully qualified path of the #included files in a comment & it
does macro expansion.
This is useful for debugging because
1) You can double check whether the files you expect to be included are the ones
that are being included ( e.g. double check that you aren't going to the i386 asm directory ).
2) Check that macro definitions aren't clashing with typedefs,
3) Check that definitions aren't being used before they are being included.
4) Helps put the line emitting the error under the microscope if it contains macros.
For convenience the Linux kernel's makefile will do preprocessing automatically for you
by suffixing the file you want built with .i ( instead of .o )
e.g.
from the linux directory type
make arch/s390/kernel/signal.i
this will build
s390-gcc -D__KERNEL__ -I/home1/barrow/linux/include -Wall -Wstrict-prototypes -O2 -fomit-frame-pointer
-fno-strict-aliasing -D__SMP__ -pipe -fno-strength-reduce -E arch/s390/kernel/signal.c
> arch/s390/kernel/signal.i
Now look at signal.i you should see something like.
# 1 "/home1/barrow/linux/include/asm/types.h" 1
typedef unsigned short umode_t;
typedef __signed__ char __s8;
typedef unsigned char __u8;
typedef __signed__ short __s16;
typedef unsigned short __u16;
If instead you are getting errors further down e.g.
unknown instruction:2515 "move.l" or better still unknown instruction:2515
"Fixme not implemented yet, call Martin" you are probably are attempting to compile some code
meant for another architecture or code that is simply not implemented, with a fixme statement
stuck into the inline assembly code so that the author of the file now knows he has work to do.
To look at the assembly emitted by gcc just before it is about to call gas ( the gnu assembler )
use the -S option.
Again for your convenience the Linux kernel's Makefile will hold your hand &
do all this donkey work for you also by building the file with the .s suffix.
e.g.
from the Linux directory type
make arch/s390/kernel/signal.s
s390-gcc -D__KERNEL__ -I/home1/barrow/linux/include -Wall -Wstrict-prototypes -O2 -fomit-frame-pointer
-fno-strict-aliasing -D__SMP__ -pipe -fno-strength-reduce -S arch/s390/kernel/signal.c
-o arch/s390/kernel/signal.s
This will output something like, ( please note the constant pool & the useful comments
in the prologue to give you a hand at interpreting it ).
.LC54:
.string "misaligned (__u16 *) in __xchg\n"
.LC57:
.string "misaligned (__u32 *) in __xchg\n"
.L$PG1: # Pool sys_sigsuspend
.LC192:
.long -262401
.LC193:
.long -1
.LC194:
.long schedule-.L$PG1
.LC195:
.long do_signal-.L$PG1
.align 4
.globl sys_sigsuspend
.type sys_sigsuspend,@function
sys_sigsuspend:
# leaf function 0
# automatics 16
# outgoing args 0
# need frame pointer 0
# call alloca 0
# has varargs 0
# incoming args (stack) 0
# function length 168
STM 8,15,32(15)
LR 0,15
AHI 15,-112
BASR 13,0
.L$CO1: AHI 13,.L$PG1-.L$CO1
ST 0,0(15)
LR 8,2
N 5,.LC192-.L$PG1(13)
Adding -g to the above output makes the output even more useful
e.g. typing
make CC:="s390-gcc -g" kernel/sched.s
which compiles.
s390-gcc -g -D__KERNEL__ -I/home/barrow/linux-2.3/include -Wall -Wstrict-prototypes -O2 -fomit-frame-pointer -fno-strict-aliasing -pipe -fno-strength-reduce -S kernel/sched.c -o kernel/sched.s
also outputs stabs ( debugger ) info, from this info you can find out the
offsets & sizes of various elements in structures.
e.g. the stab for the structure
struct rlimit {
unsigned long rlim_cur;
unsigned long rlim_max;
};
is
.stabs "rlimit:T(151,2)=s8rlim_cur:(0,5),0,32;rlim_max:(0,5),32,32;;",128,0,0,0
from this stab you can see that
rlimit_cur starts at bit offset 0 & is 32 bits in size
rlimit_max starts at bit offset 32 & is 32 bits in size.
Debugging Tools:
================
objdump
=======
This is a tool with many options the most useful being ( if compiled with -g).
objdump --source <victim program or object file> > <victims debug listing >
The whole kernel can be compiled like this ( Doing this will make a 17MB kernel
& a 200 MB listing ) however you have to strip it before building the image
using the strip command to make it a more reasonable size to boot it.
A source/assembly mixed dump of the kernel can be done with the line
objdump --source vmlinux > vmlinux.lst
Also, if the file isn't compiled -g, this will output as much debugging information
as it can (e.g. function names). This is very slow as it spends lots
of time searching for debugging info. The following self explanatory line should be used
instead if the code isn't compiled -g, as it is much faster:
objdump --disassemble-all --syms vmlinux > vmlinux.lst
As hard drive space is valuable most of us use the following approach.
1) Look at the emitted psw on the console to find the crash address in the kernel.
2) Look at the file System.map ( in the linux directory ) produced when building
the kernel to find the closest address less than the current PSW to find the
offending function.
3) use grep or similar to search the source tree looking for the source file
with this function if you don't know where it is.
4) rebuild this object file with -g on, as an example suppose the file was
( /arch/s390/kernel/signal.o )
5) Assuming the file with the erroneous function is signal.c Move to the base of the
Linux source tree.
6) rm /arch/s390/kernel/signal.o
7) make /arch/s390/kernel/signal.o
8) watch the gcc command line emitted
9) type it in again or alternatively cut & paste it on the console adding the -g option.
10) objdump --source arch/s390/kernel/signal.o > signal.lst
This will output the source & the assembly intermixed, as the snippet below shows
This will unfortunately output addresses which aren't the same
as the kernel ones you should be able to get around the mental arithmetic
by playing with the --adjust-vma parameter to objdump.
static inline void spin_lock(spinlock_t *lp)
{
a0: 18 34 lr %r3,%r4
a2: a7 3a 03 bc ahi %r3,956
__asm__ __volatile(" lhi 1,-1\n"
a6: a7 18 ff ff lhi %r1,-1
aa: 1f 00 slr %r0,%r0
ac: ba 01 30 00 cs %r0,%r1,0(%r3)
b0: a7 44 ff fd jm aa <sys_sigsuspend+0x2e>
saveset = current->blocked;
b4: d2 07 f0 68 mvc 104(8,%r15),972(%r4)
b8: 43 cc
return (set->sig[0] & mask) != 0;
}
6) If debugging under VM go down to that section in the document for more info.
I now have a tool which takes the pain out of --adjust-vma
& you are able to do something like
make /arch/s390/kernel/traps.lst
& it automatically generates the correctly relocated entries for
the text segment in traps.lst.
This tool is now standard in linux distro's in scripts/makelst
strace:
-------
Q. What is it ?
A. It is a tool for intercepting calls to the kernel & logging them
to a file & on the screen.
Q. What use is it ?
A. You can use it to find out what files a particular program opens.
Example 1
---------
If you wanted to know does ping work but didn't have the source
strace ping -c 1 127.0.0.1
& then look at the man pages for each of the syscalls below,
( In fact this is sometimes easier than looking at some spaghetti
source which conditionally compiles for several architectures ).
Not everything that it throws out needs to make sense immediately.
Just looking quickly you can see that it is making up a RAW socket
for the ICMP protocol.
Doing an alarm(10) for a 10 second timeout
& doing a gettimeofday call before & after each read to see
how long the replies took, & writing some text to stdout so the user
has an idea what is going on.
socket(PF_INET, SOCK_RAW, IPPROTO_ICMP) = 3
getuid() = 0
setuid(0) = 0
stat("/usr/share/locale/C/libc.cat", 0xbffff134) = -1 ENOENT (No such file or directory)
stat("/usr/share/locale/libc/C", 0xbffff134) = -1 ENOENT (No such file or directory)
stat("/usr/local/share/locale/C/libc.cat", 0xbffff134) = -1 ENOENT (No such file or directory)
getpid() = 353
setsockopt(3, SOL_SOCKET, SO_BROADCAST, [1], 4) = 0
setsockopt(3, SOL_SOCKET, SO_RCVBUF, [49152], 4) = 0
fstat(1, {st_mode=S_IFCHR|0620, st_rdev=makedev(3, 1), ...}) = 0
mmap(0, 4096, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0x40008000
ioctl(1, TCGETS, {B9600 opost isig icanon echo ...}) = 0
write(1, "PING 127.0.0.1 (127.0.0.1): 56 d"..., 42PING 127.0.0.1 (127.0.0.1): 56 data bytes
) = 42
sigaction(SIGINT, {0x8049ba0, [], SA_RESTART}, {SIG_DFL}) = 0
sigaction(SIGALRM, {0x8049600, [], SA_RESTART}, {SIG_DFL}) = 0
gettimeofday({948904719, 138951}, NULL) = 0
sendto(3, "\10\0D\201a\1\0\0\17#\2178\307\36"..., 64, 0, {sin_family=AF_INET,
sin_port=htons(0), sin_addr=inet_addr("127.0.0.1")}, 16) = 64
sigaction(SIGALRM, {0x8049600, [], SA_RESTART}, {0x8049600, [], SA_RESTART}) = 0
sigaction(SIGALRM, {0x8049ba0, [], SA_RESTART}, {0x8049600, [], SA_RESTART}) = 0
alarm(10) = 0
recvfrom(3, "E\0\0T\0005\0\0@\1|r\177\0\0\1\177"..., 192, 0,
{sin_family=AF_INET, sin_port=htons(50882), sin_addr=inet_addr("127.0.0.1")}, [16]) = 84
gettimeofday({948904719, 160224}, NULL) = 0
recvfrom(3, "E\0\0T\0006\0\0\377\1\275p\177\0"..., 192, 0,
{sin_family=AF_INET, sin_port=htons(50882), sin_addr=inet_addr("127.0.0.1")}, [16]) = 84
gettimeofday({948904719, 166952}, NULL) = 0
write(1, "64 bytes from 127.0.0.1: icmp_se"...,
5764 bytes from 127.0.0.1: icmp_seq=0 ttl=255 time=28.0 ms
Example 2
---------
strace passwd 2>&1 | grep open
produces the following output
open("/etc/ld.so.cache", O_RDONLY) = 3
open("/opt/kde/lib/libc.so.5", O_RDONLY) = -1 ENOENT (No such file or directory)
open("/lib/libc.so.5", O_RDONLY) = 3
open("/dev", O_RDONLY) = 3
open("/var/run/utmp", O_RDONLY) = 3
open("/etc/passwd", O_RDONLY) = 3
open("/etc/shadow", O_RDONLY) = 3
open("/etc/login.defs", O_RDONLY) = 4
open("/dev/tty", O_RDONLY) = 4
The 2>&1 is done to redirect stderr to stdout & grep is then filtering this input
through the pipe for each line containing the string open.
Example 3
---------
Getting sophisticated
telnetd crashes & I don't know why
Steps
-----
1) Replace the following line in /etc/inetd.conf
telnet stream tcp nowait root /usr/sbin/in.telnetd -h
with
telnet stream tcp nowait root /blah
2) Create the file /blah with the following contents to start tracing telnetd
#!/bin/bash
/usr/bin/strace -o/t1 -f /usr/sbin/in.telnetd -h
3) chmod 700 /blah to make it executable only to root
4)
killall -HUP inetd
or ps aux | grep inetd
get inetd's process id
& kill -HUP inetd to restart it.
Important options
-----------------
-o is used to tell strace to output to a file in our case t1 in the root directory
-f is to follow children i.e.
e.g in our case above telnetd will start the login process & subsequently a shell like bash.
You will be able to tell which is which from the process ID's listed on the left hand side
of the strace output.
-p<pid> will tell strace to attach to a running process, yup this can be done provided
it isn't being traced or debugged already & you have enough privileges,
the reason 2 processes cannot trace or debug the same program is that strace
becomes the parent process of the one being debugged & processes ( unlike people )
can have only one parent.
However the file /t1 will get big quite quickly
to test it telnet 127.0.0.1
now look at what files in.telnetd execve'd
413 execve("/usr/sbin/in.telnetd", ["/usr/sbin/in.telnetd", "-h"], [/* 17 vars */]) = 0
414 execve("/bin/login", ["/bin/login", "-h", "localhost", "-p"], [/* 2 vars */]) = 0
Whey it worked!.
Other hints:
------------
If the program is not very interactive ( i.e. not much keyboard input )
& is crashing in one architecture but not in another you can do
an strace of both programs under as identical a scenario as you can
on both architectures outputting to a file then.
do a diff of the two traces using the diff program
i.e.
diff output1 output2
& maybe you'll be able to see where the call paths differed, this
is possibly near the cause of the crash.
More info
---------
Look at man pages for strace & the various syscalls
e.g. man strace, man alarm, man socket.
Performance Debugging
=====================
gcc is capable of compiling in profiling code just add the -p option
to the CFLAGS, this obviously affects program size & performance.
This can be used by the gprof gnu profiling tool or the
gcov the gnu code coverage tool ( code coverage is a means of testing
code quality by checking if all the code in an executable in exercised by
a tester ).
Using top to find out where processes are sleeping in the kernel
----------------------------------------------------------------
To do this copy the System.map from the root directory where
the linux kernel was built to the /boot directory on your
linux machine.
Start top
Now type fU<return>
You should see a new field called WCHAN which
tells you where each process is sleeping here is a typical output.
6:59pm up 41 min, 1 user, load average: 0.00, 0.00, 0.00
28 processes: 27 sleeping, 1 running, 0 zombie, 0 stopped
CPU states: 0.0% user, 0.1% system, 0.0% nice, 99.8% idle
Mem: 254900K av, 45976K used, 208924K free, 0K shrd, 28636K buff
Swap: 0K av, 0K used, 0K free 8620K cached
PID USER PRI NI SIZE RSS SHARE WCHAN STAT LIB %CPU %MEM TIME COMMAND
750 root 12 0 848 848 700 do_select S 0 0.1 0.3 0:00 in.telnetd
767 root 16 0 1140 1140 964 R 0 0.1 0.4 0:00 top
1 root 8 0 212 212 180 do_select S 0 0.0 0.0 0:00 init
2 root 9 0 0 0 0 down_inte SW 0 0.0 0.0 0:00 kmcheck
The time command
----------------
Another related command is the time command which gives you an indication
of where a process is spending the majority of its time.
e.g.
time ping -c 5 nc
outputs
real 0m4.054s
user 0m0.010s
sys 0m0.010s
Debugging under VM Debugging under VM
================== ==================

View file

@ -11,201 +11,29 @@
#include <linux/types.h> #include <linux/types.h>
#include <linux/bug.h> #include <linux/bug.h>
extern void __xchg_called_with_bad_pointer(void); #define cmpxchg(ptr, o, n) \
({ \
static inline unsigned long __xchg(unsigned long x, void *ptr, int size) __typeof__(*(ptr)) __o = (o); \
{ __typeof__(*(ptr)) __n = (n); \
unsigned long addr, old; (__typeof__(*(ptr))) __sync_val_compare_and_swap((ptr),__o,__n);\
int shift;
switch (size) {
case 1:
addr = (unsigned long) ptr;
shift = (3 ^ (addr & 3)) << 3;
addr ^= addr & 3;
asm volatile(
" l %0,%4\n"
"0: lr 0,%0\n"
" nr 0,%3\n"
" or 0,%2\n"
" cs %0,0,%4\n"
" jl 0b\n"
: "=&d" (old), "=Q" (*(int *) addr)
: "d" ((x & 0xff) << shift), "d" (~(0xff << shift)),
"Q" (*(int *) addr) : "memory", "cc", "0");
return old >> shift;
case 2:
addr = (unsigned long) ptr;
shift = (2 ^ (addr & 2)) << 3;
addr ^= addr & 2;
asm volatile(
" l %0,%4\n"
"0: lr 0,%0\n"
" nr 0,%3\n"
" or 0,%2\n"
" cs %0,0,%4\n"
" jl 0b\n"
: "=&d" (old), "=Q" (*(int *) addr)
: "d" ((x & 0xffff) << shift), "d" (~(0xffff << shift)),
"Q" (*(int *) addr) : "memory", "cc", "0");
return old >> shift;
case 4:
asm volatile(
" l %0,%3\n"
"0: cs %0,%2,%3\n"
" jl 0b\n"
: "=&d" (old), "=Q" (*(int *) ptr)
: "d" (x), "Q" (*(int *) ptr)
: "memory", "cc");
return old;
#ifdef CONFIG_64BIT
case 8:
asm volatile(
" lg %0,%3\n"
"0: csg %0,%2,%3\n"
" jl 0b\n"
: "=&d" (old), "=m" (*(long *) ptr)
: "d" (x), "Q" (*(long *) ptr)
: "memory", "cc");
return old;
#endif /* CONFIG_64BIT */
}
__xchg_called_with_bad_pointer();
return x;
}
#define xchg(ptr, x) \
({ \
__typeof__(*(ptr)) __ret; \
__ret = (__typeof__(*(ptr))) \
__xchg((unsigned long)(x), (void *)(ptr), sizeof(*(ptr)));\
__ret; \
}) })
/* #define cmpxchg64 cmpxchg
* Atomic compare and exchange. Compare OLD with MEM, if identical, #define cmpxchg_local cmpxchg
* store NEW in MEM. Return the initial value in MEM. Success is #define cmpxchg64_local cmpxchg
* indicated by comparing RETURN with OLD.
*/ #define xchg(ptr, x) \
({ \
__typeof__(ptr) __ptr = (ptr); \
__typeof__(*(ptr)) __old; \
do { \
__old = *__ptr; \
} while (!__sync_bool_compare_and_swap(__ptr, __old, x)); \
__old; \
})
#define __HAVE_ARCH_CMPXCHG #define __HAVE_ARCH_CMPXCHG
extern void __cmpxchg_called_with_bad_pointer(void);
static inline unsigned long __cmpxchg(void *ptr, unsigned long old,
unsigned long new, int size)
{
unsigned long addr, prev, tmp;
int shift;
switch (size) {
case 1:
addr = (unsigned long) ptr;
shift = (3 ^ (addr & 3)) << 3;
addr ^= addr & 3;
asm volatile(
" l %0,%2\n"
"0: nr %0,%5\n"
" lr %1,%0\n"
" or %0,%3\n"
" or %1,%4\n"
" cs %0,%1,%2\n"
" jnl 1f\n"
" xr %1,%0\n"
" nr %1,%5\n"
" jnz 0b\n"
"1:"
: "=&d" (prev), "=&d" (tmp), "+Q" (*(int *) addr)
: "d" ((old & 0xff) << shift),
"d" ((new & 0xff) << shift),
"d" (~(0xff << shift))
: "memory", "cc");
return prev >> shift;
case 2:
addr = (unsigned long) ptr;
shift = (2 ^ (addr & 2)) << 3;
addr ^= addr & 2;
asm volatile(
" l %0,%2\n"
"0: nr %0,%5\n"
" lr %1,%0\n"
" or %0,%3\n"
" or %1,%4\n"
" cs %0,%1,%2\n"
" jnl 1f\n"
" xr %1,%0\n"
" nr %1,%5\n"
" jnz 0b\n"
"1:"
: "=&d" (prev), "=&d" (tmp), "+Q" (*(int *) addr)
: "d" ((old & 0xffff) << shift),
"d" ((new & 0xffff) << shift),
"d" (~(0xffff << shift))
: "memory", "cc");
return prev >> shift;
case 4:
asm volatile(
" cs %0,%3,%1\n"
: "=&d" (prev), "=Q" (*(int *) ptr)
: "0" (old), "d" (new), "Q" (*(int *) ptr)
: "memory", "cc");
return prev;
#ifdef CONFIG_64BIT
case 8:
asm volatile(
" csg %0,%3,%1\n"
: "=&d" (prev), "=Q" (*(long *) ptr)
: "0" (old), "d" (new), "Q" (*(long *) ptr)
: "memory", "cc");
return prev;
#endif /* CONFIG_64BIT */
}
__cmpxchg_called_with_bad_pointer();
return old;
}
#define cmpxchg(ptr, o, n) \
({ \
__typeof__(*(ptr)) __ret; \
__ret = (__typeof__(*(ptr))) \
__cmpxchg((ptr), (unsigned long)(o), (unsigned long)(n), \
sizeof(*(ptr))); \
__ret; \
})
#ifdef CONFIG_64BIT
#define cmpxchg64(ptr, o, n) \
({ \
cmpxchg((ptr), (o), (n)); \
})
#else /* CONFIG_64BIT */
static inline unsigned long long __cmpxchg64(void *ptr,
unsigned long long old,
unsigned long long new)
{
register_pair rp_old = {.pair = old};
register_pair rp_new = {.pair = new};
unsigned long long *ullptr = ptr;
asm volatile(
" cds %0,%2,%1"
: "+d" (rp_old), "+Q" (*ullptr)
: "d" (rp_new)
: "memory", "cc");
return rp_old.pair;
}
#define cmpxchg64(ptr, o, n) \
({ \
__typeof__(*(ptr)) __ret; \
__ret = (__typeof__(*(ptr))) \
__cmpxchg64((ptr), \
(unsigned long long)(o), \
(unsigned long long)(n)); \
__ret; \
})
#endif /* CONFIG_64BIT */
#define __cmpxchg_double_op(p1, p2, o1, o2, n1, n2, insn) \ #define __cmpxchg_double_op(p1, p2, o1, o2, n1, n2, insn) \
({ \ ({ \
register __typeof__(*(p1)) __old1 asm("2") = (o1); \ register __typeof__(*(p1)) __old1 asm("2") = (o1); \
@ -265,40 +93,4 @@ extern void __cmpxchg_double_called_with_bad_pointer(void);
#define system_has_cmpxchg_double() 1 #define system_has_cmpxchg_double() 1
#include <asm-generic/cmpxchg-local.h>
static inline unsigned long __cmpxchg_local(void *ptr,
unsigned long old,
unsigned long new, int size)
{
switch (size) {
case 1:
case 2:
case 4:
#ifdef CONFIG_64BIT
case 8:
#endif
return __cmpxchg(ptr, old, new, size);
default:
return __cmpxchg_local_generic(ptr, old, new, size);
}
return old;
}
/*
* cmpxchg_local and cmpxchg64_local are atomic wrt current CPU. Always make
* them available.
*/
#define cmpxchg_local(ptr, o, n) \
({ \
__typeof__(*(ptr)) __ret; \
__ret = (__typeof__(*(ptr))) \
__cmpxchg_local((ptr), (unsigned long)(o), \
(unsigned long)(n), sizeof(*(ptr))); \
__ret; \
})
#define cmpxchg64_local(ptr, o, n) cmpxchg64((ptr), (o), (n))
#endif /* __ASM_CMPXCHG_H */ #endif /* __ASM_CMPXCHG_H */

View file

@ -10,6 +10,8 @@
#include <linux/types.h> #include <linux/types.h>
#include <asm/div64.h> #include <asm/div64.h>
#define CPUTIME_PER_USEC 4096ULL
#define CPUTIME_PER_SEC (CPUTIME_PER_USEC * USEC_PER_SEC)
/* We want to use full resolution of the CPU timer: 2**-12 micro-seconds. */ /* We want to use full resolution of the CPU timer: 2**-12 micro-seconds. */
@ -38,24 +40,24 @@ static inline unsigned long __div(unsigned long long n, unsigned long base)
*/ */
static inline unsigned long cputime_to_jiffies(const cputime_t cputime) static inline unsigned long cputime_to_jiffies(const cputime_t cputime)
{ {
return __div((__force unsigned long long) cputime, 4096000000ULL / HZ); return __div((__force unsigned long long) cputime, CPUTIME_PER_SEC / HZ);
} }
static inline cputime_t jiffies_to_cputime(const unsigned int jif) static inline cputime_t jiffies_to_cputime(const unsigned int jif)
{ {
return (__force cputime_t)(jif * (4096000000ULL / HZ)); return (__force cputime_t)(jif * (CPUTIME_PER_SEC / HZ));
} }
static inline u64 cputime64_to_jiffies64(cputime64_t cputime) static inline u64 cputime64_to_jiffies64(cputime64_t cputime)
{ {
unsigned long long jif = (__force unsigned long long) cputime; unsigned long long jif = (__force unsigned long long) cputime;
do_div(jif, 4096000000ULL / HZ); do_div(jif, CPUTIME_PER_SEC / HZ);
return jif; return jif;
} }
static inline cputime64_t jiffies64_to_cputime64(const u64 jif) static inline cputime64_t jiffies64_to_cputime64(const u64 jif)
{ {
return (__force cputime64_t)(jif * (4096000000ULL / HZ)); return (__force cputime64_t)(jif * (CPUTIME_PER_SEC / HZ));
} }
/* /*
@ -68,7 +70,7 @@ static inline unsigned int cputime_to_usecs(const cputime_t cputime)
static inline cputime_t usecs_to_cputime(const unsigned int m) static inline cputime_t usecs_to_cputime(const unsigned int m)
{ {
return (__force cputime_t)(m * 4096ULL); return (__force cputime_t)(m * CPUTIME_PER_USEC);
} }
#define usecs_to_cputime64(m) usecs_to_cputime(m) #define usecs_to_cputime64(m) usecs_to_cputime(m)
@ -78,12 +80,12 @@ static inline cputime_t usecs_to_cputime(const unsigned int m)
*/ */
static inline unsigned int cputime_to_secs(const cputime_t cputime) static inline unsigned int cputime_to_secs(const cputime_t cputime)
{ {
return __div((__force unsigned long long) cputime, 2048000000) >> 1; return __div((__force unsigned long long) cputime, CPUTIME_PER_SEC / 2) >> 1;
} }
static inline cputime_t secs_to_cputime(const unsigned int s) static inline cputime_t secs_to_cputime(const unsigned int s)
{ {
return (__force cputime_t)(s * 4096000000ULL); return (__force cputime_t)(s * CPUTIME_PER_SEC);
} }
/* /*
@ -91,8 +93,8 @@ static inline cputime_t secs_to_cputime(const unsigned int s)
*/ */
static inline cputime_t timespec_to_cputime(const struct timespec *value) static inline cputime_t timespec_to_cputime(const struct timespec *value)
{ {
unsigned long long ret = value->tv_sec * 4096000000ULL; unsigned long long ret = value->tv_sec * CPUTIME_PER_SEC;
return (__force cputime_t)(ret + value->tv_nsec * 4096 / 1000); return (__force cputime_t)(ret + __div(value->tv_nsec * CPUTIME_PER_USEC, NSEC_PER_USEC));
} }
static inline void cputime_to_timespec(const cputime_t cputime, static inline void cputime_to_timespec(const cputime_t cputime,
@ -103,12 +105,12 @@ static inline void cputime_to_timespec(const cputime_t cputime,
register_pair rp; register_pair rp;
rp.pair = __cputime >> 1; rp.pair = __cputime >> 1;
asm ("dr %0,%1" : "+d" (rp) : "d" (2048000000UL)); asm ("dr %0,%1" : "+d" (rp) : "d" (CPUTIME_PER_SEC / 2));
value->tv_nsec = rp.subreg.even * 1000 / 4096; value->tv_nsec = rp.subreg.even * NSEC_PER_USEC / CPUTIME_PER_USEC;
value->tv_sec = rp.subreg.odd; value->tv_sec = rp.subreg.odd;
#else #else
value->tv_nsec = (__cputime % 4096000000ULL) * 1000 / 4096; value->tv_nsec = (__cputime % CPUTIME_PER_SEC) * NSEC_PER_USEC / CPUTIME_PER_USEC;
value->tv_sec = __cputime / 4096000000ULL; value->tv_sec = __cputime / CPUTIME_PER_SEC;
#endif #endif
} }
@ -119,8 +121,8 @@ static inline void cputime_to_timespec(const cputime_t cputime,
*/ */
static inline cputime_t timeval_to_cputime(const struct timeval *value) static inline cputime_t timeval_to_cputime(const struct timeval *value)
{ {
unsigned long long ret = value->tv_sec * 4096000000ULL; unsigned long long ret = value->tv_sec * CPUTIME_PER_SEC;
return (__force cputime_t)(ret + value->tv_usec * 4096ULL); return (__force cputime_t)(ret + value->tv_usec * CPUTIME_PER_USEC);
} }
static inline void cputime_to_timeval(const cputime_t cputime, static inline void cputime_to_timeval(const cputime_t cputime,
@ -131,12 +133,12 @@ static inline void cputime_to_timeval(const cputime_t cputime,
register_pair rp; register_pair rp;
rp.pair = __cputime >> 1; rp.pair = __cputime >> 1;
asm ("dr %0,%1" : "+d" (rp) : "d" (2048000000UL)); asm ("dr %0,%1" : "+d" (rp) : "d" (CPUTIME_PER_USEC / 2));
value->tv_usec = rp.subreg.even / 4096; value->tv_usec = rp.subreg.even / CPUTIME_PER_USEC;
value->tv_sec = rp.subreg.odd; value->tv_sec = rp.subreg.odd;
#else #else
value->tv_usec = (__cputime % 4096000000ULL) / 4096; value->tv_usec = (__cputime % CPUTIME_PER_SEC) / CPUTIME_PER_USEC;
value->tv_sec = __cputime / 4096000000ULL; value->tv_sec = __cputime / CPUTIME_PER_SEC;
#endif #endif
} }
@ -146,13 +148,13 @@ static inline void cputime_to_timeval(const cputime_t cputime,
static inline clock_t cputime_to_clock_t(cputime_t cputime) static inline clock_t cputime_to_clock_t(cputime_t cputime)
{ {
unsigned long long clock = (__force unsigned long long) cputime; unsigned long long clock = (__force unsigned long long) cputime;
do_div(clock, 4096000000ULL / USER_HZ); do_div(clock, CPUTIME_PER_SEC / USER_HZ);
return clock; return clock;
} }
static inline cputime_t clock_t_to_cputime(unsigned long x) static inline cputime_t clock_t_to_cputime(unsigned long x)
{ {
return (__force cputime_t)(x * (4096000000ULL / USER_HZ)); return (__force cputime_t)(x * (CPUTIME_PER_SEC / USER_HZ));
} }
/* /*
@ -161,7 +163,7 @@ static inline cputime_t clock_t_to_cputime(unsigned long x)
static inline clock_t cputime64_to_clock_t(cputime64_t cputime) static inline clock_t cputime64_to_clock_t(cputime64_t cputime)
{ {
unsigned long long clock = (__force unsigned long long) cputime; unsigned long long clock = (__force unsigned long long) cputime;
do_div(clock, 4096000000ULL / USER_HZ); do_div(clock, CPUTIME_PER_SEC / USER_HZ);
return clock; return clock;
} }

View file

@ -151,9 +151,21 @@ debug_text_event(debug_info_t* id, int level, const char* txt)
* stored in the s390dbf. See Documentation/s390/s390dbf.txt for more details! * stored in the s390dbf. See Documentation/s390/s390dbf.txt for more details!
*/ */
extern debug_entry_t * extern debug_entry_t *
debug_sprintf_event(debug_info_t* id,int level,char *string,...) __debug_sprintf_event(debug_info_t *id, int level, char *string, ...)
__attribute__ ((format(printf, 3, 4))); __attribute__ ((format(printf, 3, 4)));
#define debug_sprintf_event(_id, _level, _fmt, ...) \
({ \
debug_entry_t *__ret; \
debug_info_t *__id = _id; \
int __level = _level; \
if ((!__id) || (__level > __id->level)) \
__ret = NULL; \
else \
__ret = __debug_sprintf_event(__id, __level, \
_fmt, ## __VA_ARGS__); \
__ret; \
})
static inline debug_entry_t* static inline debug_entry_t*
debug_exception(debug_info_t* id, int level, void* data, int length) debug_exception(debug_info_t* id, int level, void* data, int length)
@ -194,9 +206,22 @@ debug_text_exception(debug_info_t* id, int level, const char* txt)
* stored in the s390dbf. See Documentation/s390/s390dbf.txt for more details! * stored in the s390dbf. See Documentation/s390/s390dbf.txt for more details!
*/ */
extern debug_entry_t * extern debug_entry_t *
debug_sprintf_exception(debug_info_t* id,int level,char *string,...) __debug_sprintf_exception(debug_info_t *id, int level, char *string, ...)
__attribute__ ((format(printf, 3, 4))); __attribute__ ((format(printf, 3, 4)));
#define debug_sprintf_exception(_id, _level, _fmt, ...) \
({ \
debug_entry_t *__ret; \
debug_info_t *__id = _id; \
int __level = _level; \
if ((!__id) || (__level > __id->level)) \
__ret = NULL; \
else \
__ret = __debug_sprintf_exception(__id, __level, \
_fmt, ## __VA_ARGS__);\
__ret; \
})
int debug_register_view(debug_info_t* id, struct debug_view* view); int debug_register_view(debug_info_t* id, struct debug_view* view);
int debug_unregister_view(debug_info_t* id, struct debug_view* view); int debug_unregister_view(debug_info_t* id, struct debug_view* view);

View file

@ -1,25 +1,69 @@
#ifndef _ASM_S390_FTRACE_H #ifndef _ASM_S390_FTRACE_H
#define _ASM_S390_FTRACE_H #define _ASM_S390_FTRACE_H
#define ARCH_SUPPORTS_FTRACE_OPS 1
#define MCOUNT_INSN_SIZE 24
#define MCOUNT_RETURN_FIXUP 18
#ifndef __ASSEMBLY__ #ifndef __ASSEMBLY__
extern void _mcount(void); #define ftrace_return_address(n) __builtin_return_address(n)
void _mcount(void);
void ftrace_caller(void);
extern char ftrace_graph_caller_end; extern char ftrace_graph_caller_end;
extern unsigned long ftrace_plt;
struct dyn_arch_ftrace { }; struct dyn_arch_ftrace { };
#define MCOUNT_ADDR ((long)_mcount) #define MCOUNT_ADDR ((unsigned long)_mcount)
#define FTRACE_ADDR ((unsigned long)ftrace_caller)
#define KPROBE_ON_FTRACE_NOP 0
#define KPROBE_ON_FTRACE_CALL 1
static inline unsigned long ftrace_call_adjust(unsigned long addr) static inline unsigned long ftrace_call_adjust(unsigned long addr)
{ {
return addr; return addr;
} }
struct ftrace_insn {
u16 opc;
s32 disp;
} __packed;
static inline void ftrace_generate_nop_insn(struct ftrace_insn *insn)
{
#ifdef CONFIG_FUNCTION_TRACER
/* jg .+24 */
insn->opc = 0xc0f4;
insn->disp = MCOUNT_INSN_SIZE / 2;
#endif
}
static inline int is_ftrace_nop(struct ftrace_insn *insn)
{
#ifdef CONFIG_FUNCTION_TRACER
if (insn->disp == MCOUNT_INSN_SIZE / 2)
return 1;
#endif
return 0;
}
static inline void ftrace_generate_call_insn(struct ftrace_insn *insn,
unsigned long ip)
{
#ifdef CONFIG_FUNCTION_TRACER
unsigned long target;
/* brasl r0,ftrace_caller */
target = is_module_addr((void *) ip) ? ftrace_plt : FTRACE_ADDR;
insn->opc = 0xc005;
insn->disp = (target - ip) / 2;
#endif
}
#endif /* __ASSEMBLY__ */ #endif /* __ASSEMBLY__ */
#define MCOUNT_INSN_SIZE 18
#define ARCH_SUPPORTS_FTRACE_OPS 1
#endif /* _ASM_S390_FTRACE_H */ #endif /* _ASM_S390_FTRACE_H */

View file

@ -9,9 +9,10 @@
#include <linux/types.h> #include <linux/types.h>
#include <linux/device.h> #include <linux/device.h>
#include <linux/seqlock.h>
struct s390_idle_data { struct s390_idle_data {
unsigned int sequence; seqcount_t seqcount;
unsigned long long idle_count; unsigned long long idle_count;
unsigned long long idle_time; unsigned long long idle_time;
unsigned long long clock_idle_enter; unsigned long long clock_idle_enter;

View file

@ -39,6 +39,15 @@ static inline void iounmap(volatile void __iomem *addr)
{ {
} }
static inline void __iomem *ioport_map(unsigned long port, unsigned int nr)
{
return NULL;
}
static inline void ioport_unmap(void __iomem *p)
{
}
/* /*
* s390 needs a private implementation of pci_iomap since ioremap with its * s390 needs a private implementation of pci_iomap since ioremap with its
* offset parameter isn't sufficient. That's because BAR spaces are not * offset parameter isn't sufficient. That's because BAR spaces are not

View file

@ -1,11 +1,11 @@
#ifndef _ASM_IRQ_H #ifndef _ASM_IRQ_H
#define _ASM_IRQ_H #define _ASM_IRQ_H
#define EXT_INTERRUPT 1 #define EXT_INTERRUPT 0
#define IO_INTERRUPT 2 #define IO_INTERRUPT 1
#define THIN_INTERRUPT 3 #define THIN_INTERRUPT 2
#define NR_IRQS_BASE 4 #define NR_IRQS_BASE 3
#ifdef CONFIG_PCI_NR_MSI #ifdef CONFIG_PCI_NR_MSI
# define NR_IRQS (NR_IRQS_BASE + CONFIG_PCI_NR_MSI) # define NR_IRQS (NR_IRQS_BASE + CONFIG_PCI_NR_MSI)
@ -13,9 +13,6 @@
# define NR_IRQS NR_IRQS_BASE # define NR_IRQS NR_IRQS_BASE
#endif #endif
/* This number is used when no interrupt has been assigned */
#define NO_IRQ 0
/* External interruption codes */ /* External interruption codes */
#define EXT_IRQ_INTERRUPT_KEY 0x0040 #define EXT_IRQ_INTERRUPT_KEY 0x0040
#define EXT_IRQ_CLK_COMP 0x1004 #define EXT_IRQ_CLK_COMP 0x1004

View file

@ -60,6 +60,7 @@ typedef u16 kprobe_opcode_t;
struct arch_specific_insn { struct arch_specific_insn {
/* copy of original instruction */ /* copy of original instruction */
kprobe_opcode_t *insn; kprobe_opcode_t *insn;
unsigned int is_ftrace_insn : 1;
}; };
struct prev_kprobe { struct prev_kprobe {

View file

@ -147,7 +147,7 @@ struct _lowcore {
__u32 softirq_pending; /* 0x02ec */ __u32 softirq_pending; /* 0x02ec */
__u32 percpu_offset; /* 0x02f0 */ __u32 percpu_offset; /* 0x02f0 */
__u32 machine_flags; /* 0x02f4 */ __u32 machine_flags; /* 0x02f4 */
__u32 ftrace_func; /* 0x02f8 */ __u8 pad_0x02f8[0x02fc-0x02f8]; /* 0x02f8 */
__u32 spinlock_lockval; /* 0x02fc */ __u32 spinlock_lockval; /* 0x02fc */
__u8 pad_0x0300[0x0e00-0x0300]; /* 0x0300 */ __u8 pad_0x0300[0x0e00-0x0300]; /* 0x0300 */
@ -297,7 +297,7 @@ struct _lowcore {
__u64 percpu_offset; /* 0x0378 */ __u64 percpu_offset; /* 0x0378 */
__u64 vdso_per_cpu_data; /* 0x0380 */ __u64 vdso_per_cpu_data; /* 0x0380 */
__u64 machine_flags; /* 0x0388 */ __u64 machine_flags; /* 0x0388 */
__u64 ftrace_func; /* 0x0390 */ __u8 pad_0x0390[0x0398-0x0390]; /* 0x0390 */
__u64 gmap; /* 0x0398 */ __u64 gmap; /* 0x0398 */
__u32 spinlock_lockval; /* 0x03a0 */ __u32 spinlock_lockval; /* 0x03a0 */
__u8 pad_0x03a0[0x0400-0x03a4]; /* 0x03a4 */ __u8 pad_0x03a0[0x0400-0x03a4]; /* 0x03a4 */

View file

@ -50,10 +50,6 @@ struct zpci_fmb {
atomic64_t unmapped_pages; atomic64_t unmapped_pages;
} __packed __aligned(16); } __packed __aligned(16);
#define ZPCI_MSI_VEC_BITS 11
#define ZPCI_MSI_VEC_MAX (1 << ZPCI_MSI_VEC_BITS)
#define ZPCI_MSI_VEC_MASK (ZPCI_MSI_VEC_MAX - 1)
enum zpci_state { enum zpci_state {
ZPCI_FN_STATE_RESERVED, ZPCI_FN_STATE_RESERVED,
ZPCI_FN_STATE_STANDBY, ZPCI_FN_STATE_STANDBY,
@ -90,6 +86,7 @@ struct zpci_dev {
/* IRQ stuff */ /* IRQ stuff */
u64 msi_addr; /* MSI address */ u64 msi_addr; /* MSI address */
unsigned int max_msi; /* maximum number of MSI's */
struct airq_iv *aibv; /* adapter interrupt bit vector */ struct airq_iv *aibv; /* adapter interrupt bit vector */
unsigned int aisb; /* number of the summary bit */ unsigned int aisb; /* number of the summary bit */

View file

@ -139,7 +139,8 @@ static inline int zpci_memcpy_fromio(void *dst,
int size, rc = 0; int size, rc = 0;
while (n > 0) { while (n > 0) {
size = zpci_get_max_write_size((u64) src, (u64) dst, n, 8); size = zpci_get_max_write_size((u64 __force) src,
(u64) dst, n, 8);
req = ZPCI_CREATE_REQ(entry->fh, entry->bar, size); req = ZPCI_CREATE_REQ(entry->fh, entry->bar, size);
rc = zpci_read_single(req, dst, offset, size); rc = zpci_read_single(req, dst, offset, size);
if (rc) if (rc)
@ -162,7 +163,8 @@ static inline int zpci_memcpy_toio(volatile void __iomem *dst,
return -EINVAL; return -EINVAL;
while (n > 0) { while (n > 0) {
size = zpci_get_max_write_size((u64) dst, (u64) src, n, 128); size = zpci_get_max_write_size((u64 __force) dst,
(u64) src, n, 128);
req = ZPCI_CREATE_REQ(entry->fh, entry->bar, size); req = ZPCI_CREATE_REQ(entry->fh, entry->bar, size);
if (size > 8) /* main path */ if (size > 8) /* main path */

View file

@ -22,8 +22,6 @@ unsigned long *page_table_alloc(struct mm_struct *);
void page_table_free(struct mm_struct *, unsigned long *); void page_table_free(struct mm_struct *, unsigned long *);
void page_table_free_rcu(struct mmu_gather *, unsigned long *, unsigned long); void page_table_free_rcu(struct mmu_gather *, unsigned long *, unsigned long);
void page_table_reset_pgste(struct mm_struct *, unsigned long, unsigned long,
bool init_skey);
int set_guest_storage_key(struct mm_struct *mm, unsigned long addr, int set_guest_storage_key(struct mm_struct *mm, unsigned long addr,
unsigned long key, bool nq); unsigned long key, bool nq);

View file

@ -133,6 +133,18 @@ extern unsigned long MODULES_END;
#define MODULES_LEN (1UL << 31) #define MODULES_LEN (1UL << 31)
#endif #endif
static inline int is_module_addr(void *addr)
{
#ifdef CONFIG_64BIT
BUILD_BUG_ON(MODULES_LEN > (1UL << 31));
if (addr < (void *)MODULES_VADDR)
return 0;
if (addr > (void *)MODULES_END)
return 0;
#endif
return 1;
}
/* /*
* A 31 bit pagetable entry of S390 has following format: * A 31 bit pagetable entry of S390 has following format:
* | PFRA | | OS | * | PFRA | | OS |
@ -479,6 +491,11 @@ static inline int mm_has_pgste(struct mm_struct *mm)
return 0; return 0;
} }
/*
* In the case that a guest uses storage keys
* faults should no longer be backed by zero pages
*/
#define mm_forbids_zeropage mm_use_skey
static inline int mm_use_skey(struct mm_struct *mm) static inline int mm_use_skey(struct mm_struct *mm)
{ {
#ifdef CONFIG_PGSTE #ifdef CONFIG_PGSTE
@ -1634,6 +1651,19 @@ static inline pmd_t pmdp_get_and_clear(struct mm_struct *mm,
return pmd; return pmd;
} }
#define __HAVE_ARCH_PMDP_GET_AND_CLEAR_FULL
static inline pmd_t pmdp_get_and_clear_full(struct mm_struct *mm,
unsigned long address,
pmd_t *pmdp, int full)
{
pmd_t pmd = *pmdp;
if (!full)
pmdp_flush_lazy(mm, address, pmdp);
pmd_clear(pmdp);
return pmd;
}
#define __HAVE_ARCH_PMDP_CLEAR_FLUSH #define __HAVE_ARCH_PMDP_CLEAR_FLUSH
static inline pmd_t pmdp_clear_flush(struct vm_area_struct *vma, static inline pmd_t pmdp_clear_flush(struct vm_area_struct *vma,
unsigned long address, pmd_t *pmdp) unsigned long address, pmd_t *pmdp)
@ -1746,7 +1776,8 @@ static inline pte_t mk_swap_pte(unsigned long type, unsigned long offset)
extern int vmem_add_mapping(unsigned long start, unsigned long size); extern int vmem_add_mapping(unsigned long start, unsigned long size);
extern int vmem_remove_mapping(unsigned long start, unsigned long size); extern int vmem_remove_mapping(unsigned long start, unsigned long size);
extern int s390_enable_sie(void); extern int s390_enable_sie(void);
extern void s390_enable_skey(void); extern int s390_enable_skey(void);
extern void s390_reset_cmma(struct mm_struct *mm);
/* /*
* No page table caches to initialise * No page table caches to initialise

View file

@ -217,8 +217,6 @@ static inline unsigned short stap(void)
*/ */
static inline void cpu_relax(void) static inline void cpu_relax(void)
{ {
if (MACHINE_HAS_DIAG44)
asm volatile("diag 0,0,68");
barrier(); barrier();
} }

View file

@ -18,14 +18,7 @@ extern int spin_retry;
static inline int static inline int
_raw_compare_and_swap(unsigned int *lock, unsigned int old, unsigned int new) _raw_compare_and_swap(unsigned int *lock, unsigned int old, unsigned int new)
{ {
unsigned int old_expected = old; return __sync_bool_compare_and_swap(lock, old, new);
asm volatile(
" cs %0,%3,%1"
: "=d" (old), "=Q" (*lock)
: "0" (old), "d" (new), "Q" (*lock)
: "cc", "memory" );
return old == old_expected;
} }
/* /*

View file

@ -121,6 +121,7 @@ static inline void pmd_free_tlb(struct mmu_gather *tlb, pmd_t *pmd,
#ifdef CONFIG_64BIT #ifdef CONFIG_64BIT
if (tlb->mm->context.asce_limit <= (1UL << 31)) if (tlb->mm->context.asce_limit <= (1UL << 31))
return; return;
pgtable_pmd_page_dtor(virt_to_page(pmd));
tlb_remove_table(tlb, pmd); tlb_remove_table(tlb, pmd);
#endif #endif
} }

View file

@ -287,7 +287,9 @@
#define __NR_getrandom 349 #define __NR_getrandom 349
#define __NR_memfd_create 350 #define __NR_memfd_create 350
#define __NR_bpf 351 #define __NR_bpf 351
#define NR_syscalls 352 #define __NR_s390_pci_mmio_write 352
#define __NR_s390_pci_mmio_read 353
#define NR_syscalls 354
/* /*
* There are some system calls that are not present on 64 bit, some * There are some system calls that are not present on 64 bit, some

View file

@ -17,8 +17,8 @@
* Make sure that the compiler is new enough. We want a compiler that * Make sure that the compiler is new enough. We want a compiler that
* is known to work with the "Q" assembler constraint. * is known to work with the "Q" assembler constraint.
*/ */
#if __GNUC__ < 3 || (__GNUC__ == 3 && __GNUC_MINOR__ < 3) #if __GNUC__ < 4 || (__GNUC__ == 4 && __GNUC_MINOR__ < 3)
#error Your compiler is too old; please use version 3.3.3 or newer #error Your compiler is too old; please use version 4.3 or newer
#endif #endif
int main(void) int main(void)
@ -156,7 +156,6 @@ int main(void)
DEFINE(__LC_INT_CLOCK, offsetof(struct _lowcore, int_clock)); DEFINE(__LC_INT_CLOCK, offsetof(struct _lowcore, int_clock));
DEFINE(__LC_MCCK_CLOCK, offsetof(struct _lowcore, mcck_clock)); DEFINE(__LC_MCCK_CLOCK, offsetof(struct _lowcore, mcck_clock));
DEFINE(__LC_MACHINE_FLAGS, offsetof(struct _lowcore, machine_flags)); DEFINE(__LC_MACHINE_FLAGS, offsetof(struct _lowcore, machine_flags));
DEFINE(__LC_FTRACE_FUNC, offsetof(struct _lowcore, ftrace_func));
DEFINE(__LC_DUMP_REIPL, offsetof(struct _lowcore, ipib)); DEFINE(__LC_DUMP_REIPL, offsetof(struct _lowcore, ipib));
BLANK(); BLANK();
DEFINE(__LC_CPU_TIMER_SAVE_AREA, offsetof(struct _lowcore, cpu_timer_save_area)); DEFINE(__LC_CPU_TIMER_SAVE_AREA, offsetof(struct _lowcore, cpu_timer_save_area));

View file

@ -434,7 +434,7 @@ static int setup_frame32(struct ksignal *ksig, sigset_t *set,
ksig->ka.sa.sa_restorer | PSW32_ADDR_AMODE; ksig->ka.sa.sa_restorer | PSW32_ADDR_AMODE;
} else { } else {
/* Signal frames without vectors registers are short ! */ /* Signal frames without vectors registers are short ! */
__u16 __user *svc = (void *) frame + frame_size - 2; __u16 __user *svc = (void __user *) frame + frame_size - 2;
if (__put_user(S390_SYSCALL_OPCODE | __NR_sigreturn, svc)) if (__put_user(S390_SYSCALL_OPCODE | __NR_sigreturn, svc))
return -EFAULT; return -EFAULT;
restorer = (unsigned long __force) svc | PSW32_ADDR_AMODE; restorer = (unsigned long __force) svc | PSW32_ADDR_AMODE;

View file

@ -218,3 +218,5 @@ COMPAT_SYSCALL_WRAP3(seccomp, unsigned int, op, unsigned int, flags, const char
COMPAT_SYSCALL_WRAP3(getrandom, char __user *, buf, size_t, count, unsigned int, flags) COMPAT_SYSCALL_WRAP3(getrandom, char __user *, buf, size_t, count, unsigned int, flags)
COMPAT_SYSCALL_WRAP2(memfd_create, const char __user *, uname, unsigned int, flags) COMPAT_SYSCALL_WRAP2(memfd_create, const char __user *, uname, unsigned int, flags)
COMPAT_SYSCALL_WRAP3(bpf, int, cmd, union bpf_attr *, attr, unsigned int, size); COMPAT_SYSCALL_WRAP3(bpf, int, cmd, union bpf_attr *, attr, unsigned int, size);
COMPAT_SYSCALL_WRAP3(s390_pci_mmio_write, const unsigned long, mmio_addr, const void __user *, user_buffer, const size_t, length);
COMPAT_SYSCALL_WRAP3(s390_pci_mmio_read, const unsigned long, mmio_addr, void __user *, user_buffer, const size_t, length);

View file

@ -1019,7 +1019,7 @@ debug_count_numargs(char *string)
*/ */
debug_entry_t* debug_entry_t*
debug_sprintf_event(debug_info_t* id, int level,char *string,...) __debug_sprintf_event(debug_info_t *id, int level, char *string, ...)
{ {
va_list ap; va_list ap;
int numargs,idx; int numargs,idx;
@ -1027,8 +1027,6 @@ debug_sprintf_event(debug_info_t* id, int level,char *string,...)
debug_sprintf_entry_t *curr_event; debug_sprintf_entry_t *curr_event;
debug_entry_t *active; debug_entry_t *active;
if((!id) || (level > id->level))
return NULL;
if (!debug_active || !id->areas) if (!debug_active || !id->areas)
return NULL; return NULL;
numargs=debug_count_numargs(string); numargs=debug_count_numargs(string);
@ -1050,14 +1048,14 @@ debug_sprintf_event(debug_info_t* id, int level,char *string,...)
return active; return active;
} }
EXPORT_SYMBOL(debug_sprintf_event); EXPORT_SYMBOL(__debug_sprintf_event);
/* /*
* debug_sprintf_exception: * debug_sprintf_exception:
*/ */
debug_entry_t* debug_entry_t*
debug_sprintf_exception(debug_info_t* id, int level,char *string,...) __debug_sprintf_exception(debug_info_t *id, int level, char *string, ...)
{ {
va_list ap; va_list ap;
int numargs,idx; int numargs,idx;
@ -1065,8 +1063,6 @@ debug_sprintf_exception(debug_info_t* id, int level,char *string,...)
debug_sprintf_entry_t *curr_event; debug_sprintf_entry_t *curr_event;
debug_entry_t *active; debug_entry_t *active;
if((!id) || (level > id->level))
return NULL;
if (!debug_active || !id->areas) if (!debug_active || !id->areas)
return NULL; return NULL;
@ -1089,7 +1085,7 @@ debug_sprintf_exception(debug_info_t* id, int level,char *string,...)
return active; return active;
} }
EXPORT_SYMBOL(debug_sprintf_exception); EXPORT_SYMBOL(__debug_sprintf_exception);
/* /*
* debug_register_view: * debug_register_view:

View file

@ -191,7 +191,8 @@ void die(struct pt_regs *regs, const char *str)
console_verbose(); console_verbose();
spin_lock_irq(&die_lock); spin_lock_irq(&die_lock);
bust_spinlocks(1); bust_spinlocks(1);
printk("%s: %04x [#%d] ", str, regs->int_code & 0xffff, ++die_counter); printk("%s: %04x ilc:%d [#%d] ", str, regs->int_code & 0xffff,
regs->int_code >> 17, ++die_counter);
#ifdef CONFIG_PREEMPT #ifdef CONFIG_PREEMPT
printk("PREEMPT "); printk("PREEMPT ");
#endif #endif

View file

@ -12,7 +12,6 @@
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/string.h> #include <linux/string.h>
#include <linux/ctype.h> #include <linux/ctype.h>
#include <linux/ftrace.h>
#include <linux/lockdep.h> #include <linux/lockdep.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/pfn.h> #include <linux/pfn.h>
@ -490,8 +489,5 @@ void __init startup_init(void)
detect_machine_facilities(); detect_machine_facilities();
setup_topology(); setup_topology();
sclp_early_detect(); sclp_early_detect();
#ifdef CONFIG_DYNAMIC_FTRACE
S390_lowcore.ftrace_func = (unsigned long)ftrace_caller;
#endif
lockdep_on(); lockdep_on();
} }

View file

@ -53,7 +53,7 @@ _PIF_WORK = (_PIF_PER_TRAP)
.macro TRACE_IRQS_ON .macro TRACE_IRQS_ON
#ifdef CONFIG_TRACE_IRQFLAGS #ifdef CONFIG_TRACE_IRQFLAGS
basr %r2,%r0 basr %r2,%r0
l %r1,BASED(.Lhardirqs_on) l %r1,BASED(.Lc_hardirqs_on)
basr %r14,%r1 # call trace_hardirqs_on_caller basr %r14,%r1 # call trace_hardirqs_on_caller
#endif #endif
.endm .endm
@ -61,7 +61,7 @@ _PIF_WORK = (_PIF_PER_TRAP)
.macro TRACE_IRQS_OFF .macro TRACE_IRQS_OFF
#ifdef CONFIG_TRACE_IRQFLAGS #ifdef CONFIG_TRACE_IRQFLAGS
basr %r2,%r0 basr %r2,%r0
l %r1,BASED(.Lhardirqs_off) l %r1,BASED(.Lc_hardirqs_off)
basr %r14,%r1 # call trace_hardirqs_off_caller basr %r14,%r1 # call trace_hardirqs_off_caller
#endif #endif
.endm .endm
@ -70,7 +70,7 @@ _PIF_WORK = (_PIF_PER_TRAP)
#ifdef CONFIG_LOCKDEP #ifdef CONFIG_LOCKDEP
tm __PT_PSW+1(%r11),0x01 # returning to user ? tm __PT_PSW+1(%r11),0x01 # returning to user ?
jz .+10 jz .+10
l %r1,BASED(.Llockdep_sys_exit) l %r1,BASED(.Lc_lockdep_sys_exit)
basr %r14,%r1 # call lockdep_sys_exit basr %r14,%r1 # call lockdep_sys_exit
#endif #endif
.endm .endm
@ -87,8 +87,8 @@ _PIF_WORK = (_PIF_PER_TRAP)
tmh %r8,0x0001 # interrupting from user ? tmh %r8,0x0001 # interrupting from user ?
jnz 1f jnz 1f
lr %r14,%r9 lr %r14,%r9
sl %r14,BASED(.Lcritical_start) sl %r14,BASED(.Lc_critical_start)
cl %r14,BASED(.Lcritical_length) cl %r14,BASED(.Lc_critical_length)
jhe 0f jhe 0f
la %r11,\savearea # inside critical section, do cleanup la %r11,\savearea # inside critical section, do cleanup
bras %r14,cleanup_critical bras %r14,cleanup_critical
@ -162,7 +162,7 @@ ENTRY(__switch_to)
lm %r6,%r15,__SF_GPRS(%r15) # load gprs of next task lm %r6,%r15,__SF_GPRS(%r15) # load gprs of next task
br %r14 br %r14
__critical_start: .L__critical_start:
/* /*
* SVC interrupt handler routine. System calls are synchronous events and * SVC interrupt handler routine. System calls are synchronous events and
* are executed with interrupts enabled. * are executed with interrupts enabled.
@ -170,145 +170,145 @@ __critical_start:
ENTRY(system_call) ENTRY(system_call)
stpt __LC_SYNC_ENTER_TIMER stpt __LC_SYNC_ENTER_TIMER
sysc_stm: .Lsysc_stm:
stm %r8,%r15,__LC_SAVE_AREA_SYNC stm %r8,%r15,__LC_SAVE_AREA_SYNC
l %r12,__LC_THREAD_INFO l %r12,__LC_THREAD_INFO
l %r13,__LC_SVC_NEW_PSW+4 l %r13,__LC_SVC_NEW_PSW+4
lhi %r14,_PIF_SYSCALL lhi %r14,_PIF_SYSCALL
sysc_per: .Lsysc_per:
l %r15,__LC_KERNEL_STACK l %r15,__LC_KERNEL_STACK
la %r11,STACK_FRAME_OVERHEAD(%r15) # pointer to pt_regs la %r11,STACK_FRAME_OVERHEAD(%r15) # pointer to pt_regs
sysc_vtime: .Lsysc_vtime:
UPDATE_VTIME %r8,%r9,__LC_SYNC_ENTER_TIMER UPDATE_VTIME %r8,%r9,__LC_SYNC_ENTER_TIMER
stm %r0,%r7,__PT_R0(%r11) stm %r0,%r7,__PT_R0(%r11)
mvc __PT_R8(32,%r11),__LC_SAVE_AREA_SYNC mvc __PT_R8(32,%r11),__LC_SAVE_AREA_SYNC
mvc __PT_PSW(8,%r11),__LC_SVC_OLD_PSW mvc __PT_PSW(8,%r11),__LC_SVC_OLD_PSW
mvc __PT_INT_CODE(4,%r11),__LC_SVC_ILC mvc __PT_INT_CODE(4,%r11),__LC_SVC_ILC
st %r14,__PT_FLAGS(%r11) st %r14,__PT_FLAGS(%r11)
sysc_do_svc: .Lsysc_do_svc:
l %r10,__TI_sysc_table(%r12) # 31 bit system call table l %r10,__TI_sysc_table(%r12) # 31 bit system call table
lh %r8,__PT_INT_CODE+2(%r11) lh %r8,__PT_INT_CODE+2(%r11)
sla %r8,2 # shift and test for svc0 sla %r8,2 # shift and test for svc0
jnz sysc_nr_ok jnz .Lsysc_nr_ok
# svc 0: system call number in %r1 # svc 0: system call number in %r1
cl %r1,BASED(.Lnr_syscalls) cl %r1,BASED(.Lnr_syscalls)
jnl sysc_nr_ok jnl .Lsysc_nr_ok
sth %r1,__PT_INT_CODE+2(%r11) sth %r1,__PT_INT_CODE+2(%r11)
lr %r8,%r1 lr %r8,%r1
sla %r8,2 sla %r8,2
sysc_nr_ok: .Lsysc_nr_ok:
xc __SF_BACKCHAIN(4,%r15),__SF_BACKCHAIN(%r15) xc __SF_BACKCHAIN(4,%r15),__SF_BACKCHAIN(%r15)
st %r2,__PT_ORIG_GPR2(%r11) st %r2,__PT_ORIG_GPR2(%r11)
st %r7,STACK_FRAME_OVERHEAD(%r15) st %r7,STACK_FRAME_OVERHEAD(%r15)
l %r9,0(%r8,%r10) # get system call addr. l %r9,0(%r8,%r10) # get system call addr.
tm __TI_flags+3(%r12),_TIF_TRACE tm __TI_flags+3(%r12),_TIF_TRACE
jnz sysc_tracesys jnz .Lsysc_tracesys
basr %r14,%r9 # call sys_xxxx basr %r14,%r9 # call sys_xxxx
st %r2,__PT_R2(%r11) # store return value st %r2,__PT_R2(%r11) # store return value
sysc_return: .Lsysc_return:
LOCKDEP_SYS_EXIT LOCKDEP_SYS_EXIT
sysc_tif: .Lsysc_tif:
tm __PT_PSW+1(%r11),0x01 # returning to user ? tm __PT_PSW+1(%r11),0x01 # returning to user ?
jno sysc_restore jno .Lsysc_restore
tm __PT_FLAGS+3(%r11),_PIF_WORK tm __PT_FLAGS+3(%r11),_PIF_WORK
jnz sysc_work jnz .Lsysc_work
tm __TI_flags+3(%r12),_TIF_WORK tm __TI_flags+3(%r12),_TIF_WORK
jnz sysc_work # check for thread work jnz .Lsysc_work # check for thread work
tm __LC_CPU_FLAGS+3,_CIF_WORK tm __LC_CPU_FLAGS+3,_CIF_WORK
jnz sysc_work jnz .Lsysc_work
sysc_restore: .Lsysc_restore:
mvc __LC_RETURN_PSW(8),__PT_PSW(%r11) mvc __LC_RETURN_PSW(8),__PT_PSW(%r11)
stpt __LC_EXIT_TIMER stpt __LC_EXIT_TIMER
lm %r0,%r15,__PT_R0(%r11) lm %r0,%r15,__PT_R0(%r11)
lpsw __LC_RETURN_PSW lpsw __LC_RETURN_PSW
sysc_done: .Lsysc_done:
# #
# One of the work bits is on. Find out which one. # One of the work bits is on. Find out which one.
# #
sysc_work: .Lsysc_work:
tm __LC_CPU_FLAGS+3,_CIF_MCCK_PENDING tm __LC_CPU_FLAGS+3,_CIF_MCCK_PENDING
jo sysc_mcck_pending jo .Lsysc_mcck_pending
tm __TI_flags+3(%r12),_TIF_NEED_RESCHED tm __TI_flags+3(%r12),_TIF_NEED_RESCHED
jo sysc_reschedule jo .Lsysc_reschedule
tm __PT_FLAGS+3(%r11),_PIF_PER_TRAP tm __PT_FLAGS+3(%r11),_PIF_PER_TRAP
jo sysc_singlestep jo .Lsysc_singlestep
tm __TI_flags+3(%r12),_TIF_SIGPENDING tm __TI_flags+3(%r12),_TIF_SIGPENDING
jo sysc_sigpending jo .Lsysc_sigpending
tm __TI_flags+3(%r12),_TIF_NOTIFY_RESUME tm __TI_flags+3(%r12),_TIF_NOTIFY_RESUME
jo sysc_notify_resume jo .Lsysc_notify_resume
tm __LC_CPU_FLAGS+3,_CIF_ASCE tm __LC_CPU_FLAGS+3,_CIF_ASCE
jo sysc_uaccess jo .Lsysc_uaccess
j sysc_return # beware of critical section cleanup j .Lsysc_return # beware of critical section cleanup
# #
# _TIF_NEED_RESCHED is set, call schedule # _TIF_NEED_RESCHED is set, call schedule
# #
sysc_reschedule: .Lsysc_reschedule:
l %r1,BASED(.Lschedule) l %r1,BASED(.Lc_schedule)
la %r14,BASED(sysc_return) la %r14,BASED(.Lsysc_return)
br %r1 # call schedule br %r1 # call schedule
# #
# _CIF_MCCK_PENDING is set, call handler # _CIF_MCCK_PENDING is set, call handler
# #
sysc_mcck_pending: .Lsysc_mcck_pending:
l %r1,BASED(.Lhandle_mcck) l %r1,BASED(.Lc_handle_mcck)
la %r14,BASED(sysc_return) la %r14,BASED(.Lsysc_return)
br %r1 # TIF bit will be cleared by handler br %r1 # TIF bit will be cleared by handler
# #
# _CIF_ASCE is set, load user space asce # _CIF_ASCE is set, load user space asce
# #
sysc_uaccess: .Lsysc_uaccess:
ni __LC_CPU_FLAGS+3,255-_CIF_ASCE ni __LC_CPU_FLAGS+3,255-_CIF_ASCE
lctl %c1,%c1,__LC_USER_ASCE # load primary asce lctl %c1,%c1,__LC_USER_ASCE # load primary asce
j sysc_return j .Lsysc_return
# #
# _TIF_SIGPENDING is set, call do_signal # _TIF_SIGPENDING is set, call do_signal
# #
sysc_sigpending: .Lsysc_sigpending:
lr %r2,%r11 # pass pointer to pt_regs lr %r2,%r11 # pass pointer to pt_regs
l %r1,BASED(.Ldo_signal) l %r1,BASED(.Lc_do_signal)
basr %r14,%r1 # call do_signal basr %r14,%r1 # call do_signal
tm __PT_FLAGS+3(%r11),_PIF_SYSCALL tm __PT_FLAGS+3(%r11),_PIF_SYSCALL
jno sysc_return jno .Lsysc_return
lm %r2,%r7,__PT_R2(%r11) # load svc arguments lm %r2,%r7,__PT_R2(%r11) # load svc arguments
l %r10,__TI_sysc_table(%r12) # 31 bit system call table l %r10,__TI_sysc_table(%r12) # 31 bit system call table
xr %r8,%r8 # svc 0 returns -ENOSYS xr %r8,%r8 # svc 0 returns -ENOSYS
clc __PT_INT_CODE+2(2,%r11),BASED(.Lnr_syscalls+2) clc __PT_INT_CODE+2(2,%r11),BASED(.Lnr_syscalls+2)
jnl sysc_nr_ok # invalid svc number -> do svc 0 jnl .Lsysc_nr_ok # invalid svc number -> do svc 0
lh %r8,__PT_INT_CODE+2(%r11) # load new svc number lh %r8,__PT_INT_CODE+2(%r11) # load new svc number
sla %r8,2 sla %r8,2
j sysc_nr_ok # restart svc j .Lsysc_nr_ok # restart svc
# #
# _TIF_NOTIFY_RESUME is set, call do_notify_resume # _TIF_NOTIFY_RESUME is set, call do_notify_resume
# #
sysc_notify_resume: .Lsysc_notify_resume:
lr %r2,%r11 # pass pointer to pt_regs lr %r2,%r11 # pass pointer to pt_regs
l %r1,BASED(.Ldo_notify_resume) l %r1,BASED(.Lc_do_notify_resume)
la %r14,BASED(sysc_return) la %r14,BASED(.Lsysc_return)
br %r1 # call do_notify_resume br %r1 # call do_notify_resume
# #
# _PIF_PER_TRAP is set, call do_per_trap # _PIF_PER_TRAP is set, call do_per_trap
# #
sysc_singlestep: .Lsysc_singlestep:
ni __PT_FLAGS+3(%r11),255-_PIF_PER_TRAP ni __PT_FLAGS+3(%r11),255-_PIF_PER_TRAP
lr %r2,%r11 # pass pointer to pt_regs lr %r2,%r11 # pass pointer to pt_regs
l %r1,BASED(.Ldo_per_trap) l %r1,BASED(.Lc_do_per_trap)
la %r14,BASED(sysc_return) la %r14,BASED(.Lsysc_return)
br %r1 # call do_per_trap br %r1 # call do_per_trap
# #
# call tracehook_report_syscall_entry/tracehook_report_syscall_exit before # call tracehook_report_syscall_entry/tracehook_report_syscall_exit before
# and after the system call # and after the system call
# #
sysc_tracesys: .Lsysc_tracesys:
l %r1,BASED(.Ltrace_enter) l %r1,BASED(.Lc_trace_enter)
lr %r2,%r11 # pass pointer to pt_regs lr %r2,%r11 # pass pointer to pt_regs
la %r3,0 la %r3,0
xr %r0,%r0 xr %r0,%r0
@ -316,22 +316,22 @@ sysc_tracesys:
st %r0,__PT_R2(%r11) st %r0,__PT_R2(%r11)
basr %r14,%r1 # call do_syscall_trace_enter basr %r14,%r1 # call do_syscall_trace_enter
cl %r2,BASED(.Lnr_syscalls) cl %r2,BASED(.Lnr_syscalls)
jnl sysc_tracenogo jnl .Lsysc_tracenogo
lr %r8,%r2 lr %r8,%r2
sll %r8,2 sll %r8,2
l %r9,0(%r8,%r10) l %r9,0(%r8,%r10)
sysc_tracego: .Lsysc_tracego:
lm %r3,%r7,__PT_R3(%r11) lm %r3,%r7,__PT_R3(%r11)
st %r7,STACK_FRAME_OVERHEAD(%r15) st %r7,STACK_FRAME_OVERHEAD(%r15)
l %r2,__PT_ORIG_GPR2(%r11) l %r2,__PT_ORIG_GPR2(%r11)
basr %r14,%r9 # call sys_xxx basr %r14,%r9 # call sys_xxx
st %r2,__PT_R2(%r11) # store return value st %r2,__PT_R2(%r11) # store return value
sysc_tracenogo: .Lsysc_tracenogo:
tm __TI_flags+3(%r12),_TIF_TRACE tm __TI_flags+3(%r12),_TIF_TRACE
jz sysc_return jz .Lsysc_return
l %r1,BASED(.Ltrace_exit) l %r1,BASED(.Lc_trace_exit)
lr %r2,%r11 # pass pointer to pt_regs lr %r2,%r11 # pass pointer to pt_regs
la %r14,BASED(sysc_return) la %r14,BASED(.Lsysc_return)
br %r1 # call do_syscall_trace_exit br %r1 # call do_syscall_trace_exit
# #
@ -341,18 +341,18 @@ ENTRY(ret_from_fork)
la %r11,STACK_FRAME_OVERHEAD(%r15) la %r11,STACK_FRAME_OVERHEAD(%r15)
l %r12,__LC_THREAD_INFO l %r12,__LC_THREAD_INFO
l %r13,__LC_SVC_NEW_PSW+4 l %r13,__LC_SVC_NEW_PSW+4
l %r1,BASED(.Lschedule_tail) l %r1,BASED(.Lc_schedule_tail)
basr %r14,%r1 # call schedule_tail basr %r14,%r1 # call schedule_tail
TRACE_IRQS_ON TRACE_IRQS_ON
ssm __LC_SVC_NEW_PSW # reenable interrupts ssm __LC_SVC_NEW_PSW # reenable interrupts
tm __PT_PSW+1(%r11),0x01 # forking a kernel thread ? tm __PT_PSW+1(%r11),0x01 # forking a kernel thread ?
jne sysc_tracenogo jne .Lsysc_tracenogo
# it's a kernel thread # it's a kernel thread
lm %r9,%r10,__PT_R9(%r11) # load gprs lm %r9,%r10,__PT_R9(%r11) # load gprs
ENTRY(kernel_thread_starter) ENTRY(kernel_thread_starter)
la %r2,0(%r10) la %r2,0(%r10)
basr %r14,%r9 basr %r14,%r9
j sysc_tracenogo j .Lsysc_tracenogo
/* /*
* Program check handler routine * Program check handler routine
@ -369,7 +369,7 @@ ENTRY(pgm_check_handler)
tmh %r8,0x4000 # PER bit set in old PSW ? tmh %r8,0x4000 # PER bit set in old PSW ?
jnz 0f # -> enabled, can't be a double fault jnz 0f # -> enabled, can't be a double fault
tm __LC_PGM_ILC+3,0x80 # check for per exception tm __LC_PGM_ILC+3,0x80 # check for per exception
jnz pgm_svcper # -> single stepped svc jnz .Lpgm_svcper # -> single stepped svc
0: CHECK_STACK STACK_SIZE,__LC_SAVE_AREA_SYNC 0: CHECK_STACK STACK_SIZE,__LC_SAVE_AREA_SYNC
ahi %r15,-(STACK_FRAME_OVERHEAD + __PT_SIZE) ahi %r15,-(STACK_FRAME_OVERHEAD + __PT_SIZE)
j 2f j 2f
@ -386,42 +386,42 @@ ENTRY(pgm_check_handler)
jz 0f jz 0f
l %r1,__TI_task(%r12) l %r1,__TI_task(%r12)
tmh %r8,0x0001 # kernel per event ? tmh %r8,0x0001 # kernel per event ?
jz pgm_kprobe jz .Lpgm_kprobe
oi __PT_FLAGS+3(%r11),_PIF_PER_TRAP oi __PT_FLAGS+3(%r11),_PIF_PER_TRAP
mvc __THREAD_per_address(4,%r1),__LC_PER_ADDRESS mvc __THREAD_per_address(4,%r1),__LC_PER_ADDRESS
mvc __THREAD_per_cause(2,%r1),__LC_PER_CODE mvc __THREAD_per_cause(2,%r1),__LC_PER_CODE
mvc __THREAD_per_paid(1,%r1),__LC_PER_ACCESS_ID mvc __THREAD_per_paid(1,%r1),__LC_PER_ACCESS_ID
0: REENABLE_IRQS 0: REENABLE_IRQS
xc __SF_BACKCHAIN(4,%r15),__SF_BACKCHAIN(%r15) xc __SF_BACKCHAIN(4,%r15),__SF_BACKCHAIN(%r15)
l %r1,BASED(.Ljump_table) l %r1,BASED(.Lc_jump_table)
la %r10,0x7f la %r10,0x7f
n %r10,__PT_INT_CODE(%r11) n %r10,__PT_INT_CODE(%r11)
je sysc_return je .Lsysc_return
sll %r10,2 sll %r10,2
l %r1,0(%r10,%r1) # load address of handler routine l %r1,0(%r10,%r1) # load address of handler routine
lr %r2,%r11 # pass pointer to pt_regs lr %r2,%r11 # pass pointer to pt_regs
basr %r14,%r1 # branch to interrupt-handler basr %r14,%r1 # branch to interrupt-handler
j sysc_return j .Lsysc_return
# #
# PER event in supervisor state, must be kprobes # PER event in supervisor state, must be kprobes
# #
pgm_kprobe: .Lpgm_kprobe:
REENABLE_IRQS REENABLE_IRQS
xc __SF_BACKCHAIN(4,%r15),__SF_BACKCHAIN(%r15) xc __SF_BACKCHAIN(4,%r15),__SF_BACKCHAIN(%r15)
l %r1,BASED(.Ldo_per_trap) l %r1,BASED(.Lc_do_per_trap)
lr %r2,%r11 # pass pointer to pt_regs lr %r2,%r11 # pass pointer to pt_regs
basr %r14,%r1 # call do_per_trap basr %r14,%r1 # call do_per_trap
j sysc_return j .Lsysc_return
# #
# single stepped system call # single stepped system call
# #
pgm_svcper: .Lpgm_svcper:
mvc __LC_RETURN_PSW(4),__LC_SVC_NEW_PSW mvc __LC_RETURN_PSW(4),__LC_SVC_NEW_PSW
mvc __LC_RETURN_PSW+4(4),BASED(.Lsysc_per) mvc __LC_RETURN_PSW+4(4),BASED(.Lc_sysc_per)
lhi %r14,_PIF_SYSCALL | _PIF_PER_TRAP lhi %r14,_PIF_SYSCALL | _PIF_PER_TRAP
lpsw __LC_RETURN_PSW # branch to sysc_per and enable irqs lpsw __LC_RETURN_PSW # branch to .Lsysc_per and enable irqs
/* /*
* IO interrupt handler routine * IO interrupt handler routine
@ -435,9 +435,9 @@ ENTRY(io_int_handler)
l %r13,__LC_SVC_NEW_PSW+4 l %r13,__LC_SVC_NEW_PSW+4
lm %r8,%r9,__LC_IO_OLD_PSW lm %r8,%r9,__LC_IO_OLD_PSW
tmh %r8,0x0001 # interrupting from user ? tmh %r8,0x0001 # interrupting from user ?
jz io_skip jz .Lio_skip
UPDATE_VTIME %r14,%r15,__LC_ASYNC_ENTER_TIMER UPDATE_VTIME %r14,%r15,__LC_ASYNC_ENTER_TIMER
io_skip: .Lio_skip:
SWITCH_ASYNC __LC_SAVE_AREA_ASYNC,__LC_ASYNC_STACK,STACK_SHIFT SWITCH_ASYNC __LC_SAVE_AREA_ASYNC,__LC_ASYNC_STACK,STACK_SHIFT
stm %r0,%r7,__PT_R0(%r11) stm %r0,%r7,__PT_R0(%r11)
mvc __PT_R8(32,%r11),__LC_SAVE_AREA_ASYNC mvc __PT_R8(32,%r11),__LC_SAVE_AREA_ASYNC
@ -446,35 +446,35 @@ io_skip:
xc __PT_FLAGS(4,%r11),__PT_FLAGS(%r11) xc __PT_FLAGS(4,%r11),__PT_FLAGS(%r11)
TRACE_IRQS_OFF TRACE_IRQS_OFF
xc __SF_BACKCHAIN(4,%r15),__SF_BACKCHAIN(%r15) xc __SF_BACKCHAIN(4,%r15),__SF_BACKCHAIN(%r15)
io_loop: .Lio_loop:
l %r1,BASED(.Ldo_IRQ) l %r1,BASED(.Lc_do_IRQ)
lr %r2,%r11 # pass pointer to pt_regs lr %r2,%r11 # pass pointer to pt_regs
lhi %r3,IO_INTERRUPT lhi %r3,IO_INTERRUPT
tm __PT_INT_CODE+8(%r11),0x80 # adapter interrupt ? tm __PT_INT_CODE+8(%r11),0x80 # adapter interrupt ?
jz io_call jz .Lio_call
lhi %r3,THIN_INTERRUPT lhi %r3,THIN_INTERRUPT
io_call: .Lio_call:
basr %r14,%r1 # call do_IRQ basr %r14,%r1 # call do_IRQ
tm __LC_MACHINE_FLAGS+2,0x10 # MACHINE_FLAG_LPAR tm __LC_MACHINE_FLAGS+2,0x10 # MACHINE_FLAG_LPAR
jz io_return jz .Lio_return
tpi 0 tpi 0
jz io_return jz .Lio_return
mvc __PT_INT_CODE(12,%r11),__LC_SUBCHANNEL_ID mvc __PT_INT_CODE(12,%r11),__LC_SUBCHANNEL_ID
j io_loop j .Lio_loop
io_return: .Lio_return:
LOCKDEP_SYS_EXIT LOCKDEP_SYS_EXIT
TRACE_IRQS_ON TRACE_IRQS_ON
io_tif: .Lio_tif:
tm __TI_flags+3(%r12),_TIF_WORK tm __TI_flags+3(%r12),_TIF_WORK
jnz io_work # there is work to do (signals etc.) jnz .Lio_work # there is work to do (signals etc.)
tm __LC_CPU_FLAGS+3,_CIF_WORK tm __LC_CPU_FLAGS+3,_CIF_WORK
jnz io_work jnz .Lio_work
io_restore: .Lio_restore:
mvc __LC_RETURN_PSW(8),__PT_PSW(%r11) mvc __LC_RETURN_PSW(8),__PT_PSW(%r11)
stpt __LC_EXIT_TIMER stpt __LC_EXIT_TIMER
lm %r0,%r15,__PT_R0(%r11) lm %r0,%r15,__PT_R0(%r11)
lpsw __LC_RETURN_PSW lpsw __LC_RETURN_PSW
io_done: .Lio_done:
# #
# There is work todo, find out in which context we have been interrupted: # There is work todo, find out in which context we have been interrupted:
@ -483,15 +483,15 @@ io_done:
# the preemption counter and if it is zero call preempt_schedule_irq # the preemption counter and if it is zero call preempt_schedule_irq
# Before any work can be done, a switch to the kernel stack is required. # Before any work can be done, a switch to the kernel stack is required.
# #
io_work: .Lio_work:
tm __PT_PSW+1(%r11),0x01 # returning to user ? tm __PT_PSW+1(%r11),0x01 # returning to user ?
jo io_work_user # yes -> do resched & signal jo .Lio_work_user # yes -> do resched & signal
#ifdef CONFIG_PREEMPT #ifdef CONFIG_PREEMPT
# check for preemptive scheduling # check for preemptive scheduling
icm %r0,15,__TI_precount(%r12) icm %r0,15,__TI_precount(%r12)
jnz io_restore # preemption disabled jnz .Lio_restore # preemption disabled
tm __TI_flags+3(%r12),_TIF_NEED_RESCHED tm __TI_flags+3(%r12),_TIF_NEED_RESCHED
jno io_restore jno .Lio_restore
# switch to kernel stack # switch to kernel stack
l %r1,__PT_R15(%r11) l %r1,__PT_R15(%r11)
ahi %r1,-(STACK_FRAME_OVERHEAD + __PT_SIZE) ahi %r1,-(STACK_FRAME_OVERHEAD + __PT_SIZE)
@ -499,20 +499,20 @@ io_work:
xc __SF_BACKCHAIN(4,%r1),__SF_BACKCHAIN(%r1) xc __SF_BACKCHAIN(4,%r1),__SF_BACKCHAIN(%r1)
la %r11,STACK_FRAME_OVERHEAD(%r1) la %r11,STACK_FRAME_OVERHEAD(%r1)
lr %r15,%r1 lr %r15,%r1
# TRACE_IRQS_ON already done at io_return, call # TRACE_IRQS_ON already done at .Lio_return, call
# TRACE_IRQS_OFF to keep things symmetrical # TRACE_IRQS_OFF to keep things symmetrical
TRACE_IRQS_OFF TRACE_IRQS_OFF
l %r1,BASED(.Lpreempt_irq) l %r1,BASED(.Lc_preempt_irq)
basr %r14,%r1 # call preempt_schedule_irq basr %r14,%r1 # call preempt_schedule_irq
j io_return j .Lio_return
#else #else
j io_restore j .Lio_restore
#endif #endif
# #
# Need to do work before returning to userspace, switch to kernel stack # Need to do work before returning to userspace, switch to kernel stack
# #
io_work_user: .Lio_work_user:
l %r1,__LC_KERNEL_STACK l %r1,__LC_KERNEL_STACK
mvc STACK_FRAME_OVERHEAD(__PT_SIZE,%r1),0(%r11) mvc STACK_FRAME_OVERHEAD(__PT_SIZE,%r1),0(%r11)
xc __SF_BACKCHAIN(4,%r1),__SF_BACKCHAIN(%r1) xc __SF_BACKCHAIN(4,%r1),__SF_BACKCHAIN(%r1)
@ -522,74 +522,74 @@ io_work_user:
# #
# One of the work bits is on. Find out which one. # One of the work bits is on. Find out which one.
# #
io_work_tif: .Lio_work_tif:
tm __LC_CPU_FLAGS+3(%r12),_CIF_MCCK_PENDING tm __LC_CPU_FLAGS+3(%r12),_CIF_MCCK_PENDING
jo io_mcck_pending jo .Lio_mcck_pending
tm __TI_flags+3(%r12),_TIF_NEED_RESCHED tm __TI_flags+3(%r12),_TIF_NEED_RESCHED
jo io_reschedule jo .Lio_reschedule
tm __TI_flags+3(%r12),_TIF_SIGPENDING tm __TI_flags+3(%r12),_TIF_SIGPENDING
jo io_sigpending jo .Lio_sigpending
tm __TI_flags+3(%r12),_TIF_NOTIFY_RESUME tm __TI_flags+3(%r12),_TIF_NOTIFY_RESUME
jo io_notify_resume jo .Lio_notify_resume
tm __LC_CPU_FLAGS+3,_CIF_ASCE tm __LC_CPU_FLAGS+3,_CIF_ASCE
jo io_uaccess jo .Lio_uaccess
j io_return # beware of critical section cleanup j .Lio_return # beware of critical section cleanup
# #
# _CIF_MCCK_PENDING is set, call handler # _CIF_MCCK_PENDING is set, call handler
# #
io_mcck_pending: .Lio_mcck_pending:
# TRACE_IRQS_ON already done at io_return # TRACE_IRQS_ON already done at .Lio_return
l %r1,BASED(.Lhandle_mcck) l %r1,BASED(.Lc_handle_mcck)
basr %r14,%r1 # TIF bit will be cleared by handler basr %r14,%r1 # TIF bit will be cleared by handler
TRACE_IRQS_OFF TRACE_IRQS_OFF
j io_return j .Lio_return
# #
# _CIF_ASCE is set, load user space asce # _CIF_ASCE is set, load user space asce
# #
io_uaccess: .Lio_uaccess:
ni __LC_CPU_FLAGS+3,255-_CIF_ASCE ni __LC_CPU_FLAGS+3,255-_CIF_ASCE
lctl %c1,%c1,__LC_USER_ASCE # load primary asce lctl %c1,%c1,__LC_USER_ASCE # load primary asce
j io_return j .Lio_return
# #
# _TIF_NEED_RESCHED is set, call schedule # _TIF_NEED_RESCHED is set, call schedule
# #
io_reschedule: .Lio_reschedule:
# TRACE_IRQS_ON already done at io_return # TRACE_IRQS_ON already done at .Lio_return
l %r1,BASED(.Lschedule) l %r1,BASED(.Lc_schedule)
ssm __LC_SVC_NEW_PSW # reenable interrupts ssm __LC_SVC_NEW_PSW # reenable interrupts
basr %r14,%r1 # call scheduler basr %r14,%r1 # call scheduler
ssm __LC_PGM_NEW_PSW # disable I/O and ext. interrupts ssm __LC_PGM_NEW_PSW # disable I/O and ext. interrupts
TRACE_IRQS_OFF TRACE_IRQS_OFF
j io_return j .Lio_return
# #
# _TIF_SIGPENDING is set, call do_signal # _TIF_SIGPENDING is set, call do_signal
# #
io_sigpending: .Lio_sigpending:
# TRACE_IRQS_ON already done at io_return # TRACE_IRQS_ON already done at .Lio_return
l %r1,BASED(.Ldo_signal) l %r1,BASED(.Lc_do_signal)
ssm __LC_SVC_NEW_PSW # reenable interrupts ssm __LC_SVC_NEW_PSW # reenable interrupts
lr %r2,%r11 # pass pointer to pt_regs lr %r2,%r11 # pass pointer to pt_regs
basr %r14,%r1 # call do_signal basr %r14,%r1 # call do_signal
ssm __LC_PGM_NEW_PSW # disable I/O and ext. interrupts ssm __LC_PGM_NEW_PSW # disable I/O and ext. interrupts
TRACE_IRQS_OFF TRACE_IRQS_OFF
j io_return j .Lio_return
# #
# _TIF_SIGPENDING is set, call do_signal # _TIF_SIGPENDING is set, call do_signal
# #
io_notify_resume: .Lio_notify_resume:
# TRACE_IRQS_ON already done at io_return # TRACE_IRQS_ON already done at .Lio_return
l %r1,BASED(.Ldo_notify_resume) l %r1,BASED(.Lc_do_notify_resume)
ssm __LC_SVC_NEW_PSW # reenable interrupts ssm __LC_SVC_NEW_PSW # reenable interrupts
lr %r2,%r11 # pass pointer to pt_regs lr %r2,%r11 # pass pointer to pt_regs
basr %r14,%r1 # call do_notify_resume basr %r14,%r1 # call do_notify_resume
ssm __LC_PGM_NEW_PSW # disable I/O and ext. interrupts ssm __LC_PGM_NEW_PSW # disable I/O and ext. interrupts
TRACE_IRQS_OFF TRACE_IRQS_OFF
j io_return j .Lio_return
/* /*
* External interrupt handler routine * External interrupt handler routine
@ -603,9 +603,9 @@ ENTRY(ext_int_handler)
l %r13,__LC_SVC_NEW_PSW+4 l %r13,__LC_SVC_NEW_PSW+4
lm %r8,%r9,__LC_EXT_OLD_PSW lm %r8,%r9,__LC_EXT_OLD_PSW
tmh %r8,0x0001 # interrupting from user ? tmh %r8,0x0001 # interrupting from user ?
jz ext_skip jz .Lext_skip
UPDATE_VTIME %r14,%r15,__LC_ASYNC_ENTER_TIMER UPDATE_VTIME %r14,%r15,__LC_ASYNC_ENTER_TIMER
ext_skip: .Lext_skip:
SWITCH_ASYNC __LC_SAVE_AREA_ASYNC,__LC_ASYNC_STACK,STACK_SHIFT SWITCH_ASYNC __LC_SAVE_AREA_ASYNC,__LC_ASYNC_STACK,STACK_SHIFT
stm %r0,%r7,__PT_R0(%r11) stm %r0,%r7,__PT_R0(%r11)
mvc __PT_R8(32,%r11),__LC_SAVE_AREA_ASYNC mvc __PT_R8(32,%r11),__LC_SAVE_AREA_ASYNC
@ -614,29 +614,29 @@ ext_skip:
mvc __PT_INT_PARM(4,%r11),__LC_EXT_PARAMS mvc __PT_INT_PARM(4,%r11),__LC_EXT_PARAMS
xc __PT_FLAGS(4,%r11),__PT_FLAGS(%r11) xc __PT_FLAGS(4,%r11),__PT_FLAGS(%r11)
TRACE_IRQS_OFF TRACE_IRQS_OFF
l %r1,BASED(.Ldo_IRQ) l %r1,BASED(.Lc_do_IRQ)
lr %r2,%r11 # pass pointer to pt_regs lr %r2,%r11 # pass pointer to pt_regs
lhi %r3,EXT_INTERRUPT lhi %r3,EXT_INTERRUPT
basr %r14,%r1 # call do_IRQ basr %r14,%r1 # call do_IRQ
j io_return j .Lio_return
/* /*
* Load idle PSW. The second "half" of this function is in cleanup_idle. * Load idle PSW. The second "half" of this function is in .Lcleanup_idle.
*/ */
ENTRY(psw_idle) ENTRY(psw_idle)
st %r3,__SF_EMPTY(%r15) st %r3,__SF_EMPTY(%r15)
basr %r1,0 basr %r1,0
la %r1,psw_idle_lpsw+4-.(%r1) la %r1,.Lpsw_idle_lpsw+4-.(%r1)
st %r1,__SF_EMPTY+4(%r15) st %r1,__SF_EMPTY+4(%r15)
oi __SF_EMPTY+4(%r15),0x80 oi __SF_EMPTY+4(%r15),0x80
stck __CLOCK_IDLE_ENTER(%r2) stck __CLOCK_IDLE_ENTER(%r2)
stpt __TIMER_IDLE_ENTER(%r2) stpt __TIMER_IDLE_ENTER(%r2)
psw_idle_lpsw: .Lpsw_idle_lpsw:
lpsw __SF_EMPTY(%r15) lpsw __SF_EMPTY(%r15)
br %r14 br %r14
psw_idle_end: .Lpsw_idle_end:
__critical_end: .L__critical_end:
/* /*
* Machine check handler routines * Machine check handler routines
@ -650,7 +650,7 @@ ENTRY(mcck_int_handler)
l %r13,__LC_SVC_NEW_PSW+4 l %r13,__LC_SVC_NEW_PSW+4
lm %r8,%r9,__LC_MCK_OLD_PSW lm %r8,%r9,__LC_MCK_OLD_PSW
tm __LC_MCCK_CODE,0x80 # system damage? tm __LC_MCCK_CODE,0x80 # system damage?
jo mcck_panic # yes -> rest of mcck code invalid jo .Lmcck_panic # yes -> rest of mcck code invalid
la %r14,__LC_CPU_TIMER_SAVE_AREA la %r14,__LC_CPU_TIMER_SAVE_AREA
mvc __LC_MCCK_ENTER_TIMER(8),0(%r14) mvc __LC_MCCK_ENTER_TIMER(8),0(%r14)
tm __LC_MCCK_CODE+5,0x02 # stored cpu timer value valid? tm __LC_MCCK_CODE+5,0x02 # stored cpu timer value valid?
@ -668,22 +668,22 @@ ENTRY(mcck_int_handler)
2: spt 0(%r14) 2: spt 0(%r14)
mvc __LC_MCCK_ENTER_TIMER(8),0(%r14) mvc __LC_MCCK_ENTER_TIMER(8),0(%r14)
3: tm __LC_MCCK_CODE+2,0x09 # mwp + ia of old psw valid? 3: tm __LC_MCCK_CODE+2,0x09 # mwp + ia of old psw valid?
jno mcck_panic # no -> skip cleanup critical jno .Lmcck_panic # no -> skip cleanup critical
tm %r8,0x0001 # interrupting from user ? tm %r8,0x0001 # interrupting from user ?
jz mcck_skip jz .Lmcck_skip
UPDATE_VTIME %r14,%r15,__LC_MCCK_ENTER_TIMER UPDATE_VTIME %r14,%r15,__LC_MCCK_ENTER_TIMER
mcck_skip: .Lmcck_skip:
SWITCH_ASYNC __LC_GPREGS_SAVE_AREA+32,__LC_PANIC_STACK,PAGE_SHIFT SWITCH_ASYNC __LC_GPREGS_SAVE_AREA+32,__LC_PANIC_STACK,PAGE_SHIFT
stm %r0,%r7,__PT_R0(%r11) stm %r0,%r7,__PT_R0(%r11)
mvc __PT_R8(32,%r11),__LC_GPREGS_SAVE_AREA+32 mvc __PT_R8(32,%r11),__LC_GPREGS_SAVE_AREA+32
stm %r8,%r9,__PT_PSW(%r11) stm %r8,%r9,__PT_PSW(%r11)
xc __PT_FLAGS(4,%r11),__PT_FLAGS(%r11) xc __PT_FLAGS(4,%r11),__PT_FLAGS(%r11)
xc __SF_BACKCHAIN(4,%r15),__SF_BACKCHAIN(%r15) xc __SF_BACKCHAIN(4,%r15),__SF_BACKCHAIN(%r15)
l %r1,BASED(.Ldo_machine_check) l %r1,BASED(.Lc_do_machine_check)
lr %r2,%r11 # pass pointer to pt_regs lr %r2,%r11 # pass pointer to pt_regs
basr %r14,%r1 # call s390_do_machine_check basr %r14,%r1 # call s390_do_machine_check
tm __PT_PSW+1(%r11),0x01 # returning to user ? tm __PT_PSW+1(%r11),0x01 # returning to user ?
jno mcck_return jno .Lmcck_return
l %r1,__LC_KERNEL_STACK # switch to kernel stack l %r1,__LC_KERNEL_STACK # switch to kernel stack
mvc STACK_FRAME_OVERHEAD(__PT_SIZE,%r1),0(%r11) mvc STACK_FRAME_OVERHEAD(__PT_SIZE,%r1),0(%r11)
xc __SF_BACKCHAIN(4,%r1),__SF_BACKCHAIN(%r1) xc __SF_BACKCHAIN(4,%r1),__SF_BACKCHAIN(%r1)
@ -691,12 +691,12 @@ mcck_skip:
lr %r15,%r1 lr %r15,%r1
ssm __LC_PGM_NEW_PSW # turn dat on, keep irqs off ssm __LC_PGM_NEW_PSW # turn dat on, keep irqs off
tm __LC_CPU_FLAGS+3,_CIF_MCCK_PENDING tm __LC_CPU_FLAGS+3,_CIF_MCCK_PENDING
jno mcck_return jno .Lmcck_return
TRACE_IRQS_OFF TRACE_IRQS_OFF
l %r1,BASED(.Lhandle_mcck) l %r1,BASED(.Lc_handle_mcck)
basr %r14,%r1 # call s390_handle_mcck basr %r14,%r1 # call s390_handle_mcck
TRACE_IRQS_ON TRACE_IRQS_ON
mcck_return: .Lmcck_return:
mvc __LC_RETURN_MCCK_PSW(8),__PT_PSW(%r11) # move return PSW mvc __LC_RETURN_MCCK_PSW(8),__PT_PSW(%r11) # move return PSW
tm __LC_RETURN_MCCK_PSW+1,0x01 # returning to user ? tm __LC_RETURN_MCCK_PSW+1,0x01 # returning to user ?
jno 0f jno 0f
@ -706,15 +706,15 @@ mcck_return:
0: lm %r0,%r15,__PT_R0(%r11) 0: lm %r0,%r15,__PT_R0(%r11)
lpsw __LC_RETURN_MCCK_PSW lpsw __LC_RETURN_MCCK_PSW
mcck_panic: .Lmcck_panic:
l %r14,__LC_PANIC_STACK l %r14,__LC_PANIC_STACK
slr %r14,%r15 slr %r14,%r15
sra %r14,PAGE_SHIFT sra %r14,PAGE_SHIFT
jz 0f jz 0f
l %r15,__LC_PANIC_STACK l %r15,__LC_PANIC_STACK
j mcck_skip j .Lmcck_skip
0: ahi %r15,-(STACK_FRAME_OVERHEAD + __PT_SIZE) 0: ahi %r15,-(STACK_FRAME_OVERHEAD + __PT_SIZE)
j mcck_skip j .Lmcck_skip
# #
# PSW restart interrupt handler # PSW restart interrupt handler
@ -764,58 +764,58 @@ stack_overflow:
1: .long kernel_stack_overflow 1: .long kernel_stack_overflow
#endif #endif
cleanup_table: .Lcleanup_table:
.long system_call + 0x80000000 .long system_call + 0x80000000
.long sysc_do_svc + 0x80000000 .long .Lsysc_do_svc + 0x80000000
.long sysc_tif + 0x80000000 .long .Lsysc_tif + 0x80000000
.long sysc_restore + 0x80000000 .long .Lsysc_restore + 0x80000000
.long sysc_done + 0x80000000 .long .Lsysc_done + 0x80000000
.long io_tif + 0x80000000 .long .Lio_tif + 0x80000000
.long io_restore + 0x80000000 .long .Lio_restore + 0x80000000
.long io_done + 0x80000000 .long .Lio_done + 0x80000000
.long psw_idle + 0x80000000 .long psw_idle + 0x80000000
.long psw_idle_end + 0x80000000 .long .Lpsw_idle_end + 0x80000000
cleanup_critical: cleanup_critical:
cl %r9,BASED(cleanup_table) # system_call cl %r9,BASED(.Lcleanup_table) # system_call
jl 0f jl 0f
cl %r9,BASED(cleanup_table+4) # sysc_do_svc cl %r9,BASED(.Lcleanup_table+4) # .Lsysc_do_svc
jl cleanup_system_call jl .Lcleanup_system_call
cl %r9,BASED(cleanup_table+8) # sysc_tif cl %r9,BASED(.Lcleanup_table+8) # .Lsysc_tif
jl 0f jl 0f
cl %r9,BASED(cleanup_table+12) # sysc_restore cl %r9,BASED(.Lcleanup_table+12) # .Lsysc_restore
jl cleanup_sysc_tif jl .Lcleanup_sysc_tif
cl %r9,BASED(cleanup_table+16) # sysc_done cl %r9,BASED(.Lcleanup_table+16) # .Lsysc_done
jl cleanup_sysc_restore jl .Lcleanup_sysc_restore
cl %r9,BASED(cleanup_table+20) # io_tif cl %r9,BASED(.Lcleanup_table+20) # .Lio_tif
jl 0f jl 0f
cl %r9,BASED(cleanup_table+24) # io_restore cl %r9,BASED(.Lcleanup_table+24) # .Lio_restore
jl cleanup_io_tif jl .Lcleanup_io_tif
cl %r9,BASED(cleanup_table+28) # io_done cl %r9,BASED(.Lcleanup_table+28) # .Lio_done
jl cleanup_io_restore jl .Lcleanup_io_restore
cl %r9,BASED(cleanup_table+32) # psw_idle cl %r9,BASED(.Lcleanup_table+32) # psw_idle
jl 0f jl 0f
cl %r9,BASED(cleanup_table+36) # psw_idle_end cl %r9,BASED(.Lcleanup_table+36) # .Lpsw_idle_end
jl cleanup_idle jl .Lcleanup_idle
0: br %r14 0: br %r14
cleanup_system_call: .Lcleanup_system_call:
# check if stpt has been executed # check if stpt has been executed
cl %r9,BASED(cleanup_system_call_insn) cl %r9,BASED(.Lcleanup_system_call_insn)
jh 0f jh 0f
mvc __LC_SYNC_ENTER_TIMER(8),__LC_ASYNC_ENTER_TIMER mvc __LC_SYNC_ENTER_TIMER(8),__LC_ASYNC_ENTER_TIMER
chi %r11,__LC_SAVE_AREA_ASYNC chi %r11,__LC_SAVE_AREA_ASYNC
je 0f je 0f
mvc __LC_SYNC_ENTER_TIMER(8),__LC_MCCK_ENTER_TIMER mvc __LC_SYNC_ENTER_TIMER(8),__LC_MCCK_ENTER_TIMER
0: # check if stm has been executed 0: # check if stm has been executed
cl %r9,BASED(cleanup_system_call_insn+4) cl %r9,BASED(.Lcleanup_system_call_insn+4)
jh 0f jh 0f
mvc __LC_SAVE_AREA_SYNC(32),0(%r11) mvc __LC_SAVE_AREA_SYNC(32),0(%r11)
0: # set up saved registers r12, and r13 0: # set up saved registers r12, and r13
st %r12,16(%r11) # r12 thread-info pointer st %r12,16(%r11) # r12 thread-info pointer
st %r13,20(%r11) # r13 literal-pool pointer st %r13,20(%r11) # r13 literal-pool pointer
# check if the user time calculation has been done # check if the user time calculation has been done
cl %r9,BASED(cleanup_system_call_insn+8) cl %r9,BASED(.Lcleanup_system_call_insn+8)
jh 0f jh 0f
l %r10,__LC_EXIT_TIMER l %r10,__LC_EXIT_TIMER
l %r15,__LC_EXIT_TIMER+4 l %r15,__LC_EXIT_TIMER+4
@ -824,7 +824,7 @@ cleanup_system_call:
st %r10,__LC_USER_TIMER st %r10,__LC_USER_TIMER
st %r15,__LC_USER_TIMER+4 st %r15,__LC_USER_TIMER+4
0: # check if the system time calculation has been done 0: # check if the system time calculation has been done
cl %r9,BASED(cleanup_system_call_insn+12) cl %r9,BASED(.Lcleanup_system_call_insn+12)
jh 0f jh 0f
l %r10,__LC_LAST_UPDATE_TIMER l %r10,__LC_LAST_UPDATE_TIMER
l %r15,__LC_LAST_UPDATE_TIMER+4 l %r15,__LC_LAST_UPDATE_TIMER+4
@ -848,20 +848,20 @@ cleanup_system_call:
# setup saved register 15 # setup saved register 15
st %r15,28(%r11) # r15 stack pointer st %r15,28(%r11) # r15 stack pointer
# set new psw address and exit # set new psw address and exit
l %r9,BASED(cleanup_table+4) # sysc_do_svc + 0x80000000 l %r9,BASED(.Lcleanup_table+4) # .Lsysc_do_svc + 0x80000000
br %r14 br %r14
cleanup_system_call_insn: .Lcleanup_system_call_insn:
.long system_call + 0x80000000 .long system_call + 0x80000000
.long sysc_stm + 0x80000000 .long .Lsysc_stm + 0x80000000
.long sysc_vtime + 0x80000000 + 36 .long .Lsysc_vtime + 0x80000000 + 36
.long sysc_vtime + 0x80000000 + 76 .long .Lsysc_vtime + 0x80000000 + 76
cleanup_sysc_tif: .Lcleanup_sysc_tif:
l %r9,BASED(cleanup_table+8) # sysc_tif + 0x80000000 l %r9,BASED(.Lcleanup_table+8) # .Lsysc_tif + 0x80000000
br %r14 br %r14
cleanup_sysc_restore: .Lcleanup_sysc_restore:
cl %r9,BASED(cleanup_sysc_restore_insn) cl %r9,BASED(.Lcleanup_sysc_restore_insn)
jhe 0f jhe 0f
l %r9,12(%r11) # get saved pointer to pt_regs l %r9,12(%r11) # get saved pointer to pt_regs
mvc __LC_RETURN_PSW(8),__PT_PSW(%r9) mvc __LC_RETURN_PSW(8),__PT_PSW(%r9)
@ -869,15 +869,15 @@ cleanup_sysc_restore:
lm %r0,%r7,__PT_R0(%r9) lm %r0,%r7,__PT_R0(%r9)
0: lm %r8,%r9,__LC_RETURN_PSW 0: lm %r8,%r9,__LC_RETURN_PSW
br %r14 br %r14
cleanup_sysc_restore_insn: .Lcleanup_sysc_restore_insn:
.long sysc_done - 4 + 0x80000000 .long .Lsysc_done - 4 + 0x80000000
cleanup_io_tif: .Lcleanup_io_tif:
l %r9,BASED(cleanup_table+20) # io_tif + 0x80000000 l %r9,BASED(.Lcleanup_table+20) # .Lio_tif + 0x80000000
br %r14 br %r14
cleanup_io_restore: .Lcleanup_io_restore:
cl %r9,BASED(cleanup_io_restore_insn) cl %r9,BASED(.Lcleanup_io_restore_insn)
jhe 0f jhe 0f
l %r9,12(%r11) # get saved r11 pointer to pt_regs l %r9,12(%r11) # get saved r11 pointer to pt_regs
mvc __LC_RETURN_PSW(8),__PT_PSW(%r9) mvc __LC_RETURN_PSW(8),__PT_PSW(%r9)
@ -885,10 +885,10 @@ cleanup_io_restore:
lm %r0,%r7,__PT_R0(%r9) lm %r0,%r7,__PT_R0(%r9)
0: lm %r8,%r9,__LC_RETURN_PSW 0: lm %r8,%r9,__LC_RETURN_PSW
br %r14 br %r14
cleanup_io_restore_insn: .Lcleanup_io_restore_insn:
.long io_done - 4 + 0x80000000 .long .Lio_done - 4 + 0x80000000
cleanup_idle: .Lcleanup_idle:
# copy interrupt clock & cpu timer # copy interrupt clock & cpu timer
mvc __CLOCK_IDLE_EXIT(8,%r2),__LC_INT_CLOCK mvc __CLOCK_IDLE_EXIT(8,%r2),__LC_INT_CLOCK
mvc __TIMER_IDLE_EXIT(8,%r2),__LC_ASYNC_ENTER_TIMER mvc __TIMER_IDLE_EXIT(8,%r2),__LC_ASYNC_ENTER_TIMER
@ -897,7 +897,7 @@ cleanup_idle:
mvc __CLOCK_IDLE_EXIT(8,%r2),__LC_MCCK_CLOCK mvc __CLOCK_IDLE_EXIT(8,%r2),__LC_MCCK_CLOCK
mvc __TIMER_IDLE_EXIT(8,%r2),__LC_MCCK_ENTER_TIMER mvc __TIMER_IDLE_EXIT(8,%r2),__LC_MCCK_ENTER_TIMER
0: # check if stck has been executed 0: # check if stck has been executed
cl %r9,BASED(cleanup_idle_insn) cl %r9,BASED(.Lcleanup_idle_insn)
jhe 1f jhe 1f
mvc __CLOCK_IDLE_ENTER(8,%r2),__CLOCK_IDLE_EXIT(%r2) mvc __CLOCK_IDLE_ENTER(8,%r2),__CLOCK_IDLE_EXIT(%r2)
mvc __TIMER_IDLE_ENTER(8,%r2),__TIMER_IDLE_EXIT(%r3) mvc __TIMER_IDLE_ENTER(8,%r2),__TIMER_IDLE_EXIT(%r3)
@ -913,12 +913,12 @@ cleanup_idle:
stm %r9,%r10,__LC_SYSTEM_TIMER stm %r9,%r10,__LC_SYSTEM_TIMER
mvc __LC_LAST_UPDATE_TIMER(8),__TIMER_IDLE_EXIT(%r2) mvc __LC_LAST_UPDATE_TIMER(8),__TIMER_IDLE_EXIT(%r2)
# prepare return psw # prepare return psw
n %r8,BASED(cleanup_idle_wait) # clear irq & wait state bits n %r8,BASED(.Lcleanup_idle_wait) # clear irq & wait state bits
l %r9,24(%r11) # return from psw_idle l %r9,24(%r11) # return from psw_idle
br %r14 br %r14
cleanup_idle_insn: .Lcleanup_idle_insn:
.long psw_idle_lpsw + 0x80000000 .long .Lpsw_idle_lpsw + 0x80000000
cleanup_idle_wait: .Lcleanup_idle_wait:
.long 0xfcfdffff .long 0xfcfdffff
/* /*
@ -933,30 +933,30 @@ cleanup_idle_wait:
/* /*
* Symbol constants * Symbol constants
*/ */
.Ldo_machine_check: .long s390_do_machine_check .Lc_do_machine_check: .long s390_do_machine_check
.Lhandle_mcck: .long s390_handle_mcck .Lc_handle_mcck: .long s390_handle_mcck
.Ldo_IRQ: .long do_IRQ .Lc_do_IRQ: .long do_IRQ
.Ldo_signal: .long do_signal .Lc_do_signal: .long do_signal
.Ldo_notify_resume: .long do_notify_resume .Lc_do_notify_resume: .long do_notify_resume
.Ldo_per_trap: .long do_per_trap .Lc_do_per_trap: .long do_per_trap
.Ljump_table: .long pgm_check_table .Lc_jump_table: .long pgm_check_table
.Lschedule: .long schedule .Lc_schedule: .long schedule
#ifdef CONFIG_PREEMPT #ifdef CONFIG_PREEMPT
.Lpreempt_irq: .long preempt_schedule_irq .Lc_preempt_irq: .long preempt_schedule_irq
#endif #endif
.Ltrace_enter: .long do_syscall_trace_enter .Lc_trace_enter: .long do_syscall_trace_enter
.Ltrace_exit: .long do_syscall_trace_exit .Lc_trace_exit: .long do_syscall_trace_exit
.Lschedule_tail: .long schedule_tail .Lc_schedule_tail: .long schedule_tail
.Lsysc_per: .long sysc_per + 0x80000000 .Lc_sysc_per: .long .Lsysc_per + 0x80000000
#ifdef CONFIG_TRACE_IRQFLAGS #ifdef CONFIG_TRACE_IRQFLAGS
.Lhardirqs_on: .long trace_hardirqs_on_caller .Lc_hardirqs_on: .long trace_hardirqs_on_caller
.Lhardirqs_off: .long trace_hardirqs_off_caller .Lc_hardirqs_off: .long trace_hardirqs_off_caller
#endif #endif
#ifdef CONFIG_LOCKDEP #ifdef CONFIG_LOCKDEP
.Llockdep_sys_exit: .long lockdep_sys_exit .Lc_lockdep_sys_exit: .long lockdep_sys_exit
#endif #endif
.Lcritical_start: .long __critical_start + 0x80000000 .Lc_critical_start: .long .L__critical_start + 0x80000000
.Lcritical_length: .long __critical_end - __critical_start .Lc_critical_length: .long .L__critical_end - .L__critical_start
.section .rodata, "a" .section .rodata, "a"
#define SYSCALL(esa,esame,emu) .long esa #define SYSCALL(esa,esame,emu) .long esa

View file

@ -74,4 +74,6 @@ struct old_sigaction;
long sys_s390_personality(unsigned int personality); long sys_s390_personality(unsigned int personality);
long sys_s390_runtime_instr(int command, int signum); long sys_s390_runtime_instr(int command, int signum);
long sys_s390_pci_mmio_write(unsigned long, const void __user *, size_t);
long sys_s390_pci_mmio_read(unsigned long, void __user *, size_t);
#endif /* _ENTRY_H */ #endif /* _ENTRY_H */

View file

@ -91,7 +91,7 @@ _PIF_WORK = (_PIF_PER_TRAP)
.if \reason==1 .if \reason==1
# Some program interrupts are suppressing (e.g. protection). # Some program interrupts are suppressing (e.g. protection).
# We must also check the instruction after SIE in that case. # We must also check the instruction after SIE in that case.
# do_protection_exception will rewind to rewind_pad # do_protection_exception will rewind to .Lrewind_pad
jh .+42 jh .+42
.else .else
jhe .+42 jhe .+42
@ -192,7 +192,7 @@ ENTRY(__switch_to)
lmg %r6,%r15,__SF_GPRS(%r15) # load gprs of next task lmg %r6,%r15,__SF_GPRS(%r15) # load gprs of next task
br %r14 br %r14
__critical_start: .L__critical_start:
/* /*
* SVC interrupt handler routine. System calls are synchronous events and * SVC interrupt handler routine. System calls are synchronous events and
* are executed with interrupts enabled. * are executed with interrupts enabled.
@ -200,15 +200,15 @@ __critical_start:
ENTRY(system_call) ENTRY(system_call)
stpt __LC_SYNC_ENTER_TIMER stpt __LC_SYNC_ENTER_TIMER
sysc_stmg: .Lsysc_stmg:
stmg %r8,%r15,__LC_SAVE_AREA_SYNC stmg %r8,%r15,__LC_SAVE_AREA_SYNC
lg %r10,__LC_LAST_BREAK lg %r10,__LC_LAST_BREAK
lg %r12,__LC_THREAD_INFO lg %r12,__LC_THREAD_INFO
lghi %r14,_PIF_SYSCALL lghi %r14,_PIF_SYSCALL
sysc_per: .Lsysc_per:
lg %r15,__LC_KERNEL_STACK lg %r15,__LC_KERNEL_STACK
la %r11,STACK_FRAME_OVERHEAD(%r15) # pointer to pt_regs la %r11,STACK_FRAME_OVERHEAD(%r15) # pointer to pt_regs
sysc_vtime: .Lsysc_vtime:
UPDATE_VTIME %r13,__LC_SYNC_ENTER_TIMER UPDATE_VTIME %r13,__LC_SYNC_ENTER_TIMER
LAST_BREAK %r13 LAST_BREAK %r13
stmg %r0,%r7,__PT_R0(%r11) stmg %r0,%r7,__PT_R0(%r11)
@ -216,39 +216,39 @@ sysc_vtime:
mvc __PT_PSW(16,%r11),__LC_SVC_OLD_PSW mvc __PT_PSW(16,%r11),__LC_SVC_OLD_PSW
mvc __PT_INT_CODE(4,%r11),__LC_SVC_ILC mvc __PT_INT_CODE(4,%r11),__LC_SVC_ILC
stg %r14,__PT_FLAGS(%r11) stg %r14,__PT_FLAGS(%r11)
sysc_do_svc: .Lsysc_do_svc:
lg %r10,__TI_sysc_table(%r12) # address of system call table lg %r10,__TI_sysc_table(%r12) # address of system call table
llgh %r8,__PT_INT_CODE+2(%r11) llgh %r8,__PT_INT_CODE+2(%r11)
slag %r8,%r8,2 # shift and test for svc 0 slag %r8,%r8,2 # shift and test for svc 0
jnz sysc_nr_ok jnz .Lsysc_nr_ok
# svc 0: system call number in %r1 # svc 0: system call number in %r1
llgfr %r1,%r1 # clear high word in r1 llgfr %r1,%r1 # clear high word in r1
cghi %r1,NR_syscalls cghi %r1,NR_syscalls
jnl sysc_nr_ok jnl .Lsysc_nr_ok
sth %r1,__PT_INT_CODE+2(%r11) sth %r1,__PT_INT_CODE+2(%r11)
slag %r8,%r1,2 slag %r8,%r1,2
sysc_nr_ok: .Lsysc_nr_ok:
xc __SF_BACKCHAIN(8,%r15),__SF_BACKCHAIN(%r15) xc __SF_BACKCHAIN(8,%r15),__SF_BACKCHAIN(%r15)
stg %r2,__PT_ORIG_GPR2(%r11) stg %r2,__PT_ORIG_GPR2(%r11)
stg %r7,STACK_FRAME_OVERHEAD(%r15) stg %r7,STACK_FRAME_OVERHEAD(%r15)
lgf %r9,0(%r8,%r10) # get system call add. lgf %r9,0(%r8,%r10) # get system call add.
tm __TI_flags+7(%r12),_TIF_TRACE tm __TI_flags+7(%r12),_TIF_TRACE
jnz sysc_tracesys jnz .Lsysc_tracesys
basr %r14,%r9 # call sys_xxxx basr %r14,%r9 # call sys_xxxx
stg %r2,__PT_R2(%r11) # store return value stg %r2,__PT_R2(%r11) # store return value
sysc_return: .Lsysc_return:
LOCKDEP_SYS_EXIT LOCKDEP_SYS_EXIT
sysc_tif: .Lsysc_tif:
tm __PT_PSW+1(%r11),0x01 # returning to user ? tm __PT_PSW+1(%r11),0x01 # returning to user ?
jno sysc_restore jno .Lsysc_restore
tm __PT_FLAGS+7(%r11),_PIF_WORK tm __PT_FLAGS+7(%r11),_PIF_WORK
jnz sysc_work jnz .Lsysc_work
tm __TI_flags+7(%r12),_TIF_WORK tm __TI_flags+7(%r12),_TIF_WORK
jnz sysc_work # check for work jnz .Lsysc_work # check for work
tm __LC_CPU_FLAGS+7,_CIF_WORK tm __LC_CPU_FLAGS+7,_CIF_WORK
jnz sysc_work jnz .Lsysc_work
sysc_restore: .Lsysc_restore:
lg %r14,__LC_VDSO_PER_CPU lg %r14,__LC_VDSO_PER_CPU
lmg %r0,%r10,__PT_R0(%r11) lmg %r0,%r10,__PT_R0(%r11)
mvc __LC_RETURN_PSW(16),__PT_PSW(%r11) mvc __LC_RETURN_PSW(16),__PT_PSW(%r11)
@ -256,101 +256,101 @@ sysc_restore:
mvc __VDSO_ECTG_BASE(16,%r14),__LC_EXIT_TIMER mvc __VDSO_ECTG_BASE(16,%r14),__LC_EXIT_TIMER
lmg %r11,%r15,__PT_R11(%r11) lmg %r11,%r15,__PT_R11(%r11)
lpswe __LC_RETURN_PSW lpswe __LC_RETURN_PSW
sysc_done: .Lsysc_done:
# #
# One of the work bits is on. Find out which one. # One of the work bits is on. Find out which one.
# #
sysc_work: .Lsysc_work:
tm __LC_CPU_FLAGS+7,_CIF_MCCK_PENDING tm __LC_CPU_FLAGS+7,_CIF_MCCK_PENDING
jo sysc_mcck_pending jo .Lsysc_mcck_pending
tm __TI_flags+7(%r12),_TIF_NEED_RESCHED tm __TI_flags+7(%r12),_TIF_NEED_RESCHED
jo sysc_reschedule jo .Lsysc_reschedule
#ifdef CONFIG_UPROBES #ifdef CONFIG_UPROBES
tm __TI_flags+7(%r12),_TIF_UPROBE tm __TI_flags+7(%r12),_TIF_UPROBE
jo sysc_uprobe_notify jo .Lsysc_uprobe_notify
#endif #endif
tm __PT_FLAGS+7(%r11),_PIF_PER_TRAP tm __PT_FLAGS+7(%r11),_PIF_PER_TRAP
jo sysc_singlestep jo .Lsysc_singlestep
tm __TI_flags+7(%r12),_TIF_SIGPENDING tm __TI_flags+7(%r12),_TIF_SIGPENDING
jo sysc_sigpending jo .Lsysc_sigpending
tm __TI_flags+7(%r12),_TIF_NOTIFY_RESUME tm __TI_flags+7(%r12),_TIF_NOTIFY_RESUME
jo sysc_notify_resume jo .Lsysc_notify_resume
tm __LC_CPU_FLAGS+7,_CIF_ASCE tm __LC_CPU_FLAGS+7,_CIF_ASCE
jo sysc_uaccess jo .Lsysc_uaccess
j sysc_return # beware of critical section cleanup j .Lsysc_return # beware of critical section cleanup
# #
# _TIF_NEED_RESCHED is set, call schedule # _TIF_NEED_RESCHED is set, call schedule
# #
sysc_reschedule: .Lsysc_reschedule:
larl %r14,sysc_return larl %r14,.Lsysc_return
jg schedule jg schedule
# #
# _CIF_MCCK_PENDING is set, call handler # _CIF_MCCK_PENDING is set, call handler
# #
sysc_mcck_pending: .Lsysc_mcck_pending:
larl %r14,sysc_return larl %r14,.Lsysc_return
jg s390_handle_mcck # TIF bit will be cleared by handler jg s390_handle_mcck # TIF bit will be cleared by handler
# #
# _CIF_ASCE is set, load user space asce # _CIF_ASCE is set, load user space asce
# #
sysc_uaccess: .Lsysc_uaccess:
ni __LC_CPU_FLAGS+7,255-_CIF_ASCE ni __LC_CPU_FLAGS+7,255-_CIF_ASCE
lctlg %c1,%c1,__LC_USER_ASCE # load primary asce lctlg %c1,%c1,__LC_USER_ASCE # load primary asce
j sysc_return j .Lsysc_return
# #
# _TIF_SIGPENDING is set, call do_signal # _TIF_SIGPENDING is set, call do_signal
# #
sysc_sigpending: .Lsysc_sigpending:
lgr %r2,%r11 # pass pointer to pt_regs lgr %r2,%r11 # pass pointer to pt_regs
brasl %r14,do_signal brasl %r14,do_signal
tm __PT_FLAGS+7(%r11),_PIF_SYSCALL tm __PT_FLAGS+7(%r11),_PIF_SYSCALL
jno sysc_return jno .Lsysc_return
lmg %r2,%r7,__PT_R2(%r11) # load svc arguments lmg %r2,%r7,__PT_R2(%r11) # load svc arguments
lg %r10,__TI_sysc_table(%r12) # address of system call table lg %r10,__TI_sysc_table(%r12) # address of system call table
lghi %r8,0 # svc 0 returns -ENOSYS lghi %r8,0 # svc 0 returns -ENOSYS
llgh %r1,__PT_INT_CODE+2(%r11) # load new svc number llgh %r1,__PT_INT_CODE+2(%r11) # load new svc number
cghi %r1,NR_syscalls cghi %r1,NR_syscalls
jnl sysc_nr_ok # invalid svc number -> do svc 0 jnl .Lsysc_nr_ok # invalid svc number -> do svc 0
slag %r8,%r1,2 slag %r8,%r1,2
j sysc_nr_ok # restart svc j .Lsysc_nr_ok # restart svc
# #
# _TIF_NOTIFY_RESUME is set, call do_notify_resume # _TIF_NOTIFY_RESUME is set, call do_notify_resume
# #
sysc_notify_resume: .Lsysc_notify_resume:
lgr %r2,%r11 # pass pointer to pt_regs lgr %r2,%r11 # pass pointer to pt_regs
larl %r14,sysc_return larl %r14,.Lsysc_return
jg do_notify_resume jg do_notify_resume
# #
# _TIF_UPROBE is set, call uprobe_notify_resume # _TIF_UPROBE is set, call uprobe_notify_resume
# #
#ifdef CONFIG_UPROBES #ifdef CONFIG_UPROBES
sysc_uprobe_notify: .Lsysc_uprobe_notify:
lgr %r2,%r11 # pass pointer to pt_regs lgr %r2,%r11 # pass pointer to pt_regs
larl %r14,sysc_return larl %r14,.Lsysc_return
jg uprobe_notify_resume jg uprobe_notify_resume
#endif #endif
# #
# _PIF_PER_TRAP is set, call do_per_trap # _PIF_PER_TRAP is set, call do_per_trap
# #
sysc_singlestep: .Lsysc_singlestep:
ni __PT_FLAGS+7(%r11),255-_PIF_PER_TRAP ni __PT_FLAGS+7(%r11),255-_PIF_PER_TRAP
lgr %r2,%r11 # pass pointer to pt_regs lgr %r2,%r11 # pass pointer to pt_regs
larl %r14,sysc_return larl %r14,.Lsysc_return
jg do_per_trap jg do_per_trap
# #
# call tracehook_report_syscall_entry/tracehook_report_syscall_exit before # call tracehook_report_syscall_entry/tracehook_report_syscall_exit before
# and after the system call # and after the system call
# #
sysc_tracesys: .Lsysc_tracesys:
lgr %r2,%r11 # pass pointer to pt_regs lgr %r2,%r11 # pass pointer to pt_regs
la %r3,0 la %r3,0
llgh %r0,__PT_INT_CODE+2(%r11) llgh %r0,__PT_INT_CODE+2(%r11)
@ -358,20 +358,20 @@ sysc_tracesys:
brasl %r14,do_syscall_trace_enter brasl %r14,do_syscall_trace_enter
lghi %r0,NR_syscalls lghi %r0,NR_syscalls
clgr %r0,%r2 clgr %r0,%r2
jnh sysc_tracenogo jnh .Lsysc_tracenogo
sllg %r8,%r2,2 sllg %r8,%r2,2
lgf %r9,0(%r8,%r10) lgf %r9,0(%r8,%r10)
sysc_tracego: .Lsysc_tracego:
lmg %r3,%r7,__PT_R3(%r11) lmg %r3,%r7,__PT_R3(%r11)
stg %r7,STACK_FRAME_OVERHEAD(%r15) stg %r7,STACK_FRAME_OVERHEAD(%r15)
lg %r2,__PT_ORIG_GPR2(%r11) lg %r2,__PT_ORIG_GPR2(%r11)
basr %r14,%r9 # call sys_xxx basr %r14,%r9 # call sys_xxx
stg %r2,__PT_R2(%r11) # store return value stg %r2,__PT_R2(%r11) # store return value
sysc_tracenogo: .Lsysc_tracenogo:
tm __TI_flags+7(%r12),_TIF_TRACE tm __TI_flags+7(%r12),_TIF_TRACE
jz sysc_return jz .Lsysc_return
lgr %r2,%r11 # pass pointer to pt_regs lgr %r2,%r11 # pass pointer to pt_regs
larl %r14,sysc_return larl %r14,.Lsysc_return
jg do_syscall_trace_exit jg do_syscall_trace_exit
# #
@ -384,13 +384,13 @@ ENTRY(ret_from_fork)
TRACE_IRQS_ON TRACE_IRQS_ON
ssm __LC_SVC_NEW_PSW # reenable interrupts ssm __LC_SVC_NEW_PSW # reenable interrupts
tm __PT_PSW+1(%r11),0x01 # forking a kernel thread ? tm __PT_PSW+1(%r11),0x01 # forking a kernel thread ?
jne sysc_tracenogo jne .Lsysc_tracenogo
# it's a kernel thread # it's a kernel thread
lmg %r9,%r10,__PT_R9(%r11) # load gprs lmg %r9,%r10,__PT_R9(%r11) # load gprs
ENTRY(kernel_thread_starter) ENTRY(kernel_thread_starter)
la %r2,0(%r10) la %r2,0(%r10)
basr %r14,%r9 basr %r14,%r9
j sysc_tracenogo j .Lsysc_tracenogo
/* /*
* Program check handler routine * Program check handler routine
@ -409,7 +409,7 @@ ENTRY(pgm_check_handler)
tmhh %r8,0x4000 # PER bit set in old PSW ? tmhh %r8,0x4000 # PER bit set in old PSW ?
jnz 0f # -> enabled, can't be a double fault jnz 0f # -> enabled, can't be a double fault
tm __LC_PGM_ILC+3,0x80 # check for per exception tm __LC_PGM_ILC+3,0x80 # check for per exception
jnz pgm_svcper # -> single stepped svc jnz .Lpgm_svcper # -> single stepped svc
0: CHECK_STACK STACK_SIZE,__LC_SAVE_AREA_SYNC 0: CHECK_STACK STACK_SIZE,__LC_SAVE_AREA_SYNC
aghi %r15,-(STACK_FRAME_OVERHEAD + __PT_SIZE) aghi %r15,-(STACK_FRAME_OVERHEAD + __PT_SIZE)
j 2f j 2f
@ -432,7 +432,7 @@ ENTRY(pgm_check_handler)
tm __LC_PGM_ILC+3,0x80 # check for per exception tm __LC_PGM_ILC+3,0x80 # check for per exception
jz 0f jz 0f
tmhh %r8,0x0001 # kernel per event ? tmhh %r8,0x0001 # kernel per event ?
jz pgm_kprobe jz .Lpgm_kprobe
oi __PT_FLAGS+7(%r11),_PIF_PER_TRAP oi __PT_FLAGS+7(%r11),_PIF_PER_TRAP
mvc __THREAD_per_address(8,%r14),__LC_PER_ADDRESS mvc __THREAD_per_address(8,%r14),__LC_PER_ADDRESS
mvc __THREAD_per_cause(2,%r14),__LC_PER_CODE mvc __THREAD_per_cause(2,%r14),__LC_PER_CODE
@ -443,31 +443,31 @@ ENTRY(pgm_check_handler)
llgh %r10,__PT_INT_CODE+2(%r11) llgh %r10,__PT_INT_CODE+2(%r11)
nill %r10,0x007f nill %r10,0x007f
sll %r10,2 sll %r10,2
je sysc_return je .Lsysc_return
lgf %r1,0(%r10,%r1) # load address of handler routine lgf %r1,0(%r10,%r1) # load address of handler routine
lgr %r2,%r11 # pass pointer to pt_regs lgr %r2,%r11 # pass pointer to pt_regs
basr %r14,%r1 # branch to interrupt-handler basr %r14,%r1 # branch to interrupt-handler
j sysc_return j .Lsysc_return
# #
# PER event in supervisor state, must be kprobes # PER event in supervisor state, must be kprobes
# #
pgm_kprobe: .Lpgm_kprobe:
REENABLE_IRQS REENABLE_IRQS
xc __SF_BACKCHAIN(8,%r15),__SF_BACKCHAIN(%r15) xc __SF_BACKCHAIN(8,%r15),__SF_BACKCHAIN(%r15)
lgr %r2,%r11 # pass pointer to pt_regs lgr %r2,%r11 # pass pointer to pt_regs
brasl %r14,do_per_trap brasl %r14,do_per_trap
j sysc_return j .Lsysc_return
# #
# single stepped system call # single stepped system call
# #
pgm_svcper: .Lpgm_svcper:
mvc __LC_RETURN_PSW(8),__LC_SVC_NEW_PSW mvc __LC_RETURN_PSW(8),__LC_SVC_NEW_PSW
larl %r14,sysc_per larl %r14,.Lsysc_per
stg %r14,__LC_RETURN_PSW+8 stg %r14,__LC_RETURN_PSW+8
lghi %r14,_PIF_SYSCALL | _PIF_PER_TRAP lghi %r14,_PIF_SYSCALL | _PIF_PER_TRAP
lpswe __LC_RETURN_PSW # branch to sysc_per and enable irqs lpswe __LC_RETURN_PSW # branch to .Lsysc_per and enable irqs
/* /*
* IO interrupt handler routine * IO interrupt handler routine
@ -483,10 +483,10 @@ ENTRY(io_int_handler)
HANDLE_SIE_INTERCEPT %r14,2 HANDLE_SIE_INTERCEPT %r14,2
SWITCH_ASYNC __LC_SAVE_AREA_ASYNC,__LC_ASYNC_STACK,STACK_SHIFT SWITCH_ASYNC __LC_SAVE_AREA_ASYNC,__LC_ASYNC_STACK,STACK_SHIFT
tmhh %r8,0x0001 # interrupting from user? tmhh %r8,0x0001 # interrupting from user?
jz io_skip jz .Lio_skip
UPDATE_VTIME %r14,__LC_ASYNC_ENTER_TIMER UPDATE_VTIME %r14,__LC_ASYNC_ENTER_TIMER
LAST_BREAK %r14 LAST_BREAK %r14
io_skip: .Lio_skip:
stmg %r0,%r7,__PT_R0(%r11) stmg %r0,%r7,__PT_R0(%r11)
mvc __PT_R8(64,%r11),__LC_SAVE_AREA_ASYNC mvc __PT_R8(64,%r11),__LC_SAVE_AREA_ASYNC
stmg %r8,%r9,__PT_PSW(%r11) stmg %r8,%r9,__PT_PSW(%r11)
@ -494,29 +494,29 @@ io_skip:
xc __PT_FLAGS(8,%r11),__PT_FLAGS(%r11) xc __PT_FLAGS(8,%r11),__PT_FLAGS(%r11)
TRACE_IRQS_OFF TRACE_IRQS_OFF
xc __SF_BACKCHAIN(8,%r15),__SF_BACKCHAIN(%r15) xc __SF_BACKCHAIN(8,%r15),__SF_BACKCHAIN(%r15)
io_loop: .Lio_loop:
lgr %r2,%r11 # pass pointer to pt_regs lgr %r2,%r11 # pass pointer to pt_regs
lghi %r3,IO_INTERRUPT lghi %r3,IO_INTERRUPT
tm __PT_INT_CODE+8(%r11),0x80 # adapter interrupt ? tm __PT_INT_CODE+8(%r11),0x80 # adapter interrupt ?
jz io_call jz .Lio_call
lghi %r3,THIN_INTERRUPT lghi %r3,THIN_INTERRUPT
io_call: .Lio_call:
brasl %r14,do_IRQ brasl %r14,do_IRQ
tm __LC_MACHINE_FLAGS+6,0x10 # MACHINE_FLAG_LPAR tm __LC_MACHINE_FLAGS+6,0x10 # MACHINE_FLAG_LPAR
jz io_return jz .Lio_return
tpi 0 tpi 0
jz io_return jz .Lio_return
mvc __PT_INT_CODE(12,%r11),__LC_SUBCHANNEL_ID mvc __PT_INT_CODE(12,%r11),__LC_SUBCHANNEL_ID
j io_loop j .Lio_loop
io_return: .Lio_return:
LOCKDEP_SYS_EXIT LOCKDEP_SYS_EXIT
TRACE_IRQS_ON TRACE_IRQS_ON
io_tif: .Lio_tif:
tm __TI_flags+7(%r12),_TIF_WORK tm __TI_flags+7(%r12),_TIF_WORK
jnz io_work # there is work to do (signals etc.) jnz .Lio_work # there is work to do (signals etc.)
tm __LC_CPU_FLAGS+7,_CIF_WORK tm __LC_CPU_FLAGS+7,_CIF_WORK
jnz io_work jnz .Lio_work
io_restore: .Lio_restore:
lg %r14,__LC_VDSO_PER_CPU lg %r14,__LC_VDSO_PER_CPU
lmg %r0,%r10,__PT_R0(%r11) lmg %r0,%r10,__PT_R0(%r11)
mvc __LC_RETURN_PSW(16),__PT_PSW(%r11) mvc __LC_RETURN_PSW(16),__PT_PSW(%r11)
@ -524,7 +524,7 @@ io_restore:
mvc __VDSO_ECTG_BASE(16,%r14),__LC_EXIT_TIMER mvc __VDSO_ECTG_BASE(16,%r14),__LC_EXIT_TIMER
lmg %r11,%r15,__PT_R11(%r11) lmg %r11,%r15,__PT_R11(%r11)
lpswe __LC_RETURN_PSW lpswe __LC_RETURN_PSW
io_done: .Lio_done:
# #
# There is work todo, find out in which context we have been interrupted: # There is work todo, find out in which context we have been interrupted:
@ -535,15 +535,15 @@ io_done:
# the preemption counter and if it is zero call preempt_schedule_irq # the preemption counter and if it is zero call preempt_schedule_irq
# Before any work can be done, a switch to the kernel stack is required. # Before any work can be done, a switch to the kernel stack is required.
# #
io_work: .Lio_work:
tm __PT_PSW+1(%r11),0x01 # returning to user ? tm __PT_PSW+1(%r11),0x01 # returning to user ?
jo io_work_user # yes -> do resched & signal jo .Lio_work_user # yes -> do resched & signal
#ifdef CONFIG_PREEMPT #ifdef CONFIG_PREEMPT
# check for preemptive scheduling # check for preemptive scheduling
icm %r0,15,__TI_precount(%r12) icm %r0,15,__TI_precount(%r12)
jnz io_restore # preemption is disabled jnz .Lio_restore # preemption is disabled
tm __TI_flags+7(%r12),_TIF_NEED_RESCHED tm __TI_flags+7(%r12),_TIF_NEED_RESCHED
jno io_restore jno .Lio_restore
# switch to kernel stack # switch to kernel stack
lg %r1,__PT_R15(%r11) lg %r1,__PT_R15(%r11)
aghi %r1,-(STACK_FRAME_OVERHEAD + __PT_SIZE) aghi %r1,-(STACK_FRAME_OVERHEAD + __PT_SIZE)
@ -551,19 +551,19 @@ io_work:
xc __SF_BACKCHAIN(8,%r1),__SF_BACKCHAIN(%r1) xc __SF_BACKCHAIN(8,%r1),__SF_BACKCHAIN(%r1)
la %r11,STACK_FRAME_OVERHEAD(%r1) la %r11,STACK_FRAME_OVERHEAD(%r1)
lgr %r15,%r1 lgr %r15,%r1
# TRACE_IRQS_ON already done at io_return, call # TRACE_IRQS_ON already done at .Lio_return, call
# TRACE_IRQS_OFF to keep things symmetrical # TRACE_IRQS_OFF to keep things symmetrical
TRACE_IRQS_OFF TRACE_IRQS_OFF
brasl %r14,preempt_schedule_irq brasl %r14,preempt_schedule_irq
j io_return j .Lio_return
#else #else
j io_restore j .Lio_restore
#endif #endif
# #
# Need to do work before returning to userspace, switch to kernel stack # Need to do work before returning to userspace, switch to kernel stack
# #
io_work_user: .Lio_work_user:
lg %r1,__LC_KERNEL_STACK lg %r1,__LC_KERNEL_STACK
mvc STACK_FRAME_OVERHEAD(__PT_SIZE,%r1),0(%r11) mvc STACK_FRAME_OVERHEAD(__PT_SIZE,%r1),0(%r11)
xc __SF_BACKCHAIN(8,%r1),__SF_BACKCHAIN(%r1) xc __SF_BACKCHAIN(8,%r1),__SF_BACKCHAIN(%r1)
@ -573,70 +573,70 @@ io_work_user:
# #
# One of the work bits is on. Find out which one. # One of the work bits is on. Find out which one.
# #
io_work_tif: .Lio_work_tif:
tm __LC_CPU_FLAGS+7,_CIF_MCCK_PENDING tm __LC_CPU_FLAGS+7,_CIF_MCCK_PENDING
jo io_mcck_pending jo .Lio_mcck_pending
tm __TI_flags+7(%r12),_TIF_NEED_RESCHED tm __TI_flags+7(%r12),_TIF_NEED_RESCHED
jo io_reschedule jo .Lio_reschedule
tm __TI_flags+7(%r12),_TIF_SIGPENDING tm __TI_flags+7(%r12),_TIF_SIGPENDING
jo io_sigpending jo .Lio_sigpending
tm __TI_flags+7(%r12),_TIF_NOTIFY_RESUME tm __TI_flags+7(%r12),_TIF_NOTIFY_RESUME
jo io_notify_resume jo .Lio_notify_resume
tm __LC_CPU_FLAGS+7,_CIF_ASCE tm __LC_CPU_FLAGS+7,_CIF_ASCE
jo io_uaccess jo .Lio_uaccess
j io_return # beware of critical section cleanup j .Lio_return # beware of critical section cleanup
# #
# _CIF_MCCK_PENDING is set, call handler # _CIF_MCCK_PENDING is set, call handler
# #
io_mcck_pending: .Lio_mcck_pending:
# TRACE_IRQS_ON already done at io_return # TRACE_IRQS_ON already done at .Lio_return
brasl %r14,s390_handle_mcck # TIF bit will be cleared by handler brasl %r14,s390_handle_mcck # TIF bit will be cleared by handler
TRACE_IRQS_OFF TRACE_IRQS_OFF
j io_return j .Lio_return
# #
# _CIF_ASCE is set, load user space asce # _CIF_ASCE is set, load user space asce
# #
io_uaccess: .Lio_uaccess:
ni __LC_CPU_FLAGS+7,255-_CIF_ASCE ni __LC_CPU_FLAGS+7,255-_CIF_ASCE
lctlg %c1,%c1,__LC_USER_ASCE # load primary asce lctlg %c1,%c1,__LC_USER_ASCE # load primary asce
j io_return j .Lio_return
# #
# _TIF_NEED_RESCHED is set, call schedule # _TIF_NEED_RESCHED is set, call schedule
# #
io_reschedule: .Lio_reschedule:
# TRACE_IRQS_ON already done at io_return # TRACE_IRQS_ON already done at .Lio_return
ssm __LC_SVC_NEW_PSW # reenable interrupts ssm __LC_SVC_NEW_PSW # reenable interrupts
brasl %r14,schedule # call scheduler brasl %r14,schedule # call scheduler
ssm __LC_PGM_NEW_PSW # disable I/O and ext. interrupts ssm __LC_PGM_NEW_PSW # disable I/O and ext. interrupts
TRACE_IRQS_OFF TRACE_IRQS_OFF
j io_return j .Lio_return
# #
# _TIF_SIGPENDING or is set, call do_signal # _TIF_SIGPENDING or is set, call do_signal
# #
io_sigpending: .Lio_sigpending:
# TRACE_IRQS_ON already done at io_return # TRACE_IRQS_ON already done at .Lio_return
ssm __LC_SVC_NEW_PSW # reenable interrupts ssm __LC_SVC_NEW_PSW # reenable interrupts
lgr %r2,%r11 # pass pointer to pt_regs lgr %r2,%r11 # pass pointer to pt_regs
brasl %r14,do_signal brasl %r14,do_signal
ssm __LC_PGM_NEW_PSW # disable I/O and ext. interrupts ssm __LC_PGM_NEW_PSW # disable I/O and ext. interrupts
TRACE_IRQS_OFF TRACE_IRQS_OFF
j io_return j .Lio_return
# #
# _TIF_NOTIFY_RESUME or is set, call do_notify_resume # _TIF_NOTIFY_RESUME or is set, call do_notify_resume
# #
io_notify_resume: .Lio_notify_resume:
# TRACE_IRQS_ON already done at io_return # TRACE_IRQS_ON already done at .Lio_return
ssm __LC_SVC_NEW_PSW # reenable interrupts ssm __LC_SVC_NEW_PSW # reenable interrupts
lgr %r2,%r11 # pass pointer to pt_regs lgr %r2,%r11 # pass pointer to pt_regs
brasl %r14,do_notify_resume brasl %r14,do_notify_resume
ssm __LC_PGM_NEW_PSW # disable I/O and ext. interrupts ssm __LC_PGM_NEW_PSW # disable I/O and ext. interrupts
TRACE_IRQS_OFF TRACE_IRQS_OFF
j io_return j .Lio_return
/* /*
* External interrupt handler routine * External interrupt handler routine
@ -652,10 +652,10 @@ ENTRY(ext_int_handler)
HANDLE_SIE_INTERCEPT %r14,3 HANDLE_SIE_INTERCEPT %r14,3
SWITCH_ASYNC __LC_SAVE_AREA_ASYNC,__LC_ASYNC_STACK,STACK_SHIFT SWITCH_ASYNC __LC_SAVE_AREA_ASYNC,__LC_ASYNC_STACK,STACK_SHIFT
tmhh %r8,0x0001 # interrupting from user ? tmhh %r8,0x0001 # interrupting from user ?
jz ext_skip jz .Lext_skip
UPDATE_VTIME %r14,__LC_ASYNC_ENTER_TIMER UPDATE_VTIME %r14,__LC_ASYNC_ENTER_TIMER
LAST_BREAK %r14 LAST_BREAK %r14
ext_skip: .Lext_skip:
stmg %r0,%r7,__PT_R0(%r11) stmg %r0,%r7,__PT_R0(%r11)
mvc __PT_R8(64,%r11),__LC_SAVE_AREA_ASYNC mvc __PT_R8(64,%r11),__LC_SAVE_AREA_ASYNC
stmg %r8,%r9,__PT_PSW(%r11) stmg %r8,%r9,__PT_PSW(%r11)
@ -669,23 +669,23 @@ ext_skip:
lgr %r2,%r11 # pass pointer to pt_regs lgr %r2,%r11 # pass pointer to pt_regs
lghi %r3,EXT_INTERRUPT lghi %r3,EXT_INTERRUPT
brasl %r14,do_IRQ brasl %r14,do_IRQ
j io_return j .Lio_return
/* /*
* Load idle PSW. The second "half" of this function is in cleanup_idle. * Load idle PSW. The second "half" of this function is in .Lcleanup_idle.
*/ */
ENTRY(psw_idle) ENTRY(psw_idle)
stg %r3,__SF_EMPTY(%r15) stg %r3,__SF_EMPTY(%r15)
larl %r1,psw_idle_lpsw+4 larl %r1,.Lpsw_idle_lpsw+4
stg %r1,__SF_EMPTY+8(%r15) stg %r1,__SF_EMPTY+8(%r15)
STCK __CLOCK_IDLE_ENTER(%r2) STCK __CLOCK_IDLE_ENTER(%r2)
stpt __TIMER_IDLE_ENTER(%r2) stpt __TIMER_IDLE_ENTER(%r2)
psw_idle_lpsw: .Lpsw_idle_lpsw:
lpswe __SF_EMPTY(%r15) lpswe __SF_EMPTY(%r15)
br %r14 br %r14
psw_idle_end: .Lpsw_idle_end:
__critical_end: .L__critical_end:
/* /*
* Machine check handler routines * Machine check handler routines
@ -701,7 +701,7 @@ ENTRY(mcck_int_handler)
lmg %r8,%r9,__LC_MCK_OLD_PSW lmg %r8,%r9,__LC_MCK_OLD_PSW
HANDLE_SIE_INTERCEPT %r14,4 HANDLE_SIE_INTERCEPT %r14,4
tm __LC_MCCK_CODE,0x80 # system damage? tm __LC_MCCK_CODE,0x80 # system damage?
jo mcck_panic # yes -> rest of mcck code invalid jo .Lmcck_panic # yes -> rest of mcck code invalid
lghi %r14,__LC_CPU_TIMER_SAVE_AREA lghi %r14,__LC_CPU_TIMER_SAVE_AREA
mvc __LC_MCCK_ENTER_TIMER(8),0(%r14) mvc __LC_MCCK_ENTER_TIMER(8),0(%r14)
tm __LC_MCCK_CODE+5,0x02 # stored cpu timer value valid? tm __LC_MCCK_CODE+5,0x02 # stored cpu timer value valid?
@ -719,13 +719,13 @@ ENTRY(mcck_int_handler)
2: spt 0(%r14) 2: spt 0(%r14)
mvc __LC_MCCK_ENTER_TIMER(8),0(%r14) mvc __LC_MCCK_ENTER_TIMER(8),0(%r14)
3: tm __LC_MCCK_CODE+2,0x09 # mwp + ia of old psw valid? 3: tm __LC_MCCK_CODE+2,0x09 # mwp + ia of old psw valid?
jno mcck_panic # no -> skip cleanup critical jno .Lmcck_panic # no -> skip cleanup critical
SWITCH_ASYNC __LC_GPREGS_SAVE_AREA+64,__LC_PANIC_STACK,PAGE_SHIFT SWITCH_ASYNC __LC_GPREGS_SAVE_AREA+64,__LC_PANIC_STACK,PAGE_SHIFT
tm %r8,0x0001 # interrupting from user ? tm %r8,0x0001 # interrupting from user ?
jz mcck_skip jz .Lmcck_skip
UPDATE_VTIME %r14,__LC_MCCK_ENTER_TIMER UPDATE_VTIME %r14,__LC_MCCK_ENTER_TIMER
LAST_BREAK %r14 LAST_BREAK %r14
mcck_skip: .Lmcck_skip:
lghi %r14,__LC_GPREGS_SAVE_AREA+64 lghi %r14,__LC_GPREGS_SAVE_AREA+64
stmg %r0,%r7,__PT_R0(%r11) stmg %r0,%r7,__PT_R0(%r11)
mvc __PT_R8(64,%r11),0(%r14) mvc __PT_R8(64,%r11),0(%r14)
@ -735,7 +735,7 @@ mcck_skip:
lgr %r2,%r11 # pass pointer to pt_regs lgr %r2,%r11 # pass pointer to pt_regs
brasl %r14,s390_do_machine_check brasl %r14,s390_do_machine_check
tm __PT_PSW+1(%r11),0x01 # returning to user ? tm __PT_PSW+1(%r11),0x01 # returning to user ?
jno mcck_return jno .Lmcck_return
lg %r1,__LC_KERNEL_STACK # switch to kernel stack lg %r1,__LC_KERNEL_STACK # switch to kernel stack
mvc STACK_FRAME_OVERHEAD(__PT_SIZE,%r1),0(%r11) mvc STACK_FRAME_OVERHEAD(__PT_SIZE,%r1),0(%r11)
xc __SF_BACKCHAIN(8,%r1),__SF_BACKCHAIN(%r1) xc __SF_BACKCHAIN(8,%r1),__SF_BACKCHAIN(%r1)
@ -743,11 +743,11 @@ mcck_skip:
lgr %r15,%r1 lgr %r15,%r1
ssm __LC_PGM_NEW_PSW # turn dat on, keep irqs off ssm __LC_PGM_NEW_PSW # turn dat on, keep irqs off
tm __LC_CPU_FLAGS+7,_CIF_MCCK_PENDING tm __LC_CPU_FLAGS+7,_CIF_MCCK_PENDING
jno mcck_return jno .Lmcck_return
TRACE_IRQS_OFF TRACE_IRQS_OFF
brasl %r14,s390_handle_mcck brasl %r14,s390_handle_mcck
TRACE_IRQS_ON TRACE_IRQS_ON
mcck_return: .Lmcck_return:
lg %r14,__LC_VDSO_PER_CPU lg %r14,__LC_VDSO_PER_CPU
lmg %r0,%r10,__PT_R0(%r11) lmg %r0,%r10,__PT_R0(%r11)
mvc __LC_RETURN_MCCK_PSW(16),__PT_PSW(%r11) # move return PSW mvc __LC_RETURN_MCCK_PSW(16),__PT_PSW(%r11) # move return PSW
@ -758,14 +758,14 @@ mcck_return:
0: lmg %r11,%r15,__PT_R11(%r11) 0: lmg %r11,%r15,__PT_R11(%r11)
lpswe __LC_RETURN_MCCK_PSW lpswe __LC_RETURN_MCCK_PSW
mcck_panic: .Lmcck_panic:
lg %r14,__LC_PANIC_STACK lg %r14,__LC_PANIC_STACK
slgr %r14,%r15 slgr %r14,%r15
srag %r14,%r14,PAGE_SHIFT srag %r14,%r14,PAGE_SHIFT
jz 0f jz 0f
lg %r15,__LC_PANIC_STACK lg %r15,__LC_PANIC_STACK
0: aghi %r15,-(STACK_FRAME_OVERHEAD + __PT_SIZE) 0: aghi %r15,-(STACK_FRAME_OVERHEAD + __PT_SIZE)
j mcck_skip j .Lmcck_skip
# #
# PSW restart interrupt handler # PSW restart interrupt handler
@ -815,69 +815,69 @@ stack_overflow:
#endif #endif
.align 8 .align 8
cleanup_table: .Lcleanup_table:
.quad system_call .quad system_call
.quad sysc_do_svc .quad .Lsysc_do_svc
.quad sysc_tif .quad .Lsysc_tif
.quad sysc_restore .quad .Lsysc_restore
.quad sysc_done .quad .Lsysc_done
.quad io_tif .quad .Lio_tif
.quad io_restore .quad .Lio_restore
.quad io_done .quad .Lio_done
.quad psw_idle .quad psw_idle
.quad psw_idle_end .quad .Lpsw_idle_end
cleanup_critical: cleanup_critical:
clg %r9,BASED(cleanup_table) # system_call clg %r9,BASED(.Lcleanup_table) # system_call
jl 0f jl 0f
clg %r9,BASED(cleanup_table+8) # sysc_do_svc clg %r9,BASED(.Lcleanup_table+8) # .Lsysc_do_svc
jl cleanup_system_call jl .Lcleanup_system_call
clg %r9,BASED(cleanup_table+16) # sysc_tif clg %r9,BASED(.Lcleanup_table+16) # .Lsysc_tif
jl 0f jl 0f
clg %r9,BASED(cleanup_table+24) # sysc_restore clg %r9,BASED(.Lcleanup_table+24) # .Lsysc_restore
jl cleanup_sysc_tif jl .Lcleanup_sysc_tif
clg %r9,BASED(cleanup_table+32) # sysc_done clg %r9,BASED(.Lcleanup_table+32) # .Lsysc_done
jl cleanup_sysc_restore jl .Lcleanup_sysc_restore
clg %r9,BASED(cleanup_table+40) # io_tif clg %r9,BASED(.Lcleanup_table+40) # .Lio_tif
jl 0f jl 0f
clg %r9,BASED(cleanup_table+48) # io_restore clg %r9,BASED(.Lcleanup_table+48) # .Lio_restore
jl cleanup_io_tif jl .Lcleanup_io_tif
clg %r9,BASED(cleanup_table+56) # io_done clg %r9,BASED(.Lcleanup_table+56) # .Lio_done
jl cleanup_io_restore jl .Lcleanup_io_restore
clg %r9,BASED(cleanup_table+64) # psw_idle clg %r9,BASED(.Lcleanup_table+64) # psw_idle
jl 0f jl 0f
clg %r9,BASED(cleanup_table+72) # psw_idle_end clg %r9,BASED(.Lcleanup_table+72) # .Lpsw_idle_end
jl cleanup_idle jl .Lcleanup_idle
0: br %r14 0: br %r14
cleanup_system_call: .Lcleanup_system_call:
# check if stpt has been executed # check if stpt has been executed
clg %r9,BASED(cleanup_system_call_insn) clg %r9,BASED(.Lcleanup_system_call_insn)
jh 0f jh 0f
mvc __LC_SYNC_ENTER_TIMER(8),__LC_ASYNC_ENTER_TIMER mvc __LC_SYNC_ENTER_TIMER(8),__LC_ASYNC_ENTER_TIMER
cghi %r11,__LC_SAVE_AREA_ASYNC cghi %r11,__LC_SAVE_AREA_ASYNC
je 0f je 0f
mvc __LC_SYNC_ENTER_TIMER(8),__LC_MCCK_ENTER_TIMER mvc __LC_SYNC_ENTER_TIMER(8),__LC_MCCK_ENTER_TIMER
0: # check if stmg has been executed 0: # check if stmg has been executed
clg %r9,BASED(cleanup_system_call_insn+8) clg %r9,BASED(.Lcleanup_system_call_insn+8)
jh 0f jh 0f
mvc __LC_SAVE_AREA_SYNC(64),0(%r11) mvc __LC_SAVE_AREA_SYNC(64),0(%r11)
0: # check if base register setup + TIF bit load has been done 0: # check if base register setup + TIF bit load has been done
clg %r9,BASED(cleanup_system_call_insn+16) clg %r9,BASED(.Lcleanup_system_call_insn+16)
jhe 0f jhe 0f
# set up saved registers r10 and r12 # set up saved registers r10 and r12
stg %r10,16(%r11) # r10 last break stg %r10,16(%r11) # r10 last break
stg %r12,32(%r11) # r12 thread-info pointer stg %r12,32(%r11) # r12 thread-info pointer
0: # check if the user time update has been done 0: # check if the user time update has been done
clg %r9,BASED(cleanup_system_call_insn+24) clg %r9,BASED(.Lcleanup_system_call_insn+24)
jh 0f jh 0f
lg %r15,__LC_EXIT_TIMER lg %r15,__LC_EXIT_TIMER
slg %r15,__LC_SYNC_ENTER_TIMER slg %r15,__LC_SYNC_ENTER_TIMER
alg %r15,__LC_USER_TIMER alg %r15,__LC_USER_TIMER
stg %r15,__LC_USER_TIMER stg %r15,__LC_USER_TIMER
0: # check if the system time update has been done 0: # check if the system time update has been done
clg %r9,BASED(cleanup_system_call_insn+32) clg %r9,BASED(.Lcleanup_system_call_insn+32)
jh 0f jh 0f
lg %r15,__LC_LAST_UPDATE_TIMER lg %r15,__LC_LAST_UPDATE_TIMER
slg %r15,__LC_EXIT_TIMER slg %r15,__LC_EXIT_TIMER
@ -904,21 +904,21 @@ cleanup_system_call:
# setup saved register r15 # setup saved register r15
stg %r15,56(%r11) # r15 stack pointer stg %r15,56(%r11) # r15 stack pointer
# set new psw address and exit # set new psw address and exit
larl %r9,sysc_do_svc larl %r9,.Lsysc_do_svc
br %r14 br %r14
cleanup_system_call_insn: .Lcleanup_system_call_insn:
.quad system_call .quad system_call
.quad sysc_stmg .quad .Lsysc_stmg
.quad sysc_per .quad .Lsysc_per
.quad sysc_vtime+18 .quad .Lsysc_vtime+18
.quad sysc_vtime+42 .quad .Lsysc_vtime+42
cleanup_sysc_tif: .Lcleanup_sysc_tif:
larl %r9,sysc_tif larl %r9,.Lsysc_tif
br %r14 br %r14
cleanup_sysc_restore: .Lcleanup_sysc_restore:
clg %r9,BASED(cleanup_sysc_restore_insn) clg %r9,BASED(.Lcleanup_sysc_restore_insn)
je 0f je 0f
lg %r9,24(%r11) # get saved pointer to pt_regs lg %r9,24(%r11) # get saved pointer to pt_regs
mvc __LC_RETURN_PSW(16),__PT_PSW(%r9) mvc __LC_RETURN_PSW(16),__PT_PSW(%r9)
@ -926,15 +926,15 @@ cleanup_sysc_restore:
lmg %r0,%r7,__PT_R0(%r9) lmg %r0,%r7,__PT_R0(%r9)
0: lmg %r8,%r9,__LC_RETURN_PSW 0: lmg %r8,%r9,__LC_RETURN_PSW
br %r14 br %r14
cleanup_sysc_restore_insn: .Lcleanup_sysc_restore_insn:
.quad sysc_done - 4 .quad .Lsysc_done - 4
cleanup_io_tif: .Lcleanup_io_tif:
larl %r9,io_tif larl %r9,.Lio_tif
br %r14 br %r14
cleanup_io_restore: .Lcleanup_io_restore:
clg %r9,BASED(cleanup_io_restore_insn) clg %r9,BASED(.Lcleanup_io_restore_insn)
je 0f je 0f
lg %r9,24(%r11) # get saved r11 pointer to pt_regs lg %r9,24(%r11) # get saved r11 pointer to pt_regs
mvc __LC_RETURN_PSW(16),__PT_PSW(%r9) mvc __LC_RETURN_PSW(16),__PT_PSW(%r9)
@ -942,10 +942,10 @@ cleanup_io_restore:
lmg %r0,%r7,__PT_R0(%r9) lmg %r0,%r7,__PT_R0(%r9)
0: lmg %r8,%r9,__LC_RETURN_PSW 0: lmg %r8,%r9,__LC_RETURN_PSW
br %r14 br %r14
cleanup_io_restore_insn: .Lcleanup_io_restore_insn:
.quad io_done - 4 .quad .Lio_done - 4
cleanup_idle: .Lcleanup_idle:
# copy interrupt clock & cpu timer # copy interrupt clock & cpu timer
mvc __CLOCK_IDLE_EXIT(8,%r2),__LC_INT_CLOCK mvc __CLOCK_IDLE_EXIT(8,%r2),__LC_INT_CLOCK
mvc __TIMER_IDLE_EXIT(8,%r2),__LC_ASYNC_ENTER_TIMER mvc __TIMER_IDLE_EXIT(8,%r2),__LC_ASYNC_ENTER_TIMER
@ -954,7 +954,7 @@ cleanup_idle:
mvc __CLOCK_IDLE_EXIT(8,%r2),__LC_MCCK_CLOCK mvc __CLOCK_IDLE_EXIT(8,%r2),__LC_MCCK_CLOCK
mvc __TIMER_IDLE_EXIT(8,%r2),__LC_MCCK_ENTER_TIMER mvc __TIMER_IDLE_EXIT(8,%r2),__LC_MCCK_ENTER_TIMER
0: # check if stck & stpt have been executed 0: # check if stck & stpt have been executed
clg %r9,BASED(cleanup_idle_insn) clg %r9,BASED(.Lcleanup_idle_insn)
jhe 1f jhe 1f
mvc __CLOCK_IDLE_ENTER(8,%r2),__CLOCK_IDLE_EXIT(%r2) mvc __CLOCK_IDLE_ENTER(8,%r2),__CLOCK_IDLE_EXIT(%r2)
mvc __TIMER_IDLE_ENTER(8,%r2),__TIMER_IDLE_EXIT(%r2) mvc __TIMER_IDLE_ENTER(8,%r2),__TIMER_IDLE_EXIT(%r2)
@ -973,17 +973,17 @@ cleanup_idle:
nihh %r8,0xfcfd # clear irq & wait state bits nihh %r8,0xfcfd # clear irq & wait state bits
lg %r9,48(%r11) # return from psw_idle lg %r9,48(%r11) # return from psw_idle
br %r14 br %r14
cleanup_idle_insn: .Lcleanup_idle_insn:
.quad psw_idle_lpsw .quad .Lpsw_idle_lpsw
/* /*
* Integer constants * Integer constants
*/ */
.align 8 .align 8
.Lcritical_start: .Lcritical_start:
.quad __critical_start .quad .L__critical_start
.Lcritical_length: .Lcritical_length:
.quad __critical_end - __critical_start .quad .L__critical_end - .L__critical_start
#if IS_ENABLED(CONFIG_KVM) #if IS_ENABLED(CONFIG_KVM)
@ -1000,25 +1000,25 @@ ENTRY(sie64a)
lmg %r0,%r13,0(%r3) # load guest gprs 0-13 lmg %r0,%r13,0(%r3) # load guest gprs 0-13
lg %r14,__LC_GMAP # get gmap pointer lg %r14,__LC_GMAP # get gmap pointer
ltgr %r14,%r14 ltgr %r14,%r14
jz sie_gmap jz .Lsie_gmap
lctlg %c1,%c1,__GMAP_ASCE(%r14) # load primary asce lctlg %c1,%c1,__GMAP_ASCE(%r14) # load primary asce
sie_gmap: .Lsie_gmap:
lg %r14,__SF_EMPTY(%r15) # get control block pointer lg %r14,__SF_EMPTY(%r15) # get control block pointer
oi __SIE_PROG0C+3(%r14),1 # we are going into SIE now oi __SIE_PROG0C+3(%r14),1 # we are going into SIE now
tm __SIE_PROG20+3(%r14),1 # last exit... tm __SIE_PROG20+3(%r14),1 # last exit...
jnz sie_done jnz .Lsie_done
LPP __SF_EMPTY(%r15) # set guest id LPP __SF_EMPTY(%r15) # set guest id
sie 0(%r14) sie 0(%r14)
sie_done: .Lsie_done:
LPP __SF_EMPTY+16(%r15) # set host id LPP __SF_EMPTY+16(%r15) # set host id
ni __SIE_PROG0C+3(%r14),0xfe # no longer in SIE ni __SIE_PROG0C+3(%r14),0xfe # no longer in SIE
lctlg %c1,%c1,__LC_USER_ASCE # load primary asce lctlg %c1,%c1,__LC_USER_ASCE # load primary asce
# some program checks are suppressing. C code (e.g. do_protection_exception) # some program checks are suppressing. C code (e.g. do_protection_exception)
# will rewind the PSW by the ILC, which is 4 bytes in case of SIE. Other # will rewind the PSW by the ILC, which is 4 bytes in case of SIE. Other
# instructions between sie64a and sie_done should not cause program # instructions between sie64a and .Lsie_done should not cause program
# interrupts. So lets use a nop (47 00 00 00) as a landing pad. # interrupts. So lets use a nop (47 00 00 00) as a landing pad.
# See also HANDLE_SIE_INTERCEPT # See also HANDLE_SIE_INTERCEPT
rewind_pad: .Lrewind_pad:
nop 0 nop 0
.globl sie_exit .globl sie_exit
sie_exit: sie_exit:
@ -1027,19 +1027,19 @@ sie_exit:
lmg %r6,%r14,__SF_GPRS(%r15) # restore kernel registers lmg %r6,%r14,__SF_GPRS(%r15) # restore kernel registers
lg %r2,__SF_EMPTY+24(%r15) # return exit reason code lg %r2,__SF_EMPTY+24(%r15) # return exit reason code
br %r14 br %r14
sie_fault: .Lsie_fault:
lghi %r14,-EFAULT lghi %r14,-EFAULT
stg %r14,__SF_EMPTY+24(%r15) # set exit reason code stg %r14,__SF_EMPTY+24(%r15) # set exit reason code
j sie_exit j sie_exit
.align 8 .align 8
.Lsie_critical: .Lsie_critical:
.quad sie_gmap .quad .Lsie_gmap
.Lsie_critical_length: .Lsie_critical_length:
.quad sie_done - sie_gmap .quad .Lsie_done - .Lsie_gmap
EX_TABLE(rewind_pad,sie_fault) EX_TABLE(.Lrewind_pad,.Lsie_fault)
EX_TABLE(sie_exit,sie_fault) EX_TABLE(sie_exit,.Lsie_fault)
#endif #endif
.section .rodata, "a" .section .rodata, "a"

View file

@ -7,6 +7,7 @@
* Martin Schwidefsky <schwidefsky@de.ibm.com> * Martin Schwidefsky <schwidefsky@de.ibm.com>
*/ */
#include <linux/moduleloader.h>
#include <linux/hardirq.h> #include <linux/hardirq.h>
#include <linux/uaccess.h> #include <linux/uaccess.h>
#include <linux/ftrace.h> #include <linux/ftrace.h>
@ -15,60 +16,39 @@
#include <linux/kprobes.h> #include <linux/kprobes.h>
#include <trace/syscall.h> #include <trace/syscall.h>
#include <asm/asm-offsets.h> #include <asm/asm-offsets.h>
#include <asm/cacheflush.h>
#include "entry.h" #include "entry.h"
void mcount_replace_code(void);
void ftrace_disable_code(void);
void ftrace_enable_insn(void);
/* /*
* The mcount code looks like this: * The mcount code looks like this:
* stg %r14,8(%r15) # offset 0 * stg %r14,8(%r15) # offset 0
* larl %r1,<&counter> # offset 6 * larl %r1,<&counter> # offset 6
* brasl %r14,_mcount # offset 12 * brasl %r14,_mcount # offset 12
* lg %r14,8(%r15) # offset 18 * lg %r14,8(%r15) # offset 18
* Total length is 24 bytes. The complete mcount block initially gets replaced * Total length is 24 bytes. Only the first instruction will be patched
* by ftrace_make_nop. Subsequent calls to ftrace_make_call / ftrace_make_nop * by ftrace_make_call / ftrace_make_nop.
* only patch the jg/lg instruction within the block.
* Note: we do not patch the first instruction to an unconditional branch,
* since that would break kprobes/jprobes. It is easier to leave the larl
* instruction in and only modify the second instruction.
* The enabled ftrace code block looks like this: * The enabled ftrace code block looks like this:
* larl %r0,.+24 # offset 0 * > brasl %r0,ftrace_caller # offset 0
* > lg %r1,__LC_FTRACE_FUNC # offset 6 * larl %r1,<&counter> # offset 6
* br %r1 # offset 12 * brasl %r14,_mcount # offset 12
* brcl 0,0 # offset 14 * lg %r14,8(%r15) # offset 18
* brc 0,0 # offset 20
* The ftrace function gets called with a non-standard C function call ABI * The ftrace function gets called with a non-standard C function call ABI
* where r0 contains the return address. It is also expected that the called * where r0 contains the return address. It is also expected that the called
* function only clobbers r0 and r1, but restores r2-r15. * function only clobbers r0 and r1, but restores r2-r15.
* For module code we can't directly jump to ftrace caller, but need a
* trampoline (ftrace_plt), which clobbers also r1.
* The return point of the ftrace function has offset 24, so execution * The return point of the ftrace function has offset 24, so execution
* continues behind the mcount block. * continues behind the mcount block.
* larl %r0,.+24 # offset 0 * The disabled ftrace code block looks like this:
* > jg .+18 # offset 6 * > jg .+24 # offset 0
* br %r1 # offset 12 * larl %r1,<&counter> # offset 6
* brcl 0,0 # offset 14 * brasl %r14,_mcount # offset 12
* brc 0,0 # offset 20 * lg %r14,8(%r15) # offset 18
* The jg instruction branches to offset 24 to skip as many instructions * The jg instruction branches to offset 24 to skip as many instructions
* as possible. * as possible.
*/ */
asm(
" .align 4\n"
"mcount_replace_code:\n"
" larl %r0,0f\n"
"ftrace_disable_code:\n"
" jg 0f\n"
" br %r1\n"
" brcl 0,0\n"
" brc 0,0\n"
"0:\n"
" .align 4\n"
"ftrace_enable_insn:\n"
" lg %r1,"__stringify(__LC_FTRACE_FUNC)"\n");
#define MCOUNT_BLOCK_SIZE 24 unsigned long ftrace_plt;
#define MCOUNT_INSN_OFFSET 6
#define FTRACE_INSN_SIZE 6
int ftrace_modify_call(struct dyn_ftrace *rec, unsigned long old_addr, int ftrace_modify_call(struct dyn_ftrace *rec, unsigned long old_addr,
unsigned long addr) unsigned long addr)
@ -79,24 +59,62 @@ int ftrace_modify_call(struct dyn_ftrace *rec, unsigned long old_addr,
int ftrace_make_nop(struct module *mod, struct dyn_ftrace *rec, int ftrace_make_nop(struct module *mod, struct dyn_ftrace *rec,
unsigned long addr) unsigned long addr)
{ {
/* Initial replacement of the whole mcount block */ struct ftrace_insn insn;
if (addr == MCOUNT_ADDR) { unsigned short op;
if (probe_kernel_write((void *) rec->ip - MCOUNT_INSN_OFFSET, void *from, *to;
mcount_replace_code, size_t size;
MCOUNT_BLOCK_SIZE))
return -EPERM; ftrace_generate_nop_insn(&insn);
return 0; size = sizeof(insn);
from = &insn;
to = (void *) rec->ip;
if (probe_kernel_read(&op, (void *) rec->ip, sizeof(op)))
return -EFAULT;
/*
* If we find a breakpoint instruction, a kprobe has been placed
* at the beginning of the function. We write the constant
* KPROBE_ON_FTRACE_NOP into the remaining four bytes of the original
* instruction so that the kprobes handler can execute a nop, if it
* reaches this breakpoint.
*/
if (op == BREAKPOINT_INSTRUCTION) {
size -= 2;
from += 2;
to += 2;
insn.disp = KPROBE_ON_FTRACE_NOP;
} }
if (probe_kernel_write((void *) rec->ip, ftrace_disable_code, if (probe_kernel_write(to, from, size))
MCOUNT_INSN_SIZE))
return -EPERM; return -EPERM;
return 0; return 0;
} }
int ftrace_make_call(struct dyn_ftrace *rec, unsigned long addr) int ftrace_make_call(struct dyn_ftrace *rec, unsigned long addr)
{ {
if (probe_kernel_write((void *) rec->ip, ftrace_enable_insn, struct ftrace_insn insn;
FTRACE_INSN_SIZE)) unsigned short op;
void *from, *to;
size_t size;
ftrace_generate_call_insn(&insn, rec->ip);
size = sizeof(insn);
from = &insn;
to = (void *) rec->ip;
if (probe_kernel_read(&op, (void *) rec->ip, sizeof(op)))
return -EFAULT;
/*
* If we find a breakpoint instruction, a kprobe has been placed
* at the beginning of the function. We write the constant
* KPROBE_ON_FTRACE_CALL into the remaining four bytes of the original
* instruction so that the kprobes handler can execute a brasl if it
* reaches this breakpoint.
*/
if (op == BREAKPOINT_INSTRUCTION) {
size -= 2;
from += 2;
to += 2;
insn.disp = KPROBE_ON_FTRACE_CALL;
}
if (probe_kernel_write(to, from, size))
return -EPERM; return -EPERM;
return 0; return 0;
} }
@ -111,13 +129,30 @@ int __init ftrace_dyn_arch_init(void)
return 0; return 0;
} }
static int __init ftrace_plt_init(void)
{
unsigned int *ip;
ftrace_plt = (unsigned long) module_alloc(PAGE_SIZE);
if (!ftrace_plt)
panic("cannot allocate ftrace plt\n");
ip = (unsigned int *) ftrace_plt;
ip[0] = 0x0d10e310; /* basr 1,0; lg 1,10(1); br 1 */
ip[1] = 0x100a0004;
ip[2] = 0x07f10000;
ip[3] = FTRACE_ADDR >> 32;
ip[4] = FTRACE_ADDR & 0xffffffff;
set_memory_ro(ftrace_plt, 1);
return 0;
}
device_initcall(ftrace_plt_init);
#ifdef CONFIG_FUNCTION_GRAPH_TRACER #ifdef CONFIG_FUNCTION_GRAPH_TRACER
/* /*
* Hook the return address and push it in the stack of return addresses * Hook the return address and push it in the stack of return addresses
* in current thread info. * in current thread info.
*/ */
unsigned long __kprobes prepare_ftrace_return(unsigned long parent, unsigned long prepare_ftrace_return(unsigned long parent, unsigned long ip)
unsigned long ip)
{ {
struct ftrace_graph_ent trace; struct ftrace_graph_ent trace;
@ -137,6 +172,7 @@ unsigned long __kprobes prepare_ftrace_return(unsigned long parent,
out: out:
return parent; return parent;
} }
NOKPROBE_SYMBOL(prepare_ftrace_return);
/* /*
* Patch the kernel code at ftrace_graph_caller location. The instruction * Patch the kernel code at ftrace_graph_caller location. The instruction

View file

@ -19,7 +19,7 @@
static DEFINE_PER_CPU(struct s390_idle_data, s390_idle); static DEFINE_PER_CPU(struct s390_idle_data, s390_idle);
void __kprobes enabled_wait(void) void enabled_wait(void)
{ {
struct s390_idle_data *idle = this_cpu_ptr(&s390_idle); struct s390_idle_data *idle = this_cpu_ptr(&s390_idle);
unsigned long long idle_time; unsigned long long idle_time;
@ -35,31 +35,32 @@ void __kprobes enabled_wait(void)
/* Call the assembler magic in entry.S */ /* Call the assembler magic in entry.S */
psw_idle(idle, psw_mask); psw_idle(idle, psw_mask);
trace_hardirqs_off();
/* Account time spent with enabled wait psw loaded as idle time. */ /* Account time spent with enabled wait psw loaded as idle time. */
idle->sequence++; write_seqcount_begin(&idle->seqcount);
smp_wmb();
idle_time = idle->clock_idle_exit - idle->clock_idle_enter; idle_time = idle->clock_idle_exit - idle->clock_idle_enter;
idle->clock_idle_enter = idle->clock_idle_exit = 0ULL; idle->clock_idle_enter = idle->clock_idle_exit = 0ULL;
idle->idle_time += idle_time; idle->idle_time += idle_time;
idle->idle_count++; idle->idle_count++;
account_idle_time(idle_time); account_idle_time(idle_time);
smp_wmb(); write_seqcount_end(&idle->seqcount);
idle->sequence++;
} }
NOKPROBE_SYMBOL(enabled_wait);
static ssize_t show_idle_count(struct device *dev, static ssize_t show_idle_count(struct device *dev,
struct device_attribute *attr, char *buf) struct device_attribute *attr, char *buf)
{ {
struct s390_idle_data *idle = &per_cpu(s390_idle, dev->id); struct s390_idle_data *idle = &per_cpu(s390_idle, dev->id);
unsigned long long idle_count; unsigned long long idle_count;
unsigned int sequence; unsigned int seq;
do { do {
sequence = ACCESS_ONCE(idle->sequence); seq = read_seqcount_begin(&idle->seqcount);
idle_count = ACCESS_ONCE(idle->idle_count); idle_count = ACCESS_ONCE(idle->idle_count);
if (ACCESS_ONCE(idle->clock_idle_enter)) if (ACCESS_ONCE(idle->clock_idle_enter))
idle_count++; idle_count++;
} while ((sequence & 1) || (ACCESS_ONCE(idle->sequence) != sequence)); } while (read_seqcount_retry(&idle->seqcount, seq));
return sprintf(buf, "%llu\n", idle_count); return sprintf(buf, "%llu\n", idle_count);
} }
DEVICE_ATTR(idle_count, 0444, show_idle_count, NULL); DEVICE_ATTR(idle_count, 0444, show_idle_count, NULL);
@ -69,15 +70,15 @@ static ssize_t show_idle_time(struct device *dev,
{ {
struct s390_idle_data *idle = &per_cpu(s390_idle, dev->id); struct s390_idle_data *idle = &per_cpu(s390_idle, dev->id);
unsigned long long now, idle_time, idle_enter, idle_exit; unsigned long long now, idle_time, idle_enter, idle_exit;
unsigned int sequence; unsigned int seq;
do { do {
now = get_tod_clock(); now = get_tod_clock();
sequence = ACCESS_ONCE(idle->sequence); seq = read_seqcount_begin(&idle->seqcount);
idle_time = ACCESS_ONCE(idle->idle_time); idle_time = ACCESS_ONCE(idle->idle_time);
idle_enter = ACCESS_ONCE(idle->clock_idle_enter); idle_enter = ACCESS_ONCE(idle->clock_idle_enter);
idle_exit = ACCESS_ONCE(idle->clock_idle_exit); idle_exit = ACCESS_ONCE(idle->clock_idle_exit);
} while ((sequence & 1) || (ACCESS_ONCE(idle->sequence) != sequence)); } while (read_seqcount_retry(&idle->seqcount, seq));
idle_time += idle_enter ? ((idle_exit ? : now) - idle_enter) : 0; idle_time += idle_enter ? ((idle_exit ? : now) - idle_enter) : 0;
return sprintf(buf, "%llu\n", idle_time >> 12); return sprintf(buf, "%llu\n", idle_time >> 12);
} }
@ -87,14 +88,14 @@ cputime64_t arch_cpu_idle_time(int cpu)
{ {
struct s390_idle_data *idle = &per_cpu(s390_idle, cpu); struct s390_idle_data *idle = &per_cpu(s390_idle, cpu);
unsigned long long now, idle_enter, idle_exit; unsigned long long now, idle_enter, idle_exit;
unsigned int sequence; unsigned int seq;
do { do {
now = get_tod_clock(); now = get_tod_clock();
sequence = ACCESS_ONCE(idle->sequence); seq = read_seqcount_begin(&idle->seqcount);
idle_enter = ACCESS_ONCE(idle->clock_idle_enter); idle_enter = ACCESS_ONCE(idle->clock_idle_enter);
idle_exit = ACCESS_ONCE(idle->clock_idle_exit); idle_exit = ACCESS_ONCE(idle->clock_idle_exit);
} while ((sequence & 1) || (ACCESS_ONCE(idle->sequence) != sequence)); } while (read_seqcount_retry(&idle->seqcount, seq));
return idle_enter ? ((idle_exit ?: now) - idle_enter) : 0; return idle_enter ? ((idle_exit ?: now) - idle_enter) : 0;
} }

View file

@ -127,13 +127,10 @@ int show_interrupts(struct seq_file *p, void *v)
for_each_online_cpu(cpu) for_each_online_cpu(cpu)
seq_printf(p, "CPU%d ", cpu); seq_printf(p, "CPU%d ", cpu);
seq_putc(p, '\n'); seq_putc(p, '\n');
goto out;
} }
if (index < NR_IRQS) { if (index < NR_IRQS) {
if (index >= NR_IRQS_BASE) if (index >= NR_IRQS_BASE)
goto out; goto out;
/* Adjust index to process irqclass_main_desc array entries */
index--;
seq_printf(p, "%s: ", irqclass_main_desc[index].name); seq_printf(p, "%s: ", irqclass_main_desc[index].name);
irq = irqclass_main_desc[index].irq; irq = irqclass_main_desc[index].irq;
for_each_online_cpu(cpu) for_each_online_cpu(cpu)
@ -158,7 +155,7 @@ out:
unsigned int arch_dynirq_lower_bound(unsigned int from) unsigned int arch_dynirq_lower_bound(unsigned int from)
{ {
return from < THIN_INTERRUPT ? THIN_INTERRUPT : from; return from < NR_IRQS_BASE ? NR_IRQS_BASE : from;
} }
/* /*

View file

@ -29,6 +29,7 @@
#include <linux/module.h> #include <linux/module.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/hardirq.h> #include <linux/hardirq.h>
#include <linux/ftrace.h>
#include <asm/cacheflush.h> #include <asm/cacheflush.h>
#include <asm/sections.h> #include <asm/sections.h>
#include <asm/dis.h> #include <asm/dis.h>
@ -58,12 +59,23 @@ struct kprobe_insn_cache kprobe_dmainsn_slots = {
.insn_size = MAX_INSN_SIZE, .insn_size = MAX_INSN_SIZE,
}; };
static void __kprobes copy_instruction(struct kprobe *p) static void copy_instruction(struct kprobe *p)
{ {
unsigned long ip = (unsigned long) p->addr;
s64 disp, new_disp; s64 disp, new_disp;
u64 addr, new_addr; u64 addr, new_addr;
memcpy(p->ainsn.insn, p->addr, insn_length(p->opcode >> 8)); if (ftrace_location(ip) == ip) {
/*
* If kprobes patches the instruction that is morphed by
* ftrace make sure that kprobes always sees the branch
* "jg .+24" that skips the mcount block
*/
ftrace_generate_nop_insn((struct ftrace_insn *)p->ainsn.insn);
p->ainsn.is_ftrace_insn = 1;
} else
memcpy(p->ainsn.insn, p->addr, insn_length(*p->addr >> 8));
p->opcode = p->ainsn.insn[0];
if (!probe_is_insn_relative_long(p->ainsn.insn)) if (!probe_is_insn_relative_long(p->ainsn.insn))
return; return;
/* /*
@ -79,25 +91,14 @@ static void __kprobes copy_instruction(struct kprobe *p)
new_disp = ((addr + (disp * 2)) - new_addr) / 2; new_disp = ((addr + (disp * 2)) - new_addr) / 2;
*(s32 *)&p->ainsn.insn[1] = new_disp; *(s32 *)&p->ainsn.insn[1] = new_disp;
} }
NOKPROBE_SYMBOL(copy_instruction);
static inline int is_kernel_addr(void *addr) static inline int is_kernel_addr(void *addr)
{ {
return addr < (void *)_end; return addr < (void *)_end;
} }
static inline int is_module_addr(void *addr) static int s390_get_insn_slot(struct kprobe *p)
{
#ifdef CONFIG_64BIT
BUILD_BUG_ON(MODULES_LEN > (1UL << 31));
if (addr < (void *)MODULES_VADDR)
return 0;
if (addr > (void *)MODULES_END)
return 0;
#endif
return 1;
}
static int __kprobes s390_get_insn_slot(struct kprobe *p)
{ {
/* /*
* Get an insn slot that is within the same 2GB area like the original * Get an insn slot that is within the same 2GB area like the original
@ -111,8 +112,9 @@ static int __kprobes s390_get_insn_slot(struct kprobe *p)
p->ainsn.insn = get_insn_slot(); p->ainsn.insn = get_insn_slot();
return p->ainsn.insn ? 0 : -ENOMEM; return p->ainsn.insn ? 0 : -ENOMEM;
} }
NOKPROBE_SYMBOL(s390_get_insn_slot);
static void __kprobes s390_free_insn_slot(struct kprobe *p) static void s390_free_insn_slot(struct kprobe *p)
{ {
if (!p->ainsn.insn) if (!p->ainsn.insn)
return; return;
@ -122,8 +124,9 @@ static void __kprobes s390_free_insn_slot(struct kprobe *p)
free_insn_slot(p->ainsn.insn, 0); free_insn_slot(p->ainsn.insn, 0);
p->ainsn.insn = NULL; p->ainsn.insn = NULL;
} }
NOKPROBE_SYMBOL(s390_free_insn_slot);
int __kprobes arch_prepare_kprobe(struct kprobe *p) int arch_prepare_kprobe(struct kprobe *p)
{ {
if ((unsigned long) p->addr & 0x01) if ((unsigned long) p->addr & 0x01)
return -EINVAL; return -EINVAL;
@ -132,54 +135,79 @@ int __kprobes arch_prepare_kprobe(struct kprobe *p)
return -EINVAL; return -EINVAL;
if (s390_get_insn_slot(p)) if (s390_get_insn_slot(p))
return -ENOMEM; return -ENOMEM;
p->opcode = *p->addr;
copy_instruction(p); copy_instruction(p);
return 0; return 0;
} }
NOKPROBE_SYMBOL(arch_prepare_kprobe);
struct ins_replace_args { int arch_check_ftrace_location(struct kprobe *p)
kprobe_opcode_t *ptr;
kprobe_opcode_t opcode;
};
static int __kprobes swap_instruction(void *aref)
{ {
struct kprobe_ctlblk *kcb = get_kprobe_ctlblk();
unsigned long status = kcb->kprobe_status;
struct ins_replace_args *args = aref;
kcb->kprobe_status = KPROBE_SWAP_INST;
probe_kernel_write(args->ptr, &args->opcode, sizeof(args->opcode));
kcb->kprobe_status = status;
return 0; return 0;
} }
void __kprobes arch_arm_kprobe(struct kprobe *p) struct swap_insn_args {
{ struct kprobe *p;
struct ins_replace_args args; unsigned int arm_kprobe : 1;
};
static int swap_instruction(void *data)
{
struct kprobe_ctlblk *kcb = get_kprobe_ctlblk();
unsigned long status = kcb->kprobe_status;
struct swap_insn_args *args = data;
struct ftrace_insn new_insn, *insn;
struct kprobe *p = args->p;
size_t len;
new_insn.opc = args->arm_kprobe ? BREAKPOINT_INSTRUCTION : p->opcode;
len = sizeof(new_insn.opc);
if (!p->ainsn.is_ftrace_insn)
goto skip_ftrace;
len = sizeof(new_insn);
insn = (struct ftrace_insn *) p->addr;
if (args->arm_kprobe) {
if (is_ftrace_nop(insn))
new_insn.disp = KPROBE_ON_FTRACE_NOP;
else
new_insn.disp = KPROBE_ON_FTRACE_CALL;
} else {
ftrace_generate_call_insn(&new_insn, (unsigned long)p->addr);
if (insn->disp == KPROBE_ON_FTRACE_NOP)
ftrace_generate_nop_insn(&new_insn);
}
skip_ftrace:
kcb->kprobe_status = KPROBE_SWAP_INST;
probe_kernel_write(p->addr, &new_insn, len);
kcb->kprobe_status = status;
return 0;
}
NOKPROBE_SYMBOL(swap_instruction);
void arch_arm_kprobe(struct kprobe *p)
{
struct swap_insn_args args = {.p = p, .arm_kprobe = 1};
args.ptr = p->addr;
args.opcode = BREAKPOINT_INSTRUCTION;
stop_machine(swap_instruction, &args, NULL); stop_machine(swap_instruction, &args, NULL);
} }
NOKPROBE_SYMBOL(arch_arm_kprobe);
void __kprobes arch_disarm_kprobe(struct kprobe *p) void arch_disarm_kprobe(struct kprobe *p)
{ {
struct ins_replace_args args; struct swap_insn_args args = {.p = p, .arm_kprobe = 0};
args.ptr = p->addr;
args.opcode = p->opcode;
stop_machine(swap_instruction, &args, NULL); stop_machine(swap_instruction, &args, NULL);
} }
NOKPROBE_SYMBOL(arch_disarm_kprobe);
void __kprobes arch_remove_kprobe(struct kprobe *p) void arch_remove_kprobe(struct kprobe *p)
{ {
s390_free_insn_slot(p); s390_free_insn_slot(p);
} }
NOKPROBE_SYMBOL(arch_remove_kprobe);
static void __kprobes enable_singlestep(struct kprobe_ctlblk *kcb, static void enable_singlestep(struct kprobe_ctlblk *kcb,
struct pt_regs *regs, struct pt_regs *regs,
unsigned long ip) unsigned long ip)
{ {
struct per_regs per_kprobe; struct per_regs per_kprobe;
@ -199,10 +227,11 @@ static void __kprobes enable_singlestep(struct kprobe_ctlblk *kcb,
regs->psw.mask &= ~(PSW_MASK_IO | PSW_MASK_EXT); regs->psw.mask &= ~(PSW_MASK_IO | PSW_MASK_EXT);
regs->psw.addr = ip | PSW_ADDR_AMODE; regs->psw.addr = ip | PSW_ADDR_AMODE;
} }
NOKPROBE_SYMBOL(enable_singlestep);
static void __kprobes disable_singlestep(struct kprobe_ctlblk *kcb, static void disable_singlestep(struct kprobe_ctlblk *kcb,
struct pt_regs *regs, struct pt_regs *regs,
unsigned long ip) unsigned long ip)
{ {
/* Restore control regs and psw mask, set new psw address */ /* Restore control regs and psw mask, set new psw address */
__ctl_load(kcb->kprobe_saved_ctl, 9, 11); __ctl_load(kcb->kprobe_saved_ctl, 9, 11);
@ -210,41 +239,43 @@ static void __kprobes disable_singlestep(struct kprobe_ctlblk *kcb,
regs->psw.mask |= kcb->kprobe_saved_imask; regs->psw.mask |= kcb->kprobe_saved_imask;
regs->psw.addr = ip | PSW_ADDR_AMODE; regs->psw.addr = ip | PSW_ADDR_AMODE;
} }
NOKPROBE_SYMBOL(disable_singlestep);
/* /*
* Activate a kprobe by storing its pointer to current_kprobe. The * Activate a kprobe by storing its pointer to current_kprobe. The
* previous kprobe is stored in kcb->prev_kprobe. A stack of up to * previous kprobe is stored in kcb->prev_kprobe. A stack of up to
* two kprobes can be active, see KPROBE_REENTER. * two kprobes can be active, see KPROBE_REENTER.
*/ */
static void __kprobes push_kprobe(struct kprobe_ctlblk *kcb, struct kprobe *p) static void push_kprobe(struct kprobe_ctlblk *kcb, struct kprobe *p)
{ {
kcb->prev_kprobe.kp = __this_cpu_read(current_kprobe); kcb->prev_kprobe.kp = __this_cpu_read(current_kprobe);
kcb->prev_kprobe.status = kcb->kprobe_status; kcb->prev_kprobe.status = kcb->kprobe_status;
__this_cpu_write(current_kprobe, p); __this_cpu_write(current_kprobe, p);
} }
NOKPROBE_SYMBOL(push_kprobe);
/* /*
* Deactivate a kprobe by backing up to the previous state. If the * Deactivate a kprobe by backing up to the previous state. If the
* current state is KPROBE_REENTER prev_kprobe.kp will be non-NULL, * current state is KPROBE_REENTER prev_kprobe.kp will be non-NULL,
* for any other state prev_kprobe.kp will be NULL. * for any other state prev_kprobe.kp will be NULL.
*/ */
static void __kprobes pop_kprobe(struct kprobe_ctlblk *kcb) static void pop_kprobe(struct kprobe_ctlblk *kcb)
{ {
__this_cpu_write(current_kprobe, kcb->prev_kprobe.kp); __this_cpu_write(current_kprobe, kcb->prev_kprobe.kp);
kcb->kprobe_status = kcb->prev_kprobe.status; kcb->kprobe_status = kcb->prev_kprobe.status;
} }
NOKPROBE_SYMBOL(pop_kprobe);
void __kprobes arch_prepare_kretprobe(struct kretprobe_instance *ri, void arch_prepare_kretprobe(struct kretprobe_instance *ri, struct pt_regs *regs)
struct pt_regs *regs)
{ {
ri->ret_addr = (kprobe_opcode_t *) regs->gprs[14]; ri->ret_addr = (kprobe_opcode_t *) regs->gprs[14];
/* Replace the return addr with trampoline addr */ /* Replace the return addr with trampoline addr */
regs->gprs[14] = (unsigned long) &kretprobe_trampoline; regs->gprs[14] = (unsigned long) &kretprobe_trampoline;
} }
NOKPROBE_SYMBOL(arch_prepare_kretprobe);
static void __kprobes kprobe_reenter_check(struct kprobe_ctlblk *kcb, static void kprobe_reenter_check(struct kprobe_ctlblk *kcb, struct kprobe *p)
struct kprobe *p)
{ {
switch (kcb->kprobe_status) { switch (kcb->kprobe_status) {
case KPROBE_HIT_SSDONE: case KPROBE_HIT_SSDONE:
@ -264,8 +295,9 @@ static void __kprobes kprobe_reenter_check(struct kprobe_ctlblk *kcb,
BUG(); BUG();
} }
} }
NOKPROBE_SYMBOL(kprobe_reenter_check);
static int __kprobes kprobe_handler(struct pt_regs *regs) static int kprobe_handler(struct pt_regs *regs)
{ {
struct kprobe_ctlblk *kcb; struct kprobe_ctlblk *kcb;
struct kprobe *p; struct kprobe *p;
@ -339,6 +371,7 @@ static int __kprobes kprobe_handler(struct pt_regs *regs)
preempt_enable_no_resched(); preempt_enable_no_resched();
return 0; return 0;
} }
NOKPROBE_SYMBOL(kprobe_handler);
/* /*
* Function return probe trampoline: * Function return probe trampoline:
@ -355,8 +388,7 @@ static void __used kretprobe_trampoline_holder(void)
/* /*
* Called when the probe at kretprobe trampoline is hit * Called when the probe at kretprobe trampoline is hit
*/ */
static int __kprobes trampoline_probe_handler(struct kprobe *p, static int trampoline_probe_handler(struct kprobe *p, struct pt_regs *regs)
struct pt_regs *regs)
{ {
struct kretprobe_instance *ri; struct kretprobe_instance *ri;
struct hlist_head *head, empty_rp; struct hlist_head *head, empty_rp;
@ -444,6 +476,7 @@ static int __kprobes trampoline_probe_handler(struct kprobe *p,
*/ */
return 1; return 1;
} }
NOKPROBE_SYMBOL(trampoline_probe_handler);
/* /*
* Called after single-stepping. p->addr is the address of the * Called after single-stepping. p->addr is the address of the
@ -453,12 +486,30 @@ static int __kprobes trampoline_probe_handler(struct kprobe *p,
* single-stepped a copy of the instruction. The address of this * single-stepped a copy of the instruction. The address of this
* copy is p->ainsn.insn. * copy is p->ainsn.insn.
*/ */
static void __kprobes resume_execution(struct kprobe *p, struct pt_regs *regs) static void resume_execution(struct kprobe *p, struct pt_regs *regs)
{ {
struct kprobe_ctlblk *kcb = get_kprobe_ctlblk(); struct kprobe_ctlblk *kcb = get_kprobe_ctlblk();
unsigned long ip = regs->psw.addr & PSW_ADDR_INSN; unsigned long ip = regs->psw.addr & PSW_ADDR_INSN;
int fixup = probe_get_fixup_type(p->ainsn.insn); int fixup = probe_get_fixup_type(p->ainsn.insn);
/* Check if the kprobes location is an enabled ftrace caller */
if (p->ainsn.is_ftrace_insn) {
struct ftrace_insn *insn = (struct ftrace_insn *) p->addr;
struct ftrace_insn call_insn;
ftrace_generate_call_insn(&call_insn, (unsigned long) p->addr);
/*
* A kprobe on an enabled ftrace call site actually single
* stepped an unconditional branch (ftrace nop equivalent).
* Now we need to fixup things and pretend that a brasl r0,...
* was executed instead.
*/
if (insn->disp == KPROBE_ON_FTRACE_CALL) {
ip += call_insn.disp * 2 - MCOUNT_INSN_SIZE;
regs->gprs[0] = (unsigned long)p->addr + sizeof(*insn);
}
}
if (fixup & FIXUP_PSW_NORMAL) if (fixup & FIXUP_PSW_NORMAL)
ip += (unsigned long) p->addr - (unsigned long) p->ainsn.insn; ip += (unsigned long) p->addr - (unsigned long) p->ainsn.insn;
@ -476,8 +527,9 @@ static void __kprobes resume_execution(struct kprobe *p, struct pt_regs *regs)
disable_singlestep(kcb, regs, ip); disable_singlestep(kcb, regs, ip);
} }
NOKPROBE_SYMBOL(resume_execution);
static int __kprobes post_kprobe_handler(struct pt_regs *regs) static int post_kprobe_handler(struct pt_regs *regs)
{ {
struct kprobe_ctlblk *kcb = get_kprobe_ctlblk(); struct kprobe_ctlblk *kcb = get_kprobe_ctlblk();
struct kprobe *p = kprobe_running(); struct kprobe *p = kprobe_running();
@ -504,8 +556,9 @@ static int __kprobes post_kprobe_handler(struct pt_regs *regs)
return 1; return 1;
} }
NOKPROBE_SYMBOL(post_kprobe_handler);
static int __kprobes kprobe_trap_handler(struct pt_regs *regs, int trapnr) static int kprobe_trap_handler(struct pt_regs *regs, int trapnr)
{ {
struct kprobe_ctlblk *kcb = get_kprobe_ctlblk(); struct kprobe_ctlblk *kcb = get_kprobe_ctlblk();
struct kprobe *p = kprobe_running(); struct kprobe *p = kprobe_running();
@ -567,8 +620,9 @@ static int __kprobes kprobe_trap_handler(struct pt_regs *regs, int trapnr)
} }
return 0; return 0;
} }
NOKPROBE_SYMBOL(kprobe_trap_handler);
int __kprobes kprobe_fault_handler(struct pt_regs *regs, int trapnr) int kprobe_fault_handler(struct pt_regs *regs, int trapnr)
{ {
int ret; int ret;
@ -579,12 +633,13 @@ int __kprobes kprobe_fault_handler(struct pt_regs *regs, int trapnr)
local_irq_restore(regs->psw.mask & ~PSW_MASK_PER); local_irq_restore(regs->psw.mask & ~PSW_MASK_PER);
return ret; return ret;
} }
NOKPROBE_SYMBOL(kprobe_fault_handler);
/* /*
* Wrapper routine to for handling exceptions. * Wrapper routine to for handling exceptions.
*/ */
int __kprobes kprobe_exceptions_notify(struct notifier_block *self, int kprobe_exceptions_notify(struct notifier_block *self,
unsigned long val, void *data) unsigned long val, void *data)
{ {
struct die_args *args = (struct die_args *) data; struct die_args *args = (struct die_args *) data;
struct pt_regs *regs = args->regs; struct pt_regs *regs = args->regs;
@ -616,8 +671,9 @@ int __kprobes kprobe_exceptions_notify(struct notifier_block *self,
return ret; return ret;
} }
NOKPROBE_SYMBOL(kprobe_exceptions_notify);
int __kprobes setjmp_pre_handler(struct kprobe *p, struct pt_regs *regs) int setjmp_pre_handler(struct kprobe *p, struct pt_regs *regs)
{ {
struct jprobe *jp = container_of(p, struct jprobe, kp); struct jprobe *jp = container_of(p, struct jprobe, kp);
struct kprobe_ctlblk *kcb = get_kprobe_ctlblk(); struct kprobe_ctlblk *kcb = get_kprobe_ctlblk();
@ -635,13 +691,15 @@ int __kprobes setjmp_pre_handler(struct kprobe *p, struct pt_regs *regs)
memcpy(kcb->jprobes_stack, (void *) stack, MIN_STACK_SIZE(stack)); memcpy(kcb->jprobes_stack, (void *) stack, MIN_STACK_SIZE(stack));
return 1; return 1;
} }
NOKPROBE_SYMBOL(setjmp_pre_handler);
void __kprobes jprobe_return(void) void jprobe_return(void)
{ {
asm volatile(".word 0x0002"); asm volatile(".word 0x0002");
} }
NOKPROBE_SYMBOL(jprobe_return);
int __kprobes longjmp_break_handler(struct kprobe *p, struct pt_regs *regs) int longjmp_break_handler(struct kprobe *p, struct pt_regs *regs)
{ {
struct kprobe_ctlblk *kcb = get_kprobe_ctlblk(); struct kprobe_ctlblk *kcb = get_kprobe_ctlblk();
unsigned long stack; unsigned long stack;
@ -655,6 +713,7 @@ int __kprobes longjmp_break_handler(struct kprobe *p, struct pt_regs *regs)
preempt_enable_no_resched(); preempt_enable_no_resched();
return 1; return 1;
} }
NOKPROBE_SYMBOL(longjmp_break_handler);
static struct kprobe trampoline = { static struct kprobe trampoline = {
.addr = (kprobe_opcode_t *) &kretprobe_trampoline, .addr = (kprobe_opcode_t *) &kretprobe_trampoline,
@ -666,7 +725,8 @@ int __init arch_init_kprobes(void)
return register_kprobe(&trampoline); return register_kprobe(&trampoline);
} }
int __kprobes arch_trampoline_kprobe(struct kprobe *p) int arch_trampoline_kprobe(struct kprobe *p)
{ {
return p->addr == (kprobe_opcode_t *) &kretprobe_trampoline; return p->addr == (kprobe_opcode_t *) &kretprobe_trampoline;
} }
NOKPROBE_SYMBOL(arch_trampoline_kprobe);

View file

@ -27,6 +27,7 @@ ENTRY(ftrace_caller)
.globl ftrace_regs_caller .globl ftrace_regs_caller
.set ftrace_regs_caller,ftrace_caller .set ftrace_regs_caller,ftrace_caller
lgr %r1,%r15 lgr %r1,%r15
aghi %r0,MCOUNT_RETURN_FIXUP
aghi %r15,-STACK_FRAME_SIZE aghi %r15,-STACK_FRAME_SIZE
stg %r1,__SF_BACKCHAIN(%r15) stg %r1,__SF_BACKCHAIN(%r15)
stg %r1,(STACK_PTREGS_GPRS+15*8)(%r15) stg %r1,(STACK_PTREGS_GPRS+15*8)(%r15)

View file

@ -1383,7 +1383,6 @@ static int cpumsf_pmu_add(struct perf_event *event, int flags)
cpuhw->lsctl.ed = 1; cpuhw->lsctl.ed = 1;
/* Set in_use flag and store event */ /* Set in_use flag and store event */
event->hw.idx = 0; /* only one sampling event per CPU supported */
cpuhw->event = event; cpuhw->event = event;
cpuhw->flags |= PMU_F_IN_USE; cpuhw->flags |= PMU_F_IN_USE;

View file

@ -61,7 +61,7 @@ unsigned long thread_saved_pc(struct task_struct *tsk)
return sf->gprs[8]; return sf->gprs[8];
} }
extern void __kprobes kernel_thread_starter(void); extern void kernel_thread_starter(void);
/* /*
* Free current thread data structures etc.. * Free current thread data structures etc..
@ -153,6 +153,7 @@ int copy_thread(unsigned long clone_flags, unsigned long new_stackp,
save_fp_ctl(&p->thread.fp_regs.fpc); save_fp_ctl(&p->thread.fp_regs.fpc);
save_fp_regs(p->thread.fp_regs.fprs); save_fp_regs(p->thread.fp_regs.fprs);
p->thread.fp_regs.pad = 0; p->thread.fp_regs.pad = 0;
p->thread.vxrs = NULL;
/* Set a new TLS ? */ /* Set a new TLS ? */
if (clone_flags & CLONE_SETTLS) { if (clone_flags & CLONE_SETTLS) {
unsigned long tls = frame->childregs.gprs[6]; unsigned long tls = frame->childregs.gprs[6];

View file

@ -248,14 +248,27 @@ static unsigned long __peek_user(struct task_struct *child, addr_t addr)
*/ */
tmp = 0; tmp = 0;
} else if (addr == (addr_t) &dummy->regs.fp_regs.fpc) {
/*
* floating point control reg. is in the thread structure
*/
tmp = child->thread.fp_regs.fpc;
tmp <<= BITS_PER_LONG - 32;
} else if (addr < (addr_t) (&dummy->regs.fp_regs + 1)) { } else if (addr < (addr_t) (&dummy->regs.fp_regs + 1)) {
/* /*
* floating point regs. are stored in the thread structure * floating point regs. are either in child->thread.fp_regs
* or the child->thread.vxrs array
*/ */
offset = addr - (addr_t) &dummy->regs.fp_regs; offset = addr - (addr_t) &dummy->regs.fp_regs.fprs;
tmp = *(addr_t *)((addr_t) &child->thread.fp_regs + offset); #ifdef CONFIG_64BIT
if (addr == (addr_t) &dummy->regs.fp_regs.fpc) if (child->thread.vxrs)
tmp <<= BITS_PER_LONG - 32; tmp = *(addr_t *)
((addr_t) child->thread.vxrs + 2*offset);
else
#endif
tmp = *(addr_t *)
((addr_t) &child->thread.fp_regs.fprs + offset);
} else if (addr < (addr_t) (&dummy->regs.per_info + 1)) { } else if (addr < (addr_t) (&dummy->regs.per_info + 1)) {
/* /*
@ -383,16 +396,29 @@ static int __poke_user(struct task_struct *child, addr_t addr, addr_t data)
*/ */
return 0; return 0;
} else if (addr == (addr_t) &dummy->regs.fp_regs.fpc) {
/*
* floating point control reg. is in the thread structure
*/
if ((unsigned int) data != 0 ||
test_fp_ctl(data >> (BITS_PER_LONG - 32)))
return -EINVAL;
child->thread.fp_regs.fpc = data >> (BITS_PER_LONG - 32);
} else if (addr < (addr_t) (&dummy->regs.fp_regs + 1)) { } else if (addr < (addr_t) (&dummy->regs.fp_regs + 1)) {
/* /*
* floating point regs. are stored in the thread structure * floating point regs. are either in child->thread.fp_regs
* or the child->thread.vxrs array
*/ */
if (addr == (addr_t) &dummy->regs.fp_regs.fpc) offset = addr - (addr_t) &dummy->regs.fp_regs.fprs;
if ((unsigned int) data != 0 || #ifdef CONFIG_64BIT
test_fp_ctl(data >> (BITS_PER_LONG - 32))) if (child->thread.vxrs)
return -EINVAL; *(addr_t *)((addr_t)
offset = addr - (addr_t) &dummy->regs.fp_regs; child->thread.vxrs + 2*offset) = data;
*(addr_t *)((addr_t) &child->thread.fp_regs + offset) = data; else
#endif
*(addr_t *)((addr_t)
&child->thread.fp_regs.fprs + offset) = data;
} else if (addr < (addr_t) (&dummy->regs.per_info + 1)) { } else if (addr < (addr_t) (&dummy->regs.per_info + 1)) {
/* /*
@ -611,12 +637,26 @@ static u32 __peek_user_compat(struct task_struct *child, addr_t addr)
*/ */
tmp = 0; tmp = 0;
} else if (addr == (addr_t) &dummy32->regs.fp_regs.fpc) {
/*
* floating point control reg. is in the thread structure
*/
tmp = child->thread.fp_regs.fpc;
} else if (addr < (addr_t) (&dummy32->regs.fp_regs + 1)) { } else if (addr < (addr_t) (&dummy32->regs.fp_regs + 1)) {
/* /*
* floating point regs. are stored in the thread structure * floating point regs. are either in child->thread.fp_regs
* or the child->thread.vxrs array
*/ */
offset = addr - (addr_t) &dummy32->regs.fp_regs; offset = addr - (addr_t) &dummy32->regs.fp_regs.fprs;
tmp = *(__u32 *)((addr_t) &child->thread.fp_regs + offset); #ifdef CONFIG_64BIT
if (child->thread.vxrs)
tmp = *(__u32 *)
((addr_t) child->thread.vxrs + 2*offset);
else
#endif
tmp = *(__u32 *)
((addr_t) &child->thread.fp_regs.fprs + offset);
} else if (addr < (addr_t) (&dummy32->regs.per_info + 1)) { } else if (addr < (addr_t) (&dummy32->regs.per_info + 1)) {
/* /*
@ -722,15 +762,28 @@ static int __poke_user_compat(struct task_struct *child,
*/ */
return 0; return 0;
} else if (addr == (addr_t) &dummy32->regs.fp_regs.fpc) {
/*
* floating point control reg. is in the thread structure
*/
if (test_fp_ctl(tmp))
return -EINVAL;
child->thread.fp_regs.fpc = data;
} else if (addr < (addr_t) (&dummy32->regs.fp_regs + 1)) { } else if (addr < (addr_t) (&dummy32->regs.fp_regs + 1)) {
/* /*
* floating point regs. are stored in the thread structure * floating point regs. are either in child->thread.fp_regs
* or the child->thread.vxrs array
*/ */
if (addr == (addr_t) &dummy32->regs.fp_regs.fpc && offset = addr - (addr_t) &dummy32->regs.fp_regs.fprs;
test_fp_ctl(tmp)) #ifdef CONFIG_64BIT
return -EINVAL; if (child->thread.vxrs)
offset = addr - (addr_t) &dummy32->regs.fp_regs; *(__u32 *)((addr_t)
*(__u32 *)((addr_t) &child->thread.fp_regs + offset) = tmp; child->thread.vxrs + 2*offset) = tmp;
else
#endif
*(__u32 *)((addr_t)
&child->thread.fp_regs.fprs + offset) = tmp;
} else if (addr < (addr_t) (&dummy32->regs.per_info + 1)) { } else if (addr < (addr_t) (&dummy32->regs.per_info + 1)) {
/* /*
@ -1038,12 +1091,6 @@ static int s390_tdb_set(struct task_struct *target,
return 0; return 0;
} }
static int s390_vxrs_active(struct task_struct *target,
const struct user_regset *regset)
{
return !!target->thread.vxrs;
}
static int s390_vxrs_low_get(struct task_struct *target, static int s390_vxrs_low_get(struct task_struct *target,
const struct user_regset *regset, const struct user_regset *regset,
unsigned int pos, unsigned int count, unsigned int pos, unsigned int count,
@ -1052,6 +1099,8 @@ static int s390_vxrs_low_get(struct task_struct *target,
__u64 vxrs[__NUM_VXRS_LOW]; __u64 vxrs[__NUM_VXRS_LOW];
int i; int i;
if (!MACHINE_HAS_VX)
return -ENODEV;
if (target->thread.vxrs) { if (target->thread.vxrs) {
if (target == current) if (target == current)
save_vx_regs(target->thread.vxrs); save_vx_regs(target->thread.vxrs);
@ -1070,6 +1119,8 @@ static int s390_vxrs_low_set(struct task_struct *target,
__u64 vxrs[__NUM_VXRS_LOW]; __u64 vxrs[__NUM_VXRS_LOW];
int i, rc; int i, rc;
if (!MACHINE_HAS_VX)
return -ENODEV;
if (!target->thread.vxrs) { if (!target->thread.vxrs) {
rc = alloc_vector_registers(target); rc = alloc_vector_registers(target);
if (rc) if (rc)
@ -1095,6 +1146,8 @@ static int s390_vxrs_high_get(struct task_struct *target,
{ {
__vector128 vxrs[__NUM_VXRS_HIGH]; __vector128 vxrs[__NUM_VXRS_HIGH];
if (!MACHINE_HAS_VX)
return -ENODEV;
if (target->thread.vxrs) { if (target->thread.vxrs) {
if (target == current) if (target == current)
save_vx_regs(target->thread.vxrs); save_vx_regs(target->thread.vxrs);
@ -1112,6 +1165,8 @@ static int s390_vxrs_high_set(struct task_struct *target,
{ {
int rc; int rc;
if (!MACHINE_HAS_VX)
return -ENODEV;
if (!target->thread.vxrs) { if (!target->thread.vxrs) {
rc = alloc_vector_registers(target); rc = alloc_vector_registers(target);
if (rc) if (rc)
@ -1196,7 +1251,6 @@ static const struct user_regset s390_regsets[] = {
.n = __NUM_VXRS_LOW, .n = __NUM_VXRS_LOW,
.size = sizeof(__u64), .size = sizeof(__u64),
.align = sizeof(__u64), .align = sizeof(__u64),
.active = s390_vxrs_active,
.get = s390_vxrs_low_get, .get = s390_vxrs_low_get,
.set = s390_vxrs_low_set, .set = s390_vxrs_low_set,
}, },
@ -1205,7 +1259,6 @@ static const struct user_regset s390_regsets[] = {
.n = __NUM_VXRS_HIGH, .n = __NUM_VXRS_HIGH,
.size = sizeof(__vector128), .size = sizeof(__vector128),
.align = sizeof(__vector128), .align = sizeof(__vector128),
.active = s390_vxrs_active,
.get = s390_vxrs_high_get, .get = s390_vxrs_high_get,
.set = s390_vxrs_high_set, .set = s390_vxrs_high_set,
}, },
@ -1419,7 +1472,6 @@ static const struct user_regset s390_compat_regsets[] = {
.n = __NUM_VXRS_LOW, .n = __NUM_VXRS_LOW,
.size = sizeof(__u64), .size = sizeof(__u64),
.align = sizeof(__u64), .align = sizeof(__u64),
.active = s390_vxrs_active,
.get = s390_vxrs_low_get, .get = s390_vxrs_low_get,
.set = s390_vxrs_low_set, .set = s390_vxrs_low_set,
}, },
@ -1428,7 +1480,6 @@ static const struct user_regset s390_compat_regsets[] = {
.n = __NUM_VXRS_HIGH, .n = __NUM_VXRS_HIGH,
.size = sizeof(__vector128), .size = sizeof(__vector128),
.align = sizeof(__vector128), .align = sizeof(__vector128),
.active = s390_vxrs_active,
.get = s390_vxrs_high_get, .get = s390_vxrs_high_get,
.set = s390_vxrs_high_set, .set = s390_vxrs_high_set,
}, },

View file

@ -41,7 +41,6 @@
#include <linux/ctype.h> #include <linux/ctype.h>
#include <linux/reboot.h> #include <linux/reboot.h>
#include <linux/topology.h> #include <linux/topology.h>
#include <linux/ftrace.h>
#include <linux/kexec.h> #include <linux/kexec.h>
#include <linux/crash_dump.h> #include <linux/crash_dump.h>
#include <linux/memory.h> #include <linux/memory.h>
@ -356,7 +355,6 @@ static void __init setup_lowcore(void)
lc->steal_timer = S390_lowcore.steal_timer; lc->steal_timer = S390_lowcore.steal_timer;
lc->last_update_timer = S390_lowcore.last_update_timer; lc->last_update_timer = S390_lowcore.last_update_timer;
lc->last_update_clock = S390_lowcore.last_update_clock; lc->last_update_clock = S390_lowcore.last_update_clock;
lc->ftrace_func = S390_lowcore.ftrace_func;
restart_stack = __alloc_bootmem(ASYNC_SIZE, ASYNC_SIZE, 0); restart_stack = __alloc_bootmem(ASYNC_SIZE, ASYNC_SIZE, 0);
restart_stack += ASYNC_SIZE; restart_stack += ASYNC_SIZE;

View file

@ -371,7 +371,7 @@ static int setup_frame(int sig, struct k_sigaction *ka,
restorer = (unsigned long) ka->sa.sa_restorer | PSW_ADDR_AMODE; restorer = (unsigned long) ka->sa.sa_restorer | PSW_ADDR_AMODE;
} else { } else {
/* Signal frame without vector registers are short ! */ /* Signal frame without vector registers are short ! */
__u16 __user *svc = (void *) frame + frame_size - 2; __u16 __user *svc = (void __user *) frame + frame_size - 2;
if (__put_user(S390_SYSCALL_OPCODE | __NR_sigreturn, svc)) if (__put_user(S390_SYSCALL_OPCODE | __NR_sigreturn, svc))
return -EFAULT; return -EFAULT;
restorer = (unsigned long) svc | PSW_ADDR_AMODE; restorer = (unsigned long) svc | PSW_ADDR_AMODE;

View file

@ -236,7 +236,6 @@ static void pcpu_prepare_secondary(struct pcpu *pcpu, int cpu)
lc->percpu_offset = __per_cpu_offset[cpu]; lc->percpu_offset = __per_cpu_offset[cpu];
lc->kernel_asce = S390_lowcore.kernel_asce; lc->kernel_asce = S390_lowcore.kernel_asce;
lc->machine_flags = S390_lowcore.machine_flags; lc->machine_flags = S390_lowcore.machine_flags;
lc->ftrace_func = S390_lowcore.ftrace_func;
lc->user_timer = lc->system_timer = lc->steal_timer = 0; lc->user_timer = lc->system_timer = lc->steal_timer = 0;
__ctl_store(lc->cregs_save_area, 0, 15); __ctl_store(lc->cregs_save_area, 0, 15);
save_access_regs((unsigned int *) lc->access_regs_save_area); save_access_regs((unsigned int *) lc->access_regs_save_area);

View file

@ -360,3 +360,5 @@ SYSCALL(sys_seccomp,sys_seccomp,compat_sys_seccomp)
SYSCALL(sys_getrandom,sys_getrandom,compat_sys_getrandom) SYSCALL(sys_getrandom,sys_getrandom,compat_sys_getrandom)
SYSCALL(sys_memfd_create,sys_memfd_create,compat_sys_memfd_create) /* 350 */ SYSCALL(sys_memfd_create,sys_memfd_create,compat_sys_memfd_create) /* 350 */
SYSCALL(sys_bpf,sys_bpf,compat_sys_bpf) SYSCALL(sys_bpf,sys_bpf,compat_sys_bpf)
SYSCALL(sys_ni_syscall,sys_s390_pci_mmio_write,compat_sys_s390_pci_mmio_write)
SYSCALL(sys_ni_syscall,sys_s390_pci_mmio_read,compat_sys_s390_pci_mmio_read)

View file

@ -61,10 +61,11 @@ static DEFINE_PER_CPU(struct clock_event_device, comparators);
/* /*
* Scheduler clock - returns current time in nanosec units. * Scheduler clock - returns current time in nanosec units.
*/ */
unsigned long long notrace __kprobes sched_clock(void) unsigned long long notrace sched_clock(void)
{ {
return tod_to_ns(get_tod_clock_monotonic()); return tod_to_ns(get_tod_clock_monotonic());
} }
NOKPROBE_SYMBOL(sched_clock);
/* /*
* Monotonic_clock - returns # of nanoseconds passed since time_init() * Monotonic_clock - returns # of nanoseconds passed since time_init()

View file

@ -49,7 +49,8 @@ static inline void report_user_fault(struct pt_regs *regs, int signr)
return; return;
if (!printk_ratelimit()) if (!printk_ratelimit())
return; return;
printk("User process fault: interruption code 0x%X ", regs->int_code); printk("User process fault: interruption code %04x ilc:%d ",
regs->int_code & 0xffff, regs->int_code >> 17);
print_vma_addr("in ", regs->psw.addr & PSW_ADDR_INSN); print_vma_addr("in ", regs->psw.addr & PSW_ADDR_INSN);
printk("\n"); printk("\n");
show_regs(regs); show_regs(regs);
@ -87,16 +88,16 @@ void do_report_trap(struct pt_regs *regs, int si_signo, int si_code, char *str)
} }
} }
static void __kprobes do_trap(struct pt_regs *regs, int si_signo, int si_code, static void do_trap(struct pt_regs *regs, int si_signo, int si_code, char *str)
char *str)
{ {
if (notify_die(DIE_TRAP, str, regs, 0, if (notify_die(DIE_TRAP, str, regs, 0,
regs->int_code, si_signo) == NOTIFY_STOP) regs->int_code, si_signo) == NOTIFY_STOP)
return; return;
do_report_trap(regs, si_signo, si_code, str); do_report_trap(regs, si_signo, si_code, str);
} }
NOKPROBE_SYMBOL(do_trap);
void __kprobes do_per_trap(struct pt_regs *regs) void do_per_trap(struct pt_regs *regs)
{ {
siginfo_t info; siginfo_t info;
@ -111,6 +112,7 @@ void __kprobes do_per_trap(struct pt_regs *regs)
(void __force __user *) current->thread.per_event.address; (void __force __user *) current->thread.per_event.address;
force_sig_info(SIGTRAP, &info, current); force_sig_info(SIGTRAP, &info, current);
} }
NOKPROBE_SYMBOL(do_per_trap);
void default_trap_handler(struct pt_regs *regs) void default_trap_handler(struct pt_regs *regs)
{ {
@ -151,8 +153,6 @@ DO_ERROR_INFO(privileged_op, SIGILL, ILL_PRVOPC,
"privileged operation") "privileged operation")
DO_ERROR_INFO(special_op_exception, SIGILL, ILL_ILLOPN, DO_ERROR_INFO(special_op_exception, SIGILL, ILL_ILLOPN,
"special operation exception") "special operation exception")
DO_ERROR_INFO(translation_exception, SIGILL, ILL_ILLOPN,
"translation exception")
#ifdef CONFIG_64BIT #ifdef CONFIG_64BIT
DO_ERROR_INFO(transaction_exception, SIGILL, ILL_ILLOPN, DO_ERROR_INFO(transaction_exception, SIGILL, ILL_ILLOPN,
@ -179,7 +179,13 @@ static inline void do_fp_trap(struct pt_regs *regs, int fpc)
do_trap(regs, SIGFPE, si_code, "floating point exception"); do_trap(regs, SIGFPE, si_code, "floating point exception");
} }
void __kprobes illegal_op(struct pt_regs *regs) void translation_exception(struct pt_regs *regs)
{
/* May never happen. */
die(regs, "Translation exception");
}
void illegal_op(struct pt_regs *regs)
{ {
siginfo_t info; siginfo_t info;
__u8 opcode[6]; __u8 opcode[6];
@ -252,7 +258,7 @@ void __kprobes illegal_op(struct pt_regs *regs)
if (signal) if (signal)
do_trap(regs, signal, ILL_ILLOPC, "illegal operation"); do_trap(regs, signal, ILL_ILLOPC, "illegal operation");
} }
NOKPROBE_SYMBOL(illegal_op);
#ifdef CONFIG_MATHEMU #ifdef CONFIG_MATHEMU
void specification_exception(struct pt_regs *regs) void specification_exception(struct pt_regs *regs)
@ -469,7 +475,7 @@ void space_switch_exception(struct pt_regs *regs)
do_trap(regs, SIGILL, ILL_PRVOPC, "space switch event"); do_trap(regs, SIGILL, ILL_PRVOPC, "space switch event");
} }
void __kprobes kernel_stack_overflow(struct pt_regs * regs) void kernel_stack_overflow(struct pt_regs *regs)
{ {
bust_spinlocks(1); bust_spinlocks(1);
printk("Kernel stack overflow.\n"); printk("Kernel stack overflow.\n");
@ -477,6 +483,7 @@ void __kprobes kernel_stack_overflow(struct pt_regs * regs)
bust_spinlocks(0); bust_spinlocks(0);
panic("Corrupt kernel stack, can't continue."); panic("Corrupt kernel stack, can't continue.");
} }
NOKPROBE_SYMBOL(kernel_stack_overflow);
void __init trap_init(void) void __init trap_init(void)
{ {

View file

@ -271,7 +271,7 @@ static int kvm_s390_mem_control(struct kvm *kvm, struct kvm_device_attr *attr)
case KVM_S390_VM_MEM_CLR_CMMA: case KVM_S390_VM_MEM_CLR_CMMA:
mutex_lock(&kvm->lock); mutex_lock(&kvm->lock);
idx = srcu_read_lock(&kvm->srcu); idx = srcu_read_lock(&kvm->srcu);
page_table_reset_pgste(kvm->arch.gmap->mm, 0, TASK_SIZE, false); s390_reset_cmma(kvm->arch.gmap->mm);
srcu_read_unlock(&kvm->srcu, idx); srcu_read_unlock(&kvm->srcu, idx);
mutex_unlock(&kvm->lock); mutex_unlock(&kvm->lock);
ret = 0; ret = 0;

View file

@ -156,21 +156,25 @@ static int handle_store_cpu_address(struct kvm_vcpu *vcpu)
return 0; return 0;
} }
static void __skey_check_enable(struct kvm_vcpu *vcpu) static int __skey_check_enable(struct kvm_vcpu *vcpu)
{ {
int rc = 0;
if (!(vcpu->arch.sie_block->ictl & (ICTL_ISKE | ICTL_SSKE | ICTL_RRBE))) if (!(vcpu->arch.sie_block->ictl & (ICTL_ISKE | ICTL_SSKE | ICTL_RRBE)))
return; return rc;
s390_enable_skey(); rc = s390_enable_skey();
trace_kvm_s390_skey_related_inst(vcpu); trace_kvm_s390_skey_related_inst(vcpu);
vcpu->arch.sie_block->ictl &= ~(ICTL_ISKE | ICTL_SSKE | ICTL_RRBE); vcpu->arch.sie_block->ictl &= ~(ICTL_ISKE | ICTL_SSKE | ICTL_RRBE);
return rc;
} }
static int handle_skey(struct kvm_vcpu *vcpu) static int handle_skey(struct kvm_vcpu *vcpu)
{ {
__skey_check_enable(vcpu); int rc = __skey_check_enable(vcpu);
if (rc)
return rc;
vcpu->stat.instruction_storage_key++; vcpu->stat.instruction_storage_key++;
if (vcpu->arch.sie_block->gpsw.mask & PSW_MASK_PSTATE) if (vcpu->arch.sie_block->gpsw.mask & PSW_MASK_PSTATE)
@ -683,7 +687,10 @@ static int handle_pfmf(struct kvm_vcpu *vcpu)
} }
if (vcpu->run->s.regs.gprs[reg1] & PFMF_SK) { if (vcpu->run->s.regs.gprs[reg1] & PFMF_SK) {
__skey_check_enable(vcpu); int rc = __skey_check_enable(vcpu);
if (rc)
return rc;
if (set_guest_storage_key(current->mm, useraddr, if (set_guest_storage_key(current->mm, useraddr,
vcpu->run->s.regs.gprs[reg1] & PFMF_KEY, vcpu->run->s.regs.gprs[reg1] & PFMF_KEY,
vcpu->run->s.regs.gprs[reg1] & PFMF_NQ)) vcpu->run->s.regs.gprs[reg1] & PFMF_NQ))

View file

@ -261,8 +261,8 @@ static inline void report_user_fault(struct pt_regs *regs, long signr)
return; return;
if (!printk_ratelimit()) if (!printk_ratelimit())
return; return;
printk(KERN_ALERT "User process fault: interruption code 0x%X ", printk(KERN_ALERT "User process fault: interruption code %04x ilc:%d",
regs->int_code); regs->int_code & 0xffff, regs->int_code >> 17);
print_vma_addr(KERN_CONT "in ", regs->psw.addr & PSW_ADDR_INSN); print_vma_addr(KERN_CONT "in ", regs->psw.addr & PSW_ADDR_INSN);
printk(KERN_CONT "\n"); printk(KERN_CONT "\n");
printk(KERN_ALERT "failing address: %016lx TEID: %016lx\n", printk(KERN_ALERT "failing address: %016lx TEID: %016lx\n",
@ -548,7 +548,7 @@ out:
return fault; return fault;
} }
void __kprobes do_protection_exception(struct pt_regs *regs) void do_protection_exception(struct pt_regs *regs)
{ {
unsigned long trans_exc_code; unsigned long trans_exc_code;
int fault; int fault;
@ -574,8 +574,9 @@ void __kprobes do_protection_exception(struct pt_regs *regs)
if (unlikely(fault)) if (unlikely(fault))
do_fault_error(regs, fault); do_fault_error(regs, fault);
} }
NOKPROBE_SYMBOL(do_protection_exception);
void __kprobes do_dat_exception(struct pt_regs *regs) void do_dat_exception(struct pt_regs *regs)
{ {
int access, fault; int access, fault;
@ -584,6 +585,7 @@ void __kprobes do_dat_exception(struct pt_regs *regs)
if (unlikely(fault)) if (unlikely(fault))
do_fault_error(regs, fault); do_fault_error(regs, fault);
} }
NOKPROBE_SYMBOL(do_dat_exception);
#ifdef CONFIG_PFAULT #ifdef CONFIG_PFAULT
/* /*

View file

@ -18,6 +18,8 @@
#include <linux/rcupdate.h> #include <linux/rcupdate.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/swapops.h> #include <linux/swapops.h>
#include <linux/ksm.h>
#include <linux/mman.h>
#include <asm/pgtable.h> #include <asm/pgtable.h>
#include <asm/pgalloc.h> #include <asm/pgalloc.h>
@ -750,8 +752,7 @@ int gmap_ipte_notify(struct gmap *gmap, unsigned long gaddr, unsigned long len)
break; break;
/* Walk the process page table, lock and get pte pointer */ /* Walk the process page table, lock and get pte pointer */
ptep = get_locked_pte(gmap->mm, addr, &ptl); ptep = get_locked_pte(gmap->mm, addr, &ptl);
if (unlikely(!ptep)) VM_BUG_ON(!ptep);
continue;
/* Set notification bit in the pgste of the pte */ /* Set notification bit in the pgste of the pte */
entry = *ptep; entry = *ptep;
if ((pte_val(entry) & (_PAGE_INVALID | _PAGE_PROTECT)) == 0) { if ((pte_val(entry) & (_PAGE_INVALID | _PAGE_PROTECT)) == 0) {
@ -761,7 +762,7 @@ int gmap_ipte_notify(struct gmap *gmap, unsigned long gaddr, unsigned long len)
gaddr += PAGE_SIZE; gaddr += PAGE_SIZE;
len -= PAGE_SIZE; len -= PAGE_SIZE;
} }
spin_unlock(ptl); pte_unmap_unlock(ptep, ptl);
} }
up_read(&gmap->mm->mmap_sem); up_read(&gmap->mm->mmap_sem);
return rc; return rc;
@ -834,99 +835,6 @@ static inline void page_table_free_pgste(unsigned long *table)
__free_page(page); __free_page(page);
} }
static inline unsigned long page_table_reset_pte(struct mm_struct *mm, pmd_t *pmd,
unsigned long addr, unsigned long end, bool init_skey)
{
pte_t *start_pte, *pte;
spinlock_t *ptl;
pgste_t pgste;
start_pte = pte_offset_map_lock(mm, pmd, addr, &ptl);
pte = start_pte;
do {
pgste = pgste_get_lock(pte);
pgste_val(pgste) &= ~_PGSTE_GPS_USAGE_MASK;
if (init_skey) {
unsigned long address;
pgste_val(pgste) &= ~(PGSTE_ACC_BITS | PGSTE_FP_BIT |
PGSTE_GR_BIT | PGSTE_GC_BIT);
/* skip invalid and not writable pages */
if (pte_val(*pte) & _PAGE_INVALID ||
!(pte_val(*pte) & _PAGE_WRITE)) {
pgste_set_unlock(pte, pgste);
continue;
}
address = pte_val(*pte) & PAGE_MASK;
page_set_storage_key(address, PAGE_DEFAULT_KEY, 1);
}
pgste_set_unlock(pte, pgste);
} while (pte++, addr += PAGE_SIZE, addr != end);
pte_unmap_unlock(start_pte, ptl);
return addr;
}
static inline unsigned long page_table_reset_pmd(struct mm_struct *mm, pud_t *pud,
unsigned long addr, unsigned long end, bool init_skey)
{
unsigned long next;
pmd_t *pmd;
pmd = pmd_offset(pud, addr);
do {
next = pmd_addr_end(addr, end);
if (pmd_none_or_clear_bad(pmd))
continue;
next = page_table_reset_pte(mm, pmd, addr, next, init_skey);
} while (pmd++, addr = next, addr != end);
return addr;
}
static inline unsigned long page_table_reset_pud(struct mm_struct *mm, pgd_t *pgd,
unsigned long addr, unsigned long end, bool init_skey)
{
unsigned long next;
pud_t *pud;
pud = pud_offset(pgd, addr);
do {
next = pud_addr_end(addr, end);
if (pud_none_or_clear_bad(pud))
continue;
next = page_table_reset_pmd(mm, pud, addr, next, init_skey);
} while (pud++, addr = next, addr != end);
return addr;
}
void page_table_reset_pgste(struct mm_struct *mm, unsigned long start,
unsigned long end, bool init_skey)
{
unsigned long addr, next;
pgd_t *pgd;
down_write(&mm->mmap_sem);
if (init_skey && mm_use_skey(mm))
goto out_up;
addr = start;
pgd = pgd_offset(mm, addr);
do {
next = pgd_addr_end(addr, end);
if (pgd_none_or_clear_bad(pgd))
continue;
next = page_table_reset_pud(mm, pgd, addr, next, init_skey);
} while (pgd++, addr = next, addr != end);
if (init_skey)
current->mm->context.use_skey = 1;
out_up:
up_write(&mm->mmap_sem);
}
EXPORT_SYMBOL(page_table_reset_pgste);
int set_guest_storage_key(struct mm_struct *mm, unsigned long addr, int set_guest_storage_key(struct mm_struct *mm, unsigned long addr,
unsigned long key, bool nq) unsigned long key, bool nq)
{ {
@ -992,11 +900,6 @@ static inline unsigned long *page_table_alloc_pgste(struct mm_struct *mm)
return NULL; return NULL;
} }
void page_table_reset_pgste(struct mm_struct *mm, unsigned long start,
unsigned long end, bool init_skey)
{
}
static inline void page_table_free_pgste(unsigned long *table) static inline void page_table_free_pgste(unsigned long *table)
{ {
} }
@ -1347,12 +1250,88 @@ EXPORT_SYMBOL_GPL(s390_enable_sie);
* Enable storage key handling from now on and initialize the storage * Enable storage key handling from now on and initialize the storage
* keys with the default key. * keys with the default key.
*/ */
void s390_enable_skey(void) static int __s390_enable_skey(pte_t *pte, unsigned long addr,
unsigned long next, struct mm_walk *walk)
{ {
page_table_reset_pgste(current->mm, 0, TASK_SIZE, true); unsigned long ptev;
pgste_t pgste;
pgste = pgste_get_lock(pte);
/*
* Remove all zero page mappings,
* after establishing a policy to forbid zero page mappings
* following faults for that page will get fresh anonymous pages
*/
if (is_zero_pfn(pte_pfn(*pte))) {
ptep_flush_direct(walk->mm, addr, pte);
pte_val(*pte) = _PAGE_INVALID;
}
/* Clear storage key */
pgste_val(pgste) &= ~(PGSTE_ACC_BITS | PGSTE_FP_BIT |
PGSTE_GR_BIT | PGSTE_GC_BIT);
ptev = pte_val(*pte);
if (!(ptev & _PAGE_INVALID) && (ptev & _PAGE_WRITE))
page_set_storage_key(ptev & PAGE_MASK, PAGE_DEFAULT_KEY, 1);
pgste_set_unlock(pte, pgste);
return 0;
}
int s390_enable_skey(void)
{
struct mm_walk walk = { .pte_entry = __s390_enable_skey };
struct mm_struct *mm = current->mm;
struct vm_area_struct *vma;
int rc = 0;
down_write(&mm->mmap_sem);
if (mm_use_skey(mm))
goto out_up;
mm->context.use_skey = 1;
for (vma = mm->mmap; vma; vma = vma->vm_next) {
if (ksm_madvise(vma, vma->vm_start, vma->vm_end,
MADV_UNMERGEABLE, &vma->vm_flags)) {
mm->context.use_skey = 0;
rc = -ENOMEM;
goto out_up;
}
}
mm->def_flags &= ~VM_MERGEABLE;
walk.mm = mm;
walk_page_range(0, TASK_SIZE, &walk);
out_up:
up_write(&mm->mmap_sem);
return rc;
} }
EXPORT_SYMBOL_GPL(s390_enable_skey); EXPORT_SYMBOL_GPL(s390_enable_skey);
/*
* Reset CMMA state, make all pages stable again.
*/
static int __s390_reset_cmma(pte_t *pte, unsigned long addr,
unsigned long next, struct mm_walk *walk)
{
pgste_t pgste;
pgste = pgste_get_lock(pte);
pgste_val(pgste) &= ~_PGSTE_GPS_USAGE_MASK;
pgste_set_unlock(pte, pgste);
return 0;
}
void s390_reset_cmma(struct mm_struct *mm)
{
struct mm_walk walk = { .pte_entry = __s390_reset_cmma };
down_write(&mm->mmap_sem);
walk.mm = mm;
walk_page_range(0, TASK_SIZE, &walk);
up_write(&mm->mmap_sem);
}
EXPORT_SYMBOL_GPL(s390_reset_cmma);
/* /*
* Test and reset if a guest page is dirty * Test and reset if a guest page is dirty
*/ */

View file

@ -3,4 +3,4 @@
# #
obj-$(CONFIG_PCI) += pci.o pci_dma.o pci_clp.o pci_sysfs.o \ obj-$(CONFIG_PCI) += pci.o pci_dma.o pci_clp.o pci_sysfs.o \
pci_event.o pci_debug.o pci_insn.o pci_event.o pci_debug.o pci_insn.o pci_mmio.o

View file

@ -369,8 +369,7 @@ int arch_setup_msi_irqs(struct pci_dev *pdev, int nvec, int type)
if (type == PCI_CAP_ID_MSI && nvec > 1) if (type == PCI_CAP_ID_MSI && nvec > 1)
return 1; return 1;
msi_vecs = min(nvec, ZPCI_MSI_VEC_MAX); msi_vecs = min_t(unsigned int, nvec, zdev->max_msi);
msi_vecs = min_t(unsigned int, msi_vecs, CONFIG_PCI_NR_MSI);
/* Allocate adapter summary indicator bit */ /* Allocate adapter summary indicator bit */
rc = -EIO; rc = -EIO;
@ -474,7 +473,8 @@ static void zpci_map_resources(struct zpci_dev *zdev)
len = pci_resource_len(pdev, i); len = pci_resource_len(pdev, i);
if (!len) if (!len)
continue; continue;
pdev->resource[i].start = (resource_size_t) pci_iomap(pdev, i, 0); pdev->resource[i].start =
(resource_size_t __force) pci_iomap(pdev, i, 0);
pdev->resource[i].end = pdev->resource[i].start + len - 1; pdev->resource[i].end = pdev->resource[i].start + len - 1;
} }
} }
@ -489,7 +489,8 @@ static void zpci_unmap_resources(struct zpci_dev *zdev)
len = pci_resource_len(pdev, i); len = pci_resource_len(pdev, i);
if (!len) if (!len)
continue; continue;
pci_iounmap(pdev, (void *) pdev->resource[i].start); pci_iounmap(pdev, (void __iomem __force *)
pdev->resource[i].start);
} }
} }

View file

@ -62,6 +62,7 @@ static void clp_store_query_pci_fngrp(struct zpci_dev *zdev,
zdev->tlb_refresh = response->refresh; zdev->tlb_refresh = response->refresh;
zdev->dma_mask = response->dasm; zdev->dma_mask = response->dasm;
zdev->msi_addr = response->msia; zdev->msi_addr = response->msia;
zdev->max_msi = response->noi;
zdev->fmb_update = response->mui; zdev->fmb_update = response->mui;
switch (response->version) { switch (response->version) {

View file

@ -158,10 +158,7 @@ int __init zpci_debug_init(void)
void zpci_debug_exit(void) void zpci_debug_exit(void)
{ {
if (pci_debug_msg_id) debug_unregister(pci_debug_msg_id);
debug_unregister(pci_debug_msg_id); debug_unregister(pci_debug_err_id);
if (pci_debug_err_id)
debug_unregister(pci_debug_err_id);
debugfs_remove(debugfs_root); debugfs_remove(debugfs_root);
} }

115
arch/s390/pci/pci_mmio.c Normal file
View file

@ -0,0 +1,115 @@
/*
* Access to PCI I/O memory from user space programs.
*
* Copyright IBM Corp. 2014
* Author(s): Alexey Ishchuk <aishchuk@linux.vnet.ibm.com>
*/
#include <linux/kernel.h>
#include <linux/syscalls.h>
#include <linux/init.h>
#include <linux/mm.h>
#include <linux/errno.h>
#include <linux/pci.h>
static long get_pfn(unsigned long user_addr, unsigned long access,
unsigned long *pfn)
{
struct vm_area_struct *vma;
long ret;
down_read(&current->mm->mmap_sem);
ret = -EINVAL;
vma = find_vma(current->mm, user_addr);
if (!vma)
goto out;
ret = -EACCES;
if (!(vma->vm_flags & access))
goto out;
ret = follow_pfn(vma, user_addr, pfn);
out:
up_read(&current->mm->mmap_sem);
return ret;
}
SYSCALL_DEFINE3(s390_pci_mmio_write, unsigned long, mmio_addr,
const void __user *, user_buffer, size_t, length)
{
u8 local_buf[64];
void __iomem *io_addr;
void *buf;
unsigned long pfn;
long ret;
if (!zpci_is_enabled())
return -ENODEV;
if (length <= 0 || PAGE_SIZE - (mmio_addr & ~PAGE_MASK) < length)
return -EINVAL;
if (length > 64) {
buf = kmalloc(length, GFP_KERNEL);
if (!buf)
return -ENOMEM;
} else
buf = local_buf;
ret = get_pfn(mmio_addr, VM_WRITE, &pfn);
if (ret)
goto out;
io_addr = (void *)((pfn << PAGE_SHIFT) | (mmio_addr & ~PAGE_MASK));
ret = -EFAULT;
if ((unsigned long) io_addr < ZPCI_IOMAP_ADDR_BASE)
goto out;
if (copy_from_user(buf, user_buffer, length))
goto out;
memcpy_toio(io_addr, buf, length);
ret = 0;
out:
if (buf != local_buf)
kfree(buf);
return ret;
}
SYSCALL_DEFINE3(s390_pci_mmio_read, unsigned long, mmio_addr,
void __user *, user_buffer, size_t, length)
{
u8 local_buf[64];
void __iomem *io_addr;
void *buf;
unsigned long pfn;
long ret;
if (!zpci_is_enabled())
return -ENODEV;
if (length <= 0 || PAGE_SIZE - (mmio_addr & ~PAGE_MASK) < length)
return -EINVAL;
if (length > 64) {
buf = kmalloc(length, GFP_KERNEL);
if (!buf)
return -ENOMEM;
} else
buf = local_buf;
ret = get_pfn(mmio_addr, VM_READ, &pfn);
if (ret)
goto out;
io_addr = (void *)((pfn << PAGE_SHIFT) | (mmio_addr & ~PAGE_MASK));
ret = -EFAULT;
if ((unsigned long) io_addr < ZPCI_IOMAP_ADDR_BASE)
goto out;
memcpy_fromio(buf, io_addr, length);
if (copy_to_user(user_buffer, buf, length))
goto out;
ret = 0;
out:
if (buf != local_buf)
kfree(buf);
return ret;
}

View file

@ -1377,6 +1377,20 @@ int dasd_term_IO(struct dasd_ccw_req *cqr)
"I/O error, retry"); "I/O error, retry");
break; break;
case -EINVAL: case -EINVAL:
/*
* device not valid so no I/O could be running
* handle CQR as termination successful
*/
cqr->status = DASD_CQR_CLEARED;
cqr->stopclk = get_tod_clock();
cqr->starttime = 0;
/* no retries for invalid devices */
cqr->retries = -1;
DBF_DEV_EVENT(DBF_ERR, device, "%s",
"EINVAL, handle as terminated");
/* fake rc to success */
rc = 0;
break;
case -EBUSY: case -EBUSY:
DBF_DEV_EVENT(DBF_ERR, device, "%s", DBF_DEV_EVENT(DBF_ERR, device, "%s",
"device busy, retry later"); "device busy, retry later");
@ -1683,11 +1697,8 @@ void dasd_int_handler(struct ccw_device *cdev, unsigned long intparm,
if (cqr->status == DASD_CQR_CLEAR_PENDING && if (cqr->status == DASD_CQR_CLEAR_PENDING &&
scsw_fctl(&irb->scsw) & SCSW_FCTL_CLEAR_FUNC) { scsw_fctl(&irb->scsw) & SCSW_FCTL_CLEAR_FUNC) {
cqr->status = DASD_CQR_CLEARED; cqr->status = DASD_CQR_CLEARED;
if (cqr->callback_data == DASD_SLEEPON_START_TAG)
cqr->callback_data = DASD_SLEEPON_END_TAG;
dasd_device_clear_timer(device); dasd_device_clear_timer(device);
wake_up(&dasd_flush_wq); wake_up(&dasd_flush_wq);
wake_up(&generic_waitq);
dasd_schedule_device_bh(device); dasd_schedule_device_bh(device);
return; return;
} }
@ -2326,21 +2337,11 @@ retry:
return -EAGAIN; return -EAGAIN;
/* normal recovery for basedev IO */ /* normal recovery for basedev IO */
if (__dasd_sleep_on_erp(cqr)) { if (__dasd_sleep_on_erp(cqr))
/* handle erp first */
goto retry; goto retry;
/* remember that ERP was needed */
rc = 1;
/* skip processing for active cqr */
if (cqr->status != DASD_CQR_TERMINATED &&
cqr->status != DASD_CQR_NEED_ERP)
break;
}
} }
/* start ERP requests in upper loop */
if (rc)
goto retry;
return 0; return 0;
} }

View file

@ -99,15 +99,37 @@ void dasd_gendisk_free(struct dasd_block *block)
int dasd_scan_partitions(struct dasd_block *block) int dasd_scan_partitions(struct dasd_block *block)
{ {
struct block_device *bdev; struct block_device *bdev;
int retry, rc;
retry = 5;
bdev = bdget_disk(block->gdp, 0); bdev = bdget_disk(block->gdp, 0);
if (!bdev || blkdev_get(bdev, FMODE_READ, NULL) < 0) if (!bdev) {
DBF_DEV_EVENT(DBF_ERR, block->base, "%s",
"scan partitions error, bdget returned NULL");
return -ENODEV; return -ENODEV;
}
rc = blkdev_get(bdev, FMODE_READ, NULL);
if (rc < 0) {
DBF_DEV_EVENT(DBF_ERR, block->base,
"scan partitions error, blkdev_get returned %d",
rc);
return -ENODEV;
}
/* /*
* See fs/partition/check.c:register_disk,rescan_partitions * See fs/partition/check.c:register_disk,rescan_partitions
* Can't call rescan_partitions directly. Use ioctl. * Can't call rescan_partitions directly. Use ioctl.
*/ */
ioctl_by_bdev(bdev, BLKRRPART, 0); rc = ioctl_by_bdev(bdev, BLKRRPART, 0);
while (rc == -EBUSY && retry > 0) {
schedule();
rc = ioctl_by_bdev(bdev, BLKRRPART, 0);
retry--;
DBF_DEV_EVENT(DBF_ERR, block->base,
"scan partitions error, retry %d rc %d",
retry, rc);
}
/* /*
* Since the matching blkdev_put call to the blkdev_get in * Since the matching blkdev_put call to the blkdev_get in
* this function is not called before dasd_destroy_partitions * this function is not called before dasd_destroy_partitions

View file

@ -10,6 +10,7 @@
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/spinlock.h> #include <linux/spinlock.h>
#include <linux/mempool.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/blkdev.h> #include <linux/blkdev.h>
#include <linux/genhd.h> #include <linux/genhd.h>
@ -20,13 +21,18 @@
debug_info_t *scm_debug; debug_info_t *scm_debug;
static int scm_major; static int scm_major;
static mempool_t *aidaw_pool;
static DEFINE_SPINLOCK(list_lock); static DEFINE_SPINLOCK(list_lock);
static LIST_HEAD(inactive_requests); static LIST_HEAD(inactive_requests);
static unsigned int nr_requests = 64; static unsigned int nr_requests = 64;
static unsigned int nr_requests_per_io = 8;
static atomic_t nr_devices = ATOMIC_INIT(0); static atomic_t nr_devices = ATOMIC_INIT(0);
module_param(nr_requests, uint, S_IRUGO); module_param(nr_requests, uint, S_IRUGO);
MODULE_PARM_DESC(nr_requests, "Number of parallel requests."); MODULE_PARM_DESC(nr_requests, "Number of parallel requests.");
module_param(nr_requests_per_io, uint, S_IRUGO);
MODULE_PARM_DESC(nr_requests_per_io, "Number of requests per IO.");
MODULE_DESCRIPTION("Block driver for s390 storage class memory."); MODULE_DESCRIPTION("Block driver for s390 storage class memory.");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
MODULE_ALIAS("scm:scmdev*"); MODULE_ALIAS("scm:scmdev*");
@ -36,8 +42,8 @@ static void __scm_free_rq(struct scm_request *scmrq)
struct aob_rq_header *aobrq = to_aobrq(scmrq); struct aob_rq_header *aobrq = to_aobrq(scmrq);
free_page((unsigned long) scmrq->aob); free_page((unsigned long) scmrq->aob);
free_page((unsigned long) scmrq->aidaw);
__scm_free_rq_cluster(scmrq); __scm_free_rq_cluster(scmrq);
kfree(scmrq->request);
kfree(aobrq); kfree(aobrq);
} }
@ -53,6 +59,8 @@ static void scm_free_rqs(void)
__scm_free_rq(scmrq); __scm_free_rq(scmrq);
} }
spin_unlock_irq(&list_lock); spin_unlock_irq(&list_lock);
mempool_destroy(aidaw_pool);
} }
static int __scm_alloc_rq(void) static int __scm_alloc_rq(void)
@ -65,17 +73,17 @@ static int __scm_alloc_rq(void)
return -ENOMEM; return -ENOMEM;
scmrq = (void *) aobrq->data; scmrq = (void *) aobrq->data;
scmrq->aidaw = (void *) get_zeroed_page(GFP_DMA);
scmrq->aob = (void *) get_zeroed_page(GFP_DMA); scmrq->aob = (void *) get_zeroed_page(GFP_DMA);
if (!scmrq->aob || !scmrq->aidaw) { if (!scmrq->aob)
__scm_free_rq(scmrq); goto free;
return -ENOMEM;
}
if (__scm_alloc_rq_cluster(scmrq)) { scmrq->request = kcalloc(nr_requests_per_io, sizeof(scmrq->request[0]),
__scm_free_rq(scmrq); GFP_KERNEL);
return -ENOMEM; if (!scmrq->request)
} goto free;
if (__scm_alloc_rq_cluster(scmrq))
goto free;
INIT_LIST_HEAD(&scmrq->list); INIT_LIST_HEAD(&scmrq->list);
spin_lock_irq(&list_lock); spin_lock_irq(&list_lock);
@ -83,12 +91,19 @@ static int __scm_alloc_rq(void)
spin_unlock_irq(&list_lock); spin_unlock_irq(&list_lock);
return 0; return 0;
free:
__scm_free_rq(scmrq);
return -ENOMEM;
} }
static int scm_alloc_rqs(unsigned int nrqs) static int scm_alloc_rqs(unsigned int nrqs)
{ {
int ret = 0; int ret = 0;
aidaw_pool = mempool_create_page_pool(max(nrqs/8, 1U), 0);
if (!aidaw_pool)
return -ENOMEM;
while (nrqs-- && !ret) while (nrqs-- && !ret)
ret = __scm_alloc_rq(); ret = __scm_alloc_rq();
@ -112,6 +127,18 @@ out:
static void scm_request_done(struct scm_request *scmrq) static void scm_request_done(struct scm_request *scmrq)
{ {
unsigned long flags; unsigned long flags;
struct msb *msb;
u64 aidaw;
int i;
for (i = 0; i < nr_requests_per_io && scmrq->request[i]; i++) {
msb = &scmrq->aob->msb[i];
aidaw = msb->data_addr;
if ((msb->flags & MSB_FLAG_IDA) && aidaw &&
IS_ALIGNED(aidaw, PAGE_SIZE))
mempool_free(virt_to_page(aidaw), aidaw_pool);
}
spin_lock_irqsave(&list_lock, flags); spin_lock_irqsave(&list_lock, flags);
list_add(&scmrq->list, &inactive_requests); list_add(&scmrq->list, &inactive_requests);
@ -123,48 +150,90 @@ static bool scm_permit_request(struct scm_blk_dev *bdev, struct request *req)
return rq_data_dir(req) != WRITE || bdev->state != SCM_WR_PROHIBIT; return rq_data_dir(req) != WRITE || bdev->state != SCM_WR_PROHIBIT;
} }
static void scm_request_prepare(struct scm_request *scmrq) static inline struct aidaw *scm_aidaw_alloc(void)
{
struct page *page = mempool_alloc(aidaw_pool, GFP_ATOMIC);
return page ? page_address(page) : NULL;
}
static inline unsigned long scm_aidaw_bytes(struct aidaw *aidaw)
{
unsigned long _aidaw = (unsigned long) aidaw;
unsigned long bytes = ALIGN(_aidaw, PAGE_SIZE) - _aidaw;
return (bytes / sizeof(*aidaw)) * PAGE_SIZE;
}
struct aidaw *scm_aidaw_fetch(struct scm_request *scmrq, unsigned int bytes)
{
struct aidaw *aidaw;
if (scm_aidaw_bytes(scmrq->next_aidaw) >= bytes)
return scmrq->next_aidaw;
aidaw = scm_aidaw_alloc();
if (aidaw)
memset(aidaw, 0, PAGE_SIZE);
return aidaw;
}
static int scm_request_prepare(struct scm_request *scmrq)
{ {
struct scm_blk_dev *bdev = scmrq->bdev; struct scm_blk_dev *bdev = scmrq->bdev;
struct scm_device *scmdev = bdev->gendisk->private_data; struct scm_device *scmdev = bdev->gendisk->private_data;
struct aidaw *aidaw = scmrq->aidaw; int pos = scmrq->aob->request.msb_count;
struct msb *msb = &scmrq->aob->msb[0]; struct msb *msb = &scmrq->aob->msb[pos];
struct request *req = scmrq->request[pos];
struct req_iterator iter; struct req_iterator iter;
struct aidaw *aidaw;
struct bio_vec bv; struct bio_vec bv;
aidaw = scm_aidaw_fetch(scmrq, blk_rq_bytes(req));
if (!aidaw)
return -ENOMEM;
msb->bs = MSB_BS_4K; msb->bs = MSB_BS_4K;
scmrq->aob->request.msb_count = 1; scmrq->aob->request.msb_count++;
msb->scm_addr = scmdev->address + msb->scm_addr = scmdev->address + ((u64) blk_rq_pos(req) << 9);
((u64) blk_rq_pos(scmrq->request) << 9); msb->oc = (rq_data_dir(req) == READ) ? MSB_OC_READ : MSB_OC_WRITE;
msb->oc = (rq_data_dir(scmrq->request) == READ) ?
MSB_OC_READ : MSB_OC_WRITE;
msb->flags |= MSB_FLAG_IDA; msb->flags |= MSB_FLAG_IDA;
msb->data_addr = (u64) aidaw; msb->data_addr = (u64) aidaw;
rq_for_each_segment(bv, scmrq->request, iter) { rq_for_each_segment(bv, req, iter) {
WARN_ON(bv.bv_offset); WARN_ON(bv.bv_offset);
msb->blk_count += bv.bv_len >> 12; msb->blk_count += bv.bv_len >> 12;
aidaw->data_addr = (u64) page_address(bv.bv_page); aidaw->data_addr = (u64) page_address(bv.bv_page);
aidaw++; aidaw++;
} }
scmrq->next_aidaw = aidaw;
return 0;
}
static inline void scm_request_set(struct scm_request *scmrq,
struct request *req)
{
scmrq->request[scmrq->aob->request.msb_count] = req;
} }
static inline void scm_request_init(struct scm_blk_dev *bdev, static inline void scm_request_init(struct scm_blk_dev *bdev,
struct scm_request *scmrq, struct scm_request *scmrq)
struct request *req)
{ {
struct aob_rq_header *aobrq = to_aobrq(scmrq); struct aob_rq_header *aobrq = to_aobrq(scmrq);
struct aob *aob = scmrq->aob; struct aob *aob = scmrq->aob;
memset(scmrq->request, 0,
nr_requests_per_io * sizeof(scmrq->request[0]));
memset(aob, 0, sizeof(*aob)); memset(aob, 0, sizeof(*aob));
memset(scmrq->aidaw, 0, PAGE_SIZE);
aobrq->scmdev = bdev->scmdev; aobrq->scmdev = bdev->scmdev;
aob->request.cmd_code = ARQB_CMD_MOVE; aob->request.cmd_code = ARQB_CMD_MOVE;
aob->request.data = (u64) aobrq; aob->request.data = (u64) aobrq;
scmrq->request = req;
scmrq->bdev = bdev; scmrq->bdev = bdev;
scmrq->retries = 4; scmrq->retries = 4;
scmrq->error = 0; scmrq->error = 0;
/* We don't use all msbs - place aidaws at the end of the aob page. */
scmrq->next_aidaw = (void *) &aob->msb[nr_requests_per_io];
scm_request_cluster_init(scmrq); scm_request_cluster_init(scmrq);
} }
@ -180,9 +249,12 @@ static void scm_ensure_queue_restart(struct scm_blk_dev *bdev)
void scm_request_requeue(struct scm_request *scmrq) void scm_request_requeue(struct scm_request *scmrq)
{ {
struct scm_blk_dev *bdev = scmrq->bdev; struct scm_blk_dev *bdev = scmrq->bdev;
int i;
scm_release_cluster(scmrq); scm_release_cluster(scmrq);
blk_requeue_request(bdev->rq, scmrq->request); for (i = 0; i < nr_requests_per_io && scmrq->request[i]; i++)
blk_requeue_request(bdev->rq, scmrq->request[i]);
atomic_dec(&bdev->queued_reqs); atomic_dec(&bdev->queued_reqs);
scm_request_done(scmrq); scm_request_done(scmrq);
scm_ensure_queue_restart(bdev); scm_ensure_queue_restart(bdev);
@ -191,20 +263,41 @@ void scm_request_requeue(struct scm_request *scmrq)
void scm_request_finish(struct scm_request *scmrq) void scm_request_finish(struct scm_request *scmrq)
{ {
struct scm_blk_dev *bdev = scmrq->bdev; struct scm_blk_dev *bdev = scmrq->bdev;
int i;
scm_release_cluster(scmrq); scm_release_cluster(scmrq);
blk_end_request_all(scmrq->request, scmrq->error); for (i = 0; i < nr_requests_per_io && scmrq->request[i]; i++)
blk_end_request_all(scmrq->request[i], scmrq->error);
atomic_dec(&bdev->queued_reqs); atomic_dec(&bdev->queued_reqs);
scm_request_done(scmrq); scm_request_done(scmrq);
} }
static int scm_request_start(struct scm_request *scmrq)
{
struct scm_blk_dev *bdev = scmrq->bdev;
int ret;
atomic_inc(&bdev->queued_reqs);
if (!scmrq->aob->request.msb_count) {
scm_request_requeue(scmrq);
return -EINVAL;
}
ret = eadm_start_aob(scmrq->aob);
if (ret) {
SCM_LOG(5, "no subchannel");
scm_request_requeue(scmrq);
}
return ret;
}
static void scm_blk_request(struct request_queue *rq) static void scm_blk_request(struct request_queue *rq)
{ {
struct scm_device *scmdev = rq->queuedata; struct scm_device *scmdev = rq->queuedata;
struct scm_blk_dev *bdev = dev_get_drvdata(&scmdev->dev); struct scm_blk_dev *bdev = dev_get_drvdata(&scmdev->dev);
struct scm_request *scmrq; struct scm_request *scmrq = NULL;
struct request *req; struct request *req;
int ret;
while ((req = blk_peek_request(rq))) { while ((req = blk_peek_request(rq))) {
if (req->cmd_type != REQ_TYPE_FS) { if (req->cmd_type != REQ_TYPE_FS) {
@ -214,39 +307,64 @@ static void scm_blk_request(struct request_queue *rq)
continue; continue;
} }
if (!scm_permit_request(bdev, req)) { if (!scm_permit_request(bdev, req))
scm_ensure_queue_restart(bdev); goto out;
return;
}
scmrq = scm_request_fetch();
if (!scmrq) { if (!scmrq) {
SCM_LOG(5, "no request"); scmrq = scm_request_fetch();
scm_ensure_queue_restart(bdev); if (!scmrq) {
return; SCM_LOG(5, "no request");
goto out;
}
scm_request_init(bdev, scmrq);
} }
scm_request_init(bdev, scmrq, req); scm_request_set(scmrq, req);
if (!scm_reserve_cluster(scmrq)) { if (!scm_reserve_cluster(scmrq)) {
SCM_LOG(5, "cluster busy"); SCM_LOG(5, "cluster busy");
scm_request_set(scmrq, NULL);
if (scmrq->aob->request.msb_count)
goto out;
scm_request_done(scmrq); scm_request_done(scmrq);
return; return;
} }
if (scm_need_cluster_request(scmrq)) { if (scm_need_cluster_request(scmrq)) {
atomic_inc(&bdev->queued_reqs); if (scmrq->aob->request.msb_count) {
blk_start_request(req); /* Start cluster requests separately. */
scm_initiate_cluster_request(scmrq); scm_request_set(scmrq, NULL);
return; if (scm_request_start(scmrq))
return;
} else {
atomic_inc(&bdev->queued_reqs);
blk_start_request(req);
scm_initiate_cluster_request(scmrq);
}
scmrq = NULL;
continue;
}
if (scm_request_prepare(scmrq)) {
SCM_LOG(5, "aidaw alloc failed");
scm_request_set(scmrq, NULL);
goto out;
} }
scm_request_prepare(scmrq);
atomic_inc(&bdev->queued_reqs);
blk_start_request(req); blk_start_request(req);
ret = eadm_start_aob(scmrq->aob); if (scmrq->aob->request.msb_count < nr_requests_per_io)
if (ret) { continue;
SCM_LOG(5, "no subchannel");
scm_request_requeue(scmrq); if (scm_request_start(scmrq))
return; return;
}
scmrq = NULL;
} }
out:
if (scmrq)
scm_request_start(scmrq);
else
scm_ensure_queue_restart(bdev);
} }
static void __scmrq_log_error(struct scm_request *scmrq) static void __scmrq_log_error(struct scm_request *scmrq)
@ -443,11 +561,19 @@ void scm_blk_set_available(struct scm_blk_dev *bdev)
spin_unlock_irqrestore(&bdev->lock, flags); spin_unlock_irqrestore(&bdev->lock, flags);
} }
static bool __init scm_blk_params_valid(void)
{
if (!nr_requests_per_io || nr_requests_per_io > 64)
return false;
return scm_cluster_size_valid();
}
static int __init scm_blk_init(void) static int __init scm_blk_init(void)
{ {
int ret = -EINVAL; int ret = -EINVAL;
if (!scm_cluster_size_valid()) if (!scm_blk_params_valid())
goto out; goto out;
ret = register_blkdev(0, "scm"); ret = register_blkdev(0, "scm");

View file

@ -30,8 +30,8 @@ struct scm_blk_dev {
struct scm_request { struct scm_request {
struct scm_blk_dev *bdev; struct scm_blk_dev *bdev;
struct request *request; struct aidaw *next_aidaw;
struct aidaw *aidaw; struct request **request;
struct aob *aob; struct aob *aob;
struct list_head list; struct list_head list;
u8 retries; u8 retries;
@ -55,6 +55,8 @@ void scm_blk_irq(struct scm_device *, void *, int);
void scm_request_finish(struct scm_request *); void scm_request_finish(struct scm_request *);
void scm_request_requeue(struct scm_request *); void scm_request_requeue(struct scm_request *);
struct aidaw *scm_aidaw_fetch(struct scm_request *scmrq, unsigned int bytes);
int scm_drv_init(void); int scm_drv_init(void);
void scm_drv_cleanup(void); void scm_drv_cleanup(void);

View file

@ -57,39 +57,52 @@ void scm_request_cluster_init(struct scm_request *scmrq)
scmrq->cluster.state = CLUSTER_NONE; scmrq->cluster.state = CLUSTER_NONE;
} }
static bool clusters_intersect(struct scm_request *A, struct scm_request *B) static bool clusters_intersect(struct request *A, struct request *B)
{ {
unsigned long firstA, lastA, firstB, lastB; unsigned long firstA, lastA, firstB, lastB;
firstA = ((u64) blk_rq_pos(A->request) << 9) / CLUSTER_SIZE; firstA = ((u64) blk_rq_pos(A) << 9) / CLUSTER_SIZE;
lastA = (((u64) blk_rq_pos(A->request) << 9) + lastA = (((u64) blk_rq_pos(A) << 9) +
blk_rq_bytes(A->request) - 1) / CLUSTER_SIZE; blk_rq_bytes(A) - 1) / CLUSTER_SIZE;
firstB = ((u64) blk_rq_pos(B->request) << 9) / CLUSTER_SIZE; firstB = ((u64) blk_rq_pos(B) << 9) / CLUSTER_SIZE;
lastB = (((u64) blk_rq_pos(B->request) << 9) + lastB = (((u64) blk_rq_pos(B) << 9) +
blk_rq_bytes(B->request) - 1) / CLUSTER_SIZE; blk_rq_bytes(B) - 1) / CLUSTER_SIZE;
return (firstB <= lastA && firstA <= lastB); return (firstB <= lastA && firstA <= lastB);
} }
bool scm_reserve_cluster(struct scm_request *scmrq) bool scm_reserve_cluster(struct scm_request *scmrq)
{ {
struct request *req = scmrq->request[scmrq->aob->request.msb_count];
struct scm_blk_dev *bdev = scmrq->bdev; struct scm_blk_dev *bdev = scmrq->bdev;
struct scm_request *iter; struct scm_request *iter;
int pos, add = 1;
if (write_cluster_size == 0) if (write_cluster_size == 0)
return true; return true;
spin_lock(&bdev->lock); spin_lock(&bdev->lock);
list_for_each_entry(iter, &bdev->cluster_list, cluster.list) { list_for_each_entry(iter, &bdev->cluster_list, cluster.list) {
if (clusters_intersect(scmrq, iter) && if (iter == scmrq) {
(rq_data_dir(scmrq->request) == WRITE || /*
rq_data_dir(iter->request) == WRITE)) { * We don't have to use clusters_intersect here, since
spin_unlock(&bdev->lock); * cluster requests are always started separately.
return false; */
add = 0;
continue;
}
for (pos = 0; pos <= iter->aob->request.msb_count; pos++) {
if (clusters_intersect(req, iter->request[pos]) &&
(rq_data_dir(req) == WRITE ||
rq_data_dir(iter->request[pos]) == WRITE)) {
spin_unlock(&bdev->lock);
return false;
}
} }
} }
list_add(&scmrq->cluster.list, &bdev->cluster_list); if (add)
list_add(&scmrq->cluster.list, &bdev->cluster_list);
spin_unlock(&bdev->lock); spin_unlock(&bdev->lock);
return true; return true;
@ -114,14 +127,14 @@ void scm_blk_dev_cluster_setup(struct scm_blk_dev *bdev)
blk_queue_io_opt(bdev->rq, CLUSTER_SIZE); blk_queue_io_opt(bdev->rq, CLUSTER_SIZE);
} }
static void scm_prepare_cluster_request(struct scm_request *scmrq) static int scm_prepare_cluster_request(struct scm_request *scmrq)
{ {
struct scm_blk_dev *bdev = scmrq->bdev; struct scm_blk_dev *bdev = scmrq->bdev;
struct scm_device *scmdev = bdev->gendisk->private_data; struct scm_device *scmdev = bdev->gendisk->private_data;
struct request *req = scmrq->request; struct request *req = scmrq->request[0];
struct aidaw *aidaw = scmrq->aidaw;
struct msb *msb = &scmrq->aob->msb[0]; struct msb *msb = &scmrq->aob->msb[0];
struct req_iterator iter; struct req_iterator iter;
struct aidaw *aidaw;
struct bio_vec bv; struct bio_vec bv;
int i = 0; int i = 0;
u64 addr; u64 addr;
@ -131,11 +144,9 @@ static void scm_prepare_cluster_request(struct scm_request *scmrq)
scmrq->cluster.state = CLUSTER_READ; scmrq->cluster.state = CLUSTER_READ;
/* fall through */ /* fall through */
case CLUSTER_READ: case CLUSTER_READ:
scmrq->aob->request.msb_count = 1;
msb->bs = MSB_BS_4K; msb->bs = MSB_BS_4K;
msb->oc = MSB_OC_READ; msb->oc = MSB_OC_READ;
msb->flags = MSB_FLAG_IDA; msb->flags = MSB_FLAG_IDA;
msb->data_addr = (u64) aidaw;
msb->blk_count = write_cluster_size; msb->blk_count = write_cluster_size;
addr = scmdev->address + ((u64) blk_rq_pos(req) << 9); addr = scmdev->address + ((u64) blk_rq_pos(req) << 9);
@ -146,6 +157,12 @@ static void scm_prepare_cluster_request(struct scm_request *scmrq)
CLUSTER_SIZE)) CLUSTER_SIZE))
msb->blk_count = 2 * write_cluster_size; msb->blk_count = 2 * write_cluster_size;
aidaw = scm_aidaw_fetch(scmrq, msb->blk_count * PAGE_SIZE);
if (!aidaw)
return -ENOMEM;
scmrq->aob->request.msb_count = 1;
msb->data_addr = (u64) aidaw;
for (i = 0; i < msb->blk_count; i++) { for (i = 0; i < msb->blk_count; i++) {
aidaw->data_addr = (u64) scmrq->cluster.buf[i]; aidaw->data_addr = (u64) scmrq->cluster.buf[i];
aidaw++; aidaw++;
@ -153,6 +170,7 @@ static void scm_prepare_cluster_request(struct scm_request *scmrq)
break; break;
case CLUSTER_WRITE: case CLUSTER_WRITE:
aidaw = (void *) msb->data_addr;
msb->oc = MSB_OC_WRITE; msb->oc = MSB_OC_WRITE;
for (addr = msb->scm_addr; for (addr = msb->scm_addr;
@ -173,22 +191,29 @@ static void scm_prepare_cluster_request(struct scm_request *scmrq)
} }
break; break;
} }
return 0;
} }
bool scm_need_cluster_request(struct scm_request *scmrq) bool scm_need_cluster_request(struct scm_request *scmrq)
{ {
if (rq_data_dir(scmrq->request) == READ) int pos = scmrq->aob->request.msb_count;
if (rq_data_dir(scmrq->request[pos]) == READ)
return false; return false;
return blk_rq_bytes(scmrq->request) < CLUSTER_SIZE; return blk_rq_bytes(scmrq->request[pos]) < CLUSTER_SIZE;
} }
/* Called with queue lock held. */ /* Called with queue lock held. */
void scm_initiate_cluster_request(struct scm_request *scmrq) void scm_initiate_cluster_request(struct scm_request *scmrq)
{ {
scm_prepare_cluster_request(scmrq); if (scm_prepare_cluster_request(scmrq))
goto requeue;
if (eadm_start_aob(scmrq->aob)) if (eadm_start_aob(scmrq->aob))
scm_request_requeue(scmrq); goto requeue;
return;
requeue:
scm_request_requeue(scmrq);
} }
bool scm_test_cluster_request(struct scm_request *scmrq) bool scm_test_cluster_request(struct scm_request *scmrq)

View file

@ -102,6 +102,16 @@ config SCLP_ASYNC
want for inform other people about your kernel panics, want for inform other people about your kernel panics,
need this feature and intend to run your kernel in LPAR. need this feature and intend to run your kernel in LPAR.
config SCLP_ASYNC_ID
string "Component ID for Call Home"
depends on SCLP_ASYNC
default "000000000"
help
The Component ID for Call Home is used to identify the correct
problem reporting queue the call home records should be sent to.
If your are unsure, please use the default value "000000000".
config HMC_DRV config HMC_DRV
def_tristate m def_tristate m
prompt "Support for file transfers from HMC drive CD/DVD-ROM" prompt "Support for file transfers from HMC drive CD/DVD-ROM"

View file

@ -137,7 +137,8 @@ static int sclp_async_send_wait(char *message)
* Retain Queue * Retain Queue
* e.g. 5639CC140 500 Red Hat RHEL5 Linux for zSeries (RHEL AS) * e.g. 5639CC140 500 Red Hat RHEL5 Linux for zSeries (RHEL AS)
*/ */
strncpy(sccb->evbuf.comp_id, "000000000", sizeof(sccb->evbuf.comp_id)); strncpy(sccb->evbuf.comp_id, CONFIG_SCLP_ASYNC_ID,
sizeof(sccb->evbuf.comp_id));
sccb->evbuf.header.length = sizeof(sccb->evbuf); sccb->evbuf.header.length = sizeof(sccb->evbuf);
sccb->header.length = sizeof(sccb->evbuf) + sizeof(sccb->header); sccb->header.length = sizeof(sccb->evbuf) + sizeof(sccb->header);
sccb->header.function_code = SCLP_NORMAL_WRITE; sccb->header.function_code = SCLP_NORMAL_WRITE;

View file

@ -31,7 +31,7 @@
MODULE_DESCRIPTION("driver for s390 eadm subchannels"); MODULE_DESCRIPTION("driver for s390 eadm subchannels");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
#define EADM_TIMEOUT (5 * HZ) #define EADM_TIMEOUT (7 * HZ)
static DEFINE_SPINLOCK(list_lock); static DEFINE_SPINLOCK(list_lock);
static LIST_HEAD(eadm_list); static LIST_HEAD(eadm_list);

View file

@ -103,6 +103,17 @@ static inline pmd_t pmdp_get_and_clear(struct mm_struct *mm,
#endif /* CONFIG_TRANSPARENT_HUGEPAGE */ #endif /* CONFIG_TRANSPARENT_HUGEPAGE */
#endif #endif
#ifndef __HAVE_ARCH_PMDP_GET_AND_CLEAR_FULL
#ifdef CONFIG_TRANSPARENT_HUGEPAGE
static inline pmd_t pmdp_get_and_clear_full(struct mm_struct *mm,
unsigned long address, pmd_t *pmdp,
int full)
{
return pmdp_get_and_clear(mm, address, pmdp);
}
#endif /* CONFIG_TRANSPARENT_HUGEPAGE */
#endif
#ifndef __HAVE_ARCH_PTEP_GET_AND_CLEAR_FULL #ifndef __HAVE_ARCH_PTEP_GET_AND_CLEAR_FULL
static inline pte_t ptep_get_and_clear_full(struct mm_struct *mm, static inline pte_t ptep_get_and_clear_full(struct mm_struct *mm,
unsigned long address, pte_t *ptep, unsigned long address, pte_t *ptep,

View file

@ -335,6 +335,7 @@ extern void kprobe_ftrace_handler(unsigned long ip, unsigned long parent_ip,
extern int arch_prepare_kprobe_ftrace(struct kprobe *p); extern int arch_prepare_kprobe_ftrace(struct kprobe *p);
#endif #endif
int arch_check_ftrace_location(struct kprobe *p);
/* Get the kprobe at this addr (if any) - called with preemption disabled */ /* Get the kprobe at this addr (if any) - called with preemption disabled */
struct kprobe *get_kprobe(void *addr); struct kprobe *get_kprobe(void *addr);

View file

@ -56,6 +56,17 @@ extern int sysctl_legacy_va_layout;
#define __pa_symbol(x) __pa(RELOC_HIDE((unsigned long)(x), 0)) #define __pa_symbol(x) __pa(RELOC_HIDE((unsigned long)(x), 0))
#endif #endif
/*
* To prevent common memory management code establishing
* a zero page mapping on a read fault.
* This macro should be defined within <asm/pgtable.h>.
* s390 does this to prevent multiplexing of hardware bits
* related to the physical page in case of virtualization.
*/
#ifndef mm_forbids_zeropage
#define mm_forbids_zeropage(X) (0)
#endif
extern unsigned long sysctl_user_reserve_kbytes; extern unsigned long sysctl_user_reserve_kbytes;
extern unsigned long sysctl_admin_reserve_kbytes; extern unsigned long sysctl_admin_reserve_kbytes;

View file

@ -1410,16 +1410,10 @@ static inline int check_kprobe_rereg(struct kprobe *p)
return ret; return ret;
} }
static int check_kprobe_address_safe(struct kprobe *p, int __weak arch_check_ftrace_location(struct kprobe *p)
struct module **probed_mod)
{ {
int ret = 0;
unsigned long ftrace_addr; unsigned long ftrace_addr;
/*
* If the address is located on a ftrace nop, set the
* breakpoint to the following instruction.
*/
ftrace_addr = ftrace_location((unsigned long)p->addr); ftrace_addr = ftrace_location((unsigned long)p->addr);
if (ftrace_addr) { if (ftrace_addr) {
#ifdef CONFIG_KPROBES_ON_FTRACE #ifdef CONFIG_KPROBES_ON_FTRACE
@ -1431,7 +1425,17 @@ static int check_kprobe_address_safe(struct kprobe *p,
return -EINVAL; return -EINVAL;
#endif #endif
} }
return 0;
}
static int check_kprobe_address_safe(struct kprobe *p,
struct module **probed_mod)
{
int ret;
ret = arch_check_ftrace_location(p);
if (ret)
return ret;
jump_label_lock(); jump_label_lock();
preempt_disable(); preempt_disable();

View file

@ -169,6 +169,8 @@ cond_syscall(ppc_rtas);
cond_syscall(sys_spu_run); cond_syscall(sys_spu_run);
cond_syscall(sys_spu_create); cond_syscall(sys_spu_create);
cond_syscall(sys_subpage_prot); cond_syscall(sys_subpage_prot);
cond_syscall(sys_s390_pci_mmio_read);
cond_syscall(sys_s390_pci_mmio_write);
/* mmu depending weak syscall entries */ /* mmu depending weak syscall entries */
cond_syscall(sys_mprotect); cond_syscall(sys_mprotect);

View file

@ -804,7 +804,7 @@ int do_huge_pmd_anonymous_page(struct mm_struct *mm, struct vm_area_struct *vma,
return VM_FAULT_OOM; return VM_FAULT_OOM;
if (unlikely(khugepaged_enter(vma, vma->vm_flags))) if (unlikely(khugepaged_enter(vma, vma->vm_flags)))
return VM_FAULT_OOM; return VM_FAULT_OOM;
if (!(flags & FAULT_FLAG_WRITE) && if (!(flags & FAULT_FLAG_WRITE) && !mm_forbids_zeropage(mm) &&
transparent_hugepage_use_zero_page()) { transparent_hugepage_use_zero_page()) {
spinlock_t *ptl; spinlock_t *ptl;
pgtable_t pgtable; pgtable_t pgtable;
@ -1399,7 +1399,8 @@ int zap_huge_pmd(struct mmu_gather *tlb, struct vm_area_struct *vma,
* pgtable_trans_huge_withdraw after finishing pmdp related * pgtable_trans_huge_withdraw after finishing pmdp related
* operations. * operations.
*/ */
orig_pmd = pmdp_get_and_clear(tlb->mm, addr, pmd); orig_pmd = pmdp_get_and_clear_full(tlb->mm, addr, pmd,
tlb->fullmm);
tlb_remove_pmd_tlb_entry(tlb, pmd, addr); tlb_remove_pmd_tlb_entry(tlb, pmd, addr);
pgtable = pgtable_trans_huge_withdraw(tlb->mm, pmd); pgtable = pgtable_trans_huge_withdraw(tlb->mm, pmd);
if (is_huge_zero_pmd(orig_pmd)) { if (is_huge_zero_pmd(orig_pmd)) {

View file

@ -2627,7 +2627,7 @@ static int do_anonymous_page(struct mm_struct *mm, struct vm_area_struct *vma,
return VM_FAULT_SIGBUS; return VM_FAULT_SIGBUS;
/* Use the zero-page for reads */ /* Use the zero-page for reads */
if (!(flags & FAULT_FLAG_WRITE)) { if (!(flags & FAULT_FLAG_WRITE) && !mm_forbids_zeropage(mm)) {
entry = pte_mkspecial(pfn_pte(my_zero_pfn(address), entry = pte_mkspecial(pfn_pte(my_zero_pfn(address),
vma->vm_page_prot)); vma->vm_page_prot));
page_table = pte_offset_map_lock(mm, pmd, address, &ptl); page_table = pte_offset_map_lock(mm, pmd, address, &ptl);

View file

@ -404,7 +404,7 @@ do_file(char const *const fname)
} }
if (w2(ghdr->e_machine) == EM_S390) { if (w2(ghdr->e_machine) == EM_S390) {
reltype = R_390_64; reltype = R_390_64;
mcount_adjust_64 = -8; mcount_adjust_64 = -14;
} }
if (w2(ghdr->e_machine) == EM_MIPS) { if (w2(ghdr->e_machine) == EM_MIPS) {
reltype = R_MIPS_64; reltype = R_MIPS_64;

View file

@ -243,7 +243,7 @@ if ($arch eq "x86_64") {
} elsif ($arch eq "s390" && $bits == 64) { } elsif ($arch eq "s390" && $bits == 64) {
$mcount_regex = "^\\s*([0-9a-fA-F]+):\\s*R_390_(PC|PLT)32DBL\\s+_mcount\\+0x2\$"; $mcount_regex = "^\\s*([0-9a-fA-F]+):\\s*R_390_(PC|PLT)32DBL\\s+_mcount\\+0x2\$";
$mcount_adjust = -8; $mcount_adjust = -14;
$alignment = 8; $alignment = 8;
$type = ".quad"; $type = ".quad";
$ld .= " -m elf64_s390"; $ld .= " -m elf64_s390";