~ chicken-core (chicken-5) 6810aaac5be163fb3092f51ffc72a0debbd28459
commit 6810aaac5be163fb3092f51ffc72a0debbd28459 Author: Peter Bex <peter@more-magic.net> AuthorDate: Wed May 10 22:06:13 2017 +0200 Commit: Kooda <kooda@upyum.com> CommitDate: Mon Jun 5 23:08:30 2017 +0200 Replace file-modification-time setter with set-file-times! procedure This reduces several inconsistencies and any resulting confusion: - file-access-time and file-change-time have no associated setter - The setter for file-modification-time sets both mtime AND ctime - The getters all accept strings, ports and file descriptors; the setter only accepts a string. While at it, the new procedure also makes it possible to omit the timestamps (in which case the current time is assumed), supply only one (in which case the old behaviour stays: we set both timestamps to the supplied time) or both (in which case you can set either to a different value). If #f is supplied, the specific time is unchanged. This behaviour is maximally compatible with the "specify both or none" behaviour from SCSH's "set-file-times" procedure (note the missing bang though), and with MIT Scheme's "set-file-times!" procedure where passing in #f means to avoid modifying the corresponding time. Signed-off-by: Kooda <kooda@upyum.com> diff --git a/posix-common.scm b/posix-common.scm index e3e67393..3475dda3 100644 --- a/posix-common.scm +++ b/posix-common.scm @@ -257,22 +257,22 @@ EOF _stat_st_dev _stat_st_rdev _stat_st_blksize _stat_st_blocks) ) -(define file-modification-time - (getter-with-setter - (lambda (f) - (##sys#stat f #f #t 'file-modification-time) _stat_st_mtime) - (lambda (f t) - (##sys#check-exact-integer t 'set-file-modification-time) - (let ((r ((foreign-lambda int "set_file_mtime" c-string scheme-object) - f t))) - (when (fx< r 0) - (posix-error - #:file-error 'set-file-modification-time - "cannot set file modification-time" f t)))) - "(file-modification-time f)")) - +(define (file-modification-time f) (##sys#stat f #f #t 'file-modification-time) _stat_st_mtime) (define (file-access-time f) (##sys#stat f #f #t 'file-access-time) _stat_st_atime) (define (file-change-time f) (##sys#stat f #f #t 'file-change-time) _stat_st_ctime) + +(define (set-file-times! f . rest) + (let-optionals* rest ((atime (current-seconds)) (mtime atime)) + (when atime (##sys#check-exact-integer atime 'set-file-times!)) + (when mtime (##sys#check-exact-integer mtime 'set-file-times!)) + (let ((r ((foreign-lambda int "set_file_mtime" + c-string scheme-object scheme-object) + f atime mtime))) + (when (fx< r 0) + (apply posix-error + #:file-error + 'set-file-times! "cannot set file times" f rest))))) + (define (file-owner f) (##sys#stat f #f #t 'file-owner) _stat_st_uid) (define (file-permissions f) (##sys#stat f #f #t 'file-permissions) _stat_st_mode) (define (file-size f) (##sys#stat f #f #t 'file-size) _stat_st_size) diff --git a/posix.scm b/posix.scm index 63729d3f..0bb3818e 100644 --- a/posix.scm +++ b/posix.scm @@ -101,7 +101,8 @@ open/trunc open/write open/wronly perm/irgrp perm/iroth perm/irusr perm/irwxg perm/irwxo perm/irwxu perm/isgid perm/isuid perm/isvtx perm/iwgrp perm/iwoth perm/iwusr perm/ixgrp perm/ixoth perm/ixusr - port->fileno seek/cur seek/end seek/set set-file-position!) + port->fileno seek/cur seek/end seek/set set-file-position! + set-file-times!) (import chicken chicken.posix)) (module chicken.time.posix diff --git a/posixunix.scm b/posixunix.scm index 60f547b2..d4541a44 100644 --- a/posixunix.scm +++ b/posixunix.scm @@ -60,7 +60,7 @@ process-group-id process-run process-signal process-sleep process-wait read-symbolic-link regular-file? seconds->local-time seconds->string seconds->utc-time seek/cur seek/end seek/set set-alarm! - set-buffering-mode! set-root-directory! + set-buffering-mode! set-file-times! set-root-directory! set-signal-handler! set-signal-mask! signal-handler signal-mask signal-mask! signal-masked? signal-unmask! signal/abrt signal/alrm signal/break signal/chld signal/cont signal/fpe @@ -353,11 +353,26 @@ static int get_tty_size(int p, int *rows, int *cols) } #endif -static int set_file_mtime(char *filename, C_word tm) +static int set_file_mtime(char *filename, C_word atime, C_word mtime) { + struct stat sb; struct utimbuf tb; - tb.actime = tb.modtime = C_num_to_int(tm); + /* Only lstat if needed */ + if (atime == C_SCHEME_FALSE || mtime == C_SCHEME_FALSE) { + if (lstat(filename, &sb) == -1) return -1; + } + + if (atime == C_SCHEME_FALSE) { + tb.actime = sb.st_atime; + } else { + tb.actime = C_num_to_int(atime); + } + if (mtime == C_SCHEME_FALSE) { + tb.modtime = sb.st_mtime; + } else { + tb.modtime = C_num_to_int(mtime); + } return utime(filename, &tb); } diff --git a/posixwin.scm b/posixwin.scm index 23e2f369..9685b635 100644 --- a/posixwin.scm +++ b/posixwin.scm @@ -611,11 +611,26 @@ C_process(const char *app, const char *cmdlin, const char **env, return success; } -static int set_file_mtime(char *filename, C_word tm) +static int set_file_mtime(char *filename, C_word atime, C_word mtime) { + struct stat sb; struct _utimbuf tb; - tb.actime = tb.modtime = C_num_to_int(tm); + /* Only lstat if needed */ + if (atime == C_SCHEME_FALSE || mtime == C_SCHEME_FALSE) { + if (lstat(filename, &sb) == -1) return -1; + } + + if (atime == C_SCHEME_FALSE) { + tb.actime = sb.st_atime; + } else { + tb.actime = C_num_to_int(atime); + } + if (mtime == C_SCHEME_FALSE) { + tb.modtime = sb.st_mtime; + } else { + tb.modtime = C_num_to_int(mtime); + } return _utime(filename, &tb); } EOF @@ -655,7 +670,7 @@ EOF process-spawn process-wait read-symbolic-link regular-file? seconds->local-time seconds->string seconds->utc-time seek/cur seek/end seek/set set-alarm! set-buffering-mode! set-root-directory! - set-signal-handler! set-signal-mask! signal-handler + set-file-times! set-signal-handler! set-signal-mask! signal-handler signal-mask signal-mask! signal-masked? signal-unmask! signal/abrt signal/alrm signal/break signal/chld signal/cont signal/fpe signal/bus signal/hup signal/ill signal/int signal/io signal/kill diff --git a/types.db b/types.db index 004d531d..e7e2770d 100644 --- a/types.db +++ b/types.db @@ -2049,6 +2049,7 @@ (chicken.posix#set-alarm! (#(procedure #:clean #:enforce) chicken.posix#set-alarm! (integer) integer)) (chicken.posix#set-buffering-mode! (#(procedure #:clean #:enforce) chicken.posix#set-buffering-mode! (port symbol #!optional fixnum) undefined)) (chicken.posix#set-file-position! (#(procedure #:clean #:enforce) chicken.posix#set-file-position! ((or port fixnum) integer #!optional fixnum) undefined)) +(chicken.posix#set-file-times! (#(procedure #:clean #:enforce) chicken.posix#set-file-times! (string #!optional (or false integer) (or false integer)) undefined)) (chicken.posix#set-root-directory! (#(procedure #:clean #:enforce) chicken.posix#set-root-directory! (string) undefined)) (chicken.posix#set-signal-handler! (#(procedure #:clean #:enforce) chicken.posix#set-signal-handler! (fixnum (or false (procedure (fixnum) . *))) undefined)) (chicken.posix#set-signal-mask! (#(procedure #:clean #:enforce) chicken.posix#set-signal-mask! ((list-of fixnum)) undefined))Trap