123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144 |
- #include <errno.h>
- #include <error.h>
- #include <getopt.h>
- #include <paths.h>
- #include <pwd.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <unistd.h>
- #include <private/android_filesystem_config.h>
- void pwtoid(const char* tok, uid_t* uid, gid_t* gid) {
- struct passwd* pw = getpwnam(tok);
- if (pw) {
- if (uid) *uid = pw->pw_uid;
- if (gid) *gid = pw->pw_gid;
- } else {
- char* end;
- errno = 0;
- uid_t tmpid = strtoul(tok, &end, 10);
- if (errno != 0 || end == tok) error(1, errno, "invalid uid/gid '%s'", tok);
- if (uid) *uid = tmpid;
- if (gid) *gid = tmpid;
- }
- }
- void extract_uidgids(const char* uidgids, uid_t* uid, gid_t* gid, gid_t* gids, int* gids_count) {
- char *clobberablegids;
- char *nexttok;
- char *tok;
- int gids_found;
- if (!uidgids || !*uidgids) {
- *gid = *uid = 0;
- *gids_count = 0;
- return;
- }
- clobberablegids = strdup(uidgids);
- strcpy(clobberablegids, uidgids);
- nexttok = clobberablegids;
- tok = strsep(&nexttok, ",");
- pwtoid(tok, uid, gid);
- tok = strsep(&nexttok, ",");
- if (!tok) {
-
- *gids_count = 0;
- free(clobberablegids);
- return;
- }
- pwtoid(tok, NULL, gid);
- gids_found = 0;
- while ((gids_found < *gids_count) && (tok = strsep(&nexttok, ","))) {
- pwtoid(tok, NULL, gids);
- gids_found++;
- gids++;
- }
- if (nexttok && gids_found == *gids_count) {
- fprintf(stderr, "too many group ids\n");
- }
- *gids_count = gids_found;
- free(clobberablegids);
- }
- int main(int argc, char** argv) {
- uid_t current_uid = getuid();
- if (current_uid != AID_ROOT && current_uid != AID_SHELL) error(1, 0, "not allowed");
-
- ++argv;
- if (*argv && (strcmp(*argv, "--help") == 0 || strcmp(*argv, "-h") == 0)) {
- fprintf(stderr,
- "usage: su [WHO [COMMAND...]]\n"
- "\n"
- "Switch to WHO (default 'root') and run the given COMMAND (default sh).\n"
- "\n"
- "WHO is a comma-separated list of user, group, and supplementary groups\n"
- "in that order.\n"
- "\n");
- return 0;
- }
-
- uid_t uid = 0;
- gid_t gid = 0;
-
- if (*argv) {
- gid_t gids[10];
- int gids_count = sizeof(gids)/sizeof(gids[0]);
- extract_uidgids(*argv, &uid, &gid, gids, &gids_count);
- if (gids_count) {
- if (setgroups(gids_count, gids)) {
- error(1, errno, "setgroups failed");
- }
- }
- ++argv;
- }
- if (setgid(gid)) error(1, errno, "setgid failed");
- if (setuid(uid)) error(1, errno, "setuid failed");
-
- setenv("PATH", _PATH_DEFPATH, 1);
- unsetenv("IFS");
- struct passwd* pw = getpwuid(uid);
- if (pw) {
- setenv("LOGNAME", pw->pw_name, 1);
- setenv("USER", pw->pw_name, 1);
- } else {
- unsetenv("LOGNAME");
- unsetenv("USER");
- }
-
- char* exec_args[argc + 1];
- size_t i = 0;
- for (; *argv != NULL; ++i) {
- exec_args[i] = *argv++;
- }
-
- if (i == 0) exec_args[i++] = const_cast<char*>("/system/bin/sh");
- exec_args[i] = NULL;
- execvp(exec_args[0], exec_args);
- error(1, errno, "failed to exec %s", exec_args[0]);
- }
|