~ruther/guix-config

595ee4f0311093b15c6f25b128120b0197132134 — Rutherther 6 months ago 7163277
feat: add grub with file copy to /boot
1 files changed, 137 insertions(+), 0 deletions(-)

A modules/ruther/bootloader/grub.scm
A modules/ruther/bootloader/grub.scm => modules/ruther/bootloader/grub.scm +137 -0
@@ 0,0 1,137 @@
(define-module (ruther bootloader grub)
  #:use-module (guix gexp)
  #:use-module (gnu bootloader)
  #:use-module (gnu bootloader grub)
  #:use-module (gnu packages bootloaders)
  #:use-module (guix utils)
  #:use-module (gnu system locale)
  )

(define install-grub-efi-copy
  (with-imported-modules '((guix build utils))
      #~(lambda (bootloader efi-dir mount-point)
          (use-modules (guix build utils)
                       (ice-9 textual-ports)
                       (srfi srfi-1))

          ;; There is nothing useful to do when called in the context of a disk
          ;; image generation.
          (when efi-dir
            ;; Install GRUB onto the EFI partition mounted at EFI-DIR, for the
            ;; system whose root is mounted at MOUNT-POINT.
            (let* ((grub-install (string-append bootloader "/sbin/grub-install"))
                   (install-dir (string-append mount-point "/boot"))
                   ;; When installing Guix, it's common to mount EFI-DIR below
                   ;; MOUNT-POINT rather than /boot/efi on the live image.
                   (target-esp (if (file-exists? (string-append mount-point efi-dir))
                                   (string-append mount-point efi-dir)
                                   efi-dir))
                   (grub-cfg #$(@@ (gnu bootloader grub) grub-cfg))
                   (grub-cfg-lines (string-split (call-with-input-file grub-cfg get-string-all) #\newline)))
              ;; Tell 'grub-install' that there might be a LUKS-encrypted /boot or
              ;; root partition.
              (setenv "GRUB_ENABLE_CRYPTODISK" "y")
              (invoke/quiet grub-install "--boot-directory" install-dir
                            "--bootloader-id=Guix"
                            "--efi-directory" target-esp)

              ;; After the grub is copied, we need to copy the gnu store.
              ;; To get the entries to copy we look into the grub.cfg file,
              ;; and obtain all lines starting with # NEEDED FILE:,
              ;; after : there is likely /gnu/store path that shall be copied to
              ;; /boot/gnu/store

              ;; for each line starting with NEEDED FILE:
              ;;   Get the string after :, trim it, call it needed-file
              ;;   (mkdir-p (dirname (string-append install-dir needed-file)))
              ;;   (copy-file needed-file (string-append install-dir needed-file))
              (for-each
               (lambda (file-line)
                 (let* ((source-file (string-trim-both (substring file-line (string-length "# NEEDED FILE:"))))
                        (dest-file (string-append install-dir source-file)))
                   (mkdir-p (dirname dest-file))
                   (copy-recursively source-file dest-file)))
               (delete-duplicates (filter
                                   (lambda (line) (string-prefix? "# NEEDED FILE:" line))
                                   grub-cfg-lines)))

              ;; TODO: remove files in install-dir under /boot/gnu/store that
              ;; are not alive anymore.
              )))))

(define* (grub-copy-configuration-file config entries
                                       #:key
                                       (locale #f)
                                       (system (%current-system))
                                       (old-entries '())
                                       (store-crypto-devices '())
                                       store-directory-prefix)
  ;; To get all the needed files we need to get:
  ;; First from each entry linux and vmlinuz.
  ;; Then the image
  ;; Then
  (let* ((menu-entry-needed-files
          (lambda (menu-entry)
            #~(list
               #$(menu-entry-linux menu-entry)
               #$(menu-entry-initrd menu-entry))))

         (locales (and locale
                       ((@@ (gnu bootloader grub) grub-locale-directory) grub-efi)))

         (image ((@@ (gnu bootloader grub) grub-background-image) config))

         (layout (bootloader-configuration-keyboard-layout config))
         (keymap* (and layout
                       ((@@ (gnu bootloader grub) keyboard-layout-file) layout #:grub grub-efi)))

         (needed-files-gexps
          #~(cons*
             #$image
             #$locales
             #$keymap*
             (apply append
                    (list #$@(map menu-entry-needed-files (append entries old-entries))))))

         (modify-devices
          (lambda (orig-menu-entry)
            (menu-entry
             (inherit orig-menu-entry)
             (device #f)))) ;; This will force grub.scm to use search --file. Which will work, since the file is on one of the disks already!
         (modified-entries (map modify-devices entries))
         (modified-old-entries (map modify-devices old-entries))

         (original-grub-cfg
          ((@@ (gnu bootloader grub) make-grub-configuration)
           grub-efi config modified-entries
           #:locale locale
           #:system system
           #:old-entries modified-old-entries
           #:store-crypto-devices '()
           #:store-directory-prefix #f))

         (builder
          #~(call-with-output-file #$output
              (lambda (port)
                ;; First copy the original file.
                (use-modules (ice-9 textual-ports))
                (display (call-with-input-file #$original-grub-cfg get-string-all) port)

                (display "\n\n" port)

                (for-each
                 (lambda (file)
                   (display (string-append "# NEEDED FILE: " file "\n") port))
                 #$needed-files-gexps)))))
    (computed-file "grub.cfg" builder
                   #:options '(#:local-build? #t
                               #:substitutable? #f))))

(define-public grub-efi-copy-bootloader
  (bootloader
   (name 'grub-copy-efi)
   (package grub-efi)
   (installer install-grub-efi-copy)
   (disk-image-installer #f)
   (configuration-file (@@ (gnu bootloader grub) grub-cfg))
   (configuration-file-generator grub-copy-configuration-file)))

Do not follow this link