mirror of
https://git.in.rschanz.org/ryan77627/guix.git
synced 2025-01-03 18:09:18 -05:00
d58e52b071
Fixes: <https://issues.guix.gnu.org/50723>. The keymap xml contains optional shortDescription and description fields. The assoc-ref call on those fields can return false, handle it correctly. * gnu/installer/keymap.scm (xkb-rules->models+layouts): Introduce a new "maybe-empty" helper to deal with optional fields. Use it for shortDescription and description fields.
187 lines
6.8 KiB
Scheme
187 lines
6.8 KiB
Scheme
;;; GNU Guix --- Functional package management for GNU
|
|
;;; Copyright © 2018, 2021 Mathieu Othacehe <othacehe@gnu.org>
|
|
;;; Copyright © 2020 Florian Pelz <pelzflorian@pelzflorian.de>
|
|
;;;
|
|
;;; 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 installer keymap)
|
|
#:use-module (guix records)
|
|
#:use-module (sxml match)
|
|
#:use-module (sxml simple)
|
|
#:use-module (ice-9 binary-ports)
|
|
#:use-module (ice-9 ftw)
|
|
#:use-module (ice-9 match)
|
|
#:use-module (ice-9 regex)
|
|
#:export (<x11-keymap-model>
|
|
x11-keymap-model
|
|
make-x11-keymap-model
|
|
x11-keymap-model?
|
|
x11-keymap-model-name
|
|
x11-keymap-model-description
|
|
|
|
<x11-keymap-layout>
|
|
x11-keymap-layout
|
|
make-x11-keymap-layout
|
|
x11-keymap-layout?
|
|
x11-keymap-layout-name
|
|
x11-keymap-layout-synopsis
|
|
x11-keymap-layout-description
|
|
x11-keymap-layout-variants
|
|
|
|
<x11-keymap-variant>
|
|
x11-keymap-variant
|
|
make-x11-keymap-variant
|
|
x11-keymap-variant?
|
|
x11-keymap-variant-name
|
|
x11-keymap-variant-description
|
|
|
|
default-keyboard-model
|
|
xkb-rules->models+layouts
|
|
kmscon-update-keymap))
|
|
|
|
(define-record-type* <x11-keymap-model>
|
|
x11-keymap-model make-x11-keymap-model
|
|
x11-keymap-model?
|
|
(name x11-keymap-model-name) ;string
|
|
(description x11-keymap-model-description)) ;string
|
|
|
|
(define-record-type* <x11-keymap-layout>
|
|
x11-keymap-layout make-x11-keymap-layout
|
|
x11-keymap-layout?
|
|
(name x11-keymap-layout-name) ;string
|
|
(synopsis x11-keymap-layout-synopsis) ;string (e.g., "en")
|
|
(description x11-keymap-layout-description) ;string (a whole phrase)
|
|
(variants x11-keymap-layout-variants)) ;list of <x11-keymap-variant>
|
|
|
|
(define-record-type* <x11-keymap-variant>
|
|
x11-keymap-variant make-x11-keymap-variant
|
|
x11-keymap-variant?
|
|
(name x11-keymap-variant-name) ;string
|
|
(description x11-keymap-variant-description)) ;string
|
|
|
|
;; Assume all modern keyboards have this model.
|
|
(define default-keyboard-model (make-parameter "pc105"))
|
|
|
|
(define (xkb-rules->models+layouts file)
|
|
"Parse FILE and return two values, the list of supported X11-KEYMAP-MODEL
|
|
and X11-KEYMAP-LAYOUT records. FILE is an XML file from the X Keyboard
|
|
Configuration Database, describing possible XKB configurations."
|
|
(define maybe-empty
|
|
(match-lambda
|
|
((x) x)
|
|
(#f "")))
|
|
|
|
(define (model m)
|
|
(sxml-match m
|
|
[(model
|
|
(configItem
|
|
(name ,name)
|
|
(description ,description)
|
|
. ,rest))
|
|
(x11-keymap-model
|
|
(name name)
|
|
(description description))]))
|
|
|
|
(define (variant v)
|
|
(sxml-match v
|
|
[(variant
|
|
;; According to xbd-rules DTD, the definition of a
|
|
;; configItem is: <!ELEMENT configItem
|
|
;; (name,shortDescription*,description*,vendor?,
|
|
;; countryList?,languageList?,hwList?)>
|
|
;;
|
|
;; shortDescription and description are optional elements
|
|
;; but sxml-match does not support default values for
|
|
;; elements (only attributes). So to avoid writing as many
|
|
;; patterns as existing possibilities, gather all the
|
|
;; remaining elements but name in REST-VARIANT.
|
|
(configItem
|
|
(name ,name)
|
|
. ,rest-variant))
|
|
(x11-keymap-variant
|
|
(name name)
|
|
(description (maybe-empty
|
|
(assoc-ref rest-variant 'description))))]))
|
|
|
|
(define (layout l)
|
|
(sxml-match l
|
|
[(layout
|
|
(configItem
|
|
(name ,name)
|
|
. ,rest-layout)
|
|
(variantList ,[variant -> v] ...))
|
|
(x11-keymap-layout
|
|
(name name)
|
|
(synopsis (maybe-empty
|
|
(assoc-ref rest-layout 'shortDescription)))
|
|
(description (maybe-empty
|
|
(assoc-ref rest-layout 'description)))
|
|
(variants (list v ...)))]
|
|
[(layout
|
|
(configItem
|
|
(name ,name)
|
|
. ,rest-layout))
|
|
(x11-keymap-layout
|
|
(name name)
|
|
(synopsis (maybe-empty
|
|
(assoc-ref rest-layout 'shortDescription)))
|
|
(description (maybe-empty
|
|
(assoc-ref rest-layout 'description)))
|
|
(variants '()))]))
|
|
|
|
(let ((sxml (call-with-input-file file
|
|
(lambda (port)
|
|
(xml->sxml port #:trim-whitespace? #t)))))
|
|
(match
|
|
(sxml-match sxml
|
|
[(*TOP*
|
|
,pi
|
|
(xkbConfigRegistry
|
|
(@ . ,ignored)
|
|
(modelList ,[model -> m] ...)
|
|
(layoutList ,[layout -> l] ...)
|
|
. ,rest))
|
|
(list
|
|
(list m ...)
|
|
(list l ...))])
|
|
((models layouts)
|
|
(values models layouts)))))
|
|
|
|
(define (kmscon-update-keymap model layout variant options)
|
|
"Update kmscon keymap with the provided MODEL, LAYOUT, VARIANT and OPTIONS."
|
|
(and=>
|
|
(getenv "KEYMAP_UPDATE")
|
|
(lambda (keymap-file)
|
|
(unless (file-exists? keymap-file)
|
|
(error "Unable to locate keymap update file"))
|
|
|
|
;; See file gnu/packages/patches/kmscon-runtime-keymap-switch.patch.
|
|
;; This dirty hack makes possible to update kmscon keymap at runtime by
|
|
;; writing an X11 keyboard model, layout and variant to a named pipe
|
|
;; referred by KEYMAP_UPDATE environment variable.
|
|
(call-with-output-file keymap-file
|
|
(lambda (port)
|
|
(format port model)
|
|
(put-u8 port 0)
|
|
|
|
(format port layout)
|
|
(put-u8 port 0)
|
|
|
|
(format port (or variant ""))
|
|
(put-u8 port 0)
|
|
|
|
(format port (or options ""))
|
|
(put-u8 port 0))))))
|