2018-11-03 05:55:57 +00:00
|
|
|
{ config, lib, pkgs, ... }:
|
2013-02-28 18:40:07 +00:00
|
|
|
let
|
|
|
|
|
|
|
|
cfg = config.hardware.pulseaudio;
|
|
|
|
|
2016-07-03 22:32:04 +00:00
|
|
|
hasZeroconf = let z = cfg.zeroconf; in z.publish.enable || z.discovery.enable;
|
|
|
|
|
|
|
|
overriddenPackage = cfg.package.override
|
2024-08-24 20:05:25 +00:00
|
|
|
(lib.optionalAttrs hasZeroconf { zeroconfSupport = true; });
|
|
|
|
binary = "${lib.getBin overriddenPackage}/bin/pulseaudio";
|
2016-07-03 22:32:04 +00:00
|
|
|
binaryNoDaemon = "${binary} --daemonize=no";
|
2014-04-21 09:50:49 +00:00
|
|
|
|
2021-06-10 07:27:31 +00:00
|
|
|
# Forces 32bit pulseaudio and alsa-plugins to be built/supported for apps
|
2014-09-11 03:54:00 +00:00
|
|
|
# using 32bit alsa on 64bit linux.
|
2024-08-24 20:05:25 +00:00
|
|
|
enable32BitAlsaPlugins = cfg.support32Bit && pkgs.stdenv.hostPlatform.isx86_64 && (pkgs.pkgsi686Linux.alsa-lib != null && pkgs.pkgsi686Linux.libpulseaudio != null);
|
2014-09-11 03:54:00 +00:00
|
|
|
|
2016-07-03 22:32:04 +00:00
|
|
|
|
|
|
|
myConfigFile =
|
|
|
|
let
|
2024-08-24 20:05:25 +00:00
|
|
|
addModuleIf = cond: mod: lib.optionalString cond "load-module ${mod}";
|
|
|
|
allAnon = lib.optional cfg.tcp.anonymousClients.allowAll "auth-anonymous=1";
|
2016-07-03 22:32:04 +00:00
|
|
|
ipAnon = let a = cfg.tcp.anonymousClients.allowedIpRanges;
|
2024-08-24 20:05:25 +00:00
|
|
|
in lib.optional (a != []) ''auth-ip-acl=${lib.concatStringsSep ";" a}'';
|
|
|
|
in pkgs.writeTextFile {
|
2016-07-03 22:32:04 +00:00
|
|
|
name = "default.pa";
|
|
|
|
text = ''
|
|
|
|
.include ${cfg.configFile}
|
|
|
|
${addModuleIf cfg.zeroconf.publish.enable "module-zeroconf-publish"}
|
|
|
|
${addModuleIf cfg.zeroconf.discovery.enable "module-zeroconf-discover"}
|
2024-08-24 20:05:25 +00:00
|
|
|
${addModuleIf cfg.tcp.enable (lib.concatStringsSep " "
|
2016-08-25 17:34:33 +00:00
|
|
|
([ "module-native-protocol-tcp" ] ++ allAnon ++ ipAnon))}
|
2020-07-18 11:32:48 +00:00
|
|
|
${addModuleIf config.services.jack.jackd.enable "module-jack-sink"}
|
|
|
|
${addModuleIf config.services.jack.jackd.enable "module-jack-source"}
|
2016-07-03 22:32:04 +00:00
|
|
|
${cfg.extraConfig}
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2014-09-02 11:16:13 +00:00
|
|
|
ids = config.ids;
|
2013-02-28 18:40:07 +00:00
|
|
|
|
2014-09-02 11:16:13 +00:00
|
|
|
uid = ids.uids.pulseaudio;
|
|
|
|
gid = ids.gids.pulseaudio;
|
|
|
|
|
2017-11-27 07:49:38 +00:00
|
|
|
stateDir = "/run/pulse";
|
2013-02-28 18:40:07 +00:00
|
|
|
|
|
|
|
# Create pulse/client.conf even if PulseAudio is disabled so
|
|
|
|
# that we can disable the autospawn feature in programs that
|
|
|
|
# are built with PulseAudio support (like KDE).
|
2024-08-24 20:05:25 +00:00
|
|
|
clientConf = pkgs.writeText "client.conf" ''
|
2019-10-12 00:02:25 +00:00
|
|
|
autospawn=no
|
2016-06-25 12:34:34 +00:00
|
|
|
${cfg.extraClientConf}
|
2013-02-28 18:40:07 +00:00
|
|
|
'';
|
|
|
|
|
|
|
|
# Write an /etc/asound.conf that causes all ALSA applications to
|
|
|
|
# be re-routed to the PulseAudio server through ALSA's Pulse
|
|
|
|
# plugin.
|
2024-07-13 10:56:18 +00:00
|
|
|
alsaConf = ''
|
2013-02-28 18:40:07 +00:00
|
|
|
pcm_type.pulse {
|
2021-06-10 07:27:31 +00:00
|
|
|
libs.native = ${pkgs.alsa-plugins}/lib/alsa-lib/libasound_module_pcm_pulse.so ;
|
2014-09-11 03:54:00 +00:00
|
|
|
${lib.optionalString enable32BitAlsaPlugins
|
2021-06-10 07:27:31 +00:00
|
|
|
"libs.32Bit = ${pkgs.pkgsi686Linux.alsa-plugins}/lib/alsa-lib/libasound_module_pcm_pulse.so ;"}
|
2013-02-28 18:40:07 +00:00
|
|
|
}
|
|
|
|
pcm.!default {
|
|
|
|
type pulse
|
|
|
|
hint.description "Default Audio Device (via PulseAudio)"
|
|
|
|
}
|
|
|
|
ctl_type.pulse {
|
2021-06-10 07:27:31 +00:00
|
|
|
libs.native = ${pkgs.alsa-plugins}/lib/alsa-lib/libasound_module_ctl_pulse.so ;
|
2014-09-11 03:54:00 +00:00
|
|
|
${lib.optionalString enable32BitAlsaPlugins
|
2021-06-10 07:27:31 +00:00
|
|
|
"libs.32Bit = ${pkgs.pkgsi686Linux.alsa-plugins}/lib/alsa-lib/libasound_module_ctl_pulse.so ;"}
|
2013-02-28 18:40:07 +00:00
|
|
|
}
|
|
|
|
ctl.!default {
|
|
|
|
type pulse
|
|
|
|
}
|
2024-07-13 10:56:18 +00:00
|
|
|
'';
|
2013-02-28 18:40:07 +00:00
|
|
|
|
|
|
|
in {
|
2011-07-26 01:51:56 +00:00
|
|
|
|
|
|
|
options = {
|
2011-09-14 18:20:50 +00:00
|
|
|
|
2013-02-28 18:40:07 +00:00
|
|
|
hardware.pulseaudio = {
|
2024-08-24 20:05:25 +00:00
|
|
|
enable = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
2013-02-28 18:40:07 +00:00
|
|
|
default = false;
|
2022-07-19 13:05:45 +00:00
|
|
|
description = ''
|
2013-02-28 18:40:07 +00:00
|
|
|
Whether to enable the PulseAudio sound server.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
systemWide = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
2013-02-28 18:40:07 +00:00
|
|
|
default = false;
|
2022-07-19 13:05:45 +00:00
|
|
|
description = ''
|
2013-02-28 18:40:07 +00:00
|
|
|
If false, a PulseAudio server is launched automatically for
|
|
|
|
each user that tries to use the sound system. The server runs
|
2019-10-30 15:58:57 +00:00
|
|
|
with user privileges. If true, one system-wide PulseAudio
|
2017-11-30 06:37:01 +00:00
|
|
|
server is launched on boot, running as the user "pulse", and
|
2022-08-18 09:03:58 +00:00
|
|
|
only users in the "pulse-access" group will have access to the server.
|
2013-02-28 18:40:07 +00:00
|
|
|
Please read the PulseAudio documentation for more details.
|
2019-10-30 15:58:57 +00:00
|
|
|
|
|
|
|
Don't enable this option unless you know what you are doing.
|
2013-02-28 18:40:07 +00:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
support32Bit = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
2015-06-10 18:48:52 +00:00
|
|
|
default = false;
|
2022-07-19 13:05:45 +00:00
|
|
|
description = ''
|
2017-01-26 18:00:43 +00:00
|
|
|
Whether to include the 32-bit pulseaudio libraries in the system or not.
|
2015-06-11 20:44:57 +00:00
|
|
|
This is only useful on 64-bit systems and currently limited to x86_64-linux.
|
|
|
|
'';
|
2015-06-10 18:48:52 +00:00
|
|
|
};
|
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
configFile = lib.mkOption {
|
|
|
|
type = lib.types.nullOr lib.types.path;
|
2022-07-19 13:05:45 +00:00
|
|
|
description = ''
|
2016-07-03 22:32:04 +00:00
|
|
|
The path to the default configuration options the PulseAudio server
|
2013-02-28 18:40:07 +00:00
|
|
|
should use. By default, the "default.pa" configuration
|
|
|
|
from the PulseAudio distribution is used.
|
2013-10-30 16:37:45 +00:00
|
|
|
'';
|
2013-02-28 18:40:07 +00:00
|
|
|
};
|
2013-10-30 16:37:45 +00:00
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
extraConfig = lib.mkOption {
|
|
|
|
type = lib.types.lines;
|
2016-07-03 22:32:04 +00:00
|
|
|
default = "";
|
2022-07-19 13:05:45 +00:00
|
|
|
description = ''
|
|
|
|
Literal string to append to `configFile`
|
2016-07-03 22:32:04 +00:00
|
|
|
and the config file generated by the pulseaudio module.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
extraClientConf = lib.mkOption {
|
|
|
|
type = lib.types.lines;
|
2016-06-25 12:34:34 +00:00
|
|
|
default = "";
|
2022-07-19 13:05:45 +00:00
|
|
|
description = ''
|
2016-06-25 12:34:34 +00:00
|
|
|
Extra configuration appended to pulse/client.conf file.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
package = lib.mkOption {
|
|
|
|
type = lib.types.package;
|
2020-07-18 11:32:48 +00:00
|
|
|
default = if config.services.jack.jackd.enable
|
|
|
|
then pkgs.pulseaudioFull
|
|
|
|
else pkgs.pulseaudio;
|
2024-08-24 20:05:25 +00:00
|
|
|
defaultText = lib.literalExpression "pkgs.pulseaudio";
|
|
|
|
example = lib.literalExpression "pkgs.pulseaudioFull";
|
2022-07-19 13:05:45 +00:00
|
|
|
description = ''
|
2015-06-04 10:57:12 +00:00
|
|
|
The PulseAudio derivation to use. This can be used to enable
|
|
|
|
features (such as JACK support, Bluetooth) via the
|
2022-07-19 13:05:45 +00:00
|
|
|
`pulseaudioFull` package.
|
2013-02-28 18:40:07 +00:00
|
|
|
'';
|
|
|
|
};
|
2014-04-21 09:50:49 +00:00
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
extraModules = lib.mkOption {
|
|
|
|
type = lib.types.listOf lib.types.package;
|
2018-10-17 19:05:46 +00:00
|
|
|
default = [];
|
2024-08-24 20:05:25 +00:00
|
|
|
example = lib.literalExpression "[ pkgs.pulseaudio-modules-bt ]";
|
2022-07-19 13:05:45 +00:00
|
|
|
description = ''
|
2018-10-17 19:05:46 +00:00
|
|
|
Extra pulseaudio modules to use. This is intended for out-of-tree
|
|
|
|
pulseaudio modules like extra bluetooth codecs.
|
|
|
|
|
|
|
|
Extra modules take precedence over built-in pulseaudio modules.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2014-04-21 09:50:49 +00:00
|
|
|
daemon = {
|
2024-08-24 20:05:25 +00:00
|
|
|
logLevel = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2014-04-21 09:50:49 +00:00
|
|
|
default = "notice";
|
2022-07-19 13:05:45 +00:00
|
|
|
description = ''
|
2014-04-21 09:50:49 +00:00
|
|
|
The log level that the system-wide pulseaudio daemon should use,
|
|
|
|
if activated.
|
|
|
|
'';
|
|
|
|
};
|
2017-01-14 21:58:16 +00:00
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
config = lib.mkOption {
|
|
|
|
type = lib.types.attrsOf lib.types.unspecified;
|
2017-01-14 21:58:16 +00:00
|
|
|
default = {};
|
2022-07-19 13:05:45 +00:00
|
|
|
description = "Config of the pulse daemon. See `man pulse-daemon.conf`.";
|
2024-08-24 20:05:25 +00:00
|
|
|
example = lib.literalExpression ''{ realtime-scheduling = "yes"; }'';
|
2017-01-14 21:58:16 +00:00
|
|
|
};
|
2014-04-21 09:50:49 +00:00
|
|
|
};
|
2016-07-03 22:32:04 +00:00
|
|
|
|
|
|
|
zeroconf = {
|
|
|
|
discovery.enable =
|
2024-08-24 20:05:25 +00:00
|
|
|
lib.mkEnableOption "discovery of pulseaudio sinks in the local network";
|
2016-07-03 22:32:04 +00:00
|
|
|
publish.enable =
|
2024-08-24 20:05:25 +00:00
|
|
|
lib.mkEnableOption "publishing the pulseaudio sink in the local network";
|
2016-07-03 22:32:04 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
# TODO: enable by default?
|
|
|
|
tcp = {
|
2024-08-24 20:05:25 +00:00
|
|
|
enable = lib.mkEnableOption "tcp streaming support";
|
2016-07-03 22:32:04 +00:00
|
|
|
|
|
|
|
anonymousClients = {
|
2024-08-24 20:05:25 +00:00
|
|
|
allowAll = lib.mkEnableOption "all anonymous clients to stream to the server";
|
|
|
|
allowedIpRanges = lib.mkOption {
|
|
|
|
type = lib.types.listOf lib.types.str;
|
2016-07-03 22:32:04 +00:00
|
|
|
default = [];
|
2024-08-24 20:05:25 +00:00
|
|
|
example = lib.literalExpression ''[ "127.0.0.1" "192.168.1.0/24" ]'';
|
2022-07-19 13:05:45 +00:00
|
|
|
description = ''
|
2016-07-03 22:32:04 +00:00
|
|
|
A list of IP subnets that are allowed to stream to the server.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2011-08-01 13:26:51 +00:00
|
|
|
};
|
|
|
|
|
2011-07-26 01:51:56 +00:00
|
|
|
};
|
2011-09-14 18:20:50 +00:00
|
|
|
|
2011-07-26 01:51:56 +00:00
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
config = lib.mkIf cfg.enable (lib.mkMerge [
|
2013-02-28 18:40:07 +00:00
|
|
|
{
|
2023-10-26 20:16:02 +00:00
|
|
|
environment.etc."pulse/client.conf".source = clientConf;
|
2013-02-28 18:40:07 +00:00
|
|
|
|
2016-07-03 22:32:04 +00:00
|
|
|
environment.systemPackages = [ overriddenPackage ];
|
2013-02-28 18:40:07 +00:00
|
|
|
|
2019-09-14 17:51:29 +00:00
|
|
|
environment.etc = {
|
2024-07-13 10:56:18 +00:00
|
|
|
"alsa/conf.d/99-pulseaudio.conf".text = alsaConf;
|
2017-01-14 21:58:16 +00:00
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
"pulse/daemon.conf".source = pkgs.writeText "daemon.conf"
|
2019-09-14 17:51:29 +00:00
|
|
|
(lib.generators.toKeyValue {} cfg.daemon.config);
|
2017-11-27 07:49:38 +00:00
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
"openal/alsoft.conf".source = pkgs.writeText "alsoft.conf" "drivers=pulse";
|
2017-11-27 07:49:38 +00:00
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
"libao.conf".source = pkgs.writeText "libao.conf" "default_driver=pulse";
|
2019-09-14 17:51:29 +00:00
|
|
|
};
|
2013-02-28 18:40:07 +00:00
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
hardware.pulseaudio.configFile = lib.mkDefault "${lib.getBin overriddenPackage}/etc/pulse/default.pa";
|
2023-10-26 20:16:02 +00:00
|
|
|
|
2019-01-27 19:24:12 +00:00
|
|
|
# Disable flat volumes to enable relative ones
|
2024-08-24 20:05:25 +00:00
|
|
|
hardware.pulseaudio.daemon.config.flat-volumes = lib.mkDefault "no";
|
2019-01-27 19:24:12 +00:00
|
|
|
|
2019-05-31 15:04:33 +00:00
|
|
|
# Upstream defaults to speex-float-1 which results in audible artifacts
|
2024-08-24 20:05:25 +00:00
|
|
|
hardware.pulseaudio.daemon.config.resample-method = lib.mkDefault "speex-float-5";
|
2019-05-31 15:04:33 +00:00
|
|
|
|
2013-02-28 18:40:07 +00:00
|
|
|
# Allow PulseAudio to get realtime priority using rtkit.
|
|
|
|
security.rtkit.enable = true;
|
2016-07-03 22:32:04 +00:00
|
|
|
|
2017-08-03 01:17:19 +00:00
|
|
|
systemd.packages = [ overriddenPackage ];
|
2020-02-07 21:50:07 +00:00
|
|
|
|
|
|
|
# PulseAudio is packaged with udev rules to handle various audio device quirks
|
|
|
|
services.udev.packages = [ overriddenPackage ];
|
2023-10-26 20:16:02 +00:00
|
|
|
}
|
2016-07-03 22:32:04 +00:00
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
(lib.mkIf (cfg.extraModules != []) {
|
2018-10-17 19:05:46 +00:00
|
|
|
hardware.pulseaudio.daemon.config.dl-search-path = let
|
|
|
|
overriddenModules = builtins.map
|
|
|
|
(drv: drv.override { pulseaudio = overriddenPackage; })
|
|
|
|
cfg.extraModules;
|
|
|
|
modulePaths = builtins.map
|
2022-08-27 20:02:19 +00:00
|
|
|
(drv: "${drv}/lib/pulseaudio/modules")
|
2018-10-17 19:05:46 +00:00
|
|
|
# User-provided extra modules take precedence
|
|
|
|
(overriddenModules ++ [ overriddenPackage ]);
|
|
|
|
in lib.concatStringsSep ":" modulePaths;
|
|
|
|
})
|
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
(lib.mkIf hasZeroconf {
|
2016-07-03 22:32:04 +00:00
|
|
|
services.avahi.enable = true;
|
|
|
|
})
|
2024-08-24 20:05:25 +00:00
|
|
|
(lib.mkIf cfg.zeroconf.publish.enable {
|
2016-07-03 22:32:04 +00:00
|
|
|
services.avahi.publish.enable = true;
|
|
|
|
services.avahi.publish.userServices = true;
|
2013-02-28 18:40:07 +00:00
|
|
|
})
|
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
(lib.mkIf (!cfg.systemWide) {
|
2019-09-14 17:51:29 +00:00
|
|
|
environment.etc = {
|
|
|
|
"pulse/default.pa".source = myConfigFile;
|
2013-02-28 18:40:07 +00:00
|
|
|
};
|
2016-04-02 15:05:21 +00:00
|
|
|
systemd.user = {
|
|
|
|
services.pulseaudio = {
|
2017-05-25 17:33:13 +00:00
|
|
|
restartIfChanged = true;
|
2016-04-02 15:05:21 +00:00
|
|
|
serviceConfig = {
|
2016-07-20 23:18:06 +00:00
|
|
|
RestartSec = "500ms";
|
2017-05-25 17:33:13 +00:00
|
|
|
PassEnvironment = "DISPLAY";
|
2016-04-02 15:05:21 +00:00
|
|
|
};
|
2024-08-24 20:05:25 +00:00
|
|
|
} // lib.optionalAttrs config.services.jack.jackd.enable {
|
2020-07-18 11:32:48 +00:00
|
|
|
environment.JACK_PROMISCUOUS_SERVER = "jackaudio";
|
2017-05-25 17:33:13 +00:00
|
|
|
};
|
|
|
|
sockets.pulseaudio = {
|
|
|
|
wantedBy = [ "sockets.target" ];
|
2016-04-02 15:05:21 +00:00
|
|
|
};
|
|
|
|
};
|
2013-02-28 18:40:07 +00:00
|
|
|
})
|
|
|
|
|
2024-08-24 20:05:25 +00:00
|
|
|
(lib.mkIf cfg.systemWide {
|
2018-06-29 23:58:35 +00:00
|
|
|
users.users.pulse = {
|
2013-02-28 18:40:07 +00:00
|
|
|
# For some reason, PulseAudio wants UID == GID.
|
|
|
|
uid = assert uid == gid; uid;
|
|
|
|
group = "pulse";
|
|
|
|
extraGroups = [ "audio" ];
|
|
|
|
description = "PulseAudio system service user";
|
2014-09-02 11:16:13 +00:00
|
|
|
home = stateDir;
|
2023-11-30 00:59:15 +00:00
|
|
|
homeMode = "755";
|
2014-09-02 11:16:13 +00:00
|
|
|
createHome = true;
|
2021-03-07 13:54:00 +00:00
|
|
|
isSystemUser = true;
|
2013-02-28 18:40:07 +00:00
|
|
|
};
|
2013-10-30 16:37:45 +00:00
|
|
|
|
2018-06-29 23:58:35 +00:00
|
|
|
users.groups.pulse.gid = gid;
|
2022-08-18 09:03:58 +00:00
|
|
|
users.groups.pulse-access = {};
|
2013-10-30 16:37:45 +00:00
|
|
|
|
2013-02-28 18:40:07 +00:00
|
|
|
systemd.services.pulseaudio = {
|
2013-11-09 19:06:01 +00:00
|
|
|
description = "PulseAudio System-Wide Server";
|
2013-02-28 18:40:07 +00:00
|
|
|
wantedBy = [ "sound.target" ];
|
|
|
|
before = [ "sound.target" ];
|
2014-04-21 09:50:49 +00:00
|
|
|
environment.PULSE_RUNTIME_PATH = stateDir;
|
|
|
|
serviceConfig = {
|
2016-04-02 15:05:21 +00:00
|
|
|
Type = "notify";
|
2016-07-03 22:32:04 +00:00
|
|
|
ExecStart = "${binaryNoDaemon} --log-level=${cfg.daemon.logLevel} --system -n --file=${myConfigFile}";
|
2016-04-02 15:05:21 +00:00
|
|
|
Restart = "on-failure";
|
2016-07-20 23:18:06 +00:00
|
|
|
RestartSec = "500ms";
|
2014-04-21 09:50:49 +00:00
|
|
|
};
|
2013-02-28 18:40:07 +00:00
|
|
|
};
|
2016-12-04 21:05:15 +00:00
|
|
|
|
|
|
|
environment.variables.PULSE_COOKIE = "${stateDir}/.config/pulse/cookie";
|
2013-02-28 18:40:07 +00:00
|
|
|
})
|
2023-10-26 20:16:02 +00:00
|
|
|
]);
|
2011-07-26 01:51:56 +00:00
|
|
|
|
|
|
|
}
|