serialization: 'read-byte-string' makes a single read(2) call.

On "guix build libreoffice -nd", this reduces the number of read(2)
system calls from 10,434 to 8092.

* guix/serialization.scm (sub-bytevector): New procedure.
(read-byte-string): Make a single 'get-bytevector-n*' call and use
'sub-bytevector'.
This commit is contained in:
Ludovic Courtès 2020-01-24 22:39:47 +01:00
parent 09238d618a
commit 01e5d63c87
No known key found for this signature in database
GPG key ID: 090B11993D9AEBB5

View file

@ -27,6 +27,7 @@ (define-module (guix serialization)
#:use-module ((ice-9 rdelim) #:prefix rdelim:) #:use-module ((ice-9 rdelim) #:prefix rdelim:)
#:use-module (ice-9 match) #:use-module (ice-9 match)
#:use-module (ice-9 ftw) #:use-module (ice-9 ftw)
#:use-module (system foreign)
#:export (write-int read-int #:export (write-int read-int
write-long-long read-long-long write-long-long read-long-long
write-padding write-padding
@ -80,6 +81,17 @@ (define (get-bytevector-n* port count)
(port port))))) (port port)))))
bv)) bv))
(define (sub-bytevector bv len)
"Return a bytevector that aliases the first LEN bytes of BV."
(define max (bytevector-length bv))
(cond ((= len max) bv)
((< len max)
;; Yes, this is safe because the result of each conversion procedure
;; has its life cycle synchronized with that of its argument.
(pointer->bytevector (bytevector->pointer bv) len))
(else
(error "sub-bytevector called to get a super bytevector"))))
(define (write-int n p) (define (write-int n p)
(let ((b (make-bytevector 8 0))) (let ((b (make-bytevector 8 0)))
(bytevector-u32-set! b 0 n (endianness little)) (bytevector-u32-set! b 0 n (endianness little))
@ -119,10 +131,9 @@ (define (write-string s p)
(define (read-byte-string p) (define (read-byte-string p)
(let* ((len (read-int p)) (let* ((len (read-int p))
(m (modulo len 8)) (m (modulo len 8))
(bv (get-bytevector-n* p len))) (pad (if (zero? m) 0 (- 8 m)))
(or (zero? m) (bv (get-bytevector-n* p (+ len pad))))
(get-bytevector-n* p (- 8 m))) (sub-bytevector bv len)))
bv))
(define (read-string p) (define (read-string p)
(utf8->string (read-byte-string p))) (utf8->string (read-byte-string p)))