1*cf84ac9aSAndroid Build Coastguard Worker /*
2*cf84ac9aSAndroid Build Coastguard Worker * Copyright (c) 2002 Andi Kleen <[email protected]>
3*cf84ac9aSAndroid Build Coastguard Worker * Copyright (c) 2002 Michal Ludvig <[email protected]>
4*cf84ac9aSAndroid Build Coastguard Worker * Copyright (c) 2002 Roland McGrath <[email protected]>
5*cf84ac9aSAndroid Build Coastguard Worker * Copyright (c) 2008-2013 Denys Vlasenko <[email protected]>
6*cf84ac9aSAndroid Build Coastguard Worker * Copyright (c) 2012 H.J. Lu <[email protected]>
7*cf84ac9aSAndroid Build Coastguard Worker * Copyright (c) 2010-2015 Dmitry V. Levin <[email protected]>
8*cf84ac9aSAndroid Build Coastguard Worker * Copyright (c) 2015-2018 The strace developers.
9*cf84ac9aSAndroid Build Coastguard Worker * All rights reserved.
10*cf84ac9aSAndroid Build Coastguard Worker *
11*cf84ac9aSAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without
12*cf84ac9aSAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions
13*cf84ac9aSAndroid Build Coastguard Worker * are met:
14*cf84ac9aSAndroid Build Coastguard Worker * 1. Redistributions of source code must retain the above copyright
15*cf84ac9aSAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer.
16*cf84ac9aSAndroid Build Coastguard Worker * 2. Redistributions in binary form must reproduce the above copyright
17*cf84ac9aSAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer in the
18*cf84ac9aSAndroid Build Coastguard Worker * documentation and/or other materials provided with the distribution.
19*cf84ac9aSAndroid Build Coastguard Worker * 3. The name of the author may not be used to endorse or promote products
20*cf84ac9aSAndroid Build Coastguard Worker * derived from this software without specific prior written permission.
21*cf84ac9aSAndroid Build Coastguard Worker *
22*cf84ac9aSAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
23*cf84ac9aSAndroid Build Coastguard Worker * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
24*cf84ac9aSAndroid Build Coastguard Worker * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
25*cf84ac9aSAndroid Build Coastguard Worker * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
26*cf84ac9aSAndroid Build Coastguard Worker * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
27*cf84ac9aSAndroid Build Coastguard Worker * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
28*cf84ac9aSAndroid Build Coastguard Worker * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
29*cf84ac9aSAndroid Build Coastguard Worker * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
30*cf84ac9aSAndroid Build Coastguard Worker * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
31*cf84ac9aSAndroid Build Coastguard Worker * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32*cf84ac9aSAndroid Build Coastguard Worker */
33*cf84ac9aSAndroid Build Coastguard Worker
34*cf84ac9aSAndroid Build Coastguard Worker #ifdef X86_64
35*cf84ac9aSAndroid Build Coastguard Worker # define X32_PERSONALITY_NUMBER 2
36*cf84ac9aSAndroid Build Coastguard Worker #else
37*cf84ac9aSAndroid Build Coastguard Worker # define X32_PERSONALITY_NUMBER 0
38*cf84ac9aSAndroid Build Coastguard Worker #endif
39*cf84ac9aSAndroid Build Coastguard Worker
40*cf84ac9aSAndroid Build Coastguard Worker /* Return codes: 1 - ok, 0 - ignore, other - error. */
41*cf84ac9aSAndroid Build Coastguard Worker static int
arch_get_scno(struct tcb * tcp)42*cf84ac9aSAndroid Build Coastguard Worker arch_get_scno(struct tcb *tcp)
43*cf84ac9aSAndroid Build Coastguard Worker {
44*cf84ac9aSAndroid Build Coastguard Worker kernel_ulong_t scno = 0;
45*cf84ac9aSAndroid Build Coastguard Worker unsigned int currpers;
46*cf84ac9aSAndroid Build Coastguard Worker
47*cf84ac9aSAndroid Build Coastguard Worker #ifndef __X32_SYSCALL_BIT
48*cf84ac9aSAndroid Build Coastguard Worker # define __X32_SYSCALL_BIT 0x40000000
49*cf84ac9aSAndroid Build Coastguard Worker #endif
50*cf84ac9aSAndroid Build Coastguard Worker
51*cf84ac9aSAndroid Build Coastguard Worker #if 1
52*cf84ac9aSAndroid Build Coastguard Worker /*
53*cf84ac9aSAndroid Build Coastguard Worker * GETREGSET of NT_PRSTATUS tells us regset size,
54*cf84ac9aSAndroid Build Coastguard Worker * which unambiguously detects i386.
55*cf84ac9aSAndroid Build Coastguard Worker *
56*cf84ac9aSAndroid Build Coastguard Worker * Linux kernel distinguishes x86-64 and x32 processes
57*cf84ac9aSAndroid Build Coastguard Worker * solely by looking at __X32_SYSCALL_BIT:
58*cf84ac9aSAndroid Build Coastguard Worker * arch/x86/include/asm/compat.h::is_x32_task():
59*cf84ac9aSAndroid Build Coastguard Worker * if (task_pt_regs(current)->orig_ax & __X32_SYSCALL_BIT)
60*cf84ac9aSAndroid Build Coastguard Worker * return true;
61*cf84ac9aSAndroid Build Coastguard Worker */
62*cf84ac9aSAndroid Build Coastguard Worker if (x86_io.iov_len == sizeof(i386_regs)) {
63*cf84ac9aSAndroid Build Coastguard Worker scno = i386_regs.orig_eax;
64*cf84ac9aSAndroid Build Coastguard Worker currpers = 1;
65*cf84ac9aSAndroid Build Coastguard Worker } else {
66*cf84ac9aSAndroid Build Coastguard Worker scno = x86_64_regs.orig_rax;
67*cf84ac9aSAndroid Build Coastguard Worker currpers = 0;
68*cf84ac9aSAndroid Build Coastguard Worker if (scno & __X32_SYSCALL_BIT) {
69*cf84ac9aSAndroid Build Coastguard Worker /*
70*cf84ac9aSAndroid Build Coastguard Worker * Syscall number -1 requires special treatment:
71*cf84ac9aSAndroid Build Coastguard Worker * it might be a side effect of SECCOMP_RET_ERRNO
72*cf84ac9aSAndroid Build Coastguard Worker * filtering that sets orig_rax to -1
73*cf84ac9aSAndroid Build Coastguard Worker * in some versions of linux kernel.
74*cf84ac9aSAndroid Build Coastguard Worker * If that is the case, then
75*cf84ac9aSAndroid Build Coastguard Worker * __X32_SYSCALL_BIT logic does not apply.
76*cf84ac9aSAndroid Build Coastguard Worker */
77*cf84ac9aSAndroid Build Coastguard Worker if ((long long) x86_64_regs.orig_rax != -1) {
78*cf84ac9aSAndroid Build Coastguard Worker currpers = 2;
79*cf84ac9aSAndroid Build Coastguard Worker } else {
80*cf84ac9aSAndroid Build Coastguard Worker # ifdef X32
81*cf84ac9aSAndroid Build Coastguard Worker currpers = 2;
82*cf84ac9aSAndroid Build Coastguard Worker # endif
83*cf84ac9aSAndroid Build Coastguard Worker }
84*cf84ac9aSAndroid Build Coastguard Worker }
85*cf84ac9aSAndroid Build Coastguard Worker }
86*cf84ac9aSAndroid Build Coastguard Worker
87*cf84ac9aSAndroid Build Coastguard Worker #elif 0
88*cf84ac9aSAndroid Build Coastguard Worker /*
89*cf84ac9aSAndroid Build Coastguard Worker * cs = 0x33 for long mode (native 64 bit and x32)
90*cf84ac9aSAndroid Build Coastguard Worker * cs = 0x23 for compatibility mode (32 bit)
91*cf84ac9aSAndroid Build Coastguard Worker * ds = 0x2b for x32 mode (x86-64 in 32 bit)
92*cf84ac9aSAndroid Build Coastguard Worker */
93*cf84ac9aSAndroid Build Coastguard Worker scno = x86_64_regs.orig_rax;
94*cf84ac9aSAndroid Build Coastguard Worker switch (x86_64_regs.cs) {
95*cf84ac9aSAndroid Build Coastguard Worker case 0x23:
96*cf84ac9aSAndroid Build Coastguard Worker currpers = 1;
97*cf84ac9aSAndroid Build Coastguard Worker break;
98*cf84ac9aSAndroid Build Coastguard Worker case 0x33:
99*cf84ac9aSAndroid Build Coastguard Worker if (x86_64_regs.ds == 0x2b) {
100*cf84ac9aSAndroid Build Coastguard Worker currpers = 2;
101*cf84ac9aSAndroid Build Coastguard Worker } else
102*cf84ac9aSAndroid Build Coastguard Worker currpers = 0;
103*cf84ac9aSAndroid Build Coastguard Worker break;
104*cf84ac9aSAndroid Build Coastguard Worker default:
105*cf84ac9aSAndroid Build Coastguard Worker error_msg("Unknown value CS=0x%08X while "
106*cf84ac9aSAndroid Build Coastguard Worker "detecting personality of process PID=%d",
107*cf84ac9aSAndroid Build Coastguard Worker (int)x86_64_regs.cs, tcp->pid);
108*cf84ac9aSAndroid Build Coastguard Worker currpers = current_personality;
109*cf84ac9aSAndroid Build Coastguard Worker break;
110*cf84ac9aSAndroid Build Coastguard Worker }
111*cf84ac9aSAndroid Build Coastguard Worker #elif 0
112*cf84ac9aSAndroid Build Coastguard Worker /*
113*cf84ac9aSAndroid Build Coastguard Worker * This version analyzes the opcode of a syscall instruction.
114*cf84ac9aSAndroid Build Coastguard Worker * (int 0x80 on i386 vs. syscall on x86-64)
115*cf84ac9aSAndroid Build Coastguard Worker * It works, but is too complicated, and strictly speaking, unreliable.
116*cf84ac9aSAndroid Build Coastguard Worker */
117*cf84ac9aSAndroid Build Coastguard Worker unsigned long call, rip = x86_64_regs.rip;
118*cf84ac9aSAndroid Build Coastguard Worker /* sizeof(syscall) == sizeof(int 0x80) == 2 */
119*cf84ac9aSAndroid Build Coastguard Worker rip -= 2;
120*cf84ac9aSAndroid Build Coastguard Worker errno = 0;
121*cf84ac9aSAndroid Build Coastguard Worker call = ptrace(PTRACE_PEEKTEXT, tcp->pid, (char *)rip, (char *)0);
122*cf84ac9aSAndroid Build Coastguard Worker if (errno)
123*cf84ac9aSAndroid Build Coastguard Worker perror_msg("ptrace_peektext failed");
124*cf84ac9aSAndroid Build Coastguard Worker switch (call & 0xffff) {
125*cf84ac9aSAndroid Build Coastguard Worker /* x86-64: syscall = 0x0f 0x05 */
126*cf84ac9aSAndroid Build Coastguard Worker case 0x050f:
127*cf84ac9aSAndroid Build Coastguard Worker currpers = 0;
128*cf84ac9aSAndroid Build Coastguard Worker break;
129*cf84ac9aSAndroid Build Coastguard Worker /* i386: int 0x80 = 0xcd 0x80 */
130*cf84ac9aSAndroid Build Coastguard Worker case 0x80cd:
131*cf84ac9aSAndroid Build Coastguard Worker currpers = 1;
132*cf84ac9aSAndroid Build Coastguard Worker break;
133*cf84ac9aSAndroid Build Coastguard Worker default:
134*cf84ac9aSAndroid Build Coastguard Worker currpers = current_personality;
135*cf84ac9aSAndroid Build Coastguard Worker error_msg("Unknown syscall opcode (0x%04X) while "
136*cf84ac9aSAndroid Build Coastguard Worker "detecting personality of process PID=%d",
137*cf84ac9aSAndroid Build Coastguard Worker (int)call, tcp->pid);
138*cf84ac9aSAndroid Build Coastguard Worker break;
139*cf84ac9aSAndroid Build Coastguard Worker }
140*cf84ac9aSAndroid Build Coastguard Worker #endif
141*cf84ac9aSAndroid Build Coastguard Worker
142*cf84ac9aSAndroid Build Coastguard Worker #ifdef X32
143*cf84ac9aSAndroid Build Coastguard Worker /*
144*cf84ac9aSAndroid Build Coastguard Worker * If we are built for a x32 system, then personality 0 is x32
145*cf84ac9aSAndroid Build Coastguard Worker * (not x86_64), and stracing of x86_64 apps is not supported.
146*cf84ac9aSAndroid Build Coastguard Worker * Stracing of i386 apps is still supported.
147*cf84ac9aSAndroid Build Coastguard Worker */
148*cf84ac9aSAndroid Build Coastguard Worker if (currpers == 0) {
149*cf84ac9aSAndroid Build Coastguard Worker error_msg("syscall_%" PRI_klu "(...) in unsupported "
150*cf84ac9aSAndroid Build Coastguard Worker "64-bit mode of process PID=%d", scno, tcp->pid);
151*cf84ac9aSAndroid Build Coastguard Worker return 0;
152*cf84ac9aSAndroid Build Coastguard Worker }
153*cf84ac9aSAndroid Build Coastguard Worker currpers &= ~2; /* map 2,1 to 0,1 */
154*cf84ac9aSAndroid Build Coastguard Worker #endif /* X32 */
155*cf84ac9aSAndroid Build Coastguard Worker
156*cf84ac9aSAndroid Build Coastguard Worker update_personality(tcp, currpers);
157*cf84ac9aSAndroid Build Coastguard Worker tcp->scno = scno;
158*cf84ac9aSAndroid Build Coastguard Worker return 1;
159*cf84ac9aSAndroid Build Coastguard Worker }
160