123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938 |
- ;;; GNU Guix --- Functional package management for GNU
- ;;; Copyright © 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2019, 2020, 2021 Ludovic Courtès <ludo@gnu.org>
- ;;; Copyright © 2018 Mark H Weaver <mhw@netris.org>
- ;;; Copyright © 2020 Brendan Tildesley <mail@brendan.scot>
- ;;; Copyright © 2021 Maxim Cournoyer <maxim.cournoyer@gmail.com>
- ;;;
- ;;; This file is part of GNU Guix.
- ;;;
- ;;; GNU Guix 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.
- ;;;
- ;;; GNU Guix 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 GNU Guix. If not, see <http://www.gnu.org/licenses/>.
- (define-module (guix build gnu-build-system)
- #:use-module (guix build utils)
- #:use-module (guix build gremlin)
- #:use-module (guix elf)
- #:use-module (ice-9 ftw)
- #:use-module (ice-9 match)
- #:use-module (ice-9 regex)
- #:use-module (ice-9 format)
- #:use-module (ice-9 ftw)
- #:use-module (srfi srfi-1)
- #:use-module (srfi srfi-19)
- #:use-module (srfi srfi-34)
- #:use-module (srfi srfi-35)
- #:use-module (srfi srfi-26)
- #:use-module (rnrs io ports)
- #:export (%standard-phases
- %license-file-regexp
- %bootstrap-scripts
- dump-file-contents
- gnu-build))
- ;; Commentary:
- ;;
- ;; Standard build procedure for packages using the GNU Build System or
- ;; something compatible ("./configure && make && make install"). This is the
- ;; builder-side code.
- ;;
- ;; Code:
- (cond-expand
- (guile-2.2
- ;; Guile 2.2.2 has a bug whereby 'time-monotonic' objects have seconds and
- ;; nanoseconds swapped (fixed in Guile commit 886ac3e). Work around it.
- (define time-monotonic time-tai))
- (else #t))
- (define* (set-SOURCE-DATE-EPOCH #:rest _)
- "Set the 'SOURCE_DATE_EPOCH' environment variable. This is used by tools
- that incorporate timestamps as a way to tell them to use a fixed timestamp.
- See https://reproducible-builds.org/specs/source-date-epoch/."
- (setenv "SOURCE_DATE_EPOCH" "1"))
- (define (first-subdirectory directory)
- "Return the file name of the first sub-directory of DIRECTORY or false, when
- there are none."
- (match (scandir directory
- (lambda (file)
- (and (not (member file '("." "..")))
- (file-is-directory? (string-append directory "/"
- file)))))
- ((first . _) first)
- (_ #f)))
- (define* (set-paths #:key target inputs native-inputs
- (search-paths '()) (native-search-paths '())
- #:allow-other-keys)
- (define input-directories
- ;; The "source" input can be a directory, but we don't want it for search
- ;; paths. See <https://issues.guix.gnu.org/44924>.
- (match (alist-delete "source" inputs)
- (((_ . dir) ...)
- dir)))
- (define native-input-directories
- (match native-inputs
- (((_ . dir) ...)
- dir)
- (#f ; not cross compiling
- '())))
- ;; Tell 'ld-wrapper' to disallow non-store libraries.
- (setenv "GUIX_LD_WRAPPER_ALLOW_IMPURITIES" "no")
- ;; When cross building, $PATH must refer only to native (host) inputs since
- ;; target inputs are not executable.
- (set-path-environment-variable "PATH" '("bin" "sbin")
- (append native-input-directories
- (if target
- '()
- input-directories)))
- (for-each (match-lambda
- ((env-var (files ...) separator type pattern)
- (set-path-environment-variable env-var files
- input-directories
- #:separator separator
- #:type type
- #:pattern pattern)))
- search-paths)
- (when native-search-paths
- ;; Search paths for native inputs, when cross building.
- (for-each (match-lambda
- ((env-var (files ...) separator type pattern)
- (set-path-environment-variable env-var files
- native-input-directories
- #:separator separator
- #:type type
- #:pattern pattern)))
- native-search-paths)))
- (define* (install-locale #:key
- (locale "en_US.utf8")
- (locale-category LC_ALL)
- #:allow-other-keys)
- "Try to install LOCALE; emit a warning if that fails. The main goal is to
- use a UTF-8 locale so that Guile correctly interprets UTF-8 file names.
- This phase must typically happen after 'set-paths' so that $LOCPATH has a
- chance to be set."
- (catch 'system-error
- (lambda ()
- (setlocale locale-category locale)
- ;; While we're at it, pass it to sub-processes.
- (setenv (locale-category->string locale-category) locale)
- (format (current-error-port) "using '~a' locale for category ~s~%"
- locale (locale-category->string locale-category)))
- (lambda args
- ;; This is known to fail for instance in early bootstrap where locales
- ;; are not available.
- (format (current-error-port)
- "warning: failed to install '~a' locale: ~a~%"
- locale (strerror (system-error-errno args))))))
- (define* (unpack #:key source #:allow-other-keys)
- "Unpack SOURCE in the working directory, and change directory within the
- source. When SOURCE is a directory, copy it in a sub-directory of the current
- working directory."
- (if (file-is-directory? source)
- (begin
- (mkdir "source")
- (chdir "source")
- ;; Preserve timestamps (set to the Epoch) on the copied tree so that
- ;; things work deterministically.
- (copy-recursively source "."
- #:keep-mtime? #t)
- ;; Make the source checkout files writable, for convenience.
- (for-each (lambda (f)
- (false-if-exception (make-file-writable f)))
- (find-files ".")))
- (begin
- (cond
- ((string-suffix? ".zip" source)
- (invoke "unzip" source))
- ((tarball? source)
- (invoke "tar" "xvf" source))
- (else
- (let ((name (strip-store-file-name source))
- (command (compressor source)))
- (copy-file source name)
- (when command
- (invoke command "--decompress" name)))))
- ;; Attempt to change into child directory.
- (and=> (first-subdirectory ".") chdir))))
- (define %bootstrap-scripts
- ;; Typical names of Autotools "bootstrap" scripts.
- '("bootstrap" "bootstrap.sh" "autogen.sh"))
- (define* (bootstrap #:key (bootstrap-scripts %bootstrap-scripts)
- #:allow-other-keys)
- "If the code uses Autotools and \"configure\" is missing, run
- \"autoreconf\". Otherwise do nothing."
- ;; Note: Run that right after 'unpack' so that the generated files are
- ;; visible when the 'patch-source-shebangs' phase runs.
- (define (script-exists? file)
- (and (file-exists? file)
- (not (file-is-directory? file))))
- (if (not (script-exists? "configure"))
- ;; First try one of the BOOTSTRAP-SCRIPTS. If none exists, and it's
- ;; clearly an Autoconf-based project, run 'autoreconf'. Otherwise, do
- ;; nothing (perhaps the user removed or overrode the 'configure' phase.)
- (let ((script (find script-exists? bootstrap-scripts)))
- ;; GNU packages often invoke the 'git-version-gen' script from
- ;; 'configure.ac' so make sure it has a valid shebang.
- (false-if-file-not-found
- (patch-shebang "build-aux/git-version-gen"))
- (if script
- (let ((script (string-append "./" script)))
- (setenv "NOCONFIGURE" "true")
- (format #t "running '~a'~%" script)
- (if (executable-file? script)
- (begin
- (patch-shebang script)
- (invoke script))
- (invoke "sh" script))
- ;; Let's clean up after ourselves.
- (unsetenv "NOCONFIGURE"))
- (if (or (file-exists? "configure.ac")
- (file-exists? "configure.in"))
- (invoke "autoreconf" "-vif")
- (format #t "no 'configure.ac' or anything like that, \
- doing nothing~%"))))
- (format #t "GNU build system bootstrapping not needed~%")))
- ;; See <http://bugs.gnu.org/17840>.
- (define* (patch-usr-bin-file #:key native-inputs inputs
- (patch-/usr/bin/file? #t)
- #:allow-other-keys)
- "Patch occurrences of \"/usr/bin/file\" in all the executable 'configure'
- files found in the source tree. This works around Libtool's Autoconf macros,
- which generates invocations of \"/usr/bin/file\" that are used to determine
- things like the ABI being used."
- (when patch-/usr/bin/file?
- (for-each (lambda (file)
- (when (executable-file? file)
- (patch-/usr/bin/file file)))
- (find-files "." "^configure$"))))
- (define* (patch-source-shebangs #:key source #:allow-other-keys)
- "Patch shebangs in all source files; this includes non-executable
- files such as `.in' templates. Most scripts honor $SHELL and
- $CONFIG_SHELL, but some don't, such as `mkinstalldirs' or Automake's
- `missing' script."
- (for-each patch-shebang
- (find-files "."
- (lambda (file stat)
- ;; Filter out symlinks.
- (eq? 'regular (stat:type stat)))
- #:stat lstat)))
- (define (patch-generated-file-shebangs . rest)
- "Patch shebangs in generated files, including `SHELL' variables in
- makefiles."
- ;; Patch executable regular files, some of which might have been generated
- ;; by `configure'.
- (for-each patch-shebang
- (find-files "."
- (lambda (file stat)
- (and (eq? 'regular (stat:type stat))
- (not (zero? (logand (stat:mode stat) #o100)))))
- #:stat lstat))
- ;; Patch `SHELL' in generated makefiles.
- (for-each patch-makefile-SHELL (find-files "." "^(GNU)?[mM]akefile$")))
- (define* (configure #:key build target native-inputs inputs outputs
- (configure-flags '()) out-of-source?
- #:allow-other-keys)
- (define (package-name)
- (let* ((out (assoc-ref outputs "out"))
- (base (basename out))
- (dash (string-rindex base #\-)))
- ;; XXX: We'd rather use `package-name->name+version' or similar.
- (string-drop (if dash
- (substring base 0 dash)
- base)
- (+ 1 (string-index base #\-)))))
- (let* ((prefix (assoc-ref outputs "out"))
- (bindir (assoc-ref outputs "bin"))
- (libdir (assoc-ref outputs "lib"))
- (includedir (assoc-ref outputs "include"))
- (docdir (assoc-ref outputs "doc"))
- (bash (or (and=> (assoc-ref (or native-inputs inputs) "bash")
- (cut string-append <> "/bin/bash"))
- "/bin/sh"))
- (flags `(,@(if target ; cross building
- '("CC_FOR_BUILD=gcc")
- '())
- ,(string-append "CONFIG_SHELL=" bash)
- ,(string-append "SHELL=" bash)
- ,(string-append "--prefix=" prefix)
- "--enable-fast-install" ; when using Libtool
- ;; Produce multiple outputs when specific output names
- ;; are recognized.
- ,@(if bindir
- (list (string-append "--bindir=" bindir "/bin"))
- '())
- ,@(if libdir
- (cons (string-append "--libdir=" libdir "/lib")
- (if includedir
- '()
- (list
- (string-append "--includedir="
- libdir "/include"))))
- '())
- ,@(if includedir
- (list (string-append "--includedir="
- includedir "/include"))
- '())
- ,@(if docdir
- (list (string-append "--docdir=" docdir
- "/share/doc/" (package-name)))
- '())
- ,@(if build
- (list (string-append "--build=" build))
- '())
- ,@(if target ; cross building
- (list (string-append "--host=" target))
- '())
- ,@configure-flags))
- (abs-srcdir (getcwd))
- (srcdir (if out-of-source?
- (string-append "../" (basename abs-srcdir))
- ".")))
- (format #t "source directory: ~s (relative from build: ~s)~%"
- abs-srcdir srcdir)
- (if out-of-source?
- (begin
- (mkdir "../build")
- (chdir "../build")))
- (format #t "build directory: ~s~%" (getcwd))
- (format #t "configure flags: ~s~%" flags)
- ;; Use BASH to reduce reliance on /bin/sh since it may not always be
- ;; reliable (see
- ;; <http://thread.gmane.org/gmane.linux.distributions.nixos/9748>
- ;; for a summary of the situation.)
- ;;
- ;; Call `configure' with a relative path. Otherwise, GCC's build system
- ;; (for instance) records absolute source file names, which typically
- ;; contain the hash part of the `.drv' file, leading to a reference leak.
- (apply invoke bash
- (string-append srcdir "/configure")
- flags)))
- (define* (build #:key (make-flags '()) (parallel-build? #t)
- #:allow-other-keys)
- (apply invoke "make"
- `(,@(if parallel-build?
- `("-j" ,(number->string (parallel-job-count)))
- '())
- ,@make-flags)))
- (define* (dump-file-contents directory file-regexp
- #:optional (port (current-error-port)))
- "Dump to PORT the contents of files in DIRECTORY that match FILE-REGEXP."
- (define (dump file)
- (let ((prefix (string-append "\n--- " file " ")))
- (display (if (< (string-length prefix) 78)
- (string-pad-right prefix 78 #\-)
- prefix)
- port)
- (display "\n\n" port)
- (call-with-input-file file
- (lambda (log)
- (dump-port log port)))
- (display "\n" port)))
- (for-each dump (find-files directory file-regexp)))
- (define %test-suite-log-regexp
- ;; Name of test suite log files as commonly found in GNU-based build systems
- ;; and CMake.
- "^(test-?suite\\.log|LastTestFailed\\.log)$")
- (define* (check #:key target (make-flags '()) (tests? (not target))
- (test-target "check") (parallel-tests? #t)
- (test-suite-log-regexp %test-suite-log-regexp)
- #:allow-other-keys)
- (if tests?
- (guard (c ((invoke-error? c)
- ;; Dump the test suite log to facilitate debugging.
- (display "\nTest suite failed, dumping logs.\n"
- (current-error-port))
- (dump-file-contents "." test-suite-log-regexp)
- (raise c)))
- (apply invoke "make" test-target
- `(,@(if parallel-tests?
- `("-j" ,(number->string (parallel-job-count)))
- '())
- ,@make-flags)))
- (format #t "test suite not run~%")))
- (define* (install #:key (make-flags '()) #:allow-other-keys)
- (apply invoke "make" "install" make-flags))
- (define* (patch-shebangs #:key inputs outputs (patch-shebangs? #t)
- #:allow-other-keys)
- (define (list-of-files dir)
- (map (cut string-append dir "/" <>)
- (or (scandir dir (lambda (f)
- (let ((s (lstat (string-append dir "/" f))))
- (eq? 'regular (stat:type s)))))
- '())))
- (define bin-directories
- (match-lambda
- ((_ . dir)
- (list (string-append dir "/bin")
- (string-append dir "/sbin")
- (string-append dir "/libexec")))))
- (define output-bindirs
- (append-map bin-directories outputs))
- (define input-bindirs
- ;; Shebangs should refer to binaries of the target system---i.e., from
- ;; "inputs", not from "native-inputs".
- (append-map bin-directories inputs))
- (when patch-shebangs?
- (let ((path (append output-bindirs input-bindirs)))
- (for-each (lambda (dir)
- (let ((files (list-of-files dir)))
- (for-each (cut patch-shebang <> path) files)))
- output-bindirs))))
- (define* (strip #:key target outputs (strip-binaries? #t)
- (strip-command (if target
- (string-append target "-strip")
- "strip"))
- (objcopy-command (if target
- (string-append target "-objcopy")
- "objcopy"))
- (strip-flags '("--strip-unneeded"
- "--enable-deterministic-archives"))
- (strip-directories '("lib" "lib64" "libexec"
- "bin" "sbin"))
- #:allow-other-keys)
- (define debug-output
- ;; If an output is called "debug", then that's where debugging information
- ;; will be stored instead of being discarded.
- (assoc-ref outputs "debug"))
- (define debug-file-extension
- ;; File name extension for debugging information.
- ".debug")
- (define (debug-file file)
- ;; Return the name of the debug file for FILE, an absolute file name.
- ;; Once installed in the user's profile, it is in $PROFILE/lib/debug/FILE,
- ;; which is where GDB looks for it (info "(gdb) Separate Debug Files").
- (string-append debug-output "/lib/debug/"
- file debug-file-extension))
- (define (make-debug-file file)
- ;; Create a file in DEBUG-OUTPUT containing the debugging info of FILE.
- (let ((debug (debug-file file)))
- (mkdir-p (dirname debug))
- (copy-file file debug)
- (invoke strip-command "--only-keep-debug" debug)
- (chmod debug #o400)))
- (define (add-debug-link file)
- ;; Add a debug link in FILE (info "(binutils) strip").
- ;; `objcopy --add-gnu-debuglink' wants to have the target of the debug
- ;; link around so it can compute a CRC of that file (see the
- ;; `bfd_fill_in_gnu_debuglink_section' function.) No reference to
- ;; DEBUG-OUTPUT is kept because bfd keeps only the basename of the debug
- ;; file.
- (invoke objcopy-command "--enable-deterministic-archives"
- (string-append "--add-gnu-debuglink="
- (debug-file file))
- file))
- (define (strip-dir dir)
- (format #t "stripping binaries in ~s with ~s and flags ~s~%"
- dir strip-command strip-flags)
- (when debug-output
- (format #t "debugging output written to ~s using ~s~%"
- debug-output objcopy-command))
- (for-each (lambda (file)
- (when (or (elf-file? file) (ar-file? file))
- ;; If an error occurs while processing a file, issue a
- ;; warning and continue to the next file.
- (guard (c ((invoke-error? c)
- (format (current-error-port)
- "warning: ~a: program ~s exited\
- ~@[ with non-zero exit status ~a~]\
- ~@[ terminated by signal ~a~]~%"
- file
- (invoke-error-program c)
- (invoke-error-exit-status c)
- (invoke-error-term-signal c))))
- (when debug-output
- (make-debug-file file))
- ;; Ensure the file is writable.
- (make-file-writable file)
- (apply invoke strip-command
- (append strip-flags (list file)))
- (when debug-output
- (add-debug-link file)))))
- (find-files dir
- (lambda (file stat)
- ;; Ignore symlinks such as:
- ;; libfoo.so -> libfoo.so.0.0.
- (eq? 'regular (stat:type stat)))
- #:stat lstat)))
- (when strip-binaries?
- (for-each
- strip-dir
- (append-map (match-lambda
- ((_ . dir)
- (filter-map (lambda (d)
- (let ((sub (string-append dir "/" d)))
- (and (directory-exists? sub) sub)))
- strip-directories)))
- outputs))))
- (define* (validate-runpath #:key
- (validate-runpath? #t)
- (elf-directories '("lib" "lib64" "libexec"
- "bin" "sbin"))
- outputs #:allow-other-keys)
- "When VALIDATE-RUNPATH? is true, validate that all the ELF files in
- ELF-DIRECTORIES have their dependencies found in their 'RUNPATH'.
- Since the ELF parser needs to have a copy of files in memory, better run this
- phase after stripping."
- (define (sub-directory parent)
- (lambda (directory)
- (let ((directory (string-append parent "/" directory)))
- (and (directory-exists? directory) directory))))
- (define (validate directory)
- (define (file=? file1 file2)
- (let ((st1 (stat file1))
- (st2 (stat file2)))
- (= (stat:ino st1) (stat:ino st2))))
- ;; There are always symlinks from '.so' to '.so.1' and so on, so delete
- ;; duplicates.
- (let ((files (delete-duplicates (find-files directory (lambda (file stat)
- (elf-file? file)))
- file=?)))
- (format (current-error-port)
- "validating RUNPATH of ~a binaries in ~s...~%"
- (length files) directory)
- (every* validate-needed-in-runpath files)))
- (if validate-runpath?
- (let ((dirs (append-map (match-lambda
- (("debug" . _)
- ;; The "debug" output is full of ELF files
- ;; that are not worth checking.
- '())
- ((name . output)
- (filter-map (sub-directory output)
- elf-directories)))
- outputs)))
- (unless (every* validate dirs)
- (error "RUNPATH validation failed")))
- (format (current-error-port) "skipping RUNPATH validation~%")))
- (define* (validate-documentation-location #:key outputs
- #:allow-other-keys)
- "Documentation should go to 'share/info' and 'share/man', not just 'info/'
- and 'man/'. This phase moves directories to the right place if needed."
- (define (validate-sub-directory output sub-directory)
- (let ((directory (string-append output "/" sub-directory)))
- (when (directory-exists? directory)
- (let ((target (string-append output "/share/" sub-directory)))
- (format #t "moving '~a' to '~a'~%" directory target)
- (mkdir-p (dirname target))
- (rename-file directory target)))))
- (define (validate-output output)
- (for-each (cut validate-sub-directory output <>)
- '("man" "info")))
- (match outputs
- (((names . directories) ...)
- (for-each validate-output directories))))
- (define* (reset-gzip-timestamps #:key outputs #:allow-other-keys)
- "Reset embedded timestamps in gzip files found in OUTPUTS."
- (define (process-directory directory)
- (let ((files (find-files directory
- (lambda (file stat)
- (and (eq? 'regular (stat:type stat))
- (or (string-suffix? ".gz" file)
- (string-suffix? ".tgz" file))
- (gzip-file? file)))
- #:stat lstat)))
- ;; Ensure the files are writable.
- (for-each make-file-writable files)
- (for-each reset-gzip-timestamp files)))
- (match outputs
- (((names . directories) ...)
- (for-each process-directory directories))))
- (define* (compress-documentation #:key outputs
- (compress-documentation? #t)
- (documentation-compressor "gzip")
- (documentation-compressor-flags
- '("--best" "--no-name"))
- (compressed-documentation-extension ".gz")
- #:allow-other-keys)
- "When COMPRESS-DOCUMENTATION? is true, compress man pages and Info files
- found in OUTPUTS using DOCUMENTATION-COMPRESSOR, called with
- DOCUMENTATION-COMPRESSOR-FLAGS."
- (define (retarget-symlink link)
- (let ((target (readlink link)))
- (delete-file link)
- (symlink (string-append target compressed-documentation-extension)
- (string-append link compressed-documentation-extension))))
- (define (has-links? file)
- ;; Return #t if FILE has hard links.
- (> (stat:nlink (lstat file)) 1))
- (define (points-to-symlink? symlink)
- ;; Return #t if SYMLINK points to another symbolic link.
- (let* ((target (readlink symlink))
- (target-absolute (if (string-prefix? "/" target)
- target
- (string-append (dirname symlink)
- "/" target))))
- (catch 'system-error
- (lambda ()
- (symbolic-link? target-absolute))
- (lambda args
- (if (= ENOENT (system-error-errno args))
- (begin
- (format (current-error-port)
- "The symbolic link '~a' target is missing: '~a'\n"
- symlink target-absolute)
- #f)
- (apply throw args))))))
- (define (maybe-compress-directory directory regexp)
- (when (directory-exists? directory)
- (match (find-files directory regexp)
- (() ;nothing to compress
- #t)
- ((files ...) ;one or more files
- (format #t
- "compressing documentation in '~a' with ~s and flags ~s~%"
- directory documentation-compressor
- documentation-compressor-flags)
- (call-with-values
- (lambda ()
- (partition symbolic-link? files))
- (lambda (symlinks regular-files)
- ;; Compress the non-symlink files, and adjust symlinks to refer
- ;; to the compressed files. Leave files that have hard links
- ;; unchanged ('gzip' would refuse to compress them anyway.)
- ;; Also, do not retarget symbolic links pointing to other
- ;; symbolic links, since these are not compressed.
- (for-each retarget-symlink
- (filter (lambda (symlink)
- (and (not (points-to-symlink? symlink))
- (string-match regexp symlink)))
- symlinks))
- (apply invoke documentation-compressor
- (append documentation-compressor-flags
- (remove has-links? regular-files)))))))))
- (define (maybe-compress output)
- (maybe-compress-directory (string-append output "/share/man")
- "\\.[0-9]+$")
- (maybe-compress-directory (string-append output "/share/info")
- "\\.info(-[0-9]+)?$"))
- (if compress-documentation?
- (match outputs
- (((names . directories) ...)
- (for-each maybe-compress directories)))
- (format #t "not compressing documentation~%")))
- (define* (delete-info-dir-file #:key outputs #:allow-other-keys)
- "Delete any 'share/info/dir' file from OUTPUTS."
- (for-each (match-lambda
- ((output . directory)
- (let ((info-dir-file (string-append directory "/share/info/dir")))
- (when (file-exists? info-dir-file)
- (delete-file info-dir-file)))))
- outputs))
- (define* (patch-dot-desktop-files #:key outputs inputs #:allow-other-keys)
- "Replace any references to executables in '.desktop' files with their
- absolute file names."
- (define bin-directories
- (append-map (match-lambda
- ((_ . directory)
- (list (string-append directory "/bin")
- (string-append directory "/sbin"))))
- outputs))
- (define (which program)
- (or (search-path bin-directories program)
- (begin
- (format (current-error-port)
- "warning: '.desktop' file refers to '~a', \
- which cannot be found~%"
- program)
- program)))
- (for-each (match-lambda
- ((_ . directory)
- (let ((applications (string-append directory
- "/share/applications")))
- (when (directory-exists? applications)
- (let ((files (find-files applications "\\.desktop$")))
- (format #t "adjusting ~a '.desktop' files in ~s~%"
- (length files) applications)
- ;; '.desktop' files contain translations and are always
- ;; UTF-8-encoded.
- (with-fluids ((%default-port-encoding "UTF-8"))
- (substitute* files
- (("^Exec=([^/[:blank:]\r\n]*)(.*)$" _ binary rest)
- (string-append "Exec=" (which binary) rest))
- (("^TryExec=([^/[:blank:]\r\n]*)(.*)$" _ binary rest)
- (string-append "TryExec="
- (which binary) rest)))))))))
- outputs))
- (define* (make-dynamic-linker-cache #:key outputs
- (make-dynamic-linker-cache? #t)
- #:allow-other-keys)
- "Create a dynamic linker cache under 'etc/ld.so.cache' in each of the
- OUTPUTS. This reduces application startup time by avoiding the 'stat' storm
- that traversing all the RUNPATH entries entails."
- (define (make-cache-for-output directory)
- (define bin-directories
- (filter-map (lambda (sub-directory)
- (let ((directory (string-append directory "/"
- sub-directory)))
- (and (directory-exists? directory)
- directory)))
- '("bin" "sbin" "libexec")))
- (define programs
- ;; Programs that can benefit from the ld.so cache.
- (append-map (lambda (directory)
- (if (directory-exists? directory)
- (find-files directory
- (lambda (file stat)
- (and (executable-file? file)
- (elf-file? file))))
- '()))
- bin-directories))
- (define library-path
- ;; Directories containing libraries that PROGRAMS depend on,
- ;; recursively.
- (delete-duplicates
- (append-map (lambda (program)
- (map dirname (file-needed/recursive program)))
- programs)))
- (define cache-file
- (string-append directory "/etc/ld.so.cache"))
- (define ld.so.conf
- (string-append (or (getenv "TMPDIR") "/tmp")
- "/ld.so.conf"))
- (unless (null? library-path)
- (mkdir-p (dirname cache-file))
- (guard (c ((invoke-error? c)
- ;; Do not treat 'ldconfig' failure as an error.
- (format (current-error-port)
- "warning: 'ldconfig' failed:~%")
- (report-invoke-error c (current-error-port))))
- ;; Create a config file to tell 'ldconfig' where to look for the
- ;; libraries that PROGRAMS need.
- (call-with-output-file ld.so.conf
- (lambda (port)
- (for-each (lambda (directory)
- (display directory port)
- (newline port))
- library-path)))
- (invoke "ldconfig" "-f" ld.so.conf "-C" cache-file)
- (format #t "created '~a' from ~a library search path entries~%"
- cache-file (length library-path)))))
- (if make-dynamic-linker-cache?
- (match outputs
- (((_ . directories) ...)
- (for-each make-cache-for-output directories)))
- (format #t "ld.so cache not built~%")))
- (define %license-file-regexp
- ;; Regexp matching license files.
- "^(COPYING.*|LICEN[CS]E.*|[Ll]icen[cs]e.*|Copy[Rr]ight(\\.(txt|md))?)$")
- (define* (install-license-files #:key outputs
- (license-file-regexp %license-file-regexp)
- out-of-source?
- #:allow-other-keys)
- "Install license files matching LICENSE-FILE-REGEXP to 'share/doc'."
- (define (find-source-directory package)
- ;; For an out-of-source build, guess the source directory location
- ;; relative to the current directory. Return #f on failure.
- (match (scandir ".."
- (lambda (file)
- (and (not (member file '("." ".." "build")))
- (file-is-directory?
- (string-append "../" file)))))
- (() ;hmm, no source
- #f)
- ((source) ;only one other file
- (string-append "../" source))
- ((directories ...) ;pick the most likely one
- ;; This happens for example with libstdc++, which lives within the GCC
- ;; source tree.
- (any (lambda (directory)
- (and (string-prefix? package directory)
- (string-append "../" directory)))
- directories))))
- (define (copy-to-directories directories sub-directory)
- (lambda (file)
- (for-each (if (file-is-directory? file)
- (cut copy-recursively file <>)
- (cut install-file file <>))
- (map (cut string-append <> "/" sub-directory)
- directories))))
- (let* ((regexp (make-regexp license-file-regexp))
- (out (or (assoc-ref outputs "out")
- (match outputs
- (((_ . output) _ ...)
- output))))
- (package (strip-store-file-name out))
- (outputs (match outputs
- (((_ . outputs) ...)
- outputs)))
- (source (if out-of-source?
- (find-source-directory
- (package-name->name+version package))
- "."))
- (files (and source
- (scandir source
- (lambda (file)
- (regexp-exec regexp file))))))
- (if files
- (begin
- (format #t "installing ~a license files from '~a'~%"
- (length files) source)
- (for-each (copy-to-directories outputs
- (string-append "share/doc/"
- package))
- (map (cut string-append source "/" <>) files)))
- (format (current-error-port)
- "failed to find license files~%"))))
- (define %standard-phases
- ;; Standard build phases, as a list of symbol/procedure pairs.
- (let-syntax ((phases (syntax-rules ()
- ((_ p ...) `((p . ,p) ...)))))
- (phases set-SOURCE-DATE-EPOCH set-paths install-locale unpack
- bootstrap
- patch-usr-bin-file
- patch-source-shebangs configure patch-generated-file-shebangs
- build check install
- patch-shebangs strip
- validate-runpath
- validate-documentation-location
- delete-info-dir-file
- patch-dot-desktop-files
- make-dynamic-linker-cache
- install-license-files
- reset-gzip-timestamps
- compress-documentation)))
- (define* (gnu-build #:key (source #f) (outputs #f) (inputs #f)
- (phases %standard-phases)
- #:allow-other-keys
- #:rest args)
- "Build from SOURCE to OUTPUTS, using INPUTS, and by running all of PHASES
- in order. Return #t if all the PHASES succeeded, #f otherwise."
- (define (elapsed-time end start)
- (let ((diff (time-difference end start)))
- (+ (time-second diff)
- (/ (time-nanosecond diff) 1e9))))
- (setvbuf (current-output-port) 'line)
- (setvbuf (current-error-port) 'line)
- ;; Encoding/decoding errors shouldn't be silent.
- (fluid-set! %default-port-conversion-strategy 'error)
- (guard (c ((invoke-error? c)
- (report-invoke-error c)
- (exit 1)))
- ;; The trick is to #:allow-other-keys everywhere, so that each procedure in
- ;; PHASES can pick the keyword arguments it's interested in.
- (for-each (match-lambda
- ((name . proc)
- (let ((start (current-time time-monotonic)))
- (define (end-of-phase success?)
- (let ((end (current-time time-monotonic)))
- (format #t "phase `~a' ~:[failed~;succeeded~] after ~,1f seconds~%"
- name success?
- (elapsed-time end start))
- ;; Dump the environment variables as a shell script,
- ;; for handy debugging.
- (system "export > $NIX_BUILD_TOP/environment-variables")))
- (format #t "starting phase `~a'~%" name)
- (with-throw-handler #t
- (lambda ()
- (apply proc args)
- (end-of-phase #t))
- (lambda args
- ;; This handler executes before the stack is unwound.
- ;; The exception is automatically re-thrown from here,
- ;; and we should get a proper backtrace.
- (format (current-error-port)
- "error: in phase '~a': uncaught exception:
- ~{~s ~}~%" name args)
- (end-of-phase #f))))))
- phases)))
|