2015-11-25 01:51:35 +01:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
2016-10-20 20:54:38 +02:00
|
|
|
with import <stockholm/lib>;
|
2015-11-25 01:51:35 +01:00
|
|
|
let
|
|
|
|
pkg = pkgs.pulseaudioLight;
|
|
|
|
runDir = "/run/pulse";
|
|
|
|
|
|
|
|
alsaConf = pkgs.writeText "asound.conf" ''
|
|
|
|
ctl_type.pulse {
|
|
|
|
libs.native = ${pkgs.alsaPlugins}/lib/alsa-lib/libasound_module_ctl_pulse.so;
|
|
|
|
}
|
|
|
|
pcm_type.pulse {
|
|
|
|
libs.native = ${pkgs.alsaPlugins}/lib/alsa-lib/libasound_module_pcm_pulse.so;
|
|
|
|
}
|
|
|
|
ctl.!default {
|
|
|
|
type pulse
|
|
|
|
}
|
|
|
|
pcm.!default {
|
|
|
|
type pulse
|
|
|
|
}
|
|
|
|
'';
|
|
|
|
|
|
|
|
clientConf = pkgs.writeText "client.conf" ''
|
|
|
|
autospawn=no
|
|
|
|
default-server = unix:${runDir}/socket
|
|
|
|
'';
|
|
|
|
|
|
|
|
configFile = pkgs.writeText "default.pa" ''
|
|
|
|
.include ${pkg}/etc/pulse/default.pa
|
|
|
|
load-module ${toString [
|
|
|
|
"module-native-protocol-unix"
|
|
|
|
"auth-anonymous=1"
|
|
|
|
"socket=${runDir}/socket"
|
|
|
|
]}
|
|
|
|
'';
|
|
|
|
in
|
|
|
|
|
|
|
|
{
|
|
|
|
environment = {
|
|
|
|
etc = {
|
|
|
|
"asound.conf".source = alsaConf;
|
2016-01-14 15:48:16 +01:00
|
|
|
# XXX mkForce is not strong enough (and neither is mkOverride) to create
|
|
|
|
# /etc/pulse/client.conf, see pulseaudio-hack below for a solution.
|
|
|
|
#"pulse/client.conf" = mkForce { source = clientConf; };
|
|
|
|
#"pulse/client.conf".source = mkForce clientConf;
|
2015-11-25 01:51:35 +01:00
|
|
|
"pulse/default.pa".source = configFile;
|
|
|
|
};
|
2016-01-14 15:51:41 +01:00
|
|
|
systemPackages = [
|
|
|
|
pkg
|
|
|
|
] ++ optionals config.services.xserver.enable [
|
|
|
|
pkgs.pavucontrol
|
|
|
|
];
|
2015-11-25 01:51:35 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
# Allow PulseAudio to get realtime priority using rtkit.
|
|
|
|
security.rtkit.enable = true;
|
|
|
|
|
2016-01-14 15:48:16 +01:00
|
|
|
system.activationScripts.pulseaudio-hack = ''
|
|
|
|
ln -fns ${clientConf} /etc/pulse/client.conf
|
|
|
|
'';
|
|
|
|
|
2015-11-25 01:51:35 +01:00
|
|
|
systemd.services.pulse = {
|
|
|
|
wantedBy = [ "sound.target" ];
|
|
|
|
before = [ "sound.target" ];
|
|
|
|
environment = {
|
|
|
|
PULSE_RUNTIME_PATH = "${runDir}/home";
|
|
|
|
};
|
|
|
|
serviceConfig = {
|
2016-11-18 10:51:18 +01:00
|
|
|
ExecStart = "${pkg}/bin/pulseaudio --exit-idle-time=-1";
|
2016-06-13 02:04:22 +02:00
|
|
|
ExecStartPre = pkgs.writeDash "pulse-start" ''
|
2016-01-14 15:45:41 +01:00
|
|
|
install -o pulse -g pulse -m 0750 -d ${runDir}
|
|
|
|
install -o pulse -g pulse -m 0700 -d ${runDir}/home
|
|
|
|
'';
|
|
|
|
PermissionsStartOnly = "true";
|
2015-11-25 01:51:35 +01:00
|
|
|
User = "pulse";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2017-03-06 13:11:21 +01:00
|
|
|
# TODO assert that pulse is the only user with "audio" in group/extraGroups
|
|
|
|
# otherwise the audio device can be hijacked while the pulse service restarts
|
|
|
|
# (e.g. when mpv is running) and then the service will fail.
|
2015-12-26 05:55:13 +01:00
|
|
|
users = {
|
|
|
|
groups.pulse.gid = config.users.users.pulse.uid;
|
2015-11-25 01:51:35 +01:00
|
|
|
users.pulse = {
|
2015-12-26 05:55:13 +01:00
|
|
|
uid = genid "pulse";
|
2015-11-25 01:51:35 +01:00
|
|
|
group = "pulse";
|
|
|
|
extraGroups = [ "audio" ];
|
|
|
|
home = "${runDir}/home";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|