Compare commits

...

19 Commits

9 changed files with 619 additions and 207 deletions

View File

@ -1,2 +1,5 @@
FROM debian:trixie FROM debian:trixie
RUN apt-get update && apt-get -y install make docker.io git RUN apt-get update && apt-get -y install build-essential libffi-dev docker.io git
RUN git clone https://github.com/ashinn/chibi-scheme.git --depth=1
RUN cd chibi-scheme && make && make install
RUN snow-chibi install --impls=chibi --always-yes retropikzel.test-r7rs

63
Jenkinsfile vendored
View File

@ -7,43 +7,60 @@ pipeline {
} }
} }
triggers {
GenericTrigger(
genericVariables: [
[key: 'ref', value: '$.ref']
],
causeString: 'Triggered on $ref',
printContributedVariables: true,
printPostContent: true,
silentResponse: false,
shouldNotFlatten: false,
regexpFilterText: '$ref',
regexpFilterExpression: 'refs/heads/' + BRANCH_NAME
)
}
options { options {
disableConcurrentBuilds() disableConcurrentBuilds()
buildDiscarder(logRotator(numToKeepStr: '10', artifactNumToKeepStr: '10')) buildDiscarder(logRotator(numToKeepStr: '10', artifactNumToKeepStr: '10'))
} }
parameters { environment {
string(name: 'R7RS_SCHEMES', defaultValue: 'chibi chicken gauche guile kawa mosh racket sagittarius stklos ypsilon', description: '') R7RS_SCHEMES="capyscheme chibi chicken gauche kawa mosh racket sagittarius stklos ypsilon"
string(name: 'R6RS_SCHEMES', defaultValue: 'chezscheme guile ikarus ironscheme mosh racket sagittarius ypsilon', description: '') R6RS_SCHEMES="chezscheme guile ikarus ironscheme mosh racket sagittarius ypsilon"
string(name: 'SRFIS', defaultValue: '106 170', description: '') SRFIS="170"
} }
stages { stages {
stage('Tests') { stage('R6RS Debian') {
stage('R6RS x86_64 Debian') { steps {
steps { script {
script { env.SRFIS.split().each { SRFI ->
params.SRFIS.split().each { SRFI -> env.R6RS_SCHEMES.split().each { SCHEME ->
params.R6RS_SCHEMES.split().each { SCHEME -> stage("${SCHEME} ${SRFI}") {
stage("${SCHEME} - ${SRFI}") { catchError(buildResult: 'SUCCESS', stageResult: 'FAILURE') {
catchError(buildResult: 'SUCCESS', stageResult: 'FAILURE') { sh "make SCHEME=${SCHEME} SRFI=${SRFI} RNRS=r6rs test-docker"
sh "timeout 600 make SCHEME=${SCHEME} SRFI=${SRFI} RNRS=r6rs run-test-docker"
}
} }
} }
} }
} }
} }
} }
stage('R7RS x86_64 Debian') { }
steps { stage('R7RS Debian') {
script { steps {
params.SRFIS.split().each { SRFI -> script {
params.R7RS_SCHEMES.split().each { SCHEME -> env.SRFIS.split().each { SRFI ->
stage("${SCHEME} - ${SRFI}") { env.R7RS_SCHEMES.split().each { SCHEME ->
catchError(buildResult: 'SUCCESS', stageResult: 'FAILURE') { stage("${SCHEME} ${SRFI}") {
sh "timeout 600 make SCHEME=${SCHEME} SRFI=${SRFI} RNRS=r6rs run-test-docker" catchError(buildResult: 'SUCCESS', stageResult: 'FAILURE') {
} sh "make SCHEME=${SCHEME} SRFI=${SRFI} RNRS=r7rs test-docker"
} }
} }
} }

View File

@ -13,19 +13,18 @@ TMPDIR=.tmp/${SCHEME}
PKG=srfi-${SRFI}-${VERSION}.tgz PKG=srfi-${SRFI}-${VERSION}.tgz
SFX=scm SFX=scm
AKKU_PACKAGES=
ifeq "${RNRS}" "r6rs" ifeq "${RNRS}" "r6rs"
SFX=sps SFX=sps
AKKU_PACKAGES="akku-r7rs"
endif endif
DOCKER_TAG=head DOCKER_TAG=head
ifeq "${SCHEME}" "chicken"
DOCKER_TAG=5
endif
all: package all: package
package: srfi/${SRFI}/LICENSE srfi/${SRFI}/VERSION package: srfi/${SRFI}/LICENSE srfi/${SRFI}/VERSION
echo "<pre>$$(cat srfi/${SRFI}/README.md)</pre>" > ${README} echo "<pre>$$(cat README.md)</pre>" > README.html
snow-chibi package \ snow-chibi package \
--version=${VERSION} \ --version=${VERSION} \
--authors=${AUTHOR} \ --authors=${AUTHOR} \
@ -36,23 +35,31 @@ package: srfi/${SRFI}/LICENSE srfi/${SRFI}/VERSION
install: install:
snow-chibi install --impls=${SCHEME} ${PKG} snow-chibi install --impls=${SCHEME} ${PKG}
testfiles: ${PKG}: package
testfiles: ${PKG}
rm -rf .tmp rm -rf .tmp
mkdir -p .tmp mkdir -p .tmp
cp ${PKG} .tmp cp ${PKG} .tmp
cp -r srfi .tmp/ cp -r srfi .tmp/
cat test-headers.${SFX} ${TESTFILE} | sed 's/SRFI/${SRFI}/' > .tmp/test.${SFX} cat test-headers.${SFX} ${TESTFILE} \
| sed 's/SRFI/${SRFI}/' > .tmp/test.${SFX}
cat ${TESTFILE} >> run-test.${SFX} cat ${TESTFILE} >> run-test.${SFX}
if [ "${RNRS}" = "r6rs" ]; then if [ -d ../foreign-c ]; then cp -r ../foreign-c/foreign .tmp/; fi; fi if [ "${RNRS}" = "r6rs" ]; \
then if [ -d ../foreign-c ]; \
then cp -r ../foreign-c/foreign .tmp/; \
fi; \
fi
test: testfiles package test: testfiles
cd .tmp && COMPILE_R7RS=${SCHEME} compile-r7rs -o test-program test.${SFX} cd .tmp && COMPILE_R7RS=${SCHEME} compile-r7rs -o test-program test.${SFX}
cd .tmp && ./test-program cd .tmp && ./test-program
test-docker: testfiles package test-docker: testfiles
cd .tmp && \ cd .tmp && \
DOCKER_TAG=${DOCKER_TAG} \ DOCKER_TAG=${DOCKER_TAG} \
SNOW_PACKAGES="srfi.64 srfi.60 srfi.145 srfi.180 retropikzel.mouth foreign.c ${PKG}" \ SNOW_PACKAGES="srfi.64 ${PKG}" \
AKKU_PACKAGES=${AKKU_PACKAGES} \
APT_PACKAGES="libcurl4-openssl-dev" \ APT_PACKAGES="libcurl4-openssl-dev" \
COMPILE_R7RS=${SCHEME} \ COMPILE_R7RS=${SCHEME} \
TEST_R7RS_DEBUG=1 \ TEST_R7RS_DEBUG=1 \

View File

@ -4,6 +4,7 @@
"string.h" "string.h"
"dirent.h" "dirent.h"
"sys/stat.h" "sys/stat.h"
"sys/statvfs.h"
"sys/types.h" "sys/types.h"
"unistd.h" "unistd.h"
"pwd.h" "pwd.h"
@ -21,6 +22,7 @@
(define-c-procedure c-lstat libc 'stat 'int '(pointer pointer)) (define-c-procedure c-lstat libc 'stat 'int '(pointer pointer))
(define-c-procedure c-open libc 'open 'int '(pointer int)) (define-c-procedure c-open libc 'open 'int '(pointer int))
(define-c-procedure c-opendir libc 'opendir 'pointer '(pointer)) (define-c-procedure c-opendir libc 'opendir 'pointer '(pointer))
(define-c-procedure c-dirfd libc 'dirfd 'int '(pointer))
(define-c-procedure c-readdir libc 'readdir 'pointer '(pointer)) (define-c-procedure c-readdir libc 'readdir 'pointer '(pointer))
(define-c-procedure c-close libc 'close 'int '(int)) (define-c-procedure c-close libc 'close 'int '(int))
(define-c-procedure c-closedir libc 'closedir 'int '(pointer)) (define-c-procedure c-closedir libc 'closedir 'int '(pointer))
@ -47,10 +49,50 @@
(define-c-procedure c-link libc 'link 'int '(pointer pointer)) (define-c-procedure c-link libc 'link 'int '(pointer pointer))
(define-c-procedure c-slink libc 'link 'int '(pointer pointer)) (define-c-procedure c-slink libc 'link 'int '(pointer pointer))
(define-c-procedure c-chown libc 'chown 'int '(pointer int int)) (define-c-procedure c-chown libc 'chown 'int '(pointer int int))
(define-c-procedure c-clock-gettime libc 'clock_gettime 'int '(int pointer))
(define-c-procedure c-nice libc 'nice 'int '(int))
(define-c-procedure c-umask libc 'umask 'uint '(int))
(define-c-procedure
c-utimensat libc 'utimensat 'int '(int pointer pointer int))
(define-c-procedure c-truncate libc 'truncate 'int '(pointer int))
(define-c-procedure c-statvfs libc 'statvfs 'int '(pointer pointer))
(define slash (cond-expand (windows "\\") (else "/"))) (define slash (cond-expand (windows "\\") (else "/")))
(define randomized? #f) (define randomized? #f)
(define (string-split str mark)
(let* ((str-l (string->list str))
(res (list))
(last-index 0)
(index 0)
(splitter
(lambda (c)
(cond ((char=? c mark)
(begin
(set! res
(append res
(list (string-copy str last-index index))))
(set! last-index (+ index 1))))
((equal? (length str-l) (+ index 1))
(set! res
(append res
(list (string-copy str
last-index
(+ index 1)))))))
(set! index (+ index 1)))))
(for-each splitter str-l)
res))
(define (string-char-replace replace-in replace-this replace-with)
(let ((result ""))
(string-for-each
(lambda (c)
(if (char=? c replace-this)
(set! result (string-append result replace-with))
(set! result (string-append result (string c)))))
replace-in)
result))
(define (random-to max) (define (random-to max)
(when (not randomized?) (when (not randomized?)
(c-srand (c-time (c-bytevector-null))) (c-srand (c-time (c-bytevector-null)))
@ -75,8 +117,22 @@
(random-to 128))))))))) (random-to 128)))))))))
(looper "" (random-to 128)))) (looper "" (random-to 128))))
(define-record-type file-info-record (define-record-type <file-info>
(make-file-info-record device inode mode nlinks uid gid rdev size blksize blocks atime mtime ctime fname/port follow?) (make-file-info device
inode
mode
nlinks
uid
gid
rdev
size
blksize
blocks
atime
mtime
ctime
fname/port
follow?)
file-info? file-info?
(device file-info:device) (device file-info:device)
(inode file-info:inode) (inode file-info:inode)
@ -95,42 +151,76 @@
(follow? file-info:follow?)) (follow? file-info:follow?))
(define (file-info-directory? file-info) (define (file-info-directory? file-info)
(let ((handle (c-open (string->c-bytevector (file-info:fname/port file-info)) 2))) (let ((handle (c-open (string->c-bytevector
(file-info:fname/port file-info)) 2)))
(cond ((> handle 0) (c-close handle) #f) (cond ((> handle 0) (c-close handle) #f)
(else #t)))) (else #t))))
(define-c-struct-type stat-struct
`((st_dev int)
(st_ino uint)
(st_mode uint)
(st_nlink int)
(st_uid uint)
(st_gid uint)
(st_rdev int)
(st_size int)
(st_blksize int)
(st_blocks int)
(st_atim.tv_sec long)
(st_atim.tv_nsec long)
(st_mtim.tv_sec long)
(st_mtim.tv_nsec long)
(st_ctim.tv_sec long)
(st_ctim.tv_nsec long)))
;;> The file-info procedure returns a file-info record containing useful
;;> information about a file. If the follow? flag is true the procedure will
;;> follow symlinks and report on the file to which they refer. If follow? is
;;> false the procedure checks the actual file itself, even if it's a symlink.
;;> The follow? flag is ignored if the file argument is a port.
(define (file-info fname/port follow?) (define (file-info fname/port follow?)
(when (port? fname/port) (when (port? fname/port)
(error "file-info implementation does not support ports as arguments")) (error "file-info implementation does not support ports as arguments"))
(let* ((fname-pointer (string->c-bytevector fname/port)) (let* ((fname-pointer (string->c-bytevector fname/port))
(stat-pointer (make-c-bytevector 256)) (stat-pointer (make-c-bytevector (c-type-size stat-struct)))
(result (if follow? (result (if follow?
(c-stat fname-pointer stat-pointer) (c-stat fname-pointer stat-pointer)
(c-lstat fname-pointer stat-pointer))) (c-lstat fname-pointer stat-pointer))))
(error-message "file-info error")
(error-pointer (string->c-bytevector error-message)))
(when (< result 0) (when (< result 0)
(c-perror error-pointer) (let* ((error-message "file-info error")
(error-pointer (string->c-bytevector error-message)))
(c-perror error-pointer)
(c-bytevector-free fname-pointer)
(c-bytevector-free stat-pointer)
(c-bytevector-free error-pointer)
(error error-message fname/port)))
(let ((fi (make-file-info
(c-bytevector-ref stat-pointer stat-struct 'st_dev)
(c-bytevector-ref stat-pointer stat-struct 'st_ino)
(c-bytevector-ref stat-pointer stat-struct 'st_mode)
(c-bytevector-ref stat-pointer stat-struct 'st_nlink)
(c-bytevector-ref stat-pointer stat-struct 'st_uid)
(c-bytevector-ref stat-pointer stat-struct 'st_gid)
(c-bytevector-ref stat-pointer stat-struct 'st_rdev)
(c-bytevector-ref stat-pointer stat-struct 'st_size)
(c-bytevector-ref stat-pointer stat-struct 'st_blksize)
(c-bytevector-ref stat-pointer stat-struct 'st_blocks)
(make-time time-utc
(c-bytevector-ref stat-pointer stat-struct 'st_atim.tv_sec)
(c-bytevector-ref stat-pointer stat-struct 'st_atim.tv_nsec))
(make-time time-utc
(c-bytevector-ref stat-pointer stat-struct 'st_mtim.tv_sec)
(c-bytevector-ref stat-pointer stat-struct 'st_mtim.tv_nsec))
(make-time time-utc
(c-bytevector-ref stat-pointer stat-struct 'st_ctim.tv_sec)
(c-bytevector-ref stat-pointer stat-struct 'st_ctim.tv_nsec))
fname/port
follow?)))
(c-bytevector-free fname-pointer) (c-bytevector-free fname-pointer)
(c-bytevector-free stat-pointer) (c-bytevector-free stat-pointer)
(c-bytevector-free error-pointer) fi)))
(error error-message fname/port)) ;;> The permission-bits for create-directory default to #o775 but are masked
(make-file-info-record #f ;(c-bytevector-u64-ref stat-pointer (* (c-type-size 'uint64) 0) (native-endianness)) ;;> by the current umask.
#f ;(c-bytevector-u64-ref stat-pointer (* (c-type-size 'uint64) 1) (native-endianness))
#f ;(c-bytevector-u64-ref stat-pointer (* (c-type-size 'uint64) 2) (native-endianness))
#f ;(c-bytevector-u64-ref stat-pointer (* (c-type-size 'uint64) 3) (native-endianness))
#f ;(c-bytevector-u64-ref stat-pointer (* (c-type-size 'uint64) 4) (native-endianness))
#f ;(c-bytevector-u64-ref stat-pointer (* (c-type-size 'uint64) 5) (native-endianness))
#f ;(c-bytevector-u64-ref stat-pointer (* (c-type-size 'uint64) 6) (native-endianness))
#f ;(c-bytevector-u64-ref stat-pointer (* (c-type-size 'uint64) 7) (native-endianness))
#f ;(c-bytevector-u64-ref stat-pointer (* (c-type-size 'uint64) 8) (native-endianness))
#f ;(c-bytevector-u64-ref stat-pointer (* (c-type-size 'uint64) 9) (native-endianness))
#f ;(c-bytevector-u64-ref stat-pointer (* (c-type-size 'uint64) 10) (native-endianness))
#f ;(c-bytevector-u64-ref stat-pointer (* (c-type-size 'uint64) 11) (native-endianness))
#f ;(c-bytevector-u64-ref stat-pointer (* (c-type-size 'uint64) 12) (native-endianness))
fname/port
follow?)))
(define create-directory (define create-directory
(lambda (fname . permission-bits) (lambda (fname . permission-bits)
(let* ((fname-pointer (string->c-bytevector fname)) (let* ((fname-pointer (string->c-bytevector fname))
@ -150,6 +240,8 @@
(error error-message)) (error error-message))
(c-bytevector-free error-pointer)))) (c-bytevector-free error-pointer))))
;;> The permission-bits for create-directory default to #o664, but are masked
;;> by the current umask.
(define (create-fifo fname . permission-bits) (define (create-fifo fname . permission-bits)
(let* ((fname-pointer (string->c-bytevector fname)) (let* ((fname-pointer (string->c-bytevector fname))
(mode (if (null? permission-bits) (mode (if (null? permission-bits)
@ -197,28 +289,115 @@
(c-bytevector-free buffer) (c-bytevector-free buffer)
name))))) name)))))
;;> Return the filename referenced by the symlink fname.
(define (read-symlink fname) (internal-read-symlink fname 128)) (define (read-symlink fname) (internal-read-symlink fname 128))
;;> If you override an existing object, then old-fname and new-fname must
;;> type-match — either both directories, or both non-directories.
;;> This is required by the semantics of POSIX rename().
;;>
;;> Calling rename-file on a symbolic link will rename the symbolic link,
;;> not the file it refers to.
;;>
;;> Remark: There is an unfortunate atomicity problem with the rename-file
;;> procedure: if you create file new-fname sometime between rename-file's
;;> existence check and the actual rename operation, your file will be
;;> clobbered with old-fname. There is no way to prevent this problem; at
;;> least it is highly unlikely to occur in practice.
(define (rename-file old-fname new-fname) (define (rename-file old-fname new-fname)
(c-rename (string->c-bytevector old-fname) (c-rename (string->c-bytevector old-fname)
(string->c-bytevector new-fname))) (string->c-bytevector new-fname)))
;;> This procedure deletes directories from the file system. An error is
;;> signaled if fname is not a directory or is not empty.
(define (delete-directory fname) (define (delete-directory fname)
(let* ((fname-pointer (string->c-bytevector fname)) (let* ((fname-pointer (string->c-bytevector fname))
(result (c-rmdir fname-pointer)) (result (c-rmdir fname-pointer)))
(error-message "delete-directory error")
(error-pointer (string->c-bytevector error-message)))
(c-bytevector-free fname-pointer) (c-bytevector-free fname-pointer)
(when (< result 0) (when (< result 0)
(c-perror error-pointer) (let* ((error-message "delete-directory error")
(c-bytevector-free error-pointer) (error-pointer (string->c-bytevector error-message)))
(error error-message)))) (c-perror error-pointer)
(c-bytevector-free error-pointer)
(error error-message)))))
;;> This procedure sets the owner and group of a file specified by supplying
;;> the filename. If the uid argument is the constant owner/unchanged, the
;;> owner is not changed; if the gid argument is the constant group/unchanged,
;;> the group is not changed. Setting file ownership usually requires root
;;> privileges. This procedure follows symlinks and changes the files to which
;;> they refer.
(define (set-file-owner fname uid gid) (define (set-file-owner fname uid gid)
(let ((fname-pointer (string->c-bytevector fname))) (let ((fname-pointer (string->c-bytevector fname)))
(c-chown fname-pointer uid gid) (c-chown fname-pointer uid gid)
(c-bytevector-free fname-pointer))) (c-bytevector-free fname-pointer)))
(define-c-array-type timespec-array 'long)
;;> \procedure{(set-file-times fname [access-time-object modify-time-object])}
;;> This procedure sets the access and modified times for the file fname to
;;> the supplied time object values. It is an error if they are not of type
;;> time-utc. If neither time argument is supplied, they are both taken to be
;;> the current time. The constants time/now and time/unchanged are bound to
;;> values used to specify the current time and an unchanged time
;;> respectively. It is an error if exactly one time is provided. This
;;> procedure will follow symlinks and set the times of the file to which it
;;> refers. If the procedure completes successfully, the file's time of last
;;> status-change (ctime) is set to the current time.
(define (set-file-times fname . args)
(when (and (not (= (length args) 0))
(not (= (length args) 2)))
(error
(string-append "set-file-times error: "
"It is an error if exactly one time is provided")))
(let* ((current-time (posix-time))
(access-time-object (if (null? args)
current-time
(car args)))
(modify-time-object (if (or (null? args)
(< (length args) 2))
current-time
(cadr args)))
(fname-cbv (string->c-bytevector fname))
(timespecs-cbv (make-c-bytevector (c-type-size* 'long 4)))
(current-dir-cbv (string->c-bytevector (current-directory)))
(current-dir-stream (c-opendir current-dir-cbv))
(current-dir-fd (c-dirfd current-dir-stream)))
(c-bytevector-set!
timespecs-cbv timespec-array 0 (time-second access-time-object))
(c-bytevector-set!
timespecs-cbv timespec-array 1 (time-nanosecond access-time-object))
(c-bytevector-set!
timespecs-cbv timespec-array 2 (time-second modify-time-object))
(c-bytevector-set!
timespecs-cbv timespec-array 3 (time-nanosecond modify-time-object))
(let ((result (c-utimensat current-dir-fd fname-cbv timespecs-cbv 0)))
(c-bytevector-free fname-cbv)
(c-bytevector-free timespecs-cbv)
(c-bytevector-free current-dir-cbv)
(c-bytevector-free current-dir-stream)
(when (< result 0)
(let* ((error-message "set-file-times error")
(error-pointer (string->c-bytevector error-message)))
(c-perror error-pointer)
(c-bytevector-free error-pointer)
(error error-message))))))
;;> The specified file is truncated to len bytes in length.
(define (truncate-file fname/port len)
(when (not (exact-integer? len))
(error "truncate-file error: len must be exact-integer"))
(when (not (string? fname/port))
(error "truncate-file error: ports not supported yet"))
(let* ((fname/port-cbv (string->c-bytevector fname/port))
(result (c-truncate fname/port-cbv len)))
(c-bytevector-free fname/port-cbv)
(when (< result 0)
(let* ((error-message "truncate-file error")
(error-pointer (string->c-bytevector error-message)))
(c-perror error-pointer)
(c-bytevector-free error-pointer)
(error error-message)))))
(define (pointer-string-read pointer offset) (define (pointer-string-read pointer offset)
(letrec* ((looper (lambda (c index result) (letrec* ((looper (lambda (c index result)
(if (char=? c #\null) (if (char=? c #\null)
@ -233,6 +412,26 @@
; struct dirent d_name offset on linux ; struct dirent d_name offset on linux
(define d-name-offset 19) (define d-name-offset 19)
;;> Return a list of filenames in directory dir. The dotfiles? flag
;;> (default #f) causes files beginning with . to be included in the list.
;;> Regardless of the value of dotfiles?, the two files . and .. are never
;;> returned.
;;> The directory dir is not prepended to each filename in the result list.
;;> That is,
;;>
;;> (directory-files "/etc")
;;>
;;>returns
;;>
;;> ("chown" "exports" "fstab" ...)
;;>
;;>not
;;>
;;> ("/etc/chown" "/etc/exports" "/etc/fstab" ...)
;;>
;;> To use the filenames in the returned list, the programmer can either
;;> manually prepend the directory, or change to the directory before using
;;> the filenames.
(define directory-files (define directory-files
(lambda (dir . dotfiles?) (lambda (dir . dotfiles?)
(letrec* ((include-dotfiles? (if (null? dotfiles?) #f (car dotfiles?))) (letrec* ((include-dotfiles? (if (null? dotfiles?) #f (car dotfiles?)))
@ -268,15 +467,9 @@
(c-closedir directory-pointer) (c-closedir directory-pointer)
files)))) files))))
(define real-path ;;> This procedure sets the mode bits of a file specified by supplying the
(lambda (path) ;;> filename. This procedure follows symlinks and changes the files to which
(let* ((path-pointer (string->c-bytevector path)) ;;> they refer.
(real-path-pointer (c-realpath path-pointer (c-bytevector-null)))
(real-path (string-copy (c-bytevector->string real-path-pointer))))
(c-bytevector-free path-pointer)
(c-bytevector-free real-path-pointer)
real-path)))
(define (set-file-mode path mode) (define (set-file-mode path mode)
(c-chmod (string->c-bytevector path) (c-chmod (string->c-bytevector path)
(string->number (string-append "#o" (number->string mode))))) (string->number (string-append "#o" (number->string mode)))))
@ -287,12 +480,24 @@
(handle directory:handle) (handle directory:handle)
(dot-files? directory:dot-files?)) (dot-files? directory:dot-files?))
;;> Opens the directory with the specified pathname for reading, returning an
;;> opaque directory object.
;;> The dot-files? argument controls whether filenames beginning with "." are
;;> returned. If it is #f, which is the default, they are not. The filenames
;;> . and .. are never returned.
(define (open-directory path . dot-files?) (define (open-directory path . dot-files?)
(make-directory (c-opendir (string->c-bytevector path)) (make-directory (c-opendir (string->c-bytevector path))
(if (null? dot-files?) (if (null? dot-files?)
#f #f
(car dot-files?)))) (car dot-files?))))
;;> Returns the name of the next available file, or the end-of-file object if
;;> there are no more files.
;;> The dot-files? argument controls whether filenames beginning with "." are
;;> returned. If it is #f, which is the default, they are not. The filenames
;;> . and .. are never returned.
(define (read-directory directory-object) (define (read-directory directory-object)
(let ((directory-entity (c-readdir (directory:handle directory-object)))) (let ((directory-entity (c-readdir (directory:handle directory-object))))
(if (c-bytevector-null? directory-entity) (if (c-bytevector-null? directory-entity)
@ -308,9 +513,26 @@
(read-directory directory-object)) (read-directory directory-object))
(else name)))))) (else name))))))
;;> Closes a directory object.
(define (close-directory directory-object) (define (close-directory directory-object)
(c-closedir (directory:handle directory-object))) (c-closedir (directory:handle directory-object)))
;;> Returns an absolute pathname derived from pathname that names the same
;;> file and whose resolution does not involve dot (.), dot-dot (..), or
;;> symlinks.
(define real-path
(lambda (path)
(let* ((path-pointer (string->c-bytevector path))
(real-path-pointer (c-realpath path-pointer (c-bytevector-null)))
(real-path (string-copy (c-bytevector->string real-path-pointer))))
(c-bytevector-free path-pointer)
(c-bytevector-free real-path-pointer)
real-path)))
;;> SRFI 39 or R7RS parameter that returns a string when invoked. Its initial
;;> value is the value of the environment variable TMPDIR concatenated with
;;> "/pid" if TMPDIR is set and to "/tmp/pid" otherwise, where pid is the id
;;> of the current process.
(define temp-file-prefix (define temp-file-prefix
(make-parameter (make-parameter
(if (get-environment-variable "TMPDIR") (if (get-environment-variable "TMPDIR")
@ -322,14 +544,34 @@
slash slash
(number->string (c-getpid)))))) (number->string (c-getpid))))))
;;> Creates a new temporary file and returns its name. The optional argument
;;> specifies the filename prefix to use, and defaults to the result of
;;> invoking temp-file-prefix. The procedure generates a sequence of filenames
;;> that have prefix as a common prefix, looking for a filename that doesn't
;;> already exist in the file system. When it finds one, it creates it with
;;> permission #o600 and returns the filename. (The file permission can be
;;> changed to a more permissive permission with set-file-mode after being
;;> created.)
;;> This file is guaranteed to be brand new. No other process will have it
;;> open. This procedure does not simply return a filename that is very likely
;;> to be unused. It returns a filename that definitely did not exist at the
;;> moment create-temp-file created it.
;;> It is not necessary for the process's pid to be a part of the filename
;;> for the uniqueness guarantees to hold. The pid component of the default
;;> prefix simply serves to scatter the name searches into sparse regions, so
;;> that collisions are less likely to occur. This speeds things up, but does
;;> not affect correctness.
(define create-temp-file (define create-temp-file
(lambda prefix (lambda prefix
(let* ((tmpdir (cond-expand (let* ((tmpdir (cond-expand
(windows (get-environment-variable "TMP")) (windows (get-environment-variable "TMP"))
(else "/tmp"))) (else "/tmp")))
(real-prefix (if (null? prefix) (real-prefix
(string-append tmpdir slash (number->string (c-getpid))) (if (null? prefix)
(car prefix))) (string-append tmpdir slash (number->string (c-getpid)))
(car prefix)))
(path (string-append real-prefix "-" (random-string 6)))) (path (string-append real-prefix "-" (random-string 6))))
(if (file-exists? path) (if (file-exists? path)
(create-temp-file real-prefix) (create-temp-file real-prefix)
@ -338,15 +580,79 @@
(set-file-mode path 600) (set-file-mode path 600)
path))))) path)))))
;;> This procedure can be used to perform certain atomic transactions on the
;;> file system involving filenames. Some examples:
;;>
;;> Linking a file to a fresh backup temp name.
;;> Creating and opening an unused, secure temp file.
;;> Creating an unused temporary directory.
;;>
;;> This procedure uses prefix to generate a series of trial filenames. Prefix
;;> is a string, and defaults to the value of invoking temp-file-prefix. File
;;> names are generated by concatenating prefix with a varying string.
;;>
;;> The maker procedure is called serially on each filename generated. It must
;;> return at least one value; it may return multiple values. If the first
;;> return value is #f or if maker signals an exception indicating that the
;;> file exists, call-with-temporary-filename will loop, generating a new
;;> filename and calling maker again. If the first return value is true, the
;;> loop is terminated, returning whatever value(s) maker returned.
;;>
;;> After a number of unsuccessful trials, call-with-temporary-filename may
;;> give up, in which case an exception is signaled or propagated.
;;>
;;> To rename a file to a temporary name:
;;>
;;> (call-with-temporary-filename
;;> (lambda (backup)
;;> (create-hard-link old-file backup)
;;> backup)
;;> ".temp.") ; Keep link in current working directory
;;> (delete-file old-file)
;;>
;;> Recall that this SRFI reports procedure failure by signaling an error.
;;> This is critical for this example — the programmer can assume that if the
;;> call-with-temporary-filename call returns, it returns successfully. So the
;;> following delete-file call can be reliably invoked, safe in the knowledge
;;> that the backup link has definitely been established.
;;>
;;> To create a unique temporary directory:
;;>
;;> (call-with-temporary-filename
;;> (lambda (dir)
;;> (create-directory dir)
;;> dir)
;;> "/tmp/tempdir.")
;;>
;;> Similar operations can be used to generate unique fifos, or to return
;;> values other than the new filename (for example, an open port).
(define (call-with-temporary-filename maker . prefix) (define (call-with-temporary-filename maker . prefix)
(let* ((tmpdir (cond-expand (windows (get-environment-variable "TMP")) (let* ((tmpdir (cond-expand (windows (get-environment-variable "TMP"))
(else "/tmp"))) (else "/tmp")))
(real-prefix (if (null? prefix) (real-prefix (if (null? prefix)
(string-append tmpdir slash (number->string (c-getpid))) (string-append tmpdir
slash
(number->string (c-getpid)))
(car prefix))) (car prefix)))
(path (string-append real-prefix "-" (random-string 6)))) (path (string-append real-prefix "-" (random-string 6))))
(apply maker (list path)))) (apply maker (list path))))
;;> Returns the current file protection mask, or umask, as an exact integer.
;;> Whenever a file is created, the specified or default permissions are
;;> bitwise-anded with the complement of the umask before they are used.
(define (umask)
(let ((mask (c-umask 0)))
(c-umask mask)
mask))
;;> Sets the file protection mask to the exact integer umask and returns an
;;> unspecified value.
(define (set-umask! umask)
(c-umask umask))
;;> Returns the current directory as a string containing an absolute pathname.
;;> Whenever a file is referenced with a relative path, it is interpreted as
;;> relative to this directory.
(define (current-directory) (define (current-directory)
(let* ((path-pointer (make-c-bytevector 1024)) (let* ((path-pointer (make-c-bytevector 1024))
(path (begin (path (begin
@ -355,23 +661,34 @@
(c-bytevector-free path-pointer) (c-bytevector-free path-pointer)
path)) path))
;;> Sets the current directory to new-directory and returns an unspecified
;;> value.
(define (set-current-directory! path) (define (set-current-directory! path)
(c-chdir (string->c-bytevector path))) (c-chdir (string->c-bytevector path)))
(define (pid) ;;> Retrieves the process id for the current process.
(c-getpid)) (define (pid) (c-getpid))
(define (user-uid) ;;> Increments the niceness of the current process by delta. The lower the
(c-getuid)) ;;> niceness value is, the more the process is favored during scheduling.
;;> If delta is not specified, the increment is 1.
(define (user-gid) ;;> Real-time processes are not affected by nice.
(c-getgid)) (define nice
(lambda args
(define (user-effective-uid) (let ((result (if (null? args) (c-nice 1) (c-nice (car args)))))
(c-geteuid)) (when (< result 0)
(let* ((error-message "nice error")
(define (user-effective-gid) (error-pointer (string->c-bytevector error-message)))
(c-getegid)) (c-perror error-pointer)
(c-bytevector-free timespec)
(c-bytevector-free error-pointer)
(error error-message)))
result)))
(define (user-uid) (c-getuid))
(define (user-gid) (c-getgid))
(define (user-effective-uid) (c-geteuid))
(define (user-effective-gid) (c-getegid))
(define (groups-loop max-count count groups-pointer result) (define (groups-loop max-count count groups-pointer result)
(if (>= count max-count) (if (>= count max-count)
@ -394,13 +711,70 @@
(define-record-type <user-info> (define-record-type <user-info>
(make-user-info name uid gid home-dir shell full-name) (make-user-info name uid gid home-dir shell full-name)
user-info? user-info?
(name user-info:name) (name internal-user-info:name)
(uid user-info:uid) (uid internal-user-info:uid)
(gid user-info:gid) (gid internal-user-info:gid)
(home-dir user-info:home-dir) (home-dir internal-user-info:home-dir)
(shell user-info:shell) (shell internal-user-info:shell)
(full-name user-info:full-name)) (full-name internal-user-info:full-name))
;;> Returns the user name stored in user-info respectively.
;;> An implementation returns #f for any unavailable items.
(define (user-info:name user-info) (internal-user-info:name user-info))
;;> Returns the user uid stored in user-info respectively.
;;> An implementation returns #f for any unavailable items.
(define (user-info:uid user-info) (internal-user-info:uid user-info))
;;> Returns the user gid stored in user-info respectively.
;;> An implementation returns #f for any unavailable items.
(define (user-info:gid user-info) (internal-user-info:gid user-info))
;;> Returns the user home directory stored in user-info respectively.
;;> An implementation returns #f for any unavailable items.
(define (user-info:home-dir user-info) (internal-user-info:home-dir user-info))
;;> Returns the shell path stored in user-info respectively.
;;> An implementation returns #f for any unavailable items.
(define (user-info:shell user-info) (internal-user-info:shell user-info))
;;> Returns the contents of the pw_gecos field stored in user-info. Although
;;> this field is not part of POSIX, it has been part of all Unix variants
;;> since at least the Sixth Edition of Research Unix. It normally contains
;;> the user's full name, but may contain additional system-specific
;;> information; on Windows, it contains exactly the full name.
(define (user-info:full-name user-info)
(internal-user-info:full-name user-info))
;;> Returns a parsed and expanded version of the raw string returned by
;;> user-info:full-name. The raw value is split on commas, creating a list of
;;> strings to be returned. All ampersands in the first element of the list
;;> are replaced by user-info:name, which is capitalized if it starts with an
;;> ASCII lowercase letter.
;;> However, on Windows the implementation is completely different:
;;> user-info:parsed-full-name returns a list with a single element, the
;;> result of user-info:full-name. No comma splitting or ampersand
;;> substitution is performed.
;;> The meaning of the first element of the returned list is the user's full
;;> name on all known systems. The remaining elements have varying meaning.
;;> For example, on BSD systems, the second through fourth elements are the
;;> user's work location, the user's work phone number, and the user's home
;;> phone number, respectively. On Cygwin, the second element is the Windows
;;> SID corresponding to this user; further elements depend on Cygwin-specific
;;> entries in the /etc/nsswitch.conf file.
(define (user-info:parsed-full-name user-info)
(let* ((parsed-list
(string-split (internal-user-info:full-name user-info) #\,))
(first
(string-append
(string (char-upcase (string-ref (car parsed-list) 0)))
(string-copy (car parsed-list) 1))))
(cons (string-char-replace first #\& (user-info:name user-info))
(cdr parsed-list))))
;;> Return a user-info record giving the recorded information for a particular
;;> user. The uid/name argument is either an exact integer user id or a string
;;> user name. If uid/name does not identify an existing user, #f is returned;
;;> this does not constitute an error situation, and callers must be prepared
;;> to handle it.
(define (user-info uid/name) (define (user-info uid/name)
(let ((password-struct (if (number? uid/name) (let ((password-struct (if (number? uid/name)
(c-getpwuid uid/name) (c-getpwuid uid/name)
@ -445,8 +819,76 @@
'int 'int
(* (c-type-size 'pointer) 2))))) (* (c-type-size 'pointer) 2)))))
;;> Change the value of the environment variable name to be value. Both name
;;> and value are strings. If name is not defined at the time of call, a new
;;> variable is added; if name is defined, its old value is discarded and
;;> replaced by value. If name or value are invalid according to the operating
;;> system, an exception is signaled. Mutating name or value after the call
;;> must not change the name or value of the environment variable.
(define (set-environment-variable! name value) (define (set-environment-variable! name value)
(when (not (string? name))
(error "set-environment-variable! error: name must be string"))
(when (not (string? value))
(error "set-environment-variable! error: value must be string"))
(c-setenv (string->c-bytevector name) (string->c-bytevector value) 1)) (c-setenv (string->c-bytevector name) (string->c-bytevector value) 1))
;;> Remove the environment variable name such that a subsequent
;;> (get-environment-variable name) would return #f. If the variable cannot
;;> be removed, an exception is signaled. If name does not currently have a
;;> value, the call silently succeeds.
(define (delete-environment-variable! name) (define (delete-environment-variable! name)
(when (not (string? name))
(error "delete-environment-variable! error: Name must be string"))
(c-unsetenv (string->c-bytevector name))) (c-unsetenv (string->c-bytevector name)))
(define CLOCK_REALTIME 0)
(define CLOCK_MONOTONIC 1)
(define tv_sec-type 'long)
(define tv_nsec-type 'long)
(define timespec (make-c-bytevector (c-type-size+ tv_sec-type tv_nsec-type)))
;;> Returns the current time as a time object of type time-utc, which
;;> represents the time since the POSIX epoch (midnight January 1, 1970
;;> Universal Time), excluding leap seconds. It uses the POSIX CLOCK_REALTIME
;;> clock.
(define (posix-time)
(let* ((result (c-clock-gettime CLOCK_REALTIME timespec)))
(cond
((< result 0)
(let* ((error-message "posix-time error")
(error-pointer (string->c-bytevector error-message)))
(c-perror error-pointer)
(c-bytevector-free timespec)
(c-bytevector-free error-pointer)
(error error-message)))
(else
(make-time time-utc
(c-bytevector-ref timespec
tv_nsec-type
(c-type-size tv_sec-type))
(c-bytevector-ref timespec tv_sec-type 0))))))
;;> Returns the current time as a time object of type time-monotonic, which
;;> represents the time since an arbitrary epoch. This epoch is arbitrary,
;;> but cannot change after the current program begins to run. It is
;;> guaranteed that a call to monotonic-time cannot return a time earlier
;;> than a previous call to monotonic-time. This is not guaranteed for
;;> posix-time because the system's POSIX clock is sometimes turned backward
;;> to correct local clock drift. It uses the POSIX CLOCK_MONOTONIC clock.
(define (monotonic-time)
(let* ((result (c-clock-gettime CLOCK_MONOTONIC timespec)))
(cond
((< result 0)
(let* ((error-message "posix-time error")
(error-pointer (string->c-bytevector error-message)))
(c-perror error-pointer)
(c-bytevector-free timespec)
(c-bytevector-free error-pointer)
(error error-message)))
(else
(make-time time-utc
(c-bytevector-ref timespec
tv_nsec-type
(c-type-size tv_sec-type))
(c-bytevector-ref timespec tv_sec-type 0))))))

View File

@ -5,7 +5,8 @@
(scheme write) (scheme write)
(scheme file) (scheme file)
(scheme process-context) (scheme process-context)
(foreign c)) (foreign c)
(srfi 19))
(export ;posix-error? (export ;posix-error?
;posix-error-name ;posix-error-name
;posix-error-message ;posix-error-message
@ -19,23 +20,23 @@
rename-file rename-file
delete-directory delete-directory
set-file-owner set-file-owner
;set-file-times set-file-times
;truncate-file truncate-file
file-info file-info
file-info? file-info?
;file-info:device file-info:device
;file-info:inode file-info:inode
;file-info:mode file-info:mode
;file-info:nlinks file-info:nlinks
;file-info:uid file-info:uid
;file-info:gid file-info:gid
;file-info:rdev file-info:rdev
;file-info:size file-info:size
;file-info:blksize file-info:blksize
;file-info:blocks file-info:blocks
;file-info:atime file-info:atime
;file-info:mtime file-info:mtime
;file-info:ctime file-info:ctime
file-info-directory? file-info-directory?
;file-info-fifo? ;file-info-fifo?
;file-info-symlink? ;file-info-symlink?
@ -48,17 +49,17 @@
open-directory open-directory
read-directory read-directory
close-directory close-directory
;real-path real-path
;file-space ;file-space
temp-file-prefix temp-file-prefix
create-temp-file create-temp-file
call-with-temporary-filename call-with-temporary-filename
;umask umask
;set-umask! set-umask!
current-directory current-directory
set-current-directory! set-current-directory!
pid pid
;nice nice
user-uid user-uid
user-gid user-gid
user-effective-uid user-effective-uid
@ -72,13 +73,13 @@
user-info:home-dir user-info:home-dir
user-info:shell user-info:shell
user-info:full-name user-info:full-name
;user-info:parsed-full-name user-info:parsed-full-name
group-info group-info
group-info? group-info?
group-info:name group-info:name
group-info:gid group-info:gid
;posix-time posix-time
;monotonic-time monotonic-time
set-environment-variable! set-environment-variable!
delete-environment-variable! delete-environment-variable!
;terminal? ;terminal?

View File

@ -1,83 +1,5 @@
Implementation of [SRFI 170](https://srfi.schemers.org/srfi-170/srfi-170.html) Implementation of SRFI 170 - POSIX API using (foreign c)
POSIX API using (foreign c)].
Currently only supports Linux. Currently only supports Linux.
Uncommented things here are implemented. Not everything is implemented yet, see 170.sld for commented out parts
;;;;posix-error?
;;;;posix-error-name
;;;;posix-error-message
;;;;open-file
;;;;fd->port
create-directory
;;;;create-fifo
create-hard-link
create-symlink
;;;;read-symlink
;;;;rename-file
delete-directory
;;;;set-file-owner
;;;;set-file-times
;;;;truncate-file
file-info
file-info?
;;;;file-info:device
;;;;file-info:inode
;;;;file-info:mode
;;;;file-info:nlinks
;;;;file-info:uid
;;;;file-info:gid
;;;;file-info:rdev
;;;;file-info:size
;;;;file-info:blksize
;;;;file-info:blocks
;;;;file-info:atime
;;;;file-info:mtime
;;;;file-info:ctime
file-info-directory?
;;;;file-info-fifo?
;;;;file-info-symlink?
;;;;file-info-regular?
;;;;file-info-socket?
;;;;file-info-device?
set-file-mode
directory-files
;;;;make-directory-files-generator
open-directory
read-directory
close-directory
real-path
;;;;file-space
temp-file-prefix
create-temp-file
call-with-temporary-filename
;;;;umask
;;;;set-umask!
current-directory
set-current-directory!
pid
;;;;nice
user-uid
user-gid
user-effective-uid
user-effective-gid
user-supplementary-gids
user-info
user-info?
user-info:name
user-info:uid
user-info:gid
user-info:home-dir
user-info:shell
user-info:full-name
;;;;user-info:parsed-full-name
group-info
group-info?
group-info:name
group-info:gid
;;;;posix-time
;;;;monotonic-time
set-environment-variable!
delete-environment-variable!
;;;;terminal?

View File

@ -1 +1 @@
0.1.4 0.2.0

View File

@ -1,12 +1,30 @@
(test-begin "srfi-170") (test-begin "srfi-170")
;(display (real-path "Makefile")) (write (posix-time))
;(newline) (newline)
;(exit 0) (write (monotonic-time))
(newline)
(define niceness (nice 1))
(test-assert (number? niceness))
(test-assert (> niceness 0))
(define fi (file-info "/tmp" #f))
(write fi)
(newline)
(write (file-info:mode fi))
(newline)
#|
(define tmp-dir "/tmp/foreign-c-srfi-170") (define tmp-dir "/tmp/foreign-c-srfi-170")
(when (file-exists? tmp-dir) (delete-directory tmp-dir)) (for-each
(lambda (file)
(delete-file (string-append tmp-dir "/" file)))
(directory-files tmp-dir #t))
(when (file-exists? tmp-dir)
(delete-directory tmp-dir))
(create-directory tmp-dir) (create-directory tmp-dir)
(define tmp-file (string-append tmp-dir "/test.txt")) (define tmp-file (string-append tmp-dir "/test.txt"))
@ -145,5 +163,7 @@
(display "file-info-directory? on file: ") (display "file-info-directory? on file: ")
(write (file-info-directory? tmp-file-info)) (write (file-info-directory? tmp-file-info))
(newline) (newline)
|#
(test-begin "srfi-170")
(test-end "srfi-170")

View File

@ -1,4 +1,4 @@
(import (rnrs) (import (rnrs)
(srfi :64) (srfi :64)
(foreign c) (foreign c)
(srfi SRFI)) (srfi :SRFI))