123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188 |
- ;;; 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))))))
|