2015-12-17 20:56:48 +03:00
/*
* Copyright ( c ) 1999 - 2000 Wichert Akkerman < wichert @ cistron . nl >
* Copyright ( c ) 2002 - 2005 Roland McGrath < roland @ redhat . com >
* Copyright ( c ) 2008 Jan Kratochvil < jan . kratochvil @ redhat . com >
* Copyright ( c ) 2009 - 2013 Denys Vlasenko < dvlasenk @ redhat . com >
* Copyright ( c ) 2006 - 2015 Dmitry V . Levin < ldv @ altlinux . org >
2018-02-14 01:00:00 +03:00
* Copyright ( c ) 2014 - 2018 The strace developers .
2015-12-17 20:56:48 +03:00
* All rights reserved .
*
2018-12-10 03:00:00 +03:00
* SPDX - License - Identifier : LGPL - 2.1 - or - later
2015-12-17 20:56:48 +03:00
*/
2014-12-11 22:25:02 +03:00
# include "defs.h"
2015-06-17 21:36:53 +03:00
# include <sched.h>
2017-01-01 01:12:08 +03:00
# include <asm/unistd.h>
2015-06-17 21:36:53 +03:00
# ifndef CSIGNAL
# define CSIGNAL 0x000000ff
# endif
2014-12-11 22:25:02 +03:00
# include "xlat/clone_flags.h"
2016-10-18 21:17:33 +03:00
# include "xlat/setns_types.h"
2016-10-19 21:09:58 +03:00
# include "xlat/unshare_flags.h"
2014-12-11 22:25:02 +03:00
# if defined IA64
# define ARG_FLAGS 0
# define ARG_STACK 1
2016-08-09 17:35:06 +03:00
# define ARG_STACKSIZE (tcp->scno == __NR_clone2 ? 2 : -1)
# define ARG_PTID (tcp->scno == __NR_clone2 ? 3 : 2)
# define ARG_CTID (tcp->scno == __NR_clone2 ? 4 : 3)
# define ARG_TLS (tcp->scno == __NR_clone2 ? 5 : 4)
2018-01-25 04:55:08 +03:00
# elif defined S390 || defined S390X
2014-12-11 22:25:02 +03:00
# define ARG_STACK 0
# define ARG_FLAGS 1
# define ARG_PTID 2
# define ARG_CTID 3
# define ARG_TLS 4
# elif defined X86_64 || defined X32
/* x86 personality processes have the last two arguments flipped. */
# define ARG_FLAGS 0
# define ARG_STACK 1
# define ARG_PTID 2
# define ARG_CTID ((current_personality != 1) ? 3 : 4)
# define ARG_TLS ((current_personality != 1) ? 4 : 3)
2016-08-19 16:16:41 +03:00
# elif defined ALPHA || defined TILE || defined OR1K || defined RISCV
2014-12-11 22:25:02 +03:00
# define ARG_FLAGS 0
# define ARG_STACK 1
# define ARG_PTID 2
# define ARG_CTID 3
# define ARG_TLS 4
# else
# define ARG_FLAGS 0
# define ARG_STACK 1
# define ARG_PTID 2
# define ARG_TLS 3
# define ARG_CTID 4
# endif
2016-12-21 21:01:21 +03:00
static void
2016-12-26 13:26:03 +03:00
print_tls_arg ( struct tcb * const tcp , const kernel_ulong_t addr )
2016-12-21 21:01:21 +03:00
{
# ifdef HAVE_STRUCT_USER_DESC
# if SUPPORTED_PERSONALITIES > 1
if ( current_personality = = 1 )
# endif
{
2018-01-08 20:41:30 +03:00
print_user_desc ( tcp , tcp - > u_arg [ ARG_TLS ] , USER_DESC_BOTH ) ;
2016-12-21 21:01:21 +03:00
}
# if SUPPORTED_PERSONALITIES > 1
else
# endif
# endif /* HAVE_STRUCT_USER_DESC */
{
printaddr ( tcp - > u_arg [ ARG_TLS ] ) ;
}
}
2014-12-11 22:25:02 +03:00
2015-04-07 04:36:50 +03:00
SYS_FUNC ( clone )
2014-12-11 22:25:02 +03:00
{
if ( exiting ( tcp ) ) {
const char * sep = " | " ;
2016-12-26 13:26:03 +03:00
kernel_ulong_t flags = tcp - > u_arg [ ARG_FLAGS ] ;
2016-06-11 04:28:21 +03:00
tprints ( " child_stack= " ) ;
printaddr ( tcp - > u_arg [ ARG_STACK ] ) ;
tprints ( " , " ) ;
2014-12-11 22:25:02 +03:00
# ifdef ARG_STACKSIZE
if ( ARG_STACKSIZE ! = - 1 )
2016-12-26 13:16:35 +03:00
tprintf ( " stack_size=%# " PRI_klx " , " ,
2014-12-11 22:25:02 +03:00
tcp - > u_arg [ ARG_STACKSIZE ] ) ;
# endif
tprints ( " flags= " ) ;
2017-06-18 01:23:09 +03:00
if ( ! printflags64 ( clone_flags , flags & ~ CSIGNAL , NULL ) )
2014-12-11 22:25:02 +03:00
sep = " " ;
2018-10-07 06:12:42 +03:00
if ( ( flags & CSIGNAL ) ! = 0 ) {
tprints ( sep ) ;
printsignal ( flags & CSIGNAL ) ;
}
2014-12-11 22:25:02 +03:00
if ( ( flags & ( CLONE_PARENT_SETTID | CLONE_CHILD_SETTID
| CLONE_CHILD_CLEARTID | CLONE_SETTLS ) ) = = 0 )
return 0 ;
2016-06-11 04:28:21 +03:00
if ( flags & CLONE_PARENT_SETTID ) {
tprints ( " , parent_tidptr= " ) ;
printaddr ( tcp - > u_arg [ ARG_PTID ] ) ;
}
2014-12-11 22:25:02 +03:00
if ( flags & CLONE_SETTLS ) {
2016-12-21 21:01:21 +03:00
tprints ( " , tls= " ) ;
print_tls_arg ( tcp , tcp - > u_arg [ ARG_TLS ] ) ;
2016-06-11 04:28:21 +03:00
}
if ( flags & ( CLONE_CHILD_SETTID | CLONE_CHILD_CLEARTID ) ) {
tprints ( " , child_tidptr= " ) ;
printaddr ( tcp - > u_arg [ ARG_CTID ] ) ;
2014-12-11 22:25:02 +03:00
}
}
/* TODO on syscall entry:
* We can clear CLONE_PTRACE here since it is an ancient hack
* to allow us to catch children , and we use another hack for that .
* But CLONE_PTRACE can conceivably be used by malicious programs
* to subvert us . By clearing this bit , we can defend against it :
* in untraced execution , CLONE_PTRACE should have no effect .
*
* We can also clear CLONE_UNTRACED , since it allows to start
* children outside of our control . At the moment
* I ' m trying to figure out whether there is a * legitimate *
* use of this flag which we should respect .
*/
return 0 ;
}
2015-04-07 04:36:50 +03:00
SYS_FUNC ( setns )
2014-12-11 22:25:02 +03:00
{
2015-07-20 20:16:56 +03:00
printfd ( tcp , tcp - > u_arg [ 0 ] ) ;
tprints ( " , " ) ;
2016-10-18 21:17:33 +03:00
printxval ( setns_types , tcp - > u_arg [ 1 ] , " CLONE_NEW??? " ) ;
2015-07-20 20:16:56 +03:00
return RVAL_DECODED ;
2014-12-11 22:25:02 +03:00
}
2015-04-07 04:36:50 +03:00
SYS_FUNC ( unshare )
2014-12-11 22:25:02 +03:00
{
2016-12-26 05:28:04 +03:00
printflags64 ( unshare_flags , tcp - > u_arg [ 0 ] , " CLONE_??? " ) ;
2015-07-20 20:16:56 +03:00
return RVAL_DECODED ;
2014-12-11 22:25:02 +03:00
}
2015-04-07 04:36:50 +03:00
SYS_FUNC ( fork )
2014-12-11 22:25:02 +03:00
{
2018-03-29 19:00:18 +03:00
return RVAL_DECODED ;
2014-12-11 22:25:02 +03:00
}