2014-04-11 07:38:11 -04:00
|
|
|
|
;;; GNU Guix --- Functional package management for GNU
|
2020-03-26 18:21:11 -04:00
|
|
|
|
;;; Copyright © 2013, 2014, 2015, 2016, 2017, 2018, 2019, 2020 Ludovic Courtès <ludo@gnu.org>
|
2021-08-15 14:15:37 -04:00
|
|
|
|
;;; Copyright © 2016 Christine Lemmer-Webber <cwebber@dustycloud.org>
|
2017-12-02 20:17:45 -05:00
|
|
|
|
;;; Copyright © 2016, 2017 Leo Famulari <leo@famulari.name>
|
2017-04-15 08:22:24 -04:00
|
|
|
|
;;; Copyright © 2017 Mathieu Othacehe <m.othacehe@gmail.com>
|
2017-04-11 04:47:38 -04:00
|
|
|
|
;;; Copyright © 2017 Marius Bakke <mbakke@fastmail.com>
|
2018-03-15 00:09:11 -04:00
|
|
|
|
;;; Copyright © 2018 Chris Marusich <cmmarusich@gmail.com>
|
2020-04-06 11:48:21 -04:00
|
|
|
|
;;; Copyright © 2020 Tobias Geerinckx-Rice <me@tobias.gr>
|
2014-04-11 07:38:11 -04:00
|
|
|
|
;;;
|
|
|
|
|
;;; 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/>.
|
|
|
|
|
|
2014-09-03 04:47:05 -04:00
|
|
|
|
(define-module (gnu build vm)
|
2014-04-11 07:38:11 -04:00
|
|
|
|
#:use-module (guix build utils)
|
2014-09-04 17:05:12 -04:00
|
|
|
|
#:use-module (guix build store-copy)
|
2017-04-06 02:55:45 -04:00
|
|
|
|
#:use-module (guix build syscalls)
|
2018-10-27 18:17:08 -04:00
|
|
|
|
#:use-module (guix store database)
|
2020-04-28 10:17:59 -04:00
|
|
|
|
#:use-module (gnu build bootloader)
|
2014-09-03 05:14:12 -04:00
|
|
|
|
#:use-module (gnu build linux-boot)
|
2014-09-03 04:47:05 -04:00
|
|
|
|
#:use-module (gnu build install)
|
file-systems: Introduce (gnu system uuid).
* gnu/build/file-systems.scm (sub-bytevector)
(latin1->string, %fat32-endianness, fat32-uuid->string)
(%iso9660-uuid-rx, string->iso9660-uuid)
(iso9660-uuid->string, %network-byte-order)
(dce-uuid->string, %uuid-rx, string->dce-uuid)
(string->ext2-uuid, string->ext3-uuid, string->ext4-uuid)
(vhashq, %uuid-parsers, %uuid-printers, string->uuid)
(uuid->string): Move to...
* gnu/system/uuid.scm: ... here. New file.
* gnu/system/file-systems.scm (uuid): Move to the above file.
* gnu/system/vm.scm: Adjust accordingly.
* gnu/local.mk (GNU_SYSTEM_MODULES): Add uuid.scm.
2017-09-05 15:51:12 -04:00
|
|
|
|
#:use-module (gnu system uuid)
|
2015-07-25 17:57:52 -04:00
|
|
|
|
#:use-module (guix records)
|
2017-06-29 18:13:54 -04:00
|
|
|
|
#:use-module ((guix combinators) #:select (fold2))
|
2017-05-07 09:31:30 -04:00
|
|
|
|
#:use-module (ice-9 format)
|
2020-04-06 09:16:09 -04:00
|
|
|
|
#:use-module (ice-9 ftw)
|
2014-04-11 12:44:53 -04:00
|
|
|
|
#:use-module (ice-9 match)
|
2014-07-13 10:07:26 -04:00
|
|
|
|
#:use-module (ice-9 regex)
|
2018-06-26 07:47:30 -04:00
|
|
|
|
#:use-module (ice-9 popen)
|
2015-07-25 17:57:52 -04:00
|
|
|
|
#:use-module (srfi srfi-1)
|
|
|
|
|
#:use-module (srfi srfi-9)
|
2019-04-20 16:34:28 -04:00
|
|
|
|
#:use-module (srfi srfi-19)
|
2014-04-11 12:44:53 -04:00
|
|
|
|
#:use-module (srfi srfi-26)
|
2014-07-13 10:07:26 -04:00
|
|
|
|
#:export (qemu-command
|
|
|
|
|
load-in-linux-vm
|
2014-05-21 17:31:46 -04:00
|
|
|
|
format-partition
|
2015-07-25 17:57:52 -04:00
|
|
|
|
|
|
|
|
|
partition
|
|
|
|
|
partition?
|
|
|
|
|
partition-device
|
|
|
|
|
partition-size
|
|
|
|
|
partition-file-system
|
|
|
|
|
partition-label
|
2017-04-11 04:47:38 -04:00
|
|
|
|
partition-flags
|
2015-07-25 17:57:52 -04:00
|
|
|
|
partition-initializer
|
|
|
|
|
|
2017-06-29 18:04:38 -04:00
|
|
|
|
estimated-partition-size
|
2015-07-25 17:57:52 -04:00
|
|
|
|
root-partition-initializer
|
2014-05-21 17:31:46 -04:00
|
|
|
|
initialize-partition-table
|
2020-04-28 10:37:57 -04:00
|
|
|
|
initialize-hard-disk))
|
2014-04-11 07:38:11 -04:00
|
|
|
|
|
|
|
|
|
;;; Commentary:
|
|
|
|
|
;;;
|
|
|
|
|
;;; This module provides supporting code to run virtual machines and build
|
|
|
|
|
;;; virtual machine images using QEMU.
|
|
|
|
|
;;;
|
|
|
|
|
;;; Code:
|
|
|
|
|
|
2014-07-13 10:07:26 -04:00
|
|
|
|
(define* (qemu-command #:optional (system %host-type))
|
|
|
|
|
"Return the default name of the QEMU command for SYSTEM."
|
2016-06-20 17:03:08 -04:00
|
|
|
|
(let ((cpu (substring system 0
|
|
|
|
|
(string-index system #\-))))
|
2014-07-13 10:07:26 -04:00
|
|
|
|
(string-append "qemu-system-"
|
2019-12-01 10:46:00 -05:00
|
|
|
|
(cond
|
|
|
|
|
((string-match "^i[3456]86$" cpu) "i386")
|
|
|
|
|
((string-match "armhf" cpu) "arm")
|
|
|
|
|
(else cpu)))))
|
2014-04-11 07:38:11 -04:00
|
|
|
|
|
|
|
|
|
(define* (load-in-linux-vm builder
|
|
|
|
|
#:key
|
|
|
|
|
output
|
|
|
|
|
(qemu (qemu-command)) (memory-size 512)
|
|
|
|
|
linux initrd
|
2017-06-29 17:11:23 -04:00
|
|
|
|
make-disk-image?
|
2017-07-18 15:38:16 -04:00
|
|
|
|
single-file-output?
|
2017-06-29 17:11:23 -04:00
|
|
|
|
(disk-image-size (* 100 (expt 2 20)))
|
2014-05-22 16:30:13 -04:00
|
|
|
|
(disk-image-format "qcow2")
|
2014-04-11 07:38:11 -04:00
|
|
|
|
(references-graphs '()))
|
|
|
|
|
"Run BUILDER, a Scheme file, into a VM running LINUX with INITRD, and copy
|
2017-07-18 15:38:16 -04:00
|
|
|
|
the result to OUTPUT. If SINGLE-FILE-OUTPUT? is true, copy a single file from
|
|
|
|
|
/xchg to OUTPUT. Otherwise, copy the contents of /xchg to a new directory
|
|
|
|
|
OUTPUT.
|
2014-04-11 07:38:11 -04:00
|
|
|
|
|
|
|
|
|
When MAKE-DISK-IMAGE? is true, OUTPUT will contain a VM image of
|
2017-06-29 17:11:23 -04:00
|
|
|
|
DISK-IMAGE-SIZE bytes resulting from the execution of BUILDER, which may
|
|
|
|
|
access it via /dev/hda.
|
2014-04-11 07:38:11 -04:00
|
|
|
|
|
|
|
|
|
REFERENCES-GRAPHS can specify a list of reference-graph files as produced by
|
|
|
|
|
the #:references-graphs parameter of 'derivation'."
|
2017-12-05 05:34:01 -05:00
|
|
|
|
|
2020-05-27 17:04:48 -04:00
|
|
|
|
(define target-arm32?
|
|
|
|
|
(string-prefix? "arm-" %host-type))
|
|
|
|
|
|
|
|
|
|
(define target-aarch64?
|
|
|
|
|
(string-prefix? "aarch64-" %host-type))
|
|
|
|
|
|
|
|
|
|
(define target-arm?
|
|
|
|
|
(or target-arm32? target-aarch64?))
|
2019-08-20 12:00:40 -04:00
|
|
|
|
|
2017-12-05 05:34:01 -05:00
|
|
|
|
(define arch-specific-flags
|
|
|
|
|
`(;; On ARM, a machine has to be specified. Use "virt" machine to avoid
|
|
|
|
|
;; hardware limits imposed by other machines.
|
2019-08-20 12:00:40 -04:00
|
|
|
|
,@(if target-arm?
|
|
|
|
|
'("-M" "virt")
|
|
|
|
|
'())
|
2017-12-05 05:34:01 -05:00
|
|
|
|
|
2019-08-23 03:51:33 -04:00
|
|
|
|
;; On ARM32, if the kernel is built without LPAE support, ECAM conflicts
|
|
|
|
|
;; with VIRT_PCIE_MMIO causing PCI devices not to show up. Disable
|
|
|
|
|
;; explicitely highmem to fix it.
|
|
|
|
|
;; See: https://bugs.launchpad.net/qemu/+bug/1790975.
|
|
|
|
|
,@(if target-arm32?
|
|
|
|
|
'("-machine" "highmem=off")
|
|
|
|
|
'())
|
|
|
|
|
|
2017-12-05 05:34:01 -05:00
|
|
|
|
;; Only enable kvm if we see /dev/kvm exists. This allows users without
|
|
|
|
|
;; hardware virtualization to still use these commands. KVM support is
|
2019-08-20 12:00:40 -04:00
|
|
|
|
;; still buggy on some ARM boards. Do not use it even if available.
|
2017-12-05 05:34:01 -05:00
|
|
|
|
,@(if (and (file-exists? "/dev/kvm")
|
2019-08-20 12:00:40 -04:00
|
|
|
|
(not target-arm?))
|
2017-12-05 05:34:01 -05:00
|
|
|
|
'("-enable-kvm")
|
|
|
|
|
'())
|
2018-05-25 06:04:22 -04:00
|
|
|
|
|
|
|
|
|
;; Pass "panic=1" so that the guest dies upon error.
|
2017-12-05 05:34:01 -05:00
|
|
|
|
"-append"
|
2018-05-25 06:04:22 -04:00
|
|
|
|
,(string-append "panic=1 --load=" builder
|
|
|
|
|
|
|
|
|
|
;; The serial port name differs between emulated
|
|
|
|
|
;; architectures/machines.
|
|
|
|
|
" console="
|
2019-12-06 14:53:08 -05:00
|
|
|
|
(if target-arm? "ttyAMA0" "ttyS0"))))
|
2017-12-05 05:34:01 -05:00
|
|
|
|
|
2014-04-11 07:38:11 -04:00
|
|
|
|
(when make-disk-image?
|
2017-06-29 17:26:35 -04:00
|
|
|
|
(format #t "creating ~a image of ~,2f MiB...~%"
|
|
|
|
|
disk-image-format (/ disk-image-size (expt 2 20)))
|
|
|
|
|
(force-output)
|
2018-03-15 11:30:41 -04:00
|
|
|
|
(invoke "qemu-img" "create" "-f" disk-image-format output
|
|
|
|
|
(number->string disk-image-size)))
|
2014-04-11 07:38:11 -04:00
|
|
|
|
|
|
|
|
|
(mkdir "xchg")
|
2018-03-15 00:09:12 -04:00
|
|
|
|
(mkdir "tmp")
|
2014-04-11 07:38:11 -04:00
|
|
|
|
|
|
|
|
|
(match references-graphs
|
|
|
|
|
((graph-files ...)
|
|
|
|
|
;; Copy the reference-graph files under xchg/ so EXP can access it.
|
|
|
|
|
(map (lambda (file)
|
|
|
|
|
(copy-file file (string-append "xchg/" file)))
|
|
|
|
|
graph-files))
|
|
|
|
|
(_ #f))
|
|
|
|
|
|
2018-03-15 11:30:41 -04:00
|
|
|
|
(apply invoke qemu "-nographic" "-no-reboot"
|
2019-08-20 12:00:40 -04:00
|
|
|
|
;; CPU "max" behaves as "host" when KVM is enabled, and like a system
|
|
|
|
|
;; CPU with the maximum possible feature set otherwise.
|
|
|
|
|
"-cpu" "max"
|
2018-03-15 11:30:41 -04:00
|
|
|
|
"-m" (number->string memory-size)
|
2019-12-06 14:53:08 -05:00
|
|
|
|
"-nic" "user,model=virtio-net-pci"
|
2020-09-20 15:14:52 -04:00
|
|
|
|
"-object" "rng-random,filename=/dev/urandom,id=guix-vm-rng"
|
|
|
|
|
"-device" "virtio-rng-pci,rng=guix-vm-rng"
|
2018-03-15 11:30:41 -04:00
|
|
|
|
"-virtfs"
|
|
|
|
|
(string-append "local,id=store_dev,path="
|
|
|
|
|
(%store-directory)
|
|
|
|
|
",security_model=none,mount_tag=store")
|
|
|
|
|
"-virtfs"
|
|
|
|
|
(string-append "local,id=xchg_dev,path=xchg"
|
|
|
|
|
",security_model=none,mount_tag=xchg")
|
2018-03-15 00:09:12 -04:00
|
|
|
|
"-virtfs"
|
|
|
|
|
;; Some programs require more space in /tmp than is normally
|
|
|
|
|
;; available in the guest. Accommodate such programs by sharing a
|
|
|
|
|
;; temporary directory.
|
|
|
|
|
(string-append "local,id=tmp_dev,path=tmp"
|
|
|
|
|
",security_model=none,mount_tag=tmp")
|
2018-03-15 11:30:41 -04:00
|
|
|
|
"-kernel" linux
|
|
|
|
|
"-initrd" initrd
|
|
|
|
|
(append
|
|
|
|
|
(if make-disk-image?
|
|
|
|
|
`("-device" "virtio-blk,drive=myhd"
|
|
|
|
|
"-drive" ,(string-append "if=none,file=" output
|
|
|
|
|
",format=" disk-image-format
|
|
|
|
|
",id=myhd"))
|
|
|
|
|
'())
|
|
|
|
|
arch-specific-flags))
|
2014-04-11 07:38:11 -04:00
|
|
|
|
|
2020-03-26 18:21:11 -04:00
|
|
|
|
(unless (file-exists? "xchg/.exit-status")
|
|
|
|
|
(error "VM did not produce an exit code"))
|
|
|
|
|
|
|
|
|
|
(match (call-with-input-file "xchg/.exit-status" read)
|
|
|
|
|
(0 #t)
|
|
|
|
|
(status (error "guest VM code exited with a non-zero status" status)))
|
|
|
|
|
|
|
|
|
|
(delete-file "xchg/.exit-status")
|
|
|
|
|
|
2016-11-28 16:56:41 -05:00
|
|
|
|
;; When MAKE-DISK-IMAGE? is true, the image is in OUTPUT already.
|
|
|
|
|
(unless make-disk-image?
|
2017-07-18 15:38:16 -04:00
|
|
|
|
(if single-file-output?
|
|
|
|
|
(let ((graph? (lambda (name stat)
|
|
|
|
|
(member (basename name) references-graphs))))
|
|
|
|
|
(match (find-files "xchg" (negate graph?))
|
|
|
|
|
((result)
|
|
|
|
|
(copy-file result output))
|
|
|
|
|
(x
|
|
|
|
|
(error "did not find a single result file" x))))
|
|
|
|
|
(begin
|
|
|
|
|
(mkdir output)
|
|
|
|
|
(copy-recursively "xchg" output)))))
|
2014-04-11 07:38:11 -04:00
|
|
|
|
|
2018-10-27 18:17:08 -04:00
|
|
|
|
(define* (register-closure prefix closure
|
|
|
|
|
#:key
|
|
|
|
|
(schema (sql-schema)))
|
|
|
|
|
"Register CLOSURE in PREFIX, where PREFIX is the directory name of the
|
|
|
|
|
target store and CLOSURE is the name of a file containing a reference graph as
|
2020-12-10 15:25:39 -05:00
|
|
|
|
produced by #:references-graphs."
|
2018-10-27 18:17:08 -04:00
|
|
|
|
(let ((items (call-with-input-file closure read-reference-graph)))
|
2020-06-18 05:51:44 -04:00
|
|
|
|
(parameterize ((sql-schema schema))
|
|
|
|
|
(with-database (store-database-file #:prefix prefix) db
|
|
|
|
|
(register-items db items
|
|
|
|
|
#:prefix prefix
|
|
|
|
|
#:registration-time %epoch)))))
|
2018-10-27 18:17:08 -04:00
|
|
|
|
|
2015-07-25 17:57:52 -04:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Partitions.
|
|
|
|
|
;;;
|
|
|
|
|
|
|
|
|
|
(define-record-type* <partition> partition make-partition
|
|
|
|
|
partition?
|
|
|
|
|
(device partition-device (default #f))
|
|
|
|
|
(size partition-size)
|
|
|
|
|
(file-system partition-file-system (default "ext4"))
|
2020-04-01 09:08:11 -04:00
|
|
|
|
(file-system-options partition-file-system-options ;passed to 'mkfs.FS'
|
|
|
|
|
(default '()))
|
2015-07-25 17:57:52 -04:00
|
|
|
|
(label partition-label (default #f))
|
2017-07-19 18:15:43 -04:00
|
|
|
|
(uuid partition-uuid (default #f))
|
2017-04-11 04:47:38 -04:00
|
|
|
|
(flags partition-flags (default '()))
|
2015-07-25 17:57:52 -04:00
|
|
|
|
(initializer partition-initializer (default (const #t))))
|
|
|
|
|
|
2017-06-29 18:04:38 -04:00
|
|
|
|
(define (estimated-partition-size graphs)
|
|
|
|
|
"Return the estimated size of a partition that can store the store items
|
|
|
|
|
given by GRAPHS, a list of file names produced by #:references-graphs."
|
2017-07-18 10:30:14 -04:00
|
|
|
|
;; Simply add a 25% overhead.
|
|
|
|
|
(round (* 1.25 (closure-size graphs))))
|
2017-06-29 18:04:38 -04:00
|
|
|
|
|
2015-07-25 17:57:52 -04:00
|
|
|
|
(define* (initialize-partition-table device partitions
|
2014-04-11 12:44:53 -04:00
|
|
|
|
#:key
|
|
|
|
|
(label-type "msdos")
|
2014-05-21 17:31:46 -04:00
|
|
|
|
(offset (expt 2 20)))
|
2015-07-25 17:57:52 -04:00
|
|
|
|
"Create on DEVICE a partition table of type LABEL-TYPE, containing the given
|
|
|
|
|
PARTITIONS (a list of <partition> objects), starting at OFFSET bytes. On
|
|
|
|
|
success, return PARTITIONS with their 'device' field changed to reflect their
|
|
|
|
|
actual /dev name based on DEVICE."
|
|
|
|
|
(define (partition-options part offset index)
|
|
|
|
|
(cons* "mkpart" "primary" "ext2"
|
|
|
|
|
(format #f "~aB" offset)
|
|
|
|
|
(format #f "~aB" (+ offset (partition-size part)))
|
2017-04-11 04:47:38 -04:00
|
|
|
|
(append-map (lambda (flag)
|
|
|
|
|
(list "set" (number->string index)
|
|
|
|
|
(symbol->string flag) "on"))
|
|
|
|
|
(partition-flags part))))
|
2015-07-25 17:57:52 -04:00
|
|
|
|
|
|
|
|
|
(define (options partitions offset)
|
|
|
|
|
(let loop ((partitions partitions)
|
|
|
|
|
(offset offset)
|
|
|
|
|
(index 1)
|
|
|
|
|
(result '()))
|
|
|
|
|
(match partitions
|
|
|
|
|
(()
|
|
|
|
|
(concatenate (reverse result)))
|
|
|
|
|
((head tail ...)
|
|
|
|
|
(loop tail
|
|
|
|
|
;; Leave one sector (512B) between partitions to placate
|
|
|
|
|
;; Parted.
|
|
|
|
|
(+ offset 512 (partition-size head))
|
|
|
|
|
(+ 1 index)
|
|
|
|
|
(cons (partition-options head offset index)
|
|
|
|
|
result))))))
|
|
|
|
|
|
2017-06-29 17:26:35 -04:00
|
|
|
|
(format #t "creating partition table with ~a partitions (~a)...\n"
|
|
|
|
|
(length partitions)
|
|
|
|
|
(string-join (map (compose (cut string-append <> " MiB")
|
|
|
|
|
number->string
|
|
|
|
|
(lambda (size)
|
|
|
|
|
(round (/ size (expt 2. 20))))
|
|
|
|
|
partition-size)
|
|
|
|
|
partitions)
|
|
|
|
|
", "))
|
2018-03-15 11:30:41 -04:00
|
|
|
|
(apply invoke "parted" "--script"
|
|
|
|
|
device "mklabel" label-type
|
|
|
|
|
(options partitions offset))
|
2015-07-25 17:57:52 -04:00
|
|
|
|
|
|
|
|
|
;; Set the 'device' field of each partition.
|
|
|
|
|
(reverse
|
|
|
|
|
(fold2 (lambda (part result index)
|
|
|
|
|
(values (cons (partition
|
|
|
|
|
(inherit part)
|
|
|
|
|
(device (string-append device
|
|
|
|
|
(number->string index))))
|
|
|
|
|
result)
|
|
|
|
|
(+ 1 index)))
|
|
|
|
|
'()
|
|
|
|
|
1
|
|
|
|
|
partitions)))
|
2014-04-11 12:44:53 -04:00
|
|
|
|
|
2014-05-15 16:55:14 -04:00
|
|
|
|
(define MS_BIND 4096) ; <sys/mounts.h> again!
|
|
|
|
|
|
2017-05-06 16:53:58 -04:00
|
|
|
|
(define* (create-ext-file-system partition type
|
2020-04-01 09:08:11 -04:00
|
|
|
|
#:key label uuid (options '()))
|
2018-01-19 10:25:13 -05:00
|
|
|
|
"Create an ext-family file system of TYPE on PARTITION. If LABEL is true,
|
2017-07-19 18:15:43 -04:00
|
|
|
|
use that as the volume name. If UUID is true, use it as the partition UUID."
|
2018-05-23 04:14:20 -04:00
|
|
|
|
(format #t "creating ~a partition... ~@[label: ~s~] ~@[uuid: ~s~]\n"
|
|
|
|
|
type label (and uuid (uuid->string uuid)))
|
2018-03-15 11:30:41 -04:00
|
|
|
|
(apply invoke (string-append "mkfs." type)
|
|
|
|
|
"-F" partition
|
|
|
|
|
`(,@(if label
|
|
|
|
|
`("-L" ,label)
|
|
|
|
|
'())
|
|
|
|
|
,@(if uuid
|
|
|
|
|
`("-U" ,(uuid->string uuid))
|
2020-04-01 09:08:11 -04:00
|
|
|
|
'())
|
|
|
|
|
,@options)))
|
2014-05-15 16:55:14 -04:00
|
|
|
|
|
2017-05-06 16:53:58 -04:00
|
|
|
|
(define* (create-fat-file-system partition
|
2020-04-01 09:08:11 -04:00
|
|
|
|
#:key label uuid (options '()))
|
2018-01-19 10:25:13 -05:00
|
|
|
|
"Create a FAT file system on PARTITION. The number of File Allocation Tables
|
|
|
|
|
will be determined based on file system size. If LABEL is true, use that as the
|
2017-05-06 16:53:58 -04:00
|
|
|
|
volume name."
|
2017-07-19 18:15:43 -04:00
|
|
|
|
;; FIXME: UUID is ignored!
|
2017-05-06 16:53:58 -04:00
|
|
|
|
(format #t "creating FAT partition...\n")
|
2018-03-15 11:30:41 -04:00
|
|
|
|
(apply invoke "mkfs.fat" partition
|
2020-04-01 09:08:11 -04:00
|
|
|
|
(append (if label `("-n" ,label) '()) options)))
|
2017-05-06 16:53:58 -04:00
|
|
|
|
|
|
|
|
|
(define* (format-partition partition type
|
2020-04-01 09:08:11 -04:00
|
|
|
|
#:key label uuid (options '()))
|
2017-05-06 16:53:58 -04:00
|
|
|
|
"Create a file system TYPE on PARTITION. If LABEL is true, use that as the
|
2020-04-01 09:08:11 -04:00
|
|
|
|
volume name. Options is a list of command-line options passed to 'mkfs.FS'."
|
2017-05-06 16:53:58 -04:00
|
|
|
|
(cond ((string-prefix? "ext" type)
|
2020-04-01 09:08:11 -04:00
|
|
|
|
(create-ext-file-system partition type #:label label #:uuid uuid
|
|
|
|
|
#:options options))
|
2017-05-06 16:53:58 -04:00
|
|
|
|
((or (string-prefix? "fat" type) (string= "vfat" type))
|
2020-04-01 09:08:11 -04:00
|
|
|
|
(create-fat-file-system partition #:label label #:uuid uuid
|
|
|
|
|
#:options options))
|
2017-05-06 16:53:58 -04:00
|
|
|
|
(else (error "Unsupported file system."))))
|
|
|
|
|
|
2015-07-25 17:57:52 -04:00
|
|
|
|
(define (initialize-partition partition)
|
|
|
|
|
"Format PARTITION, a <partition> object with a non-#f 'device' field, mount
|
|
|
|
|
it, run its initializer, and unmount it."
|
|
|
|
|
(let ((target "/fs"))
|
|
|
|
|
(format-partition (partition-device partition)
|
|
|
|
|
(partition-file-system partition)
|
2017-07-19 18:15:43 -04:00
|
|
|
|
#:label (partition-label partition)
|
2020-04-01 09:08:11 -04:00
|
|
|
|
#:uuid (partition-uuid partition)
|
|
|
|
|
#:options (partition-file-system-options partition))
|
2015-07-25 17:57:52 -04:00
|
|
|
|
(mkdir-p target)
|
|
|
|
|
(mount (partition-device partition) target
|
|
|
|
|
(partition-file-system partition))
|
|
|
|
|
|
|
|
|
|
((partition-initializer partition) target)
|
|
|
|
|
|
|
|
|
|
(umount target)
|
|
|
|
|
partition))
|
|
|
|
|
|
|
|
|
|
(define* (root-partition-initializer #:key (closures '())
|
|
|
|
|
copy-closures?
|
|
|
|
|
(register-closures? #t)
|
2018-03-15 00:09:11 -04:00
|
|
|
|
system-directory
|
2020-04-01 09:03:10 -04:00
|
|
|
|
(deduplicate? #t)
|
2020-04-07 04:21:48 -04:00
|
|
|
|
(make-device-nodes
|
|
|
|
|
make-essential-device-nodes)
|
2020-04-01 09:03:10 -04:00
|
|
|
|
(extra-directives '()))
|
2015-07-25 17:57:52 -04:00
|
|
|
|
"Return a procedure to initialize a root partition.
|
|
|
|
|
|
2018-03-15 00:09:11 -04:00
|
|
|
|
If REGISTER-CLOSURES? is true, register all of CLOSURES in the partition's
|
|
|
|
|
store. If DEDUPLICATE? is true, then also deduplicate files common to
|
|
|
|
|
CLOSURES and the rest of the store when registering the closures. If
|
|
|
|
|
COPY-CLOSURES? is true, copy all of CLOSURES to the partition.
|
2020-04-01 09:03:10 -04:00
|
|
|
|
SYSTEM-DIRECTORY is the name of the directory of the 'system' derivation.
|
|
|
|
|
|
|
|
|
|
EXTRA-DIRECTIVES is an optional list of directives to populate the root file
|
|
|
|
|
system that is passed to 'populate-root-file-system'."
|
2015-07-25 17:57:52 -04:00
|
|
|
|
(lambda (target)
|
|
|
|
|
(define target-store
|
|
|
|
|
(string-append target (%store-directory)))
|
|
|
|
|
|
|
|
|
|
(when copy-closures?
|
|
|
|
|
;; Populate the store.
|
|
|
|
|
(populate-store (map (cut string-append "/xchg/" <>) closures)
|
2020-12-10 09:12:34 -05:00
|
|
|
|
target
|
|
|
|
|
#:deduplicate? deduplicate?))
|
2015-07-25 17:57:52 -04:00
|
|
|
|
|
|
|
|
|
;; Populate /dev.
|
2020-04-07 04:21:48 -04:00
|
|
|
|
(make-device-nodes target)
|
2015-07-25 17:57:52 -04:00
|
|
|
|
|
|
|
|
|
;; Optionally, register the inputs in the image's store.
|
|
|
|
|
(when register-closures?
|
|
|
|
|
(unless copy-closures?
|
2018-06-08 05:03:31 -04:00
|
|
|
|
;; XXX: 'register-closure' wants to palpate the things it registers, so
|
2015-07-25 17:57:52 -04:00
|
|
|
|
;; bind-mount the store on the target.
|
|
|
|
|
(mkdir-p target-store)
|
|
|
|
|
(mount (%store-directory) target-store "" MS_BIND))
|
|
|
|
|
|
|
|
|
|
(display "registering closures...\n")
|
|
|
|
|
(for-each (lambda (closure)
|
|
|
|
|
(register-closure target
|
2020-12-10 15:25:39 -05:00
|
|
|
|
(string-append "/xchg/" closure)))
|
2015-07-25 17:57:52 -04:00
|
|
|
|
closures)
|
|
|
|
|
(unless copy-closures?
|
|
|
|
|
(umount target-store)))
|
|
|
|
|
|
|
|
|
|
;; Add the non-store directories and files.
|
|
|
|
|
(display "populating...\n")
|
2020-04-01 09:03:10 -04:00
|
|
|
|
(populate-root-file-system system-directory target
|
|
|
|
|
#:extras extra-directives)
|
2015-07-25 17:57:52 -04:00
|
|
|
|
|
2018-06-08 05:03:31 -04:00
|
|
|
|
;; 'register-closure' resets timestamps and everything, so no need to do it
|
2015-07-25 17:57:52 -04:00
|
|
|
|
;; once more in that case.
|
|
|
|
|
(unless register-closures?
|
2020-04-06 09:16:09 -04:00
|
|
|
|
;; 'reset-timestamps' also resets file permissions; do that everywhere
|
|
|
|
|
;; except on /dev so that /dev/null remains writable, etc.
|
|
|
|
|
(for-each (lambda (directory)
|
|
|
|
|
(reset-timestamps (string-append target "/" directory)))
|
|
|
|
|
(scandir target
|
|
|
|
|
(match-lambda
|
|
|
|
|
((or "." ".." "dev") #f)
|
|
|
|
|
(_ #t))))
|
|
|
|
|
(reset-timestamps (string-append target "/dev")
|
|
|
|
|
#:preserve-permissions? #t))))
|
2014-05-21 17:31:46 -04:00
|
|
|
|
|
2017-05-09 04:52:02 -04:00
|
|
|
|
(define (register-bootcfg-root target bootcfg)
|
2017-04-15 08:22:24 -04:00
|
|
|
|
"On file system TARGET, register BOOTCFG as a GC root."
|
2014-12-09 05:06:22 -05:00
|
|
|
|
(let ((directory (string-append target "/var/guix/gcroots")))
|
2014-12-04 17:52:28 -05:00
|
|
|
|
(mkdir-p directory)
|
2017-05-09 04:52:02 -04:00
|
|
|
|
(symlink bootcfg (string-append directory "/bootcfg"))))
|
2014-12-04 17:52:28 -05:00
|
|
|
|
|
2014-05-21 17:31:46 -04:00
|
|
|
|
(define* (initialize-hard-disk device
|
|
|
|
|
#:key
|
2017-05-09 04:52:02 -04:00
|
|
|
|
bootloader-package
|
|
|
|
|
bootcfg
|
|
|
|
|
bootcfg-location
|
|
|
|
|
bootloader-installer
|
2017-05-07 09:31:30 -04:00
|
|
|
|
(grub-efi #f)
|
2015-07-25 17:57:52 -04:00
|
|
|
|
(partitions '()))
|
|
|
|
|
"Initialize DEVICE as a disk containing all the <partition> objects listed
|
2017-04-15 08:22:24 -04:00
|
|
|
|
in PARTITIONS, and using BOOTCFG as its bootloader configuration file.
|
2014-05-21 17:31:46 -04:00
|
|
|
|
|
2015-07-25 17:57:52 -04:00
|
|
|
|
Each partition is initialized by calling its 'initializer' procedure,
|
|
|
|
|
passing it a directory name where it is mounted."
|
2017-04-11 04:47:38 -04:00
|
|
|
|
|
|
|
|
|
(define (partition-bootable? partition)
|
|
|
|
|
"Return the first partition found with the boot flag set."
|
|
|
|
|
(member 'boot (partition-flags partition)))
|
|
|
|
|
|
2017-05-07 09:31:30 -04:00
|
|
|
|
(define (partition-esp? partition)
|
|
|
|
|
"Return the first EFI System Partition."
|
|
|
|
|
(member 'esp (partition-flags partition)))
|
|
|
|
|
|
2015-07-25 17:57:52 -04:00
|
|
|
|
(let* ((partitions (initialize-partition-table device partitions))
|
|
|
|
|
(root (find partition-bootable? partitions))
|
2017-05-07 09:31:30 -04:00
|
|
|
|
(esp (find partition-esp? partitions))
|
2015-07-25 17:57:52 -04:00
|
|
|
|
(target "/fs"))
|
|
|
|
|
(unless root
|
|
|
|
|
(error "no bootable partition specified" partitions))
|
2014-04-11 12:44:53 -04:00
|
|
|
|
|
2015-07-25 17:57:52 -04:00
|
|
|
|
(for-each initialize-partition partitions)
|
2014-04-11 12:44:53 -04:00
|
|
|
|
|
2015-07-25 17:57:52 -04:00
|
|
|
|
(display "mounting root partition...\n")
|
|
|
|
|
(mkdir-p target)
|
|
|
|
|
(mount (partition-device root) target (partition-file-system root))
|
2017-05-09 04:52:02 -04:00
|
|
|
|
(install-boot-config bootcfg bootcfg-location target)
|
|
|
|
|
(when bootloader-installer
|
2017-05-07 09:31:30 -04:00
|
|
|
|
(display "installing bootloader...\n")
|
2017-05-09 04:52:02 -04:00
|
|
|
|
(bootloader-installer bootloader-package device target))
|
2014-12-04 17:52:28 -05:00
|
|
|
|
|
2017-05-07 09:31:30 -04:00
|
|
|
|
(when esp
|
|
|
|
|
;; Mount the ESP somewhere and install GRUB UEFI image.
|
2020-04-28 10:17:59 -04:00
|
|
|
|
(let ((mount-point (string-append target "/boot/efi")))
|
2017-05-07 09:31:30 -04:00
|
|
|
|
(display "mounting EFI system partition...\n")
|
|
|
|
|
(mkdir-p mount-point)
|
|
|
|
|
(mount (partition-device esp) mount-point
|
|
|
|
|
(partition-file-system esp))
|
|
|
|
|
|
|
|
|
|
(display "creating EFI firmware image...")
|
2020-04-28 10:17:59 -04:00
|
|
|
|
(install-efi-loader grub-efi mount-point)
|
2017-05-07 09:31:30 -04:00
|
|
|
|
(display "done.\n")
|
|
|
|
|
|
|
|
|
|
(umount mount-point)))
|
|
|
|
|
|
2017-05-09 04:52:02 -04:00
|
|
|
|
;; Register BOOTCFG as a GC root.
|
|
|
|
|
(register-bootcfg-root target bootcfg)
|
2014-04-11 12:44:53 -04:00
|
|
|
|
|
2015-07-25 17:57:52 -04:00
|
|
|
|
(umount target)))
|
2014-04-11 12:44:53 -04:00
|
|
|
|
|
2014-04-11 07:38:11 -04:00
|
|
|
|
;;; vm.scm ends here
|