/* vi: set sw=4 ts=4: */ /* * Mini chown implementation for busybox * * Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org> * * Licensed under GPLv2 or later, see file LICENSE in this tarball for details. */ /* BB_AUDIT SUSv3 defects - unsupported options -H, -L, and -P. */ /* BB_AUDIT GNU defects - unsupported long options. */ /* http://www.opengroup.org/onlinepubs/007904975/utilities/chown.html */ #include "busybox.h" static struct bb_uidgid_t ugid = { -1, -1 }; static int (*chown_func)(const char *, uid_t, gid_t) = chown; #define OPT_RECURSE (option_mask32 & 1) #define OPT_NODEREF (option_mask32 & 2) #define OPT_VERBOSE (USE_DESKTOP(option_mask32 & 4) SKIP_DESKTOP(0)) #define OPT_CHANGED (USE_DESKTOP(option_mask32 & 8) SKIP_DESKTOP(0)) #define OPT_QUIET (USE_DESKTOP(option_mask32 & 0x10) SKIP_DESKTOP(0)) #define OPT_STR ("Rh" USE_DESKTOP("vcf")) /* TODO: * -H if a command line argument is a symbolic link to a directory, traverse it * -L traverse every symbolic link to a directory encountered * -P do not traverse any symbolic links (default) */ static int fileAction(const char *fileName, struct stat *statbuf, void ATTRIBUTE_UNUSED *junk, int depth) { // TODO: -H/-L/-P // if (depth ... && S_ISLNK(statbuf->st_mode)) .... if (!chown_func(fileName, (ugid.uid == (uid_t)-1) ? statbuf->st_uid : ugid.uid, (ugid.gid == (gid_t)-1) ? statbuf->st_gid : ugid.gid) ) { if (OPT_VERBOSE || (OPT_CHANGED && (statbuf->st_uid != ugid.uid || statbuf->st_gid != ugid.gid)) ) { printf("changed ownership of '%s' to %u:%u\n", fileName, ugid.uid, ugid.gid); } return TRUE; } if (!OPT_QUIET) bb_perror_msg("%s", fileName); /* A filename can have % in it... */ return FALSE; } int chown_main(int argc, char **argv) { char *groupName; int retval = EXIT_SUCCESS; opt_complementary = "-2"; getopt32(argc, argv, OPT_STR); argv += optind; if (OPT_NODEREF) chown_func = lchown; /* First, check if there is a group name here */ groupName = strchr(*argv, '.'); /* deprecated? */ if (!groupName) groupName = strchr(*argv, ':'); else *groupName = ':'; /* replace '.' with ':' */ /* First, try parsing "user[:[group]]" */ if (!groupName) { /* "user" */ ugid.uid = get_ug_id(*argv, xuname2uid); } else if (groupName == *argv) { /* ":group" */ ugid.gid = get_ug_id(groupName + 1, xgroup2gid); } else { if (!groupName[1]) /* "user:" */ *groupName = '\0'; if (!get_uidgid(&ugid, *argv, 1)) bb_error_msg_and_die("unknown user/group %s", *argv); } /* Ok, ready to do the deed now */ argv++; do { if (!recursive_action(*argv, OPT_RECURSE, // recurse FALSE, // follow links: TODO: -H/-L/-P FALSE, // depth first fileAction, // file action fileAction, // dir action NULL, // user data 0) // depth ) { retval = EXIT_FAILURE; } } while (*++argv); return retval; }