~ruther/guix-config

guix-config/modules/ruther/bootloader/grub.scm -rw-r--r-- 7.3 KiB
0eb49edb — Rutherther feat: add simple iso for making openpgp keys 5 days ago
                                                                                
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
(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)))

(define* (grub-lvm-configuration-file config entries
                                       #:key
                                       (locale #f)
                                       (system (%current-system))
                                       (old-entries '())
                                       (store-crypto-devices '())
                                       store-directory-prefix)
  (let* ((original-grub-cfg
          ((@@ (gnu bootloader grub) make-grub-configuration)
           grub-efi config entries
           #:locale locale
           #:system system
           #:old-entries old-entries
           #:store-crypto-devices store-crypto-devices
           #:store-directory-prefix store-directory-prefix))

         (builder
          #~(call-with-output-file #$output
              (lambda (port)
                (use-modules (ice-9 textual-ports))
                ;; Sneek in insmod lvm at beginning of the file
                (display "insmod lvm\n" port)
                ;; After, copy the original file.
                (display (call-with-input-file #$original-grub-cfg get-string-all) port)))))
    (computed-file "grub.cfg" builder
                   #:options '(#:local-build? #t
                               #:substitutable? #f))))

(define-public grub-lvm-bootloader
  (bootloader
   (inherit grub-efi-removable-bootloader)
   (configuration-file-generator grub-lvm-configuration-file)))
Do not follow this link