blob: 5fd637983ff4a9bdef3902f78e1118268d1a532b [file] [log] [blame]
Juan Cespedesd44c6b81998-09-25 14:48:42 +02001#include "config.h"
Juan Cespedesd44c6b81998-09-25 14:48:42 +02002
Juan Cespedes5e01f651998-03-08 22:31:44 +01003#include <stdio.h>
Juan Cespedes504a3852003-02-04 23:24:38 +01004#include <stdlib.h>
Juan Cespedese1887051998-03-10 00:08:41 +01005#include <sys/types.h>
6#include <sys/stat.h>
Juan Cespedes5e01f651998-03-08 22:31:44 +01007#include <unistd.h>
8#include <errno.h>
9#include <string.h>
Juan Cespedese1887051998-03-10 00:08:41 +010010#include <pwd.h>
11#include <grp.h>
Juan Cespedes5e01f651998-03-08 22:31:44 +010012
Juan Cespedesf7281232009-06-25 16:11:21 +020013#include "common.h"
Juan Cespedes5e01f651998-03-08 22:31:44 +010014
Juan Cespedesf1350522008-12-16 18:19:58 +010015static void
Juan Cespedesa8909f72009-04-28 20:02:41 +020016change_uid(Process *proc) {
Juan Cespedese1887051998-03-10 00:08:41 +010017 uid_t run_uid, run_euid;
18 gid_t run_gid, run_egid;
19
Juan Cespedesce377d52008-12-16 19:38:10 +010020 if (options.user) {
Juan Cespedese1887051998-03-10 00:08:41 +010021 struct passwd *pent;
22
23 if (getuid() != 0 || geteuid() != 0) {
Ian Wienand2d45b1a2006-02-20 22:48:07 +010024 fprintf(stderr,
25 "you must be root to use the -u option\n");
Juan Cespedese1887051998-03-10 00:08:41 +010026 exit(1);
27 }
Juan Cespedesce377d52008-12-16 19:38:10 +010028 if ((pent = getpwnam(options.user)) == NULL) {
29 fprintf(stderr, "cannot find user `%s'\n", options.user);
Juan Cespedese1887051998-03-10 00:08:41 +010030 exit(1);
31 }
32 run_uid = pent->pw_uid;
33 run_gid = pent->pw_gid;
34
Juan Cespedesce377d52008-12-16 19:38:10 +010035 if (initgroups(options.user, run_gid) < 0) {
Juan Cespedese3eb9aa1999-04-03 03:21:52 +020036 perror("ltrace: initgroups");
Juan Cespedese1887051998-03-10 00:08:41 +010037 exit(1);
38 }
39 } else {
40 run_uid = getuid();
41 run_gid = getgid();
42 }
Juan Cespedesce377d52008-12-16 19:38:10 +010043 if (options.user || !geteuid()) {
Juan Cespedese1887051998-03-10 00:08:41 +010044 struct stat statbuf;
45 run_euid = run_uid;
46 run_egid = run_gid;
47
48 if (!stat(proc->filename, &statbuf)) {
49 if (statbuf.st_mode & S_ISUID) {
50 run_euid = statbuf.st_uid;
51 }
52 if (statbuf.st_mode & S_ISGID) {
Juan Cespedes666da8b1998-04-29 20:21:48 +020053 run_egid = statbuf.st_gid;
Juan Cespedese1887051998-03-10 00:08:41 +010054 }
55 }
56 if (setregid(run_gid, run_egid) < 0) {
Juan Cespedese3eb9aa1999-04-03 03:21:52 +020057 perror("ltrace: setregid");
Juan Cespedese1887051998-03-10 00:08:41 +010058 exit(1);
59 }
60 if (setreuid(run_uid, run_euid) < 0) {
Juan Cespedese3eb9aa1999-04-03 03:21:52 +020061 perror("ltrace: setreuid");
Juan Cespedese1887051998-03-10 00:08:41 +010062 exit(1);
63 }
64 }
65}
Juan Cespedescac15c32003-01-31 18:58:58 +010066
Juan Cespedesf1350522008-12-16 18:19:58 +010067void
Juan Cespedesa8909f72009-04-28 20:02:41 +020068execute_program(Process *sp, char **argv) {
Juan Cespedescac15c32003-01-31 18:58:58 +010069 pid_t pid;
70
71 debug(1, "Executing `%s'...", sp->filename);
72
73 pid = fork();
Ian Wienand2d45b1a2006-02-20 22:48:07 +010074 if (pid < 0) {
Juan Cespedescac15c32003-01-31 18:58:58 +010075 perror("ltrace: fork");
76 exit(1);
77 } else if (!pid) { /* child */
78 change_uid(sp);
79 trace_me();
80 execvp(sp->filename, argv);
Ian Wienand2d45b1a2006-02-20 22:48:07 +010081 fprintf(stderr, "Can't execute `%s': %s\n", sp->filename,
82 strerror(errno));
Juan Cespedese672ad12009-02-11 18:49:18 +010083 _exit(1);
Juan Cespedescac15c32003-01-31 18:58:58 +010084 }
85
86 debug(1, "PID=%d", pid);
87
88 sp->pid = pid;
89
90 return;
91}