diff options
-rwxr-xr-x | configure | 5 | ||||
-rw-r--r-- | sysif.c | 37 | ||||
-rw-r--r-- | txr.1 | 29 |
3 files changed, 59 insertions, 12 deletions
@@ -2278,13 +2278,14 @@ else printf "no\n" fi -printf "Checking for chmod ... " +printf "Checking for chmod/fchmod ... " cat > conftest.c <<! #include <sys/stat.h> int main(void) { - int r = chmod("a", S_IWUSR); + int r0 = chmod("a", S_IWUSR); + int r1 = fchmod(4, S_IWUSR); return 0; } ! @@ -451,17 +451,42 @@ static val mknod_wrap(val path, val mode, val dev) #if HAVE_CHMOD -static val chmod_wrap(val path, val mode) +static int get_fd(val stream, val self) { + val fd_in = if3(integerp(stream), stream, stream_get_prop(stream, fd_k)); + + if (stream && !fd_in) + uw_throwf(file_error_s, + lit("~a: stream ~s has no :fd property"), + self, stream, nao); + + if (!stream) + uw_throwf(file_error_s, + lit("~a: ~s isn't a stream object"), + self, stream, nao); + + return c_int(fd_in, self); +} + +static val chmod_wrap(val target, val mode) +{ + val self = lit("chmod"); cnum cmode = c_num(mode); - char *u8path = utf8_dup_to(c_str(path)); - int err = chmod(u8path, cmode); - free(u8path); + int err; + + if (stringp(target)) { + char *u8path = utf8_dup_to(c_str(target)); + err = chmod(u8path, cmode); + free(u8path); + } else { + int fd = get_fd(target, self); + err = fchmod(fd, cmode); + } if (err < 0) { int eno = errno; - uw_throwf(errno_to_file_error(eno), lit("chmod ~a #o~o: ~d/~s"), - path, mode, num(eno), string_utf8(strerror(eno)), nao); + uw_throwf(errno_to_file_error(eno), lit("~a ~a #o~o: ~d/~s"), + self, target, mode, num(eno), string_utf8(strerror(eno)), nao); } return t; @@ -57880,15 +57880,18 @@ from a combined device number. .coNP Function @ chmod .synb -.mets (chmod < path << mode ) +.mets (chmod < target << mode ) .syne .desc The .code chmod -function changes the permissions of the filesystem objects +function changes the permissions of the filesystem object specified by -.metn path . -It is a direct wrapper for the POSIX C library function of the same name. +.metn target . +It is implemented in terms of the POSIX functions +.code chmod +and +.codn fchmod . The permissions are specified by .metn mode , @@ -57903,6 +57906,24 @@ The function throws a exception if an error occurs, otherwise it returns .codn t . +The +.meta target +argument may be a character string, in which case it specifies a pathname in +the filesystem. In this case, the POSIX function +.code chmod +is invoked. + +The +.meta target +argument may also be an integer file descriptor, or a stream. In these two +cases, the POSIX +.code fchmod +function is invoked. For a stream +.metn target , +the integer file descriptor is retrieved from the stream using +.code fileno +function. + .TP* Example: .verb ;; Set permissions of foo.txt to "rw-r--r--" |