‘guile-static-initrd’ weighs in at 46 MiB, compared to 54 MiB for ‘guile-static-stripped’ (15% reduction). * gnu/packages/make-bootstrap.scm (make-guile-static-stripped): Add ‘directories-to-remove’ parameter and honor it. (%guile-static-initrd): New variable. * gnu/system/linux-initrd.scm (expression->initrd): Default to ‘%guile-static-initrd’. * doc/guix.texi (Initial RAM Disk): Adjust accordingly. Change-Id: I2baf06fed7a3698433e7c83b1d7726054a8c746e
		
			
				
	
	
		
			854 lines
		
	
	
	
		
			36 KiB
		
	
	
	
		
			Scheme
		
	
	
	
	
	
			
		
		
	
	
			854 lines
		
	
	
	
		
			36 KiB
		
	
	
	
		
			Scheme
		
	
	
	
	
	
| ;;; GNU Guix --- Functional package management for GNU
 | ||
| ;;; Copyright © 2012-2021, 2023-2024 Ludovic Courtès <ludo@gnu.org>
 | ||
| ;;; Copyright © 2017, 2021 Efraim Flashner <efraim@flashner.co.il>
 | ||
| ;;; Copyright © 2018 Tobias Geerinckx-Rice <me@tobias.gr>
 | ||
| ;;; Copyright © 2018, 2019 Mark H Weaver <mhw@netris.org>
 | ||
| ;;; Copyright © 2018, 2019, 2021, 2022, 2023 Jan (janneke) Nieuwenhuizen <janneke@gnu.org>
 | ||
| ;;; Copyright © 2019, 2020, 2022 Marius Bakke <marius@gnu.org>
 | ||
| ;;; Copyright © 2020 Mathieu Othacehe <m.othacehe@gmail.com>
 | ||
| ;;; Copyright © 2021 Pierre Langlois <pierre.langlois@gmx.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 (gnu packages make-bootstrap)
 | ||
|   #:use-module (guix gexp)
 | ||
|   #:use-module (guix utils)
 | ||
|   #:use-module (guix packages)
 | ||
|   #:use-module (guix memoization)
 | ||
|   #:use-module ((guix licenses) #:select (gpl3+))
 | ||
|   #:use-module (guix build-system trivial)
 | ||
|   #:use-module (guix build-system gnu)
 | ||
|   #:use-module ((gnu packages) #:select (search-patch search-patches))
 | ||
|   #:use-module (gnu packages base)
 | ||
|   #:use-module (gnu packages cross-base)
 | ||
|   #:use-module (gnu packages bash)
 | ||
|   #:use-module (gnu packages compression)
 | ||
|   #:use-module (gnu packages gawk)
 | ||
|   #:use-module (gnu packages gcc)
 | ||
|   #:use-module (gnu packages guile)
 | ||
|   #:use-module (gnu packages bdw-gc)
 | ||
|   #:use-module (gnu packages libunistring)
 | ||
|   #:use-module (gnu packages linux)
 | ||
|   #:use-module (gnu packages hurd)
 | ||
|   #:use-module (gnu packages mes)
 | ||
|   #:use-module (gnu packages multiprecision)
 | ||
|   #:use-module (ice-9 match)
 | ||
|   #:use-module (srfi srfi-1)
 | ||
|   #:export (%bootstrap-binaries-tarball
 | ||
|             %linux-libre-headers-bootstrap-tarball
 | ||
|             %binutils-bootstrap-tarball
 | ||
|             %glibc-bootstrap-tarball
 | ||
|             %gcc-bootstrap-tarball
 | ||
|             %guile-bootstrap-tarball
 | ||
|             %bootstrap-tarballs
 | ||
| 
 | ||
|             %guile-static-stripped
 | ||
|             %guile-static-initrd))
 | ||
| 
 | ||
| ;;; Commentary:
 | ||
| ;;;
 | ||
| ;;; This module provides tools to build tarballs of the "bootstrap binaries"
 | ||
| ;;; used in (gnu packages bootstrap).  These statically-linked binaries are
 | ||
| ;;; taken for granted and used as the root of the whole bootstrap procedure.
 | ||
| ;;;
 | ||
| ;;; Code:
 | ||
| 
 | ||
| (define glibc-for-bootstrap
 | ||
|   (mlambdaq (base)
 | ||
|     "Return a libc deriving from BASE whose `system' and `popen' functions looks
 | ||
| for `sh' in $PATH, and without nscd, and with static NSS modules."
 | ||
|     (package
 | ||
|       (inherit base)
 | ||
|       (source (origin (inherit (package-source base))
 | ||
|                       (patches (append (search-patches
 | ||
|                                         "glibc-bootstrap-system.patch")
 | ||
|                                    (origin-patches (package-source base))))))
 | ||
|       (arguments
 | ||
|        (substitute-keyword-arguments (package-arguments base)
 | ||
|          ((#:configure-flags flags)
 | ||
|           ;; Arrange so that getaddrinfo & co. do not contact the nscd,
 | ||
|           ;; and can use statically-linked NSS modules.
 | ||
|           `(cons* "--disable-nscd" "--disable-build-nscd"
 | ||
|                   "--enable-static-nss"
 | ||
|                   ,flags))))
 | ||
| 
 | ||
|       ;; Make sure to build glibc with the same compiler version as the rest
 | ||
|       ;; of the bootstrap.  Otherwise it fails to statically link on aarch64.
 | ||
|       (native-inputs
 | ||
|        `(("gcc" ,gcc-7)
 | ||
|          ,@(package-native-inputs base)))
 | ||
| 
 | ||
|       ;; Remove the 'debug' output to allow bit-reproducible builds (when the
 | ||
|       ;; 'debug' output is used, ELF files end up with a .gnu_debuglink, which
 | ||
|       ;; includes a CRC of the corresponding debugging symbols; those symbols
 | ||
|       ;; contain store file names, so the CRC changes at every rebuild.)
 | ||
|       (outputs (delete "debug" (package-outputs base))))))
 | ||
| 
 | ||
| (define gcc-for-bootstrap
 | ||
|   (mlambdaq (glibc)
 | ||
|     "Return a variant of GCC that uses the bootstrap variant of GLIBC."
 | ||
|     (package
 | ||
|       (inherit gcc-7)
 | ||
|       (outputs '("out")) ;all in one so libgcc_s is easily found
 | ||
|       (inputs
 | ||
|        `( ;; Distinguish the name so we can refer to it below.
 | ||
|          ("bootstrap-libc" ,(glibc-for-bootstrap glibc))
 | ||
|          ("libc:static" ,(glibc-for-bootstrap glibc) "static")
 | ||
|          ,@(package-inputs gcc-7))))))
 | ||
| 
 | ||
| (define (package-with-relocatable-glibc p)
 | ||
|   "Return a variant of P that uses the libc as defined by
 | ||
| `glibc-for-bootstrap'."
 | ||
| 
 | ||
|   (define (cross-bootstrap-libc target)
 | ||
|     (glibc-for-bootstrap
 | ||
|      ;; `cross-libc' already returns a cross libc, so clear
 | ||
|      ;; %CURRENT-TARGET-SYSTEM.
 | ||
|      (parameterize ((%current-target-system #f))
 | ||
|        (cross-libc target))))
 | ||
| 
 | ||
|   ;; Standard inputs with the above libc and corresponding GCC.
 | ||
| 
 | ||
|   (define (inputs)
 | ||
|     (if (%current-target-system)                ; is this package cross built?
 | ||
|         `(("cross-libc"
 | ||
|            ,(cross-bootstrap-libc (%current-target-system)))
 | ||
|           ("cross-libc:static"
 | ||
|            ,(cross-bootstrap-libc (%current-target-system))
 | ||
|            "static"))
 | ||
|         '()))
 | ||
| 
 | ||
|   (define (native-inputs)
 | ||
|     (if (%current-target-system)
 | ||
|         (let* ((target (%current-target-system))
 | ||
|                (xgcc (cross-gcc
 | ||
|                       target
 | ||
|                       #:xbinutils (cross-binutils target)
 | ||
|                       #:libc (cross-bootstrap-libc target))))
 | ||
|           `(("cross-gcc" ,(package
 | ||
|                             (inherit xgcc)
 | ||
|                             (search-paths
 | ||
|                              ;; Ensure the cross libc headers appears on the
 | ||
|                              ;; C++ system header search path.
 | ||
|                              (cons (search-path-specification
 | ||
|                                     (variable "CROSS_CPLUS_INCLUDE_PATH")
 | ||
|                                     (files '("include")))
 | ||
|                                    (package-search-paths gcc-7)))))
 | ||
|             ("cross-binutils" ,(cross-binutils target))
 | ||
|             ,@(%final-inputs)))
 | ||
|         `(("libc" ,(glibc-for-bootstrap glibc))
 | ||
|           ("libc:static" ,(glibc-for-bootstrap glibc) "static")
 | ||
|           ("gcc" ,(gcc-for-bootstrap glibc))
 | ||
|           ,@(fold alist-delete (%final-inputs) '("libc" "gcc")))))
 | ||
| 
 | ||
|   (package-with-explicit-inputs p inputs
 | ||
|                                 (current-source-location)
 | ||
|                                 #:native-inputs native-inputs))
 | ||
| 
 | ||
| (define %static-inputs
 | ||
|   ;; Packages that are to be used as %BOOTSTRAP-INPUTS.
 | ||
|   (let ((coreutils (package
 | ||
|                      (inherit coreutils)
 | ||
|                      (arguments
 | ||
|                       (substitute-keyword-arguments
 | ||
|                           (package-arguments coreutils)
 | ||
|                         ((#:configure-flags _ #~'())
 | ||
|                          #~'("--disable-nls"
 | ||
|                              "--disable-silent-rules"
 | ||
|                              "--enable-no-install-program=stdbuf,libstdbuf.so"
 | ||
|                              "CFLAGS=-Os -g0"   ; smaller, please
 | ||
|                              "LDFLAGS=-static -pthread"
 | ||
| 
 | ||
|                              ;; Work around a cross-compilation bug whereby libcoreutils.a
 | ||
|                              ;; would provide '__mktime_internal', which conflicts with the
 | ||
|                              ;; one in libc.a.
 | ||
|                              #$@(if (%current-target-system)
 | ||
|                                     #~("gl_cv_func_working_mktime=yes")
 | ||
|                                     #~())))
 | ||
|                         ((#:tests? _ #t)
 | ||
|                          #f)))               ;signal-related Gnulib tests fail
 | ||
| 
 | ||
|                       ;; Remove optional dependencies such as GMP.
 | ||
|                       (inputs '())
 | ||
| 
 | ||
|                       ;; Remove the 'debug' output (see above for the reason.)
 | ||
|                       (outputs '("out"))))
 | ||
|         (bzip2 (package
 | ||
|                  (inherit bzip2)
 | ||
|                  (arguments
 | ||
|                   (substitute-keyword-arguments (package-arguments bzip2)
 | ||
|                     ((#:phases phases)
 | ||
|                      #~(modify-phases #$phases
 | ||
|                          (add-before 'build 'dash-static
 | ||
|                            (lambda _
 | ||
|                              (substitute* "Makefile"
 | ||
|                                (("^LDFLAGS[[:blank:]]*=.*$")
 | ||
|                                 "LDFLAGS = -static"))))))))))
 | ||
|         (xz (package
 | ||
|               (inherit xz)
 | ||
|               (outputs '("out"))
 | ||
|               (arguments
 | ||
|                (list #:strip-flags #~'("--strip-all")
 | ||
|                      #:phases #~(modify-phases %standard-phases
 | ||
|                                   (add-before 'configure 'static-executable
 | ||
|                                     (lambda _
 | ||
|                                       ;; Ask Libtool for a static executable.
 | ||
|                                       (substitute* "src/xz/Makefile.in"
 | ||
|                                         (("^xz_LDADD =")
 | ||
|                                          "xz_LDADD = -all-static")))))))))
 | ||
|         (gawk (package
 | ||
|                 (inherit gawk)
 | ||
|                 (source (origin
 | ||
|                           (inherit (package-source gawk))
 | ||
|                           (modules '((guix build utils)))
 | ||
|                           (snippet
 | ||
|                            ;; Do not build 'getopt.c' since that leads to a
 | ||
|                            ;; link failure due to duplicate symbols with
 | ||
|                            ;; 'libc.a'.
 | ||
|                            '(substitute* "support/Makefile.in"
 | ||
|                               (("getopt\\.\\$\\(OBJEXT\\)") "")))
 | ||
|                           (patches (cons (search-patch "gawk-shell.patch")
 | ||
|                                          (origin-patches
 | ||
|                                           (package-source gawk))))))
 | ||
|                 (arguments
 | ||
|                  (substitute-keyword-arguments (package-arguments gawk)
 | ||
|                    ((#:configure-flags _ #~'())
 | ||
|                     ;; Starting from gawk 4.1.0, some of the tests for the
 | ||
|                     ;; plug-in mechanism just fail on static builds:
 | ||
|                     ;;
 | ||
|                     ;; ./fts.awk:1: error: can't open shared library `filefuncs' for reading (No such file or directory)
 | ||
|                     ;;
 | ||
|                     ;; Therefore disable extensions support.
 | ||
|                     #~(list "--disable-extensions"))
 | ||
|                    ((#:phases phases)
 | ||
|                     #~(modify-phases #$phases
 | ||
|                         (add-before 'configure 'no-export-dynamic
 | ||
|                           (lambda _
 | ||
|                             ;; Since we use `-static', remove
 | ||
|                             ;; `-export-dynamic'.
 | ||
|                             (substitute* "configure"
 | ||
|                               (("-Wl,-export-dynamic") ""))))))))
 | ||
|                 (inputs (if (%current-target-system)
 | ||
|                             (list static-bash)
 | ||
|                             '()))))
 | ||
| 	(tar (package
 | ||
|                (inherit tar)
 | ||
| 	       (arguments
 | ||
|                 (substitute-keyword-arguments (package-arguments tar)
 | ||
|                   ((#:configure-flags flags #~'())
 | ||
|                    ;; Work around a cross-compilation bug whereby libgnu.a
 | ||
|                    ;; would provide '__mktime_internal', which conflicts
 | ||
|                    ;; with the one in libc.a.
 | ||
|                    #~'("gl_cv_func_working_mktime=yes"))
 | ||
|                   ((#:phases phases)
 | ||
|                    #~(modify-phases #$phases
 | ||
|                        (replace 'set-shell-file-name
 | ||
|                          (lambda _
 | ||
|                            ;; Do not use "/bin/sh" to run programs; see
 | ||
|                            ;; <http://lists.gnu.org/archive/html/guix-devel/2016-09/msg02272.html>.
 | ||
|                            (substitute* "src/system.c"
 | ||
|                              (("/bin/sh") "sh")
 | ||
|                              (("execv ") "execvp "))))))))))
 | ||
|         ;; We don't want to retain a reference to /gnu/store in the bootstrap
 | ||
|         ;; versions of egrep/fgrep, so we remove the custom phase added since
 | ||
|         ;; grep@2.25. The effect is 'egrep' and 'fgrep' look for 'grep' in
 | ||
|         ;; $PATH.
 | ||
|         (grep (package
 | ||
|                 (inherit grep)
 | ||
|                 (inputs '())                   ;remove PCRE, which is optional
 | ||
|                 (arguments
 | ||
|                  (substitute-keyword-arguments (package-arguments grep)
 | ||
|                    ((#:phases phases)
 | ||
|                     `(modify-phases ,phases
 | ||
|                        (delete 'fix-egrep-and-fgrep)))))))
 | ||
|         (finalize (compose static-package
 | ||
|                            package-with-relocatable-glibc)))
 | ||
|     (append (map finalize
 | ||
|                  (list tar gzip bzip2 xz patch coreutils sed grep gawk))
 | ||
|         (list static-bash))))
 | ||
| 
 | ||
| (define %static-binaries
 | ||
|   (package
 | ||
|     (name "static-binaries")
 | ||
|     (version "0")
 | ||
|     (build-system trivial-build-system)
 | ||
|     (source #f)
 | ||
|     (inputs %static-inputs)
 | ||
|     (arguments
 | ||
|      `(#:modules ((guix build utils))
 | ||
|        #:builder
 | ||
|        (begin
 | ||
|          (use-modules (ice-9 ftw)
 | ||
|                       (ice-9 match)
 | ||
|                       (srfi srfi-1)
 | ||
|                       (srfi srfi-26)
 | ||
|                       (guix build utils))
 | ||
| 
 | ||
|          (let ()
 | ||
|           (define (directory-contents dir)
 | ||
|             (map (cut string-append dir "/" <>)
 | ||
|                  (scandir dir (negate (cut member <> '("." ".."))))))
 | ||
| 
 | ||
|           (define (copy-directory source destination)
 | ||
|             (for-each (lambda (file)
 | ||
|                         (format #t "copying ~s...~%" file)
 | ||
|                         (copy-file file
 | ||
|                                    (string-append destination "/"
 | ||
|                                                   (basename file))))
 | ||
|                       (directory-contents source)))
 | ||
| 
 | ||
|           (let* ((out (assoc-ref %outputs "out"))
 | ||
|                  (bin (string-append out "/bin")))
 | ||
|             (mkdir-p bin)
 | ||
| 
 | ||
|             ;; Copy Coreutils binaries.
 | ||
|             (let* ((coreutils (assoc-ref %build-inputs "coreutils"))
 | ||
|                    (source    (string-append coreutils "/bin")))
 | ||
|               (copy-directory source bin))
 | ||
| 
 | ||
|             ;; For the other inputs, copy just one binary, which has the
 | ||
|             ;; same name as the input.
 | ||
|             (for-each (match-lambda
 | ||
|                        ((name . dir)
 | ||
|                         (let* ((name   (if (string-prefix? "bash" name)
 | ||
|                                            "bash"
 | ||
|                                            name))
 | ||
|                                (source (string-append dir "/bin/" name)))
 | ||
|                           (format #t "copying ~s...~%" source)
 | ||
|                           (copy-file source
 | ||
|                                      (string-append bin "/" name)))))
 | ||
|                       (alist-delete "coreutils" %build-inputs))
 | ||
| 
 | ||
|             ;; But of course, there are exceptions to this rule.
 | ||
|             (let ((grep (assoc-ref %build-inputs "grep")))
 | ||
|               (install-file (string-append grep "/bin/fgrep") bin)
 | ||
|               (install-file (string-append grep "/bin/egrep") bin))
 | ||
| 
 | ||
|             ;; Clear references to the store path.
 | ||
|             (for-each remove-store-references
 | ||
|                       (directory-contents bin))
 | ||
| 
 | ||
|             (with-directory-excursion bin
 | ||
|               ;; Programs such as Perl's build system want these aliases.
 | ||
|               (symlink "bash" "sh")
 | ||
|               (symlink "gawk" "awk"))
 | ||
| 
 | ||
|             #t)))))
 | ||
|     (synopsis "Statically-linked bootstrap binaries")
 | ||
|     (description
 | ||
|      "Binaries used to bootstrap the distribution.")
 | ||
|     (license gpl3+)
 | ||
|     (home-page #f)))
 | ||
| 
 | ||
| (define %linux-libre-headers-stripped
 | ||
|   ;; The subset of Linux-Libre-Headers that we need.
 | ||
|   (package (inherit linux-libre-headers)
 | ||
|     (name (string-append (package-name linux-libre-headers) "-stripped"))
 | ||
|     (build-system trivial-build-system)
 | ||
|     (outputs '("out"))
 | ||
|     (arguments
 | ||
|      `(#:modules ((guix build utils)
 | ||
|                   (guix build make-bootstrap))
 | ||
|        #:builder
 | ||
|        (begin
 | ||
|          (use-modules (guix build utils)
 | ||
|                       (guix build make-bootstrap))
 | ||
| 
 | ||
|          (let* ((in  (assoc-ref %build-inputs "linux-libre-headers"))
 | ||
|                 (out (assoc-ref %outputs "out")))
 | ||
|            (copy-linux-headers out in)
 | ||
|            #t))))
 | ||
|     (inputs (list linux-libre-headers))))
 | ||
| 
 | ||
| (define %binutils-static
 | ||
|   ;; Statically-linked Binutils.
 | ||
|   (package (inherit binutils)
 | ||
|     (name "binutils-static")
 | ||
|     (arguments
 | ||
|      `(#:configure-flags (cons "--disable-gold"
 | ||
|                                ,(match (memq #:configure-flags
 | ||
|                                              (package-arguments binutils))
 | ||
|                                   ((#:configure-flags flags _ ...)
 | ||
|                                    flags)))
 | ||
|        #:make-flags ,(match (memq #:make-flags (package-arguments binutils))
 | ||
|                        ((#:make-flags flags _ ...)
 | ||
|                         flags)
 | ||
|                        (_ ''()))
 | ||
|        #:strip-flags '("--strip-all")
 | ||
|        #:phases (modify-phases %standard-phases
 | ||
|                   (add-before 'configure 'all-static
 | ||
|                     (lambda _
 | ||
|                       ;; The `-all-static' libtool flag can only be passed
 | ||
|                       ;; after `configure', since configure tests don't use
 | ||
|                       ;; libtool, and only for executables built with libtool.
 | ||
|                       (substitute* '("binutils/Makefile.in"
 | ||
|                                      "gas/Makefile.in"
 | ||
|                                      "ld/Makefile.in")
 | ||
|                         (("^LDFLAGS =(.*)$" line)
 | ||
|                          (string-append line
 | ||
|                                         "\nAM_LDFLAGS = -static -all-static\n")))
 | ||
|                       #t)))))))
 | ||
| 
 | ||
| (define %binutils-static-stripped
 | ||
|   ;; The subset of Binutils that we need.
 | ||
|   (package
 | ||
|     (inherit %binutils-static)
 | ||
|     (name (string-append (package-name %binutils-static) "-stripped"))
 | ||
|     (build-system trivial-build-system)
 | ||
|     (outputs '("out"))
 | ||
|     (arguments
 | ||
|      (list #:modules '((guix build utils))
 | ||
|            #:builder
 | ||
|            #~(begin
 | ||
|                (use-modules (guix build utils))
 | ||
| 
 | ||
|                (setvbuf (current-output-port)
 | ||
|                         (cond-expand (guile-2.0 _IOLBF) (else 'line)))
 | ||
|                (let* ((in  #$%binutils-static)
 | ||
|                       (out #$output)
 | ||
|                       (bin (string-append out "/bin")))
 | ||
|                  (mkdir-p bin)
 | ||
|                  (for-each (lambda (file)
 | ||
|                              (let ((target (string-append bin "/" file)))
 | ||
|                                (format #t "copying `~a'...~%" file)
 | ||
|                                (copy-file (string-append in "/bin/" file)
 | ||
|                                           target)
 | ||
|                                (remove-store-references target)))
 | ||
|                            '("ar" "as" "ld" "nm"  "objcopy" "objdump"
 | ||
|                              "ranlib" "readelf" "size" "strings" "strip"))))))))
 | ||
| 
 | ||
| (define (%glibc-stripped)
 | ||
|   ;; GNU libc's essential shared libraries, dynamic linker, and headers,
 | ||
|   ;; with all references to store directories stripped.  As a result,
 | ||
|   ;; libc.so is unusable and need to be patched for proper relocation.
 | ||
|   (let ((glibc (glibc-for-bootstrap glibc)))
 | ||
|     (package (inherit glibc)
 | ||
|       (name "glibc-stripped")
 | ||
|       (build-system trivial-build-system)
 | ||
|       (arguments
 | ||
|        `(#:modules ((guix build utils)
 | ||
|                     (guix build make-bootstrap))
 | ||
|          #:builder
 | ||
|          (begin
 | ||
|            (use-modules (guix build make-bootstrap))
 | ||
|            (make-stripped-libc (assoc-ref %outputs "out")
 | ||
|                                (assoc-ref %build-inputs "libc")
 | ||
|                                (assoc-ref %build-inputs "kernel-headers")))))
 | ||
|       (inputs `(("kernel-headers"
 | ||
|                  ,(if (or (and (%current-target-system)
 | ||
|                                (target-hurd? (%current-target-system)))
 | ||
|                           (string-suffix? "-hurd" (%current-system)))
 | ||
|                       gnumach-headers
 | ||
|                       linux-libre-headers))
 | ||
|                 ("libc" ,(let ((target (%current-target-system)))
 | ||
|                            (if target
 | ||
|                                (glibc-for-bootstrap
 | ||
|                                 (parameterize ((%current-target-system #f))
 | ||
|                                   (cross-libc target)))
 | ||
|                                glibc)))))
 | ||
|       (native-inputs '())
 | ||
|       (propagated-inputs '())
 | ||
| 
 | ||
|       ;; Only one output.
 | ||
|       (outputs '("out")))))
 | ||
| 
 | ||
| (define %gcc-static
 | ||
|   ;; A statically-linked GCC, with stripped-down functionality.
 | ||
|   (package-with-relocatable-glibc
 | ||
|    (package (inherit gcc-7)
 | ||
|      (name "gcc-static")
 | ||
|      (outputs '("out"))                           ; all in one
 | ||
|      (arguments
 | ||
|       (substitute-keyword-arguments (package-arguments gcc-7)
 | ||
|         ((#:modules modules %gnu-build-system-modules)
 | ||
|          `((srfi srfi-1)
 | ||
|            (srfi srfi-26)
 | ||
|            (ice-9 regex)
 | ||
|            ,@modules))
 | ||
|         ((#:guile _) #f)
 | ||
|         ((#:implicit-inputs? _) #t)
 | ||
|         ((#:configure-flags flags)
 | ||
|          `(append (list
 | ||
|                    ;; We don't need a full bootstrap here.
 | ||
|                    "--disable-bootstrap"
 | ||
| 
 | ||
|                    ;; Make sure '-static' is passed where it matters.
 | ||
|                    "--with-stage1-ldflags=-static"
 | ||
| 
 | ||
|                    ;; GCC 4.8+ requires a C++ compiler and library.
 | ||
|                    "--enable-languages=c,c++"
 | ||
| 
 | ||
|                    ;; Make sure gcc-nm doesn't require liblto_plugin.so.
 | ||
|                    "--disable-lto"
 | ||
| 
 | ||
|                    "--disable-shared"
 | ||
|                    "--disable-plugin"
 | ||
|                    "--disable-libmudflap"
 | ||
|                    "--disable-libatomic"
 | ||
|                    "--disable-libsanitizer"
 | ||
|                    "--disable-libitm"
 | ||
|                    "--disable-libgomp"
 | ||
|                    "--disable-libcilkrts"
 | ||
|                    "--disable-libvtv"
 | ||
|                    "--disable-libssp"
 | ||
|                    "--disable-libquadmath")
 | ||
|                   (remove (cut string-match "--(.*plugin|enable-languages)" <>)
 | ||
|                           ,flags)))
 | ||
|         ((#:phases phases)
 | ||
|          `(modify-phases ,phases
 | ||
|             (add-after 'pre-configure 'remove-lgcc_s
 | ||
|               (lambda _
 | ||
|                 ;; Remove the '-lgcc_s' added to GNU_USER_TARGET_LIB_SPEC in
 | ||
|                 ;; the 'pre-configure phase of our main gcc package, because
 | ||
|                 ;; that shared library is not present in this static gcc.  See
 | ||
|                 ;; <https://lists.gnu.org/archive/html/guix-devel/2015-01/msg00008.html>.
 | ||
|                 (substitute* (cons "gcc/config/rs6000/sysv4.h"
 | ||
|                                    (find-files "gcc/config"
 | ||
|                                                "^gnu-user.*\\.h$"))
 | ||
|                   ((" -lgcc_s}}") "}}"))
 | ||
|                 #t))))))
 | ||
|      (inputs
 | ||
|       `(("zlib:static" ,zlib "static")
 | ||
|         ("isl:static" ,isl "static")
 | ||
|         ,@(package-inputs gcc-7)))
 | ||
|      (native-inputs
 | ||
|       (if (%current-target-system)
 | ||
|           `(;; When doing a Canadian cross, we need GMP/MPFR/MPC both
 | ||
|             ;; as target inputs and as native inputs; the latter is
 | ||
|             ;; needed when building build-time tools ('genconstants',
 | ||
|             ;; etc.)  Failing to do that leads to misdetections of
 | ||
|             ;; declarations by 'gcc/configure', and eventually to
 | ||
|             ;; duplicate declarations as reported in
 | ||
|             ;; <https://gcc.gnu.org/bugzilla/show_bug.cgi?id=59217>.
 | ||
|             ("gmp-native" ,gmp)
 | ||
|             ("mpfr-native" ,mpfr)
 | ||
|             ("mpc-native" ,mpc)
 | ||
|             ,@(package-native-inputs gcc-7))
 | ||
|           (package-native-inputs gcc-7))))))
 | ||
| 
 | ||
| (define %gcc-stripped
 | ||
|   ;; The subset of GCC files needed for bootstrap.
 | ||
|   (package
 | ||
|     (inherit gcc-7)
 | ||
|     (name "gcc-stripped")
 | ||
|     (build-system trivial-build-system)
 | ||
|     (source #f)
 | ||
|     (outputs '("out"))                            ;only one output
 | ||
|     (arguments
 | ||
|      (list #:modules '((guix build utils))
 | ||
|            #:builder
 | ||
|            #~(begin
 | ||
|                (use-modules (srfi srfi-1)
 | ||
|                             (srfi srfi-26)
 | ||
|                             (guix build utils))
 | ||
| 
 | ||
|                (setvbuf (current-output-port)
 | ||
|                         (cond-expand (guile-2.0 _IOLBF) (else 'line)))
 | ||
|                (let* ((out        #$output)
 | ||
|                       (bindir     (string-append out "/bin"))
 | ||
|                       (libdir     (string-append out "/lib"))
 | ||
|                       (includedir (string-append out "/include"))
 | ||
|                       (libexecdir (string-append out "/libexec"))
 | ||
|                       (gcc        #$%gcc-static))
 | ||
|                  (copy-recursively (string-append gcc "/bin") bindir)
 | ||
|                  (for-each remove-store-references
 | ||
|                            (find-files bindir ".*"))
 | ||
| 
 | ||
|                  (copy-recursively (string-append gcc "/lib") libdir)
 | ||
|                  (for-each remove-store-references
 | ||
|                            (remove (cut string-suffix? ".h" <>)
 | ||
|                                    (find-files libdir ".*")))
 | ||
| 
 | ||
|                  (copy-recursively (string-append gcc "/libexec")
 | ||
|                                    libexecdir)
 | ||
|                  (for-each remove-store-references
 | ||
|                            (find-files libexecdir ".*"))
 | ||
| 
 | ||
|                  ;; Starting from GCC 4.8, helper programs built natively
 | ||
|                  ;; (‘genchecksum’, ‘gcc-nm’, etc.) rely on C++ headers.
 | ||
|                  (copy-recursively (string-append gcc "/include/c++")
 | ||
|                                    (string-append includedir "/c++"))
 | ||
| 
 | ||
|                  ;; For native builds, check whether the binaries actually work.
 | ||
|                  #$@(if (%current-target-system)
 | ||
|                         '()
 | ||
|                         '((for-each (lambda (prog)
 | ||
|                                       (invoke (string-append gcc "/bin/" prog)
 | ||
|                                               "--version"))
 | ||
|                                     '("gcc" "g++" "cpp"))))))))))
 | ||
| 
 | ||
| (define* (make-guile-static guile patches)
 | ||
|   (package-with-relocatable-glibc
 | ||
|    (static-package
 | ||
|     (package
 | ||
|       (inherit guile)
 | ||
|       (source
 | ||
|        (origin (inherit (package-source guile))
 | ||
|                (patches (append (map search-patch patches)
 | ||
|                                 (origin-patches (package-source guile))))))
 | ||
|       (name (string-append (package-name guile) "-static"))
 | ||
|       (synopsis "Statically-linked and relocatable Guile")
 | ||
| 
 | ||
|       ;; Remove the 'debug' output (see above for the reason.)
 | ||
|       (outputs (delete "debug" (package-outputs guile)))
 | ||
| 
 | ||
|       (inputs
 | ||
|        (modify-inputs (package-inputs guile)
 | ||
|          (prepend `(,libunistring "static"))))
 | ||
| 
 | ||
|       (propagated-inputs
 | ||
|        (modify-inputs (package-propagated-inputs guile)
 | ||
|          (replace "libgc" libgc/static-libs)))
 | ||
|       (arguments
 | ||
|        (substitute-keyword-arguments (package-arguments guile)
 | ||
|          ((#:configure-flags flags #~'())
 | ||
|           ;; When `configure' checks for ltdl availability, it
 | ||
|           ;; doesn't try to link using libtool, and thus fails
 | ||
|           ;; because of a missing -ldl.  Work around that.
 | ||
|           #~(list "LDFLAGS=-ldl" "--enable-mini-gmp"
 | ||
| 
 | ||
|                   ;; Guile does an LTO build by default, but in 3.0.9 it
 | ||
|                   ;; wrongfully picks 'ar' instead of 'gcc-ar', so work around
 | ||
|                   ;; it (see <https://issues.guix.gnu.org/61086>).
 | ||
|                   #$@(if (version-prefix? "3.0" (package-version guile))
 | ||
|                          #~("AR=gcc-ar" "RANLIB=gcc-ranlib")
 | ||
|                          #~())
 | ||
| 
 | ||
|                   #$@(if (target-hurd?)
 | ||
|                          #~("--disable-jit")
 | ||
|                          #~())))
 | ||
|          ((#:phases phases '%standard-phases)
 | ||
|           #~(modify-phases #$phases
 | ||
| 
 | ||
|               ;; Do not record the absolute file name of 'sh' in
 | ||
|               ;; (ice-9 popen).  This makes 'open-pipe' unusable in
 | ||
|               ;; a build chroot ('open-pipe*' is fine) but avoids
 | ||
|               ;; keeping a reference to Bash.
 | ||
|               (delete 'pre-configure)
 | ||
| 
 | ||
|               (add-before 'configure 'static-guile
 | ||
|                 (lambda _
 | ||
|                   (substitute* "libguile/Makefile.in"
 | ||
|                     ;; Create a statically-linked `guile'
 | ||
|                     ;; executable.
 | ||
|                     (("^guile_LDFLAGS =")
 | ||
|                      "guile_LDFLAGS = -all-static")
 | ||
| 
 | ||
|                     ;; Add `-ldl' *after* libguile-2.0.la.
 | ||
|                     (("^guile_LDADD =(.*)$" _ ldadd)
 | ||
|                      (string-append "guile_LDADD = "
 | ||
|                                     (string-trim-right ldadd)
 | ||
|                                     " -ldl\n")))))))
 | ||
|          ((#:tests? _ #f)
 | ||
|           ;; There are uses of `dynamic-link' in
 | ||
|           ;; {foreign,coverage}.test that don't fly here.
 | ||
|           #f)
 | ||
|          ((#:parallel-build? _ #f)
 | ||
|           ;; Work around the fact that the Guile build system is
 | ||
|           ;; not deterministic when parallel-build is enabled.
 | ||
|           #f)))))))
 | ||
| 
 | ||
| (define %guile-static
 | ||
|   ;; A statically-linked Guile that is relocatable--i.e., it can search
 | ||
|   ;; .scm and .go files relative to its installation directory, rather
 | ||
|   ;; than in hard-coded configure-time paths.
 | ||
|   (make-guile-static guile-2.0 '("guile-relocatable.patch"
 | ||
|                                  "guile-default-utf8.patch"
 | ||
|                                  "guile-linux-syscalls.patch")))
 | ||
| 
 | ||
| (define %guile-static-3.0
 | ||
|   (make-guile-static guile-3.0-latest
 | ||
|                      '("guile-2.2-default-utf8.patch"
 | ||
|                        "guile-3.0-linux-syscalls.patch"
 | ||
|                        "guile-3.0-relocatable.patch")))
 | ||
| 
 | ||
| (define* (make-guile-static-stripped static-guile
 | ||
|                                      #:optional (directories-to-remove '()))
 | ||
|   (package
 | ||
|     (inherit static-guile)
 | ||
|     (name (string-append (package-name static-guile) "-stripped"))
 | ||
|     (build-system trivial-build-system)
 | ||
|     (arguments
 | ||
|      ;; The end result should depend on nothing but itself.
 | ||
|      (list #:allowed-references '("out")
 | ||
|            #:modules '((guix build utils))
 | ||
|            #:builder
 | ||
|            #~(let ((version #$(version-major+minor (package-version static-guile))))
 | ||
|                (use-modules (guix build utils))
 | ||
| 
 | ||
|                (let* ((in     #$static-guile)
 | ||
|                       (out    #$output)
 | ||
|                       (guile1 (string-append in "/bin/guile"))
 | ||
|                       (guile2 (string-append out "/bin/guile")))
 | ||
|                  (mkdir-p (string-append out "/share/guile/" version))
 | ||
|                  (copy-recursively (string-append in "/share/guile/" version)
 | ||
|                                    (string-append out "/share/guile/" version))
 | ||
| 
 | ||
|                  (mkdir-p (string-append out "/lib/guile/" version "/ccache"))
 | ||
|                  (copy-recursively (string-append in "/lib/guile/" version "/ccache")
 | ||
|                                    (string-append out "/lib/guile/" version "/ccache"))
 | ||
| 
 | ||
|                  (mkdir (string-append out "/bin"))
 | ||
|                  (copy-file guile1 guile2)
 | ||
| 
 | ||
|                  ;; Optionally remove additional directories.
 | ||
|                  (for-each (lambda (directory)
 | ||
|                              (delete-file-recursively
 | ||
|                               (string-append out "/" directory)))
 | ||
|                            '#$directories-to-remove)
 | ||
| 
 | ||
|                  ;; Verify that the relocated Guile works.
 | ||
|                  #$@(if (%current-target-system)
 | ||
|                         '()
 | ||
|                         '((invoke guile2 "--version")))
 | ||
| 
 | ||
|                  ;; Strip store references.
 | ||
|                  (remove-store-references guile2)
 | ||
| 
 | ||
|                  ;; Verify that the stripped Guile works.  If it aborts, it could be
 | ||
|                  ;; that it tries to open iconv descriptors and fails because libc's
 | ||
|                  ;; iconv data isn't available (see `guile-default-utf8.patch'.)
 | ||
|                  #$@(if (%current-target-system)
 | ||
|                         '()
 | ||
|                         '((invoke guile2 "--version")))))))
 | ||
|     (outputs '("out"))
 | ||
|     (synopsis "Minimal statically-linked and relocatable Guile")))
 | ||
| 
 | ||
| (define %guile-static-stripped
 | ||
|   ;; A stripped static Guile 3.0 binary for use during bootstrap.
 | ||
|   (make-guile-static-stripped %guile-static-3.0))
 | ||
| 
 | ||
| (define %guile-static-initrd
 | ||
|   ;; A stripped static Guile 3.0 binary for use in initrds.  Remove various
 | ||
|   ;; modules that are useless in an initrd.  Note: Keep most of language/
 | ||
|   ;; because it is needed for Bournish.
 | ||
|   (package
 | ||
|     (inherit
 | ||
|      (make-guile-static-stripped
 | ||
|       %guile-static-3.0
 | ||
|       (append-map (lambda (directory)
 | ||
|                     (list (string-append "lib/guile/3.0/ccache/" directory)
 | ||
|                           (string-append "share/guile/3.0/" directory)))
 | ||
|                   '("language/brainfuck"
 | ||
|                     "language/ecmascript"
 | ||
|                     "language/elisp"
 | ||
|                     "oop"
 | ||
|                     "scripts"
 | ||
|                     "texinfo"
 | ||
|                     "web"))))
 | ||
|     (name "guile-static-initrd")))
 | ||
| 
 | ||
| (define (tarball-package pkg)
 | ||
|   "Return a package containing a tarball of PKG."
 | ||
|   (package
 | ||
|     (inherit pkg)
 | ||
|     (name (string-append (package-name pkg) "-tarball"))
 | ||
|     (build-system trivial-build-system)
 | ||
|     (native-inputs (list tar xz))
 | ||
|     (arguments
 | ||
|      (list #:modules '((guix build utils))
 | ||
|            #:builder
 | ||
|            #~(begin
 | ||
|                (use-modules (guix build utils))
 | ||
|                (let ((out   #$output)
 | ||
|                      (input #$pkg)
 | ||
|                      (tar   #+(this-package-native-input "tar"))
 | ||
|                      (xz    #+(this-package-native-input "xz")))
 | ||
|                  (mkdir out)
 | ||
|                  (set-path-environment-variable "PATH" '("bin") (list tar xz))
 | ||
|                  (with-directory-excursion input
 | ||
|                    (invoke "tar" "cJvf"
 | ||
|                            (string-append out "/"
 | ||
|                                           #$(package-name pkg) "-"
 | ||
|                                           #$(package-version pkg)
 | ||
|                                           "-"
 | ||
|                                           #$(or (%current-target-system)
 | ||
|                                                 (%current-system))
 | ||
|                                           ".tar.xz")
 | ||
|                            "."
 | ||
|                            ;; avoid non-determinism in the archive
 | ||
|                            "--sort=name" "--mtime=@0"
 | ||
|                            "--owner=root:0" "--group=root:0"))))))))
 | ||
| 
 | ||
| (define %bootstrap-binaries-tarball
 | ||
|   ;; A tarball with the statically-linked bootstrap binaries.
 | ||
|   (tarball-package %static-binaries))
 | ||
| 
 | ||
| (define %linux-libre-headers-bootstrap-tarball
 | ||
|   ;; A tarball with the statically-linked Linux-Libre-Headers programs.
 | ||
|   (tarball-package %linux-libre-headers-stripped))
 | ||
| 
 | ||
| (define %binutils-bootstrap-tarball
 | ||
|   ;; A tarball with the statically-linked Binutils programs.
 | ||
|   (tarball-package %binutils-static-stripped))
 | ||
| 
 | ||
| (define (%glibc-bootstrap-tarball)
 | ||
|   ;; A tarball with GNU libc's shared libraries, dynamic linker, and headers.
 | ||
|   (tarball-package (%glibc-stripped)))
 | ||
| 
 | ||
| (define %gcc-bootstrap-tarball
 | ||
|   ;; A tarball with a dynamic-linked GCC and its headers.
 | ||
|   (tarball-package %gcc-stripped))
 | ||
| 
 | ||
| (define %guile-bootstrap-tarball
 | ||
|   ;; A tarball with the statically-linked, relocatable Guile.
 | ||
|   (tarball-package %guile-static-stripped))
 | ||
| 
 | ||
| (define %bootstrap-tarballs
 | ||
|   ;; A single derivation containing all the bootstrap tarballs, for
 | ||
|   ;; convenience.
 | ||
|   (package
 | ||
|     (name "bootstrap-tarballs")
 | ||
|     (version "0")
 | ||
|     (source #f)
 | ||
|     (build-system trivial-build-system)
 | ||
|     (arguments
 | ||
|      (list #:modules '((guix build utils))
 | ||
|            #:builder
 | ||
|            #~(begin
 | ||
|                (use-modules (guix build utils)
 | ||
|                             (ice-9 match)
 | ||
|                             (srfi srfi-26))
 | ||
| 
 | ||
|                (define out #$output)
 | ||
| 
 | ||
|                (setvbuf (current-output-port)
 | ||
|                         (cond-expand (guile-2.0 _IOLBF) (else 'line)))
 | ||
|                (mkdir out)
 | ||
|                (chdir out)
 | ||
|                (for-each (match-lambda
 | ||
|                            ((name . directory)
 | ||
|                             (for-each (lambda (file)
 | ||
|                                         (format #t "~a -> ~a~%" file out)
 | ||
|                                         (symlink file (basename file)))
 | ||
|                                       (find-files directory "\\.tar\\."))))
 | ||
|                          %build-inputs))))
 | ||
|     (inputs
 | ||
|      (append (list %guile-bootstrap-tarball)
 | ||
|          (match (or (%current-target-system) (%current-system))
 | ||
|            ((or "i686-linux" "x86_64-linux")
 | ||
|             (list %linux-libre-headers-bootstrap-tarball))
 | ||
|            (_
 | ||
|             (list %gcc-bootstrap-tarball
 | ||
|                   %binutils-bootstrap-tarball
 | ||
|                   (%glibc-bootstrap-tarball)
 | ||
|                   %bootstrap-binaries-tarball)))))
 | ||
|     (synopsis "Tarballs containing all the bootstrap binaries")
 | ||
|     (description synopsis)
 | ||
|     (home-page #f)
 | ||
|     (license gpl3+)))
 | ||
| 
 | ||
| ;;; make-bootstrap.scm ends here
 |