diff options
Diffstat (limited to 'unix/tclUnixFCmd.c')
-rw-r--r-- | unix/tclUnixFCmd.c | 230 |
1 files changed, 225 insertions, 5 deletions
diff --git a/unix/tclUnixFCmd.c b/unix/tclUnixFCmd.c index c8b35eb..f396d45 100644 --- a/unix/tclUnixFCmd.c +++ b/unix/tclUnixFCmd.c @@ -10,7 +10,7 @@ * See the file "license.terms" for information on usage and redistribution * of this file, and for a DISCLAIMER OF ALL WARRANTIES. * - * RCS: @(#) $Id: tclUnixFCmd.c,v 1.4 1999/04/16 00:48:04 stanton Exp $ + * RCS: @(#) $Id: tclUnixFCmd.c,v 1.5 2000/01/12 11:13:55 hobbs Exp $ * * Portions of this code were derived from NetBSD source code which has * the following copyright notice: @@ -88,7 +88,10 @@ static int SetOwnerAttribute _ANSI_ARGS_((Tcl_Interp *interp, static int SetPermissionsAttribute _ANSI_ARGS_(( Tcl_Interp *interp, int objIndex, CONST char *fileName, Tcl_Obj *attributePtr)); - +static int GetModeFromPermString _ANSI_ARGS_(( + Tcl_Interp *interp, char *modeStringPtr, + mode_t *modePtr)); + /* * Prototype for the TraverseUnixTree callback function. */ @@ -1340,16 +1343,44 @@ SetPermissionsAttribute(interp, objIndex, fileName, attributePtr) Tcl_Obj *attributePtr; /* The attribute to set. */ { long mode; + mode_t newMode; int result; CONST char *native; Tcl_DString ds; - if (Tcl_GetLongFromObj(interp, attributePtr, &mode) != TCL_OK) { - return TCL_ERROR; + /* + * First try if the string is a number + */ + if (Tcl_GetLongFromObj(NULL, attributePtr, &mode) == TCL_OK) { + newMode = (mode_t) (mode & 0x00007FFF); + } else { + struct stat buf; + char *modeStringPtr = Tcl_GetString(attributePtr); + + /* + * Try the forms "rwxrwxrwx" and "ugo=rwx" + * + * We get the current mode of the file, in order to allow for + * ug+-=rwx style chmod strings. + */ + result = TclStat(fileName, &buf); + if (result != 0) { + Tcl_AppendResult(interp, "could not read \"", fileName, "\": ", + Tcl_PosixError(interp), (char *) NULL); + return TCL_ERROR; + } + newMode = (mode_t) (buf.st_mode & 0x00007FFF); + + if (GetModeFromPermString(NULL, modeStringPtr, &newMode) != TCL_OK) { + Tcl_AppendStringsToObj(Tcl_GetObjResult(interp), + "unknown permission string format \"", + modeStringPtr, "\"", (char *) NULL); + return TCL_ERROR; + } } native = Tcl_UtfToExternalDString(NULL, fileName, -1, &ds); - result = chmod(native, (mode_t) mode); /* INTL: Native. */ + result = chmod(native, newMode); /* INTL: Native. */ Tcl_DStringFree(&ds); if (result != 0) { Tcl_AppendStringsToObj(Tcl_GetObjResult(interp), @@ -1359,6 +1390,7 @@ SetPermissionsAttribute(interp, objIndex, fileName, attributePtr) } return TCL_OK; } + /* *--------------------------------------------------------------------------- * @@ -1388,4 +1420,192 @@ TclpListVolumes(interp) Tcl_SetStringObj(resultPtr, "/", 1); return TCL_OK; } + +/* + *---------------------------------------------------------------------- + * + * GetModeFromPermString -- + * + * This procedure is invoked to process the "file permissions" + * Tcl command, to check for a "rwxrwxrwx" or "ugoa+-=rwxst" string. + * See the user documentation for details on what it does. + * + * Results: + * A standard Tcl result. + * + * Side effects: + * See the user documentation. + * + *---------------------------------------------------------------------- + */ + +static int +GetModeFromPermString(interp, modeStringPtr, modePtr) + Tcl_Interp *interp; /* The interp we are using for errors. */ + char *modeStringPtr; /* Permissions string */ + mode_t *modePtr; /* pointer to the mode value */ +{ + mode_t newMode; + mode_t oldMode; /* Storage for the value of the old mode + * (that is passed in), to allow for the + * chmod style manipulation */ + int i,n, who, op, what, op_found, who_found; + /* + * We start off checking for an "rwxrwxrwx" style permissions string + */ + if (strlen(modeStringPtr) != 9) { + goto chmodStyleCheck; + } + + newMode = 0; + for (i = 0; i < 9; i++) { + switch (*(modeStringPtr+i)) { + case 'r': + if ((i%3) != 0) { + goto chmodStyleCheck; + } + newMode |= (1<<(8-i)); + break; + case 'w': + if ((i%3) != 1) { + goto chmodStyleCheck; + } + newMode |= (1<<(8-i)); + break; + case 'x': + if ((i%3) != 2) { + goto chmodStyleCheck; + } + newMode |= (1<<(8-i)); + break; + case 's': + if (((i%3) != 2) || (i > 5)) { + goto chmodStyleCheck; + } + newMode |= (1<<(8-i)); + newMode |= (1<<(11-(i/3))); + break; + case 'S': + if (((i%3) != 2) || (i > 5)) { + goto chmodStyleCheck; + } + newMode |= (1<<(11-(i/3))); + break; + case 't': + if (i != 8) { + goto chmodStyleCheck; + } + newMode |= (1<<(8-i)); + newMode |= (1<<9); + break; + case 'T': + if (i != 8) { + goto chmodStyleCheck; + } + newMode |= (1<<9); + break; + case '-': + break; + default: + /* + * Oops, not what we thought it was, so go on + */ + goto chmodStyleCheck; + } + } + *modePtr = newMode; + return TCL_OK; + + chmodStyleCheck: + /* + * We now check for an "ugoa+-=rwxst" style permissions string + */ + + for (n = 0 ; *(modeStringPtr+n) != '\0' ; n = n + i) { + oldMode = *modePtr; + who = op = what = op_found = who_found = 0; + for (i = 0 ; *(modeStringPtr+n+i) != '\0' ; i++ ) { + if (!who_found) { + /* who */ + switch (*(modeStringPtr+n+i)) { + case 'u' : + who |= 0x9c0; + continue; + case 'g' : + who |= 0x438; + continue; + case 'o' : + who |= 0x207; + continue; + case 'a' : + who |= 0xfff; + continue; + } + } + who_found = 1; + if (who == 0) { + who = 0xfff; + } + if (!op_found) { + /* op */ + switch (*(modeStringPtr+n+i)) { + case '+' : + op = 1; + op_found = 1; + continue; + case '-' : + op = 2; + op_found = 1; + continue; + case '=' : + op = 3; + op_found = 1; + continue; + default : + return TCL_ERROR; + break; + } + } + /* what */ + switch (*(modeStringPtr+n+i)) { + case 'r' : + what |= 0x124; + continue; + case 'w' : + what |= 0x92; + continue; + case 'x' : + what |= 0x49; + continue; + case 's' : + what |= 0xc00; + continue; + case 't' : + what |= 0x200; + continue; + case ',' : + break; + default : + return TCL_ERROR; + break; + } + if (*(modeStringPtr+n+i) == ',') { + i++; + break; + } + } + switch (op) { + case 1 : + *modePtr = oldMode | (who & what); + continue; + case 2 : + *modePtr = oldMode & ~(who & what); + continue; + case 3 : + *modePtr = (oldMode & ~who) | (who & what); + continue; + } + } + return TCL_OK; +} |