Spaces:
Sleeping
Sleeping
;;;; This file is part of LilyPond, the GNU music typesetter. | |
;;;; | |
;;;; Copyright (C) 2005--2020 Jan Nieuwenhuizen <[email protected]> | |
;;;; Han-Wen Nienhuys <[email protected]> | |
;;;; | |
;;;; LilyPond is free software: you can redistribute it and/or modify | |
;;;; it under the terms of the GNU General Public License as published by | |
;;;; the Free Software Foundation, either version 3 of the License, or | |
;;;; (at your option) any later version. | |
;;;; | |
;;;; LilyPond is distributed in the hope that it will be useful, | |
;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of | |
;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
;;;; GNU General Public License for more details. | |
;;;; | |
;;;; You should have received a copy of the GNU General Public License | |
;;;; along with LilyPond. If not, see <http://www.gnu.org/licenses/>. | |
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | |
;; backend helpers. | |
(use-modules (scm ps-to-png) | |
(scm paper-system) | |
(ice-9 optargs)) | |
(define-public (ly:system command) | |
(ly:debug (_ "Invoking `~a'...") (string-join command)) | |
(let ((status (apply ly:spawn command))) | |
(if (> status 0) | |
(begin | |
(ly:warning (_ "`~a' failed (~a)\n") command status) | |
;; hmmm. what's the best failure option? | |
(throw 'ly-file-failed))))) | |
(define-public (search-executable names) | |
(define (helper path lst) | |
(if (null? (cdr lst)) | |
(car lst) | |
(if (search-path path (car lst)) (car lst) | |
(helper path (cdr lst))))) | |
(let ((path (parse-path (getenv "PATH")))) | |
(helper path names))) | |
(define-public (ly:gs-cli args run-str) | |
(let* | |
((tmp (make-tmpfile #f)) | |
(tmp-name (port-filename tmp))) | |
(ly:debug (_ "Preparing Ghostscript command to `~a': ~a") | |
tmp-name run-str) | |
(display run-str tmp) | |
(close-port tmp) | |
(set! args (append args (list tmp-name))) | |
(ly:system args) | |
(delete-file tmp-name) | |
)) | |
(define-public (gs-cmd-args is-eps fit-page) | |
(filter string? | |
(list | |
(search-executable '("gs")) | |
(if (not (ly:get-option 'verbose)) "-q") | |
"-dNODISPLAY" | |
;; see function gs-safe-run below where we use .setsafe instead | |
"-dNOSAFER" | |
"-dNOPAUSE" | |
"-dBATCH" | |
(if (and is-eps (not fit-page)) "-dEPSCrop") | |
(if fit-page "-dEPSFitPage") | |
"-dAutoRotatePages=/None" | |
"-dPrinted=false"))) | |
(define-public (gs-safe-run input) | |
;; The PostScript Language Reference Manual says "run is a convenience | |
;; operator that combines the functions of file and exec". (This does | |
;; not seem to hold for Ghostscript, the second operator must be 'run' | |
;; for PNG output). | |
;; To enable access control (as usually done with -dSAFER), we need to | |
;; insert the operator .setsafe between file and run, _after_ we have | |
;; opened the input file. We cannot use .addcontrolpath because it was | |
;; introduced in version 9.50, but .setsafe should also work with older | |
;; releases. | |
(string-join | |
(list | |
(ly:format "(~a)" input) | |
"(r) file" | |
(if (or (ly:get-option 'gs-api) | |
(ly:get-option 'gs-load-fonts) | |
(ly:get-option 'gs-load-lily-fonts) | |
(eq? PLATFORM 'windows)) | |
"" | |
".setsafe") | |
"run") | |
" ")) | |
(define-public (postscript->pdf paper-width paper-height | |
base-name tmp-name is-eps) | |
(let* ((pdf-name (ly:format "~a.~a.pdf" tmp-name (random 1000000))) | |
(flush-name (string-append pdf-name ".flush")) | |
(dest (string-append base-name ".pdf")) | |
(output-file (string-join (string-split pdf-name #\%) "%%")) | |
(run-strings | |
(filter string? | |
(list | |
(ly:format "mark /OutputFile (~a)" output-file) | |
(if (not is-eps) | |
(ly:format "/PageSize [~$ ~$]" paper-width paper-height)) | |
"(pdfwrite) finddevice putdeviceprops pop" | |
;; `setdevice` does not set some defaults. So we use | |
;; `selectdevide` instead. | |
"(pdfwrite) selectdevice" | |
;; from Resource/Init/gs_pdfwr.ps; needed here because we | |
;; do not have the pdfwrite device initially (-dNODISPLAY). | |
"newpath fill" | |
(gs-safe-run tmp-name))) | |
)) | |
(ly:message (_ "Converting to `~a'...\n") dest) | |
((if (ly:get-option 'gs-api) ly:gs-api ly:gs-cli) | |
(gs-cmd-args is-eps #f) (string-join run-strings " ")) | |
;; for pdfwrite, the output is only finalized once a new output | |
;; file is opened. | |
(if (ly:get-option 'gs-api) | |
(begin | |
(ly:gs-api (gs-cmd-args is-eps #f) | |
(string-join | |
(list | |
(ly:format "mark /OutputFile (~a)" flush-name) | |
"(pdfwrite) finddevice putdeviceprops pop" | |
;; see above | |
"(pdfwrite) selectdevice" | |
;; see above | |
"newpath fill") | |
" ")) | |
(delete-file flush-name))) | |
(ly:rename-file pdf-name dest) | |
)) | |
(define-public (postscript->png resolution paper-width paper-height bbox | |
base-name tmp-name is-eps) | |
(let* ((verbose (ly:get-option 'verbose)) | |
(rename-page-1 #f)) | |
;; Do not try to guess the name of the png file, | |
;; GS produces PNG files like BASE-page%d.png. | |
(ly:message (_ "Converting to ~a...") "PNG") | |
;; If option `png-width` and/or `png-height` is set, `resolution` | |
;; is ignored. | |
(make-ps-images base-name tmp-name is-eps | |
#:resolution resolution | |
#:page-width paper-width | |
#:page-height paper-height | |
#:bbox bbox | |
#:rename-page-1 rename-page-1 | |
#:be-verbose verbose | |
#:anti-alias-factor (ly:get-option 'anti-alias-factor) | |
#:pixmap-format (ly:get-option 'pixmap-format) | |
#:png-width (ly:get-option 'png-width) | |
#:png-height (ly:get-option 'png-height)) | |
(ly:progress "\n"))) | |
(define-public (postscript->ps base-name tmp-name is-eps) | |
(let* ((ps-name (string-append base-name | |
(if is-eps ".eps" ".ps")))) | |
(if (not (equal? ps-name tmp-name)) | |
(begin | |
(ly:message (_ "Copying to `~a'...\n") ps-name) | |
(copy-binary-file tmp-name ps-name))))) | |
(define-public (mkdir-if-not-exist path . mode) | |
(catch | |
'system-error | |
(lambda () | |
;; mkdir: | |
;; If the directory already exists, it raises system-error. | |
(if (null? mode) | |
(mkdir path) | |
(mkdir path (car mode))) | |
#t) | |
(lambda stuff | |
;; Catch the system-error | |
(if (= EEXIST (system-error-errno stuff)) | |
;; If the directory already exists, avoid error and return #f. | |
(begin #f) | |
;; If the cause is something else, re-throw the error. | |
(throw 'system-error (cdr stuff)))))) | |
(define-public (symlink-if-not-exist oldpath newpath) | |
(catch | |
'system-error | |
(lambda () | |
;; symlink: | |
;; If the file already exists, it raises system-error. | |
(symlink oldpath newpath) | |
#t) | |
(lambda stuff | |
;; Catch the system-error | |
(if (= EEXIST (system-error-errno stuff)) | |
;; If the file already exists, avoid error and return #f. | |
(begin #f) | |
;; If the cause is something else, re-throw the error. | |
(throw 'system-error (cdr stuff)))))) | |
(define-public (close-port-rename port name) | |
(let* ((tmp (port-filename port))) | |
(close-port port) | |
(ly:rename-file tmp name))) | |
(define-public (symlink-or-copy-if-not-exist oldpath newpath) | |
(if (eq? PLATFORM 'windows) | |
(let ((port (create-file-exclusive newpath))) | |
(if port | |
(begin | |
(close port) | |
(copy-binary-file oldpath newpath) | |
#t) | |
(begin #f))) | |
(symlink-if-not-exist oldpath newpath))) | |
(define-public (create-file-exclusive path . mode) | |
(catch | |
'system-error | |
(lambda () | |
;; Exclusive file create: | |
;; If the file already exists, it raises system-error. | |
(if (null? mode) | |
(open path (logior O_WRONLY O_CREAT O_EXCL)) | |
(open path (logior O_WRONLY O_CREAT O_EXCL) (car mode)))) | |
(lambda stuff | |
;; Catch the system-error | |
(if (= EEXIST (system-error-errno stuff)) | |
;; If the file already exists, avoid error and return #f. | |
(begin #f) | |
;; If the cause is something else, re-throw the error. | |
(throw 'system-error (cdr stuff)))))) | |
(define-public (copy-binary-file from-name to-name) | |
(if (eq? PLATFORM 'windows) | |
;; MINGW hack: MinGW Guile's copy-file is broken. | |
;; It opens files by the text mode instead of the binary mode. | |
;; (It is fixed from Guile 2.0.9.) | |
;; By the text mode, copied binary files are broken. | |
;; So, we open files by the binary mode and copy by ourselves. | |
(let ((port-from (open-file from-name "rb")) | |
(port-to (open-file to-name "wb"))) | |
(let loop((c (read-char port-from))) | |
(if (eof-object? c) | |
(begin (close port-from) | |
(close port-to)) | |
(begin (write-char c port-to) | |
(loop (read-char port-from)))))) | |
;; Cygwin and other platforms: | |
;; Pass through to copy-file | |
;; TODO: this should write destination atomically. | |
(copy-file from-name to-name))) | |
(define-public (make-tmpfile basename) | |
"Returns a temp file as port. If basename is #f, a file under $TMPDIR is created." | |
(define max-try 10) | |
(define (inner basename tries) | |
(if (> tries 0) | |
(let* | |
((name (ly:format "~a-tmp-~a" basename (random 10000000))) | |
(port (create-file-exclusive name #o666)) | |
(bport #f)) | |
(if port | |
(begin | |
(set! bport (open-file name "wb")) | |
(close-port port) | |
bport) | |
(make-tmpfile basename (1- tries)))) | |
(ly:error "can't create temp file for ~a after ~a times" basename max-try) | |
)) | |
(if (not basename) | |
(set! basename (cond | |
;; MINGW hack: TMP / TEMP may include | |
;; unusable characters (Unicode etc.). | |
((eq? PLATFORM 'windows) "./tmp-") | |
;; Cygwin can handle any characters | |
;; including Unicode. | |
((eq? PLATFORM 'cygwin) (string-append | |
(or (getenv "TMP") | |
(getenv "TEMP")) | |
"/")) | |
;; Other platforms (POSIX platforms) | |
;; use TMPDIR or /tmp. | |
(else (string-append | |
(or (getenv "TMPDIR") | |
"/tmp") | |
"/lilypond"))))) | |
(inner basename max-try)) | |
(define-public (postprocess-output paper-book module formats | |
base-name tmp-name is-eps) | |
(let* ((completed (completize-formats formats is-eps))) | |
(for-each (lambda (f) | |
((eval (string->symbol (format #f "convert-to-~a" f)) module) | |
paper-book base-name tmp-name is-eps)) completed) | |
(if (and (ly:get-option 'delete-intermediate-files) | |
(or (not is-eps) | |
(not (member "ps" completed))) | |
(file-exists? tmp-name)) | |
(delete-file tmp-name)))) | |
(define-public (completize-formats formats is-eps) | |
(define new-fmts '()) | |
(if (and is-eps (member "eps" formats)) | |
(set! formats (cons "ps" formats))) | |
(if (not (or (member "pdf" formats) | |
(member "png" formats))) | |
(set! formats (cons "ps" formats))) | |
(for-each (lambda (x) | |
(if (member x formats) (set! new-fmts (cons x new-fmts)))) | |
'("ps" "pdf" "png")) | |
(uniq-list (reverse new-fmts))) | |
(define (header-to-file file-name key value) | |
(set! key (symbol->string key)) | |
(if (not (equal? "-" file-name)) | |
(set! file-name (string-append file-name "." key))) | |
(ly:message (_ "Writing header field `~a' to `~a'...") | |
key | |
(if (equal? "-" file-name) "<stdout>" file-name)) | |
(if (equal? file-name "-") | |
(display value) | |
(let ((port (open-file file-name "w"))) | |
(display value port) | |
(close-port port))) | |
(ly:progress "\n") | |
"") | |
(define-public (output-scopes scopes fields basename) | |
(define (output-scope scope) | |
(string-concatenate | |
(module-map | |
(lambda (sym var) | |
(let ((val (if (variable-bound? var) (variable-ref var) ""))) | |
(if (and (memq sym fields) (string? val)) | |
(header-to-file basename sym val)) | |
"")) | |
scope))) | |
(string-concatenate (map output-scope scopes))) | |
(define-public (relevant-book-systems book) | |
(let ((systems (ly:paper-book-systems book))) | |
;; skip booktitles. | |
(if (and (not (ly:get-option 'include-book-title-preview)) | |
(pair? systems) | |
(ly:prob-property (car systems) 'is-book-title #f)) | |
(cdr systems) | |
systems))) | |
(define-public (relevant-dump-systems systems) | |
(let ((to-dump-systems '())) | |
(for-each | |
(lambda (sys) | |
(if (or (paper-system-title? sys) | |
(not (pair? to-dump-systems)) | |
(paper-system-title? (car to-dump-systems))) | |
(set! to-dump-systems (cons sys to-dump-systems)))) | |
systems) | |
to-dump-systems)) | |
(define-public (font-name-split font-name) | |
"Return @code{(FONT-NAME . DESIGN-SIZE)} from @var{font-name} string | |
or @code{#f}." | |
(let ((match (regexp-exec (make-regexp "(.*)-([0-9]*)") font-name))) | |
(if (regexp-match? match) | |
(cons (match:substring match 1) (match:substring match 2)) | |
(cons font-name-designsize #f)))) | |
;; Example of a pango-physical-font | |
;; ("Emmentaler-11" "/home/janneke/vc/lilypond/out/share/lilypond/current/fonts/otf/emmentaler-11.otf" 0) | |
(define-public (pango-pf-font-name pango-pf) | |
"Return the font-name of the pango physical font @var{pango-pf}." | |
(list-ref pango-pf 0)) | |
(define-public (pango-pf-file-name pango-pf) | |
"Return the file-name of the pango physical font @var{pango-pf}." | |
(list-ref pango-pf 1)) | |
(define-public (pango-pf-fontindex pango-pf) | |
"Return the fontindex of the pango physical font @var{pango-pf}." | |
(list-ref pango-pf 2)) | |
(define (pango-font-name pango-font) | |
(let ((pf-fonts (ly:pango-font-physical-fonts pango-font))) | |
(if (pair? pf-fonts) | |
(pango-pf-font-name (car pf-fonts)) | |
""))) | |
(define-public (define-fonts paper define-font define-pango-pf) | |
"Return a string of all fonts used in @var{paper}, invoking the functions | |
@var{define-font} and @var{define-pango-pf} for producing the actual font | |
definition." | |
(let* ((font-list (ly:paper-fonts paper)) | |
(pango-fonts (filter ly:pango-font? font-list)) | |
(other-fonts (remove ly:pango-font? font-list)) | |
(other-font-names (map ly:font-name other-fonts)) | |
(pango-only-fonts | |
(remove (lambda (x) | |
(member (pango-font-name x) other-font-names)) | |
pango-fonts))) | |
(define (font-load-command font) | |
(let* ((font-name (ly:font-name font)) | |
(designsize (ly:font-design-size font)) | |
(magnification (* (ly:font-magnification font))) | |
(ops (ly:output-def-lookup paper 'output-scale)) | |
(scaling (* ops magnification designsize))) | |
(if (equal? font-name "unknown") | |
(display (list font font-name))) | |
(define-font font font-name scaling))) | |
(define (pango-font-load-command pango-font) | |
(let* ((pf-fonts (ly:pango-font-physical-fonts pango-font)) | |
(pango-pf (if (pair? pf-fonts) (car pf-fonts) '("" "" 0))) | |
(font-name (pango-pf-font-name pango-pf)) | |
(scaling (ly:output-def-lookup paper 'output-scale))) | |
(if (equal? font-name "unknown") | |
(display (list pango-font font-name))) | |
(define-pango-pf pango-pf font-name scaling))) | |
(string-append | |
(string-concatenate (map font-load-command other-fonts)) | |
(string-concatenate (map pango-font-load-command pango-only-fonts))))) | |