blob: 3651b66d391be99154419f46862c50dee53a6e08 [file] [log] [blame]
Juan Cespedesd44c6b81998-09-25 14:48:42 +02001#include "config.h"
Juan Cespedesd44c6b81998-09-25 14:48:42 +02002
Joe Damatoab3b72c2010-10-31 00:21:53 -07003#if defined(HAVE_LIBUNWIND)
4#include <libunwind-ptrace.h>
5#endif /* defined(HAVE_LIBUNWIND) */
6
Juan Cespedes5e01f651998-03-08 22:31:44 +01007#include <stdio.h>
Juan Cespedes504a3852003-02-04 23:24:38 +01008#include <stdlib.h>
Juan Cespedese1887051998-03-10 00:08:41 +01009#include <sys/types.h>
10#include <sys/stat.h>
Juan Cespedes5e01f651998-03-08 22:31:44 +010011#include <unistd.h>
12#include <errno.h>
13#include <string.h>
Juan Cespedese1887051998-03-10 00:08:41 +010014#include <pwd.h>
15#include <grp.h>
Juan Cespedes5e01f651998-03-08 22:31:44 +010016
Juan Cespedesf7281232009-06-25 16:11:21 +020017#include "common.h"
Juan Cespedes5e01f651998-03-08 22:31:44 +010018
Juan Cespedesf1350522008-12-16 18:19:58 +010019static void
Juan Cespedesa8909f72009-04-28 20:02:41 +020020change_uid(Process *proc) {
Juan Cespedese1887051998-03-10 00:08:41 +010021 uid_t run_uid, run_euid;
22 gid_t run_gid, run_egid;
23
Juan Cespedesce377d52008-12-16 19:38:10 +010024 if (options.user) {
Juan Cespedese1887051998-03-10 00:08:41 +010025 struct passwd *pent;
26
27 if (getuid() != 0 || geteuid() != 0) {
Ian Wienand2d45b1a2006-02-20 22:48:07 +010028 fprintf(stderr,
29 "you must be root to use the -u option\n");
Juan Cespedese1887051998-03-10 00:08:41 +010030 exit(1);
31 }
Juan Cespedesce377d52008-12-16 19:38:10 +010032 if ((pent = getpwnam(options.user)) == NULL) {
33 fprintf(stderr, "cannot find user `%s'\n", options.user);
Juan Cespedese1887051998-03-10 00:08:41 +010034 exit(1);
35 }
36 run_uid = pent->pw_uid;
37 run_gid = pent->pw_gid;
38
Juan Cespedesce377d52008-12-16 19:38:10 +010039 if (initgroups(options.user, run_gid) < 0) {
Juan Cespedese3eb9aa1999-04-03 03:21:52 +020040 perror("ltrace: initgroups");
Juan Cespedese1887051998-03-10 00:08:41 +010041 exit(1);
42 }
43 } else {
44 run_uid = getuid();
45 run_gid = getgid();
46 }
Juan Cespedesce377d52008-12-16 19:38:10 +010047 if (options.user || !geteuid()) {
Juan Cespedese1887051998-03-10 00:08:41 +010048 struct stat statbuf;
49 run_euid = run_uid;
50 run_egid = run_gid;
51
52 if (!stat(proc->filename, &statbuf)) {
53 if (statbuf.st_mode & S_ISUID) {
54 run_euid = statbuf.st_uid;
55 }
56 if (statbuf.st_mode & S_ISGID) {
Juan Cespedes666da8b1998-04-29 20:21:48 +020057 run_egid = statbuf.st_gid;
Juan Cespedese1887051998-03-10 00:08:41 +010058 }
59 }
60 if (setregid(run_gid, run_egid) < 0) {
Juan Cespedese3eb9aa1999-04-03 03:21:52 +020061 perror("ltrace: setregid");
Juan Cespedese1887051998-03-10 00:08:41 +010062 exit(1);
63 }
64 if (setreuid(run_uid, run_euid) < 0) {
Juan Cespedese3eb9aa1999-04-03 03:21:52 +020065 perror("ltrace: setreuid");
Juan Cespedese1887051998-03-10 00:08:41 +010066 exit(1);
67 }
68 }
69}
Juan Cespedescac15c32003-01-31 18:58:58 +010070
Juan Cespedesf1350522008-12-16 18:19:58 +010071void
Juan Cespedesa8909f72009-04-28 20:02:41 +020072execute_program(Process *sp, char **argv) {
Juan Cespedescac15c32003-01-31 18:58:58 +010073 pid_t pid;
74
75 debug(1, "Executing `%s'...", sp->filename);
76
77 pid = fork();
Ian Wienand2d45b1a2006-02-20 22:48:07 +010078 if (pid < 0) {
Juan Cespedescac15c32003-01-31 18:58:58 +010079 perror("ltrace: fork");
80 exit(1);
81 } else if (!pid) { /* child */
82 change_uid(sp);
83 trace_me();
84 execvp(sp->filename, argv);
Ian Wienand2d45b1a2006-02-20 22:48:07 +010085 fprintf(stderr, "Can't execute `%s': %s\n", sp->filename,
86 strerror(errno));
Juan Cespedese672ad12009-02-11 18:49:18 +010087 _exit(1);
Juan Cespedescac15c32003-01-31 18:58:58 +010088 }
89
90 debug(1, "PID=%d", pid);
91
92 sp->pid = pid;
93
Joe Damatoab3b72c2010-10-31 00:21:53 -070094#if defined(HAVE_LIBUNWIND)
95 sp->unwind_priv = _UPT_create(pid);
96#endif /* defined(HAVE_LIBUNWIND) */
97
Juan Cespedescac15c32003-01-31 18:58:58 +010098 return;
99}