257 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			257 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* getfacl.c
 | 
						|
 | 
						|
   Copyright 2000, 2001, 2002 Red Hat Inc.
 | 
						|
 | 
						|
   Written by Corinna Vinschen <vinschen@redhat.com>
 | 
						|
 | 
						|
This file is part of Cygwin.
 | 
						|
 | 
						|
This software is a copyrighted work licensed under the terms of the
 | 
						|
Cygwin license.  Please consult the file "CYGWIN_LICENSE" for
 | 
						|
details. */
 | 
						|
 | 
						|
#include <pwd.h>
 | 
						|
#include <grp.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <getopt.h>
 | 
						|
#include <sys/types.h>
 | 
						|
#include <sys/acl.h>
 | 
						|
#include <sys/stat.h>
 | 
						|
#include <string.h>
 | 
						|
 | 
						|
static const char version[] = "$Revision$";
 | 
						|
static char *prog_name;
 | 
						|
 | 
						|
char *
 | 
						|
permstr (mode_t perm)
 | 
						|
{
 | 
						|
  static char pbuf[4];
 | 
						|
 | 
						|
  pbuf[0] = (perm & S_IROTH) ? 'r' : '-';
 | 
						|
  pbuf[1] = (perm & S_IWOTH) ? 'w' : '-';
 | 
						|
  pbuf[2] = (perm & S_IXOTH) ? 'x' : '-';
 | 
						|
  pbuf[3] = '\0';
 | 
						|
  return pbuf;
 | 
						|
}
 | 
						|
 | 
						|
const char *
 | 
						|
username (uid_t uid)
 | 
						|
{
 | 
						|
  static char ubuf[256];
 | 
						|
  struct passwd *pw;
 | 
						|
 | 
						|
  if ((pw = getpwuid (uid)))
 | 
						|
    strcpy (ubuf, pw->pw_name);
 | 
						|
  else
 | 
						|
    sprintf (ubuf, "%lu <unknown>", (unsigned long)uid);
 | 
						|
  return ubuf;
 | 
						|
}
 | 
						|
 | 
						|
const char *
 | 
						|
groupname (gid_t gid)
 | 
						|
{
 | 
						|
  static char gbuf[256];
 | 
						|
  struct group *gr;
 | 
						|
 | 
						|
  if ((gr = getgrgid (gid)))
 | 
						|
    strcpy (gbuf, gr->gr_name);
 | 
						|
  else
 | 
						|
    sprintf (gbuf, "%lu <unknown>", (unsigned long)gid);
 | 
						|
  return gbuf;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
usage (FILE * stream)
 | 
						|
{
 | 
						|
  fprintf (stream, "Usage: %s [-adn] FILE [FILE2...]\n"
 | 
						|
            "Display file and directory access control lists (ACLs).\n"
 | 
						|
            "\n"
 | 
						|
            "  -a, --all      display the filename, the owner, the group, and\n"
 | 
						|
            "                 the ACL of the file\n"
 | 
						|
            "  -d, --dir      display the filename, the owner, the group, and\n"
 | 
						|
            "                 the default ACL of the directory, if it exists\n"
 | 
						|
            "  -h, --help     output usage information and exit\n"
 | 
						|
            "  -n, --noname   display user and group IDs instead of names\n"
 | 
						|
            "  -v, --version  output version information and exit\n"
 | 
						|
            "\n"
 | 
						|
            "When multiple files are specified on the command line, a blank\n"
 | 
						|
            "line separates the ACLs for each file.\n", prog_name);
 | 
						|
  if (stream == stdout) 
 | 
						|
    {
 | 
						|
      fprintf (stream, ""
 | 
						|
            "For each argument that is a regular file, special file or\n"
 | 
						|
            "directory, getfacl displays the owner, the group, and the ACL.\n"
 | 
						|
            "For directories getfacl displays additionally the default ACL.\n"
 | 
						|
            "\n"
 | 
						|
            "With no options specified, getfacl displays the filename, the\n"
 | 
						|
            "owner, the group, and both the ACL and the default ACL, if it\n"
 | 
						|
            "exists.\n"
 | 
						|
            "\n"
 | 
						|
            "The format for ACL output is as follows:\n"
 | 
						|
            "     # file: filename\n"
 | 
						|
            "     # owner: name or uid\n"
 | 
						|
            "     # group: name or uid\n"
 | 
						|
            "     user::perm\n"
 | 
						|
            "     user:name or uid:perm\n"
 | 
						|
            "     group::perm\n"
 | 
						|
            "     group:name or gid:perm\n"
 | 
						|
            "     mask:perm\n"
 | 
						|
            "     other:perm\n"
 | 
						|
            "     default:user::perm\n"
 | 
						|
            "     default:user:name or uid:perm\n"
 | 
						|
            "     default:group::perm\n"
 | 
						|
            "     default:group:name or gid:perm\n"
 | 
						|
            "     default:mask:perm\n"
 | 
						|
            "     default:other:perm\n"
 | 
						|
            "\n");
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
struct option longopts[] = {
 | 
						|
  {"all", no_argument, NULL, 'a'},
 | 
						|
  {"dir", no_argument, NULL, 'd'},
 | 
						|
  {"help", no_argument, NULL, 'h'},
 | 
						|
  {"noname", no_argument, NULL, 'n'},
 | 
						|
  {"version", no_argument, NULL, 'v'},
 | 
						|
  {0, no_argument, NULL, 0}
 | 
						|
};
 | 
						|
 | 
						|
static void
 | 
						|
print_version ()
 | 
						|
{
 | 
						|
  const char *v = strchr (version, ':');
 | 
						|
  int len;
 | 
						|
  if (!v)
 | 
						|
    {
 | 
						|
      v = "?";
 | 
						|
      len = 1;
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      v += 2;
 | 
						|
      len = strchr (v, ' ') - v;
 | 
						|
    }
 | 
						|
  printf ("\
 | 
						|
getfacl (cygwin) %.*s\n\
 | 
						|
ACL Utility\n\
 | 
						|
Copyright (c) 2000, 2001, 2002 Red Hat, Inc.\n\
 | 
						|
Compiled on %s\n\
 | 
						|
", len, v, __DATE__);
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
main (int argc, char **argv)
 | 
						|
{
 | 
						|
  int c, i;
 | 
						|
  int aopt = 0;
 | 
						|
  int dopt = 0;
 | 
						|
  int nopt = 0;
 | 
						|
  int first = 1;
 | 
						|
  struct stat st;
 | 
						|
  aclent_t acls[MAX_ACL_ENTRIES];
 | 
						|
 | 
						|
  prog_name = strrchr (argv[0], '/');
 | 
						|
  if (prog_name == NULL)
 | 
						|
    prog_name = strrchr (argv[0], '\\');
 | 
						|
  if (prog_name == NULL)
 | 
						|
    prog_name = argv[0];
 | 
						|
  else
 | 
						|
    prog_name++;
 | 
						|
 | 
						|
  while ((c = getopt_long (argc, argv, "adhnv", longopts, NULL)) != EOF)
 | 
						|
    switch (c)
 | 
						|
      {
 | 
						|
      case 'a':
 | 
						|
	aopt = 1;
 | 
						|
	break;
 | 
						|
      case 'd':
 | 
						|
	dopt = 1;
 | 
						|
	break;
 | 
						|
      case 'h':
 | 
						|
	usage (stdout);
 | 
						|
	return 0;
 | 
						|
      case 'n':
 | 
						|
	nopt = 1;
 | 
						|
	break;
 | 
						|
      case 'v':
 | 
						|
	print_version ();
 | 
						|
	return 0;
 | 
						|
      default:
 | 
						|
	usage (stderr);
 | 
						|
	return 1;
 | 
						|
      }
 | 
						|
  if (optind > argc - 1)
 | 
						|
    {
 | 
						|
      usage (stderr);
 | 
						|
      return 1;
 | 
						|
    }
 | 
						|
  while ((c = optind++) < argc)
 | 
						|
    {
 | 
						|
      if (stat (argv[c], &st))
 | 
						|
	{
 | 
						|
	  perror (argv[0]);
 | 
						|
	  continue;
 | 
						|
	}
 | 
						|
      if (!first)
 | 
						|
	putchar ('\n');
 | 
						|
      first = 0;
 | 
						|
      printf ("# file: %s\n", argv[c]);
 | 
						|
      if (nopt)
 | 
						|
        {
 | 
						|
	  printf ("# owner: %lu\n", (unsigned long)st.st_uid);
 | 
						|
	  printf ("# group: %lu\n", (unsigned long)st.st_gid);
 | 
						|
	}
 | 
						|
      else
 | 
						|
        {
 | 
						|
	  printf ("# owner: %s\n", username (st.st_uid));
 | 
						|
	  printf ("# group: %s\n", groupname (st.st_gid));
 | 
						|
	}
 | 
						|
      if ((c = acl (argv[c], GETACL, MAX_ACL_ENTRIES, acls)) < 0)
 | 
						|
	{
 | 
						|
	  perror (argv[0]);
 | 
						|
	  continue;
 | 
						|
	}
 | 
						|
      for (i = 0; i < c; ++i)
 | 
						|
	{
 | 
						|
	  if (acls[i].a_type & ACL_DEFAULT)
 | 
						|
	    {
 | 
						|
	      if (aopt)
 | 
						|
		continue;
 | 
						|
	      printf ("default:");
 | 
						|
	    }
 | 
						|
	  else if (dopt)
 | 
						|
	    continue;
 | 
						|
	  switch (acls[i].a_type & ~ACL_DEFAULT)
 | 
						|
	    {
 | 
						|
	    case USER_OBJ:
 | 
						|
	      printf ("user::");
 | 
						|
	      break;
 | 
						|
	    case USER:
 | 
						|
	      if (nopt)
 | 
						|
		printf ("user:%lu:", (unsigned long)acls[i].a_id);
 | 
						|
	      else
 | 
						|
		printf ("user:%s:", username (acls[i].a_id));
 | 
						|
	      break;
 | 
						|
	    case GROUP_OBJ:
 | 
						|
	      printf ("group::");
 | 
						|
	      break;
 | 
						|
	    case GROUP:
 | 
						|
	      if (nopt)
 | 
						|
		printf ("group:%lu:", (unsigned long)acls[i].a_id);
 | 
						|
	      else
 | 
						|
		printf ("group:%s:", groupname (acls[i].a_id));
 | 
						|
	      break;
 | 
						|
	    case CLASS_OBJ:
 | 
						|
	      printf ("mask:");
 | 
						|
	      break;
 | 
						|
	    case OTHER_OBJ:
 | 
						|
	      printf ("other:");
 | 
						|
	      break;
 | 
						|
	    }
 | 
						|
	  printf ("%s\n", permstr (acls[i].a_perm));
 | 
						|
	}
 | 
						|
    }
 | 
						|
  return 0;
 | 
						|
}
 |