2014-04-14 15:26:48 +01:00
|
|
|
{ config, lib, pkgs, ... }:
|
2010-01-11 21:56:01 +00:00
|
|
|
|
2016-06-04 12:07:09 +01:00
|
|
|
with lib;
|
|
|
|
|
2014-03-21 22:52:24 +00:00
|
|
|
let
|
2016-06-04 12:07:09 +01:00
|
|
|
cfgFile = pkgs.writeText "reader.conf" config.services.pcscd.readerConfig;
|
2014-03-21 22:52:24 +00:00
|
|
|
|
2016-06-04 12:07:09 +01:00
|
|
|
pluginEnv = pkgs.buildEnv {
|
|
|
|
name = "pcscd-plugins";
|
|
|
|
paths = map (p: "${p}/pcsc/drivers") config.services.pcscd.plugins;
|
|
|
|
};
|
2010-01-11 21:56:01 +00:00
|
|
|
|
2020-09-08 13:32:16 +01:00
|
|
|
in
|
|
|
|
{
|
2010-01-11 21:56:01 +00:00
|
|
|
|
|
|
|
###### interface
|
|
|
|
|
2020-09-08 13:32:16 +01:00
|
|
|
options.services.pcscd = {
|
|
|
|
enable = mkEnableOption "PCSC-Lite daemon";
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2020-09-08 13:32:16 +01:00
|
|
|
plugins = mkOption {
|
|
|
|
type = types.listOf types.package;
|
|
|
|
default = [ pkgs.ccid ];
|
2021-10-03 17:06:03 +01:00
|
|
|
defaultText = literalExpression "[ pkgs.ccid ]";
|
|
|
|
example = literalExpression "[ pkgs.pcsc-cyberjack ]";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Plugin packages to be used for PCSC-Lite.";
|
2020-09-08 13:32:16 +01:00
|
|
|
};
|
2016-06-04 12:07:09 +01:00
|
|
|
|
2020-09-08 13:32:16 +01:00
|
|
|
readerConfig = mkOption {
|
|
|
|
type = types.lines;
|
|
|
|
default = "";
|
|
|
|
example = ''
|
|
|
|
FRIENDLYNAME "Some serial reader"
|
|
|
|
DEVICENAME /dev/ttyS0
|
|
|
|
LIBPATH /path/to/serial_reader.so
|
|
|
|
CHANNELID 1
|
|
|
|
'';
|
2022-08-05 18:39:00 +01:00
|
|
|
description = lib.mdDoc ''
|
2020-09-08 13:32:16 +01:00
|
|
|
Configuration for devices that aren't hotpluggable.
|
|
|
|
|
2022-08-05 18:39:00 +01:00
|
|
|
See {manpage}`reader.conf(5)` for valid options.
|
2020-09-08 13:32:16 +01:00
|
|
|
'';
|
2010-01-11 21:56:01 +00:00
|
|
|
};
|
|
|
|
};
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2010-01-11 21:56:01 +00:00
|
|
|
###### implementation
|
|
|
|
|
|
|
|
config = mkIf config.services.pcscd.enable {
|
|
|
|
|
2020-09-08 13:32:16 +01:00
|
|
|
environment.etc."reader.conf".source = cfgFile;
|
|
|
|
|
2021-04-30 09:21:43 +01:00
|
|
|
environment.systemPackages = [ pkgs.pcsclite ];
|
2020-09-08 13:32:16 +01:00
|
|
|
systemd.packages = [ (getBin pkgs.pcsclite) ];
|
|
|
|
|
|
|
|
systemd.sockets.pcscd.wantedBy = [ "sockets.target" ];
|
2014-04-12 03:58:21 +01:00
|
|
|
|
2014-03-21 22:52:24 +00:00
|
|
|
systemd.services.pcscd = {
|
2016-06-04 12:07:09 +01:00
|
|
|
environment.PCSCLITE_HP_DROPDIR = pluginEnv;
|
2020-09-08 13:32:16 +01:00
|
|
|
restartTriggers = [ "/etc/reader.conf" ];
|
2021-04-29 02:17:33 +01:00
|
|
|
|
|
|
|
# If the cfgFile is empty and not specified (in which case the default
|
|
|
|
# /etc/reader.conf is assumed), pcscd will happily start going through the
|
|
|
|
# entire confdir (/etc in our case) looking for a config file and try to
|
|
|
|
# parse everything it finds. Doesn't take a lot of imagination to see how
|
|
|
|
# well that works. It really shouldn't do that to begin with, but to work
|
|
|
|
# around it, we force the path to the cfgFile.
|
|
|
|
#
|
|
|
|
# https://github.com/NixOS/nixpkgs/issues/121088
|
|
|
|
serviceConfig.ExecStart = [ "" "${getBin pkgs.pcsclite}/bin/pcscd -f -x -c ${cfgFile}" ];
|
2014-03-21 22:52:24 +00:00
|
|
|
};
|
2010-01-11 21:56:01 +00:00
|
|
|
};
|
|
|
|
}
|