| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | This file is umask.def, from which is created umask.c. | 
					
						
							|  |  |  | It implements the builtin "umask" in Bash. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | Copyright (C) 1987-2015 Free Software Foundation, Inc. | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | This file is part of GNU Bash, the Bourne Again SHell. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | Bash is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  | the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | (at your option) any later version. | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | Bash is distributed in the hope that it will be useful, | 
					
						
							|  |  |  | but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  | GNU General Public License for more details. | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  | along with Bash.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | $PRODUCES umask.c | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | $BUILTIN umask | 
					
						
							|  |  |  | $FUNCTION umask_builtin | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | $SHORT_DOC umask [-p] [-S] [mode] | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | Display or set file mode mask. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Sets the user file-creation mask to MODE.  If MODE is omitted, prints | 
					
						
							|  |  |  | the current value of the mask. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | If MODE begins with a digit, it is interpreted as an octal number; | 
					
						
							|  |  |  | otherwise it is a symbolic mode string like that accepted by chmod(1). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Options: | 
					
						
							|  |  |  |   -p	if MODE is omitted, output in a form that may be reused as input | 
					
						
							|  |  |  |   -S	makes the output symbolic; otherwise an octal number is output | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Exit Status: | 
					
						
							|  |  |  | Returns success unless MODE is invalid or an invalid option is given. | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | $END | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #include <config.h> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "../bashtypes.h" | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | #include "filecntl.h" | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #if ! defined(_MINIX) && defined (HAVE_SYS_FILE_H) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #  include <sys/file.h> | 
					
						
							|  |  |  | #endif | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if defined (HAVE_UNISTD_H) | 
					
						
							|  |  |  | #include <unistd.h> | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <stdio.h> | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #include <chartypes.h> | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #include "../bashintl.h" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #include "../shell.h" | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | #include "posixstat.h" | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #include "common.h" | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #include "bashgetopt.h" | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* **************************************************************** */ | 
					
						
							|  |  |  | /*                                                                  */ | 
					
						
							|  |  |  | /*                     UMASK Builtin and Helpers                    */ | 
					
						
							|  |  |  | /*                                                                  */ | 
					
						
							|  |  |  | /* **************************************************************** */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static void print_symbolic_umask __P((mode_t)); | 
					
						
							|  |  |  | static int symbolic_umask __P((WORD_LIST *)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Set or display the mask used by the system when creating files.  Flag | 
					
						
							|  |  |  |    of -S means display the umask in a symbolic mode. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | umask_builtin (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   int print_symbolically, opt, umask_value, pflag; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   mode_t umask_arg; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   print_symbolically = pflag = 0; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   reset_internal_getopt (); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   while ((opt = internal_getopt (list, "Sp")) != -1) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       switch (opt) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	case 'S': | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  print_symbolically++; | 
					
						
							|  |  |  | 	  break; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	case 'p': | 
					
						
							|  |  |  | 	  pflag++; | 
					
						
							|  |  |  | 	  break; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	CASE_HELPOPT; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	default: | 
					
						
							|  |  |  | 	  builtin_usage (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  return (EX_USAGE); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   list = loptend; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (list) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       if (DIGIT (*list->word->word)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  umask_value = read_octal (list->word->word); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  /* Note that other shells just let you set the umask to zero | 
					
						
							|  |  |  | 	     by specifying a number out of range.  This is a problem | 
					
						
							|  |  |  | 	     with those shells.  We don't change the umask if the input | 
					
						
							|  |  |  | 	     is lousy. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if (umask_value == -1) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	      sh_erange (list->word->word, _("octal number")); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	      return (EXECUTION_FAILURE); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  umask_value = symbolic_umask (list); | 
					
						
							|  |  |  | 	  if (umask_value == -1) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    return (EXECUTION_FAILURE); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |       umask_arg = (mode_t)umask_value; | 
					
						
							|  |  |  |       umask (umask_arg); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       if (print_symbolically) | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 	print_symbolic_umask (umask_arg); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   else				/* Display the UMASK for this user. */ | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |       umask_arg = umask (022); | 
					
						
							|  |  |  |       umask (umask_arg); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (pflag) | 
					
						
							|  |  |  | 	printf ("umask%s ", (print_symbolically ? " -S" : "")); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       if (print_symbolically) | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 	print_symbolic_umask (umask_arg); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	printf ("%04lo\n", (unsigned long)umask_arg); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   return (sh_chkwrite (EXECUTION_SUCCESS)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Print the umask in a symbolic form.  In the output, a letter is | 
					
						
							|  |  |  |    printed if the corresponding bit is clear in the umask. */ | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #if defined (__STDC__) | 
					
						
							|  |  |  | print_symbolic_umask (mode_t um) | 
					
						
							|  |  |  | #else | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | print_symbolic_umask (um) | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |      mode_t um; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #endif | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   char ubits[4], gbits[4], obits[4];		/* u=rwx,g=rwx,o=rwx */ | 
					
						
							|  |  |  |   int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   i = 0; | 
					
						
							|  |  |  |   if ((um & S_IRUSR) == 0) | 
					
						
							|  |  |  |     ubits[i++] = 'r'; | 
					
						
							|  |  |  |   if ((um & S_IWUSR) == 0) | 
					
						
							|  |  |  |     ubits[i++] = 'w'; | 
					
						
							|  |  |  |   if ((um & S_IXUSR) == 0) | 
					
						
							|  |  |  |     ubits[i++] = 'x'; | 
					
						
							|  |  |  |   ubits[i] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   i = 0; | 
					
						
							|  |  |  |   if ((um & S_IRGRP) == 0) | 
					
						
							|  |  |  |     gbits[i++] = 'r'; | 
					
						
							|  |  |  |   if ((um & S_IWGRP) == 0) | 
					
						
							|  |  |  |     gbits[i++] = 'w'; | 
					
						
							|  |  |  |   if ((um & S_IXGRP) == 0) | 
					
						
							|  |  |  |     gbits[i++] = 'x'; | 
					
						
							|  |  |  |   gbits[i] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   i = 0; | 
					
						
							|  |  |  |   if ((um & S_IROTH) == 0) | 
					
						
							|  |  |  |     obits[i++] = 'r'; | 
					
						
							|  |  |  |   if ((um & S_IWOTH) == 0) | 
					
						
							|  |  |  |     obits[i++] = 'w'; | 
					
						
							|  |  |  |   if ((um & S_IXOTH) == 0) | 
					
						
							|  |  |  |     obits[i++] = 'x'; | 
					
						
							|  |  |  |   obits[i] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   printf ("u=%s,g=%s,o=%s\n", ubits, gbits, obits); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | int | 
					
						
							|  |  |  | parse_symbolic_mode (mode, initial_bits) | 
					
						
							|  |  |  |      char *mode; | 
					
						
							|  |  |  |      int initial_bits; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   int who, op, perm, bits, c; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   char *s; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |   for (s = mode, bits = initial_bits;;) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       who = op = perm = 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* Parse the `who' portion of the symbolic mode clause. */ | 
					
						
							|  |  |  |       while (member (*s, "agou")) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  switch (c = *s++) | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	    case 'u': | 
					
						
							|  |  |  | 	      who |= S_IRWXU; | 
					
						
							|  |  |  | 	      continue; | 
					
						
							|  |  |  | 	    case 'g': | 
					
						
							|  |  |  | 	      who |= S_IRWXG; | 
					
						
							|  |  |  | 	      continue; | 
					
						
							|  |  |  | 	    case 'o': | 
					
						
							|  |  |  | 	      who |= S_IRWXO; | 
					
						
							|  |  |  | 	      continue; | 
					
						
							|  |  |  | 	    case 'a': | 
					
						
							|  |  |  | 	      who |= S_IRWXU | S_IRWXG | S_IRWXO; | 
					
						
							|  |  |  | 	      continue; | 
					
						
							|  |  |  | 	    default: | 
					
						
							|  |  |  | 	      break; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* The operation is now sitting in *s. */ | 
					
						
							|  |  |  |       op = *s++; | 
					
						
							|  |  |  |       switch (op) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	case '+': | 
					
						
							|  |  |  | 	case '-': | 
					
						
							|  |  |  | 	case '=': | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  builtin_error (_("`%c': invalid symbolic mode operator"), op); | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	  return (-1); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Parse out the `perm' section of the symbolic mode clause. */ | 
					
						
							|  |  |  |       while (member (*s, "rwx")) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  c = *s++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  switch (c) | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	    case 'r': | 
					
						
							|  |  |  | 	      perm |= S_IRUGO; | 
					
						
							|  |  |  | 	      break; | 
					
						
							|  |  |  | 	    case 'w': | 
					
						
							|  |  |  | 	      perm |= S_IWUGO; | 
					
						
							|  |  |  | 	      break; | 
					
						
							|  |  |  | 	    case 'x': | 
					
						
							|  |  |  | 	      perm |= S_IXUGO; | 
					
						
							|  |  |  | 	      break; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Now perform the operation or return an error for a | 
					
						
							|  |  |  | 	 bad permission string. */ | 
					
						
							|  |  |  |       if (!*s || *s == ',') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if (who) | 
					
						
							|  |  |  | 	    perm &= who; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  switch (op) | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	    case '+': | 
					
						
							|  |  |  | 	      bits |= perm; | 
					
						
							|  |  |  | 	      break; | 
					
						
							|  |  |  | 	    case '-': | 
					
						
							|  |  |  | 	      bits &= ~perm; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	      break; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	    case '=': | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	      if (who == 0) | 
					
						
							|  |  |  | 		who = S_IRWXU | S_IRWXG | S_IRWXO; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	      bits &= ~who; | 
					
						
							|  |  |  | 	      bits |= perm; | 
					
						
							|  |  |  | 	      break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	    /* No other values are possible. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  if (*s == '\0') | 
					
						
							|  |  |  | 	    break; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  else | 
					
						
							|  |  |  | 	    s++;	/* skip past ',' */ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  builtin_error (_("`%c': invalid symbolic mode character"), *s); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  return (-1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return (bits); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Set the umask from a symbolic mode string similar to that accepted | 
					
						
							|  |  |  |    by chmod.  If the -S argument is given, then print the umask in a | 
					
						
							|  |  |  |    symbolic form. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | symbolic_umask (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int um, bits; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Get the initial umask.  Don't change it yet. */ | 
					
						
							|  |  |  |   um = umask (022); | 
					
						
							|  |  |  |   umask (um); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* All work is done with the complement of the umask -- it's | 
					
						
							|  |  |  |      more intuitive and easier to deal with.  It is complemented | 
					
						
							|  |  |  |      again before being returned. */ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   bits = parse_symbolic_mode (list->word->word, ~um & 0777); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   if (bits == -1) | 
					
						
							|  |  |  |     return (-1); | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   um = ~bits & 0777; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   return (um); | 
					
						
							|  |  |  | } |