zlib.scm 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. ;;; Guile-zlib --- GNU Guile bindings of zlib
  2. ;;; Copyright © 2016, 2017 Ludovic Courtès <ludo@gnu.org>
  3. ;;; Copyright © 2020 Mathieu Othacehe <othacehe@gnu.org>
  4. ;;;
  5. ;;; This file is part of Guile-zlib.
  6. ;;;
  7. ;;; Guile-zlib is free software; you can redistribute it and/or modify it
  8. ;;; under the terms of the GNU General Public License as published by
  9. ;;; the Free Software Foundation; either version 3 of the License, or (at
  10. ;;; your option) any later version.
  11. ;;;
  12. ;;; Guile-zlib is distributed in the hope that it will be useful, but
  13. ;;; WITHOUT ANY WARRANTY; without even the implied warranty of
  14. ;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. ;;; GNU General Public License for more details.
  16. ;;;
  17. ;;; You should have received a copy of the GNU General Public License
  18. ;;; along with Guile-zlib. If not, see <http://www.gnu.org/licenses/>.
  19. (define-module (zlib)
  20. #:use-module (zlib config)
  21. #:use-module (rnrs bytevectors)
  22. #:use-module (ice-9 binary-ports)
  23. #:use-module (ice-9 match)
  24. #:use-module (system foreign)
  25. #:export (make-gzip-input-port
  26. make-gzip-output-port
  27. call-with-gzip-input-port
  28. call-with-gzip-output-port
  29. %default-buffer-size
  30. %default-compression-level))
  31. ;;; Commentary:
  32. ;;;
  33. ;;; This file is extracted from Guix and originally writen by Ludovic Courtès.
  34. ;;; Bindings to the gzip-related part of zlib's API. The main limitation of
  35. ;;; this API is that it requires a file descriptor as the source or sink.
  36. ;;;
  37. ;;; Code:
  38. (define %zlib
  39. (delay (dynamic-link %libz)))
  40. (define (zlib-procedure ret name parameters)
  41. "Return a procedure corresponding to C function NAME in libz, or #f if
  42. either zlib or the function could not be found."
  43. (match (false-if-exception (dynamic-func name (force %zlib)))
  44. ((? pointer? ptr)
  45. (pointer->procedure ret ptr parameters))
  46. (#f
  47. #f)))
  48. (define-wrapped-pointer-type <gzip-file>
  49. ;; Scheme counterpart of the 'gzFile' opaque type.
  50. gzip-file?
  51. pointer->gzip-file
  52. gzip-file->pointer
  53. (lambda (obj port)
  54. (format port "#<gzip-file ~a>"
  55. (number->string (object-address obj) 16))))
  56. (define gzerror
  57. (let ((proc (zlib-procedure '* "gzerror" '(* *))))
  58. (lambda (gzfile)
  59. (let* ((errnum* (make-bytevector (sizeof int)))
  60. (ptr (proc (gzip-file->pointer gzfile)
  61. (bytevector->pointer errnum*))))
  62. (values (bytevector-sint-ref errnum* 0
  63. (native-endianness) (sizeof int))
  64. (pointer->string ptr))))))
  65. (define gzdopen
  66. (let ((proc (zlib-procedure '* "gzdopen" (list int '*))))
  67. (lambda (fd mode)
  68. "Open file descriptor FD as a gzip stream with the given MODE. MODE must
  69. be a string denoting the how FD is to be opened, such as \"r\" for reading or
  70. \"w9\" for writing data compressed at level 9 to FD. Calling 'gzclose' also
  71. closes FD."
  72. (let ((result (proc fd (string->pointer mode))))
  73. (if (null-pointer? result)
  74. (throw 'zlib-error 'gzdopen)
  75. (pointer->gzip-file result))))))
  76. (define gzread!
  77. (let ((proc (zlib-procedure int "gzread" (list '* '* unsigned-int))))
  78. (lambda* (gzfile bv #:optional (start 0) (count (bytevector-length bv)))
  79. "Read up to COUNT bytes from GZFILE into BV at offset START. Return the
  80. number of uncompressed bytes actually read; it is zero if COUNT is zero or if
  81. the end-of-stream has been reached."
  82. (let ((ret (proc (gzip-file->pointer gzfile)
  83. (bytevector->pointer bv start)
  84. count)))
  85. (if (< ret 0)
  86. (throw 'zlib-error 'gzread! ret)
  87. ret)))))
  88. (define gzwrite
  89. (let ((proc (zlib-procedure int "gzwrite" (list '* '* unsigned-int))))
  90. (lambda* (gzfile bv #:optional (start 0) (count (bytevector-length bv)))
  91. "Write up to COUNT bytes from BV at offset START into GZFILE. Return
  92. the number of uncompressed bytes written, a strictly positive integer."
  93. (let ((ret (proc (gzip-file->pointer gzfile)
  94. (bytevector->pointer bv start)
  95. count)))
  96. (if (<= ret 0)
  97. (throw 'zlib-error 'gzwrite ret)
  98. ret)))))
  99. (define gzbuffer!
  100. (let ((proc (zlib-procedure int "gzbuffer" (list '* unsigned-int))))
  101. (lambda (gzfile size)
  102. "Change the internal buffer size of GZFILE to SIZE bytes."
  103. (let ((ret (proc (gzip-file->pointer gzfile) size)))
  104. (unless (zero? ret)
  105. (throw 'zlib-error 'gzbuffer! ret))))))
  106. (define gzeof?
  107. (let ((proc (zlib-procedure int "gzeof" '(*))))
  108. (lambda (gzfile)
  109. "Return true if the end-of-file has been reached on GZFILE."
  110. (not (zero? (proc (gzip-file->pointer gzfile)))))))
  111. (define gzclose
  112. (let ((proc (zlib-procedure int "gzclose" '(*))))
  113. (lambda (gzfile)
  114. "Close GZFILE."
  115. (let ((ret (proc (gzip-file->pointer gzfile))))
  116. (unless (zero? ret)
  117. (throw 'zlib-error 'gzclose ret (gzerror gzfile)))))))
  118. ;;;
  119. ;;; Port interface.
  120. ;;;
  121. (define %default-buffer-size
  122. ;; Default buffer size, as documented in <zlib.h>.
  123. 8192)
  124. (define %default-compression-level
  125. ;; Z_DEFAULT_COMPRESSION.
  126. -1)
  127. (define* (make-gzip-input-port port #:key (buffer-size %default-buffer-size))
  128. "Return an input port that decompresses data read from PORT, a file port.
  129. PORT is automatically closed when the resulting port is closed. BUFFER-SIZE
  130. is the size in bytes of the internal buffer, 8 KiB by default; using a larger
  131. buffer increases decompression speed. An error is thrown if PORT contains
  132. buffered input, which would be lost (and is lost anyway)."
  133. (define gzfile
  134. (match (drain-input port)
  135. ("" ;PORT's buffer is empty
  136. ;; 'gzclose' will eventually close the file descriptor beneath PORT.
  137. ;; 'close-port' on PORT would get EBADF if 'gzclose' already closed it,
  138. ;; so that's no good; revealed ports are no good either because they
  139. ;; leak (see <https://bugs.gnu.org/28784>); calling 'close-port' after
  140. ;; 'gzclose' doesn't work either because it leads to a race condition
  141. ;; (see <https://bugs.gnu.org/29335>). So we dup and close PORT right
  142. ;; away.
  143. (gzdopen (dup (fileno port)) "r"))
  144. (_
  145. ;; This is unrecoverable but it's better than having the buffered input
  146. ;; be lost, leading to unclear end-of-file or corrupt-data errors down
  147. ;; the path.
  148. (throw 'zlib-error 'make-gzip-input-port
  149. "port contains buffered input" port))))
  150. (define (read! bv start count)
  151. (gzread! gzfile bv start count))
  152. (unless (= buffer-size %default-buffer-size)
  153. (gzbuffer! gzfile buffer-size))
  154. (close-port port) ;we no longer need it
  155. (make-custom-binary-input-port "gzip-input" read! #f #f
  156. (lambda ()
  157. (gzclose gzfile))))
  158. (define* (make-gzip-output-port port
  159. #:key
  160. (level %default-compression-level)
  161. (buffer-size %default-buffer-size))
  162. "Return an output port that compresses data at the given LEVEL, using PORT,
  163. a file port, as its sink. PORT is automatically closed when the resulting
  164. port is closed."
  165. (define gzfile
  166. (begin
  167. (force-output port) ;empty PORT's buffer
  168. (gzdopen (dup (fileno port))
  169. (string-append "w" (number->string level)))))
  170. (define (write! bv start count)
  171. (gzwrite gzfile bv start count))
  172. (unless (= buffer-size %default-buffer-size)
  173. (gzbuffer! gzfile buffer-size))
  174. (close-port port)
  175. (make-custom-binary-output-port "gzip-output" write! #f #f
  176. (lambda ()
  177. (gzclose gzfile))))
  178. (define* (call-with-gzip-input-port port proc
  179. #:key (buffer-size %default-buffer-size))
  180. "Call PROC with a port that wraps PORT and decompresses data read from it.
  181. PORT is closed upon completion. The gzip internal buffer size is set to
  182. BUFFER-SIZE bytes."
  183. (let ((gzip (make-gzip-input-port port #:buffer-size buffer-size)))
  184. (dynamic-wind
  185. (const #t)
  186. (lambda ()
  187. (proc gzip))
  188. (lambda ()
  189. (close-port gzip)))))
  190. (define* (call-with-gzip-output-port port proc
  191. #:key
  192. (level %default-compression-level)
  193. (buffer-size %default-buffer-size))
  194. "Call PROC with an output port that wraps PORT and compresses data. PORT is
  195. close upon completion. The gzip internal buffer size is set to BUFFER-SIZE
  196. bytes."
  197. (let ((gzip (make-gzip-output-port port
  198. #:level level
  199. #:buffer-size buffer-size)))
  200. (dynamic-wind
  201. (const #t)
  202. (lambda ()
  203. (proc gzip))
  204. (lambda ()
  205. (close-port gzip)))))
  206. ;;; zlib.scm ends here