1 /* Copyright (C) 2009 Pierre-Marc Fournier
3 * This library is free software; you can redistribute it and/or
4 * modify it under the terms of the GNU Lesser General Public
5 * License as published by the Free Software Foundation; either
6 * version 2.1 of the License, or (at your option) any later version.
8 * This library is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
11 * Lesser General Public License for more details.
13 * You should have received a copy of the GNU Lesser General Public
14 * License along with this library; if not, write to the Free Software
15 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
26 #include <ust/tracectl.h>
33 static pid_t (*plibc_func
)(void) = NULL
;
34 ust_fork_info_t fork_info
;
38 if(plibc_func
== NULL
) {
39 plibc_func
= dlsym(RTLD_NEXT
, "fork");
40 if(plibc_func
== NULL
) {
41 fprintf(stderr
, "libcwrap: unable to find fork\n");
46 ust_before_fork(&fork_info
);
48 /* Do the real fork */
49 retval
= plibc_func();
53 ust_after_fork_child(&fork_info
);
56 ust_after_fork_parent(&fork_info
);
62 int execve(const char *filename
, char *const argv
[], char *const envp
[])
64 static int (*plibc_func
)(const char *filename
, char *const argv
[], char *const envp
[]) = NULL
;
68 if(plibc_func
== NULL
) {
69 plibc_func
= dlsym(RTLD_NEXT
, "execve");
70 if(plibc_func
== NULL
) {
71 fprintf(stderr
, "libinterfork: unable to find execve\n");
78 retval
= plibc_func(filename
, argv
, envp
);
83 struct interfork_clone_info
{
86 ust_fork_info_t fork_info
;
89 static int clone_fn(void *arg
)
91 struct interfork_clone_info
*info
= (struct interfork_clone_info
*)arg
;
93 /* clone is now done and we are in child */
94 ust_after_fork_child(&info
->fork_info
);
96 return info
->fn(info
->arg
);
99 int clone(int (*fn
)(void *), void *child_stack
, int flags
, void *arg
, ...)
101 static int (*plibc_func
)(int (*fn
)(void *), void *child_stack
, int flags
, void *arg
, pid_t
*ptid
, struct user_desc
*tls
, pid_t
*ctid
) = NULL
;
105 struct user_desc
*tls
;
113 ptid
= va_arg(ap
, pid_t
*);
114 tls
= va_arg(ap
, struct user_desc
*);
115 ctid
= va_arg(ap
, pid_t
*);
118 if(plibc_func
== NULL
) {
119 plibc_func
= dlsym(RTLD_NEXT
, "clone");
120 if(plibc_func
== NULL
) {
121 fprintf(stderr
, "libinterfork: unable to find clone\n");
126 if(flags
& CLONE_VM
) {
127 /* creating a thread, no need to intervene, just pass on the arguments */
128 retval
= plibc_func(fn
, child_stack
, flags
, arg
, ptid
, tls
, ctid
);
131 /* creating a real process, we need to intervene */
132 struct interfork_clone_info info
= { fn
: fn
, arg
: arg
};
134 ust_before_fork(&info
.fork_info
);
136 retval
= plibc_func(clone_fn
, child_stack
, flags
, &info
, ptid
, tls
, ctid
);
138 /* The child doesn't get here */
139 ust_after_fork_parent(&info
.fork_info
);