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;
 | |
| }
 |