You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
127 lines
2.5 KiB
127 lines
2.5 KiB
/*
|
|
* Copyright (c) 1997,2007 Andrew G. Morgan <morgan@kernel.org>
|
|
*
|
|
* This displays the capabilities of a given file.
|
|
*/
|
|
|
|
#undef _XOPEN_SOURCE
|
|
#define _XOPEN_SOURCE 500
|
|
|
|
#include <errno.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <dirent.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include <sys/capability.h>
|
|
|
|
#include <ftw.h>
|
|
|
|
static int verbose = 0;
|
|
static int recursive = 0;
|
|
static int namespace = 0;
|
|
|
|
static void usage(int code)
|
|
{
|
|
fprintf(stderr,
|
|
"usage: getcap [-h] [-l] [-n] [-r] [-v] <filename> [<filename> ...]\n"
|
|
"\n"
|
|
"\tdisplays the capabilities on the queried file(s).\n"
|
|
);
|
|
exit(code);
|
|
}
|
|
|
|
static int do_getcap(const char *fname, const struct stat *stbuf,
|
|
int tflag, struct FTW* ftwbuf)
|
|
{
|
|
cap_t cap_d;
|
|
char *result;
|
|
uid_t rootid;
|
|
|
|
if (tflag != FTW_F) {
|
|
if (verbose) {
|
|
printf("%s (Not a regular file)\n", fname);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
cap_d = cap_get_file(fname);
|
|
if (cap_d == NULL) {
|
|
if (errno != ENODATA) {
|
|
fprintf(stderr, "Failed to get capabilities of file '%s' (%s)\n",
|
|
fname, strerror(errno));
|
|
} else if (verbose) {
|
|
printf("%s\n", fname);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
result = cap_to_text(cap_d, NULL);
|
|
if (!result) {
|
|
fprintf(stderr,
|
|
"Failed to get capabilities of human readable format at '%s' (%s)\n",
|
|
fname, strerror(errno));
|
|
cap_free(cap_d);
|
|
return 0;
|
|
}
|
|
rootid = cap_get_nsowner(cap_d);
|
|
if (namespace && (rootid+1 > 1)) {
|
|
printf("%s %s [rootid=%d]\n", fname, result, rootid);
|
|
} else {
|
|
printf("%s %s\n", fname, result);
|
|
}
|
|
cap_free(cap_d);
|
|
cap_free(result);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
int i, c;
|
|
|
|
while ((c = getopt(argc, argv, "rvhnl")) > 0) {
|
|
switch(c) {
|
|
case 'r':
|
|
recursive = 1;
|
|
break;
|
|
case 'v':
|
|
verbose = 1;
|
|
break;
|
|
case 'n':
|
|
namespace = 1;
|
|
break;
|
|
case 'h':
|
|
usage(0);
|
|
case 'l':
|
|
printf("%s has a you choose license: BSD 3-clause or GPL2\n"
|
|
"Copyright (c) 1997,2007 Andrew G. Morgan"
|
|
" <morgan@kernel.org>\n", argv[0]);
|
|
exit(0);
|
|
default:
|
|
usage(1);
|
|
}
|
|
}
|
|
|
|
if (!argv[optind])
|
|
usage(1);
|
|
|
|
for (i=optind; argv[i] != NULL; i++) {
|
|
struct stat stbuf;
|
|
|
|
if (lstat(argv[i], &stbuf) != 0) {
|
|
fprintf(stderr, "%s (%s)\n", argv[i], strerror(errno));
|
|
} else if (recursive) {
|
|
nftw(argv[i], do_getcap, 20, FTW_PHYS);
|
|
} else {
|
|
int tflag = S_ISREG(stbuf.st_mode) ? FTW_F :
|
|
(S_ISLNK(stbuf.st_mode) ? FTW_SL : FTW_NS);
|
|
do_getcap(argv[i], &stbuf, tflag, 0);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|