Merge pull request #110615 from jansol/pipewire
This commit is contained in:
commit
9783fa9554
@ -321,7 +321,8 @@
|
|||||||
./services/desktops/gsignond.nix
|
./services/desktops/gsignond.nix
|
||||||
./services/desktops/gvfs.nix
|
./services/desktops/gvfs.nix
|
||||||
./services/desktops/malcontent.nix
|
./services/desktops/malcontent.nix
|
||||||
./services/desktops/pipewire.nix
|
./services/desktops/pipewire/pipewire.nix
|
||||||
|
./services/desktops/pipewire/pipewire-media-session.nix
|
||||||
./services/desktops/gnome3/at-spi2-core.nix
|
./services/desktops/gnome3/at-spi2-core.nix
|
||||||
./services/desktops/gnome3/chrome-gnome-shell.nix
|
./services/desktops/gnome3/chrome-gnome-shell.nix
|
||||||
./services/desktops/gnome3/evolution-data-server.nix
|
./services/desktops/gnome3/evolution-data-server.nix
|
||||||
|
@ -1,183 +0,0 @@
|
|||||||
# pipewire service.
|
|
||||||
{ config, lib, pkgs, ... }:
|
|
||||||
|
|
||||||
with lib;
|
|
||||||
|
|
||||||
let
|
|
||||||
cfg = config.services.pipewire;
|
|
||||||
enable32BitAlsaPlugins = cfg.alsa.support32Bit
|
|
||||||
&& pkgs.stdenv.isx86_64
|
|
||||||
&& pkgs.pkgsi686Linux.pipewire != null;
|
|
||||||
|
|
||||||
# The package doesn't output to $out/lib/pipewire directly so that the
|
|
||||||
# overlays can use the outputs to replace the originals in FHS environments.
|
|
||||||
#
|
|
||||||
# This doesn't work in general because of missing development information.
|
|
||||||
jack-libs = pkgs.runCommand "jack-libs" {} ''
|
|
||||||
mkdir -p "$out/lib"
|
|
||||||
ln -s "${cfg.package.jack}/lib" "$out/lib/pipewire"
|
|
||||||
'';
|
|
||||||
in {
|
|
||||||
|
|
||||||
meta = {
|
|
||||||
maintainers = teams.freedesktop.members;
|
|
||||||
};
|
|
||||||
|
|
||||||
###### interface
|
|
||||||
options = {
|
|
||||||
services.pipewire = {
|
|
||||||
enable = mkEnableOption "pipewire service";
|
|
||||||
|
|
||||||
package = mkOption {
|
|
||||||
type = types.package;
|
|
||||||
default = pkgs.pipewire;
|
|
||||||
defaultText = "pkgs.pipewire";
|
|
||||||
example = literalExample "pkgs.pipewire";
|
|
||||||
description = ''
|
|
||||||
The pipewire derivation to use.
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
socketActivation = mkOption {
|
|
||||||
default = true;
|
|
||||||
type = types.bool;
|
|
||||||
description = ''
|
|
||||||
Automatically run pipewire when connections are made to the pipewire socket.
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
extraConfig = mkOption {
|
|
||||||
type = types.lines;
|
|
||||||
default = "";
|
|
||||||
description = ''
|
|
||||||
Literal string to append to /etc/pipewire/pipewire.conf.
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
sessionManager = mkOption {
|
|
||||||
type = types.nullOr types.string;
|
|
||||||
default = null;
|
|
||||||
example = literalExample ''"''${pipewire}/bin/pipewire-media-session"'';
|
|
||||||
description = ''
|
|
||||||
Path to the pipewire session manager executable.
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
sessionManagerArguments = mkOption {
|
|
||||||
type = types.listOf types.string;
|
|
||||||
default = [];
|
|
||||||
example = literalExample ''[ "-p" "bluez5.msbc-support=true" ]'';
|
|
||||||
description = ''
|
|
||||||
Arguments passed to the pipewire session manager.
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
alsa = {
|
|
||||||
enable = mkEnableOption "ALSA support";
|
|
||||||
support32Bit = mkEnableOption "32-bit ALSA support on 64-bit systems";
|
|
||||||
};
|
|
||||||
|
|
||||||
jack = {
|
|
||||||
enable = mkEnableOption "JACK audio emulation";
|
|
||||||
};
|
|
||||||
|
|
||||||
pulse = {
|
|
||||||
enable = mkEnableOption "PulseAudio server emulation";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
###### implementation
|
|
||||||
config = mkIf cfg.enable {
|
|
||||||
assertions = [
|
|
||||||
{
|
|
||||||
assertion = cfg.pulse.enable -> !config.hardware.pulseaudio.enable;
|
|
||||||
message = "PipeWire based PulseAudio server emulation replaces PulseAudio. This option requires `hardware.pulseaudio.enable` to be set to false";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
assertion = cfg.jack.enable -> !config.services.jack.jackd.enable;
|
|
||||||
message = "PipeWire based JACK emulation doesn't use the JACK service. This option requires `services.jack.jackd.enable` to be set to false";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
services.pipewire.sessionManager = mkDefault "${cfg.package}/bin/pipewire-media-session";
|
|
||||||
|
|
||||||
environment.systemPackages = [ cfg.package ]
|
|
||||||
++ lib.optional cfg.jack.enable jack-libs;
|
|
||||||
|
|
||||||
systemd.packages = [ cfg.package ]
|
|
||||||
++ lib.optional cfg.pulse.enable cfg.package.pulse;
|
|
||||||
|
|
||||||
# PipeWire depends on DBUS but doesn't list it. Without this booting
|
|
||||||
# into a terminal results in the service crashing with an error.
|
|
||||||
systemd.user.sockets.pipewire.wantedBy = lib.mkIf cfg.socketActivation [ "sockets.target" ];
|
|
||||||
systemd.user.sockets.pipewire-pulse.wantedBy = lib.mkIf (cfg.socketActivation && cfg.pulse.enable) ["sockets.target"];
|
|
||||||
systemd.user.services.pipewire.bindsTo = [ "dbus.service" ];
|
|
||||||
services.udev.packages = [ cfg.package ];
|
|
||||||
|
|
||||||
# If any paths are updated here they must also be updated in the package test.
|
|
||||||
environment.etc."alsa/conf.d/49-pipewire-modules.conf" = mkIf cfg.alsa.enable {
|
|
||||||
text = ''
|
|
||||||
pcm_type.pipewire {
|
|
||||||
libs.native = ${cfg.package.lib}/lib/alsa-lib/libasound_module_pcm_pipewire.so ;
|
|
||||||
${optionalString enable32BitAlsaPlugins
|
|
||||||
"libs.32Bit = ${pkgs.pkgsi686Linux.pipewire.lib}/lib/alsa-lib/libasound_module_pcm_pipewire.so ;"}
|
|
||||||
}
|
|
||||||
ctl_type.pipewire {
|
|
||||||
libs.native = ${cfg.package.lib}/lib/alsa-lib/libasound_module_ctl_pipewire.so ;
|
|
||||||
${optionalString enable32BitAlsaPlugins
|
|
||||||
"libs.32Bit = ${pkgs.pkgsi686Linux.pipewire.lib}/lib/alsa-lib/libasound_module_ctl_pipewire.so ;"}
|
|
||||||
}
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
environment.etc."alsa/conf.d/50-pipewire.conf" = mkIf cfg.alsa.enable {
|
|
||||||
source = "${cfg.package}/share/alsa/alsa.conf.d/50-pipewire.conf";
|
|
||||||
};
|
|
||||||
environment.etc."alsa/conf.d/99-pipewire-default.conf" = mkIf cfg.alsa.enable {
|
|
||||||
source = "${cfg.package}/share/alsa/alsa.conf.d/99-pipewire-default.conf";
|
|
||||||
};
|
|
||||||
environment.sessionVariables.LD_LIBRARY_PATH =
|
|
||||||
lib.optional cfg.jack.enable "/run/current-system/sw/lib/pipewire";
|
|
||||||
|
|
||||||
environment.etc."pipewire/pipewire.conf" = {
|
|
||||||
# Adapted from src/daemon/pipewire.conf.in
|
|
||||||
text = ''
|
|
||||||
set-prop link.max-buffers 16 # version < 3 clients can't handle more
|
|
||||||
|
|
||||||
add-spa-lib audio.convert* audioconvert/libspa-audioconvert
|
|
||||||
add-spa-lib api.alsa.* alsa/libspa-alsa
|
|
||||||
add-spa-lib api.v4l2.* v4l2/libspa-v4l2
|
|
||||||
add-spa-lib api.libcamera.* libcamera/libspa-libcamera
|
|
||||||
add-spa-lib api.bluez5.* bluez5/libspa-bluez5
|
|
||||||
add-spa-lib api.vulkan.* vulkan/libspa-vulkan
|
|
||||||
add-spa-lib api.jack.* jack/libspa-jack
|
|
||||||
add-spa-lib support.* support/libspa-support
|
|
||||||
|
|
||||||
load-module libpipewire-module-rtkit # rt.prio=20 rt.time.soft=200000 rt.time.hard=200000
|
|
||||||
load-module libpipewire-module-protocol-native
|
|
||||||
load-module libpipewire-module-profiler
|
|
||||||
load-module libpipewire-module-metadata
|
|
||||||
load-module libpipewire-module-spa-device-factory
|
|
||||||
load-module libpipewire-module-spa-node-factory
|
|
||||||
load-module libpipewire-module-client-node
|
|
||||||
load-module libpipewire-module-client-device
|
|
||||||
load-module libpipewire-module-portal
|
|
||||||
load-module libpipewire-module-access
|
|
||||||
load-module libpipewire-module-adapter
|
|
||||||
load-module libpipewire-module-link-factory
|
|
||||||
load-module libpipewire-module-session-manager
|
|
||||||
|
|
||||||
create-object spa-node-factory factory.name=support.node.driver node.name=Dummy priority.driver=8000
|
|
||||||
|
|
||||||
exec ${cfg.sessionManager} ${lib.concatStringsSep " " cfg.sessionManagerArguments}
|
|
||||||
|
|
||||||
${cfg.extraConfig}
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
environment.etc."pipewire/media-session.d/with-alsa" = mkIf cfg.alsa.enable { text = ""; };
|
|
||||||
environment.etc."pipewire/media-session.d/with-pulseaudio" = mkIf cfg.pulse.enable { text = ""; };
|
|
||||||
environment.etc."pipewire/media-session.d/with-jack" = mkIf cfg.jack.enable { text = ""; };
|
|
||||||
};
|
|
||||||
}
|
|
@ -0,0 +1,340 @@
|
|||||||
|
# pipewire example session manager.
|
||||||
|
{ config, lib, pkgs, ... }:
|
||||||
|
|
||||||
|
with lib;
|
||||||
|
|
||||||
|
let
|
||||||
|
cfg = config.services.pipewire.media-session;
|
||||||
|
enable32BitAlsaPlugins = cfg.alsa.support32Bit
|
||||||
|
&& pkgs.stdenv.isx86_64
|
||||||
|
&& pkgs.pkgsi686Linux.pipewire != null;
|
||||||
|
|
||||||
|
# Helpers for generating the pipewire JSON config file
|
||||||
|
mkSPAValueString = v:
|
||||||
|
if builtins.isList v then "[${lib.concatMapStringsSep " " mkSPAValueString v}]"
|
||||||
|
else if lib.types.attrs.check v then
|
||||||
|
"{${lib.concatStringsSep " " (mkSPAKeyValue v)}}"
|
||||||
|
else lib.generators.mkValueStringDefault { } v;
|
||||||
|
|
||||||
|
mkSPAKeyValue = attrs: map (def: def.content) (
|
||||||
|
lib.sortProperties
|
||||||
|
(
|
||||||
|
lib.mapAttrsToList
|
||||||
|
(k: v: lib.mkOrder (v._priority or 1000) "${lib.escape [ "=" ] k} = ${mkSPAValueString (v._content or v)}")
|
||||||
|
attrs
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
toSPAJSON = attrs: lib.concatStringsSep "\n" (mkSPAKeyValue attrs);
|
||||||
|
in {
|
||||||
|
|
||||||
|
meta = {
|
||||||
|
maintainers = teams.freedesktop.members;
|
||||||
|
};
|
||||||
|
|
||||||
|
###### interface
|
||||||
|
options = {
|
||||||
|
services.pipewire.media-session = {
|
||||||
|
enable = mkOption {
|
||||||
|
type = types.bool;
|
||||||
|
default = true;
|
||||||
|
description = "Example pipewire session manager";
|
||||||
|
};
|
||||||
|
|
||||||
|
package = mkOption {
|
||||||
|
type = types.package;
|
||||||
|
default = pkgs.pipewire.mediaSession;
|
||||||
|
example = literalExample "pkgs.pipewire.mediaSession";
|
||||||
|
description = ''
|
||||||
|
The pipewire-media-session derivation to use.
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
|
||||||
|
config = mkOption {
|
||||||
|
type = types.attrs;
|
||||||
|
description = ''
|
||||||
|
Configuration for the media session core.
|
||||||
|
'';
|
||||||
|
default = {
|
||||||
|
# media-session config file
|
||||||
|
properties = {
|
||||||
|
# Properties to configure the session and some
|
||||||
|
# modules
|
||||||
|
#mem.mlock-all = false;
|
||||||
|
#context.profile.modules = "default,rtkit";
|
||||||
|
};
|
||||||
|
|
||||||
|
spa-libs = {
|
||||||
|
# Mapping from factory name to library.
|
||||||
|
"api.bluez5.*" = "bluez5/libspa-bluez5";
|
||||||
|
"api.alsa.*" = "alsa/libspa-alsa";
|
||||||
|
"api.v4l2.*" = "v4l2/libspa-v4l2";
|
||||||
|
"api.libcamera.*" = "libcamera/libspa-libcamera";
|
||||||
|
};
|
||||||
|
|
||||||
|
modules = {
|
||||||
|
# These are the modules that are enabled when a file with
|
||||||
|
# the key name is found in the media-session.d config directory.
|
||||||
|
# the default bundle is always enabled.
|
||||||
|
|
||||||
|
default = [
|
||||||
|
"flatpak" # manages flatpak access
|
||||||
|
"portal" # manage portal permissions
|
||||||
|
"v4l2" # video for linux udev detection
|
||||||
|
#"libcamera" # libcamera udev detection
|
||||||
|
"suspend-node" # suspend inactive nodes
|
||||||
|
"policy-node" # configure and link nodes
|
||||||
|
#"metadata" # export metadata API
|
||||||
|
#"default-nodes" # restore default nodes
|
||||||
|
#"default-profile" # restore default profiles
|
||||||
|
#"default-routes" # restore default route
|
||||||
|
#"streams-follow-default" # move streams when default changes
|
||||||
|
#"alsa-seq" # alsa seq midi support
|
||||||
|
#"alsa-monitor" # alsa udev detection
|
||||||
|
#"bluez5" # bluetooth support
|
||||||
|
#"restore-stream" # restore stream settings
|
||||||
|
];
|
||||||
|
"with-audio" = [
|
||||||
|
"metadata"
|
||||||
|
"default-nodes"
|
||||||
|
"default-profile"
|
||||||
|
"default-routes"
|
||||||
|
"alsa-seq"
|
||||||
|
"alsa-monitor"
|
||||||
|
];
|
||||||
|
"with-alsa" = [
|
||||||
|
"with-audio"
|
||||||
|
];
|
||||||
|
"with-jack" = [
|
||||||
|
"with-audio"
|
||||||
|
];
|
||||||
|
"with-pulseaudio" = [
|
||||||
|
"with-audio"
|
||||||
|
"bluez5"
|
||||||
|
"restore-stream"
|
||||||
|
"streams-follow-default"
|
||||||
|
];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
alsaMonitorConfig = mkOption {
|
||||||
|
type = types.attrs;
|
||||||
|
description = ''
|
||||||
|
Configuration for the alsa monitor.
|
||||||
|
'';
|
||||||
|
default = {
|
||||||
|
# alsa-monitor config file
|
||||||
|
properties = {
|
||||||
|
#alsa.jack-device = true
|
||||||
|
};
|
||||||
|
|
||||||
|
rules = [
|
||||||
|
# an array of matches/actions to evaluate
|
||||||
|
{
|
||||||
|
# rules for matching a device or node. It is an array of
|
||||||
|
# properties that all need to match the regexp. If any of the
|
||||||
|
# matches work, the actions are executed for the object.
|
||||||
|
matches = [
|
||||||
|
{
|
||||||
|
# this matches all cards
|
||||||
|
device.name = "~alsa_card.*";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
actions = {
|
||||||
|
# actions can update properties on the matched object.
|
||||||
|
update-props = {
|
||||||
|
api.alsa.use-acp = true;
|
||||||
|
#api.alsa.use-ucm = true;
|
||||||
|
#api.alsa.soft-mixer = false;
|
||||||
|
#api.alsa.ignore-dB = false;
|
||||||
|
#device.profile-set = "profileset-name";
|
||||||
|
#device.profile = "default profile name";
|
||||||
|
api.acp.auto-profile = false;
|
||||||
|
api.acp.auto-port = false;
|
||||||
|
#device.nick = "My Device";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
{
|
||||||
|
matches = [
|
||||||
|
{
|
||||||
|
# matches all sinks
|
||||||
|
node.name = "~alsa_input.*";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
# matches all sources
|
||||||
|
node.name = "~alsa_output.*";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
actions = {
|
||||||
|
update-props = {
|
||||||
|
#node.nick = "My Node";
|
||||||
|
#node.nick = null;
|
||||||
|
#priority.driver = 100;
|
||||||
|
#priority.session = 100;
|
||||||
|
#node.pause-on-idle = false;
|
||||||
|
#resample.quality = 4;
|
||||||
|
#channelmix.normalize = false;
|
||||||
|
#channelmix.mix-lfe = false;
|
||||||
|
#audio.channels = 2;
|
||||||
|
#audio.format = "S16LE";
|
||||||
|
#audio.rate = 44100;
|
||||||
|
#audio.position = "FL,FR";
|
||||||
|
#api.alsa.period-size = 1024;
|
||||||
|
#api.alsa.headroom = 0;
|
||||||
|
#api.alsa.disable-mmap = false;
|
||||||
|
#api.alsa.disable-batch = false;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
bluezMonitorConfig = mkOption {
|
||||||
|
type = types.attrs;
|
||||||
|
description = ''
|
||||||
|
Configuration for the bluez5 monitor.
|
||||||
|
'';
|
||||||
|
default = {
|
||||||
|
# bluez-monitor config file
|
||||||
|
properties = {
|
||||||
|
# msbc is not expected to work on all headset + adapter combinations.
|
||||||
|
#bluez5.msbc-support = true;
|
||||||
|
#bluez5.sbc-xq-support = true;
|
||||||
|
|
||||||
|
# Enabled headset roles (default: [ hsp_hs hfp_ag ]), this
|
||||||
|
# property only applies to native backend. Currently some headsets
|
||||||
|
# (Sony WH-1000XM3) are not working with both hsp_ag and hfp_ag
|
||||||
|
# enabled, disable either hsp_ag or hfp_ag to work around it.
|
||||||
|
#
|
||||||
|
# Supported headset roles: hsp_hs (HSP Headset),
|
||||||
|
# hsp_ag (HSP Audio Gateway),
|
||||||
|
# hfp_ag (HFP Audio Gateway)
|
||||||
|
#bluez5.headset-roles = [ "hsp_hs" "hsp_ag" "hfp_ag" ];
|
||||||
|
|
||||||
|
# Enabled A2DP codecs (default: all)
|
||||||
|
#bluez5.codecs = [ "sbc" "aac" "ldac" "aptx" "aptx_hd" ];
|
||||||
|
};
|
||||||
|
|
||||||
|
rules = [
|
||||||
|
# an array of matches/actions to evaluate
|
||||||
|
{
|
||||||
|
# rules for matching a device or node. It is an array of
|
||||||
|
# properties that all need to match the regexp. If any of the
|
||||||
|
# matches work, the actions are executed for the object.
|
||||||
|
matches = [
|
||||||
|
{
|
||||||
|
# this matches all cards
|
||||||
|
device.name = "~bluez_card.*";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
actions = {
|
||||||
|
# actions can update properties on the matched object.
|
||||||
|
update-props = {
|
||||||
|
#device.nick = "My Device";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
{
|
||||||
|
matches = [
|
||||||
|
{
|
||||||
|
# matches all sinks
|
||||||
|
node.name = "~bluez_input.*";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
# matches all sources
|
||||||
|
node.name = "~bluez_output.*";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
actions = {
|
||||||
|
update-props = {
|
||||||
|
#node.nick = "My Node"
|
||||||
|
#node.nick = null;
|
||||||
|
#priority.driver = 100;
|
||||||
|
#priority.session = 100;
|
||||||
|
#node.pause-on-idle = false;
|
||||||
|
#resample.quality = 4;
|
||||||
|
#channelmix.normalize = false;
|
||||||
|
#channelmix.mix-lfe = false;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
v4l2MonitorConfig = mkOption {
|
||||||
|
type = types.attrs;
|
||||||
|
description = ''
|
||||||
|
Configuration for the V4L2 monitor.
|
||||||
|
'';
|
||||||
|
default = {
|
||||||
|
# v4l2-monitor config file
|
||||||
|
properties = {
|
||||||
|
};
|
||||||
|
|
||||||
|
rules = [
|
||||||
|
# an array of matches/actions to evaluate
|
||||||
|
{
|
||||||
|
# rules for matching a device or node. It is an array of
|
||||||
|
# properties that all need to match the regexp. If any of the
|
||||||
|
# matches work, the actions are executed for the object.
|
||||||
|
matches = [
|
||||||
|
{
|
||||||
|
# this matches all devices
|
||||||
|
device.name = "~v4l2_device.*";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
actions = {
|
||||||
|
# actions can update properties on the matched object.
|
||||||
|
update-props = {
|
||||||
|
#device.nick = "My Device";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
{
|
||||||
|
matches = [
|
||||||
|
{
|
||||||
|
# matches all sinks
|
||||||
|
node.name = "~v4l2_input.*";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
# matches all sources
|
||||||
|
node.name = "~v4l2_output.*";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
actions = {
|
||||||
|
update-props = {
|
||||||
|
#node.nick = "My Node";
|
||||||
|
#node.nick = null;
|
||||||
|
#priority.driver = 100;
|
||||||
|
#priority.session = 100;
|
||||||
|
#node.pause-on-idle = true;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
###### implementation
|
||||||
|
config = mkIf cfg.enable {
|
||||||
|
environment.systemPackages = [ cfg.package ];
|
||||||
|
services.pipewire.sessionManagerExecutable = "${cfg.package}/bin/pipewire-media-session";
|
||||||
|
|
||||||
|
environment.etc."pipewire/media-session.d/media-session.conf" = { text = toSPAJSON cfg.config; };
|
||||||
|
environment.etc."pipewire/media-session.d/v4l2-monitor.conf" = { text = toSPAJSON cfg.v4l2MonitorConfig; };
|
||||||
|
|
||||||
|
environment.etc."pipewire/media-session.d/with-alsa" = mkIf config.services.pipewire.alsa.enable { text = ""; };
|
||||||
|
environment.etc."pipewire/media-session.d/alsa-monitor.conf" = mkIf config.services.pipewire.alsa.enable { text = toSPAJSON cfg.alsaMonitorConfig; };
|
||||||
|
|
||||||
|
environment.etc."pipewire/media-session.d/with-pulseaudio" = mkIf config.services.pipewire.pulse.enable { text = ""; };
|
||||||
|
environment.etc."pipewire/media-session.d/bluez-monitor.conf" = mkIf config.services.pipewire.pulse.enable { text = toSPAJSON cfg.bluezMonitorConfig; };
|
||||||
|
|
||||||
|
environment.etc."pipewire/media-session.d/with-jack" = mkIf config.services.pipewire.jack.enable { text = ""; };
|
||||||
|
};
|
||||||
|
}
|
265
nixos/modules/services/desktops/pipewire/pipewire.nix
Normal file
265
nixos/modules/services/desktops/pipewire/pipewire.nix
Normal file
@ -0,0 +1,265 @@
|
|||||||
|
# pipewire service.
|
||||||
|
{ config, lib, pkgs, ... }:
|
||||||
|
|
||||||
|
with lib;
|
||||||
|
|
||||||
|
let
|
||||||
|
cfg = config.services.pipewire;
|
||||||
|
enable32BitAlsaPlugins = cfg.alsa.support32Bit
|
||||||
|
&& pkgs.stdenv.isx86_64
|
||||||
|
&& pkgs.pkgsi686Linux.pipewire != null;
|
||||||
|
|
||||||
|
# The package doesn't output to $out/lib/pipewire directly so that the
|
||||||
|
# overlays can use the outputs to replace the originals in FHS environments.
|
||||||
|
#
|
||||||
|
# This doesn't work in general because of missing development information.
|
||||||
|
jack-libs = pkgs.runCommand "jack-libs" {} ''
|
||||||
|
mkdir -p "$out/lib"
|
||||||
|
ln -s "${cfg.package.jack}/lib" "$out/lib/pipewire"
|
||||||
|
'';
|
||||||
|
|
||||||
|
# Helpers for generating the pipewire JSON config file
|
||||||
|
mkSPAValueString = v:
|
||||||
|
if builtins.isList v then "[${lib.concatMapStringsSep " " mkSPAValueString v}]"
|
||||||
|
else if lib.types.attrs.check v then
|
||||||
|
"{${lib.concatStringsSep " " (mkSPAKeyValue v)}}"
|
||||||
|
else lib.generators.mkValueStringDefault { } v;
|
||||||
|
|
||||||
|
mkSPAKeyValue = attrs: map (def: def.content) (
|
||||||
|
lib.sortProperties
|
||||||
|
(
|
||||||
|
lib.mapAttrsToList
|
||||||
|
(k: v: lib.mkOrder (v._priority or 1000) "${lib.escape [ "=" ] k} = ${mkSPAValueString (v._content or v)}")
|
||||||
|
attrs
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
toSPAJSON = attrs: lib.concatStringsSep "\n" (mkSPAKeyValue attrs);
|
||||||
|
in {
|
||||||
|
|
||||||
|
meta = {
|
||||||
|
maintainers = teams.freedesktop.members;
|
||||||
|
};
|
||||||
|
|
||||||
|
###### interface
|
||||||
|
options = {
|
||||||
|
services.pipewire = {
|
||||||
|
enable = mkEnableOption "pipewire service";
|
||||||
|
|
||||||
|
package = mkOption {
|
||||||
|
type = types.package;
|
||||||
|
default = pkgs.pipewire;
|
||||||
|
defaultText = "pkgs.pipewire";
|
||||||
|
example = literalExample "pkgs.pipewire";
|
||||||
|
description = ''
|
||||||
|
The pipewire derivation to use.
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
|
||||||
|
socketActivation = mkOption {
|
||||||
|
default = true;
|
||||||
|
type = types.bool;
|
||||||
|
description = ''
|
||||||
|
Automatically run pipewire when connections are made to the pipewire socket.
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
|
||||||
|
config = mkOption {
|
||||||
|
type = types.attrs;
|
||||||
|
description = ''
|
||||||
|
Configuration for the pipewire daemon.
|
||||||
|
'';
|
||||||
|
default = {
|
||||||
|
properties = {
|
||||||
|
## set-prop is used to configure properties in the system
|
||||||
|
#
|
||||||
|
# "library.name.system" = "support/libspa-support";
|
||||||
|
# "context.data-loop.library.name.system" = "support/libspa-support";
|
||||||
|
"link.max-buffers" = 16; # version < 3 clients can't handle more than 16
|
||||||
|
#"mem.allow-mlock" = false;
|
||||||
|
#"mem.mlock-all" = true;
|
||||||
|
## https://gitlab.freedesktop.org/pipewire/pipewire/-/blob/master/src/pipewire/pipewire.h#L93
|
||||||
|
#"log.level" = 2; # 5 is trace, which is verbose as hell, default is 2 which is warnings, 4 is debug output, 3 is info
|
||||||
|
|
||||||
|
## Properties for the DSP configuration
|
||||||
|
#
|
||||||
|
#"default.clock.rate" = 48000;
|
||||||
|
#"default.clock.quantum" = 1024;
|
||||||
|
#"default.clock.min-quantum" = 32;
|
||||||
|
#"default.clock.max-quantum" = 8192;
|
||||||
|
#"default.video.width" = 640;
|
||||||
|
#"default.video.height" = 480;
|
||||||
|
#"default.video.rate.num" = 25;
|
||||||
|
#"default.video.rate.denom" = 1;
|
||||||
|
};
|
||||||
|
|
||||||
|
spa-libs = {
|
||||||
|
## add-spa-lib <factory-name regex> <library-name>
|
||||||
|
#
|
||||||
|
# used to find spa factory names. It maps an spa factory name
|
||||||
|
# regular expression to a library name that should contain
|
||||||
|
# that factory.
|
||||||
|
#
|
||||||
|
"audio.convert*" = "audioconvert/libspa-audioconvert";
|
||||||
|
"api.alsa.*" = "alsa/libspa-alsa";
|
||||||
|
"api.v4l2.*" = "v4l2/libspa-v4l2";
|
||||||
|
"api.libcamera.*" = "libcamera/libspa-libcamera";
|
||||||
|
"api.bluez5.*" = "bluez5/libspa-bluez5";
|
||||||
|
"api.vulkan.*" = "vulkan/libspa-vulkan";
|
||||||
|
"api.jack.*" = "jack/libspa-jack";
|
||||||
|
"support.*" = "support/libspa-support";
|
||||||
|
# "videotestsrc" = "videotestsrc/libspa-videotestsrc";
|
||||||
|
# "audiotestsrc" = "audiotestsrc/libspa-audiotestsrc";
|
||||||
|
};
|
||||||
|
|
||||||
|
modules = {
|
||||||
|
## <module-name> = { [args = "<key>=<value> ..."]
|
||||||
|
# [flags = ifexists] }
|
||||||
|
# [flags = [ifexists]|[nofail]}
|
||||||
|
#
|
||||||
|
# Loads a module with the given parameters.
|
||||||
|
# If ifexists is given, the module is ignoed when it is not found.
|
||||||
|
# If nofail is given, module initialization failures are ignored.
|
||||||
|
#
|
||||||
|
libpipewire-module-rtkit = {
|
||||||
|
args = {
|
||||||
|
#rt.prio = 20;
|
||||||
|
#rt.time.soft = 200000;
|
||||||
|
#rt.time.hard = 200000;
|
||||||
|
#nice.level = -11;
|
||||||
|
};
|
||||||
|
flags = "ifexists|nofail";
|
||||||
|
};
|
||||||
|
libpipewire-module-protocol-native = { _priority = -100; _content = "null"; };
|
||||||
|
libpipewire-module-profiler = "null";
|
||||||
|
libpipewire-module-metadata = "null";
|
||||||
|
libpipewire-module-spa-device-factory = "null";
|
||||||
|
libpipewire-module-spa-node-factory = "null";
|
||||||
|
libpipewire-module-client-node = "null";
|
||||||
|
libpipewire-module-client-device = "null";
|
||||||
|
libpipewire-module-portal = "null";
|
||||||
|
libpipewire-module-access = {
|
||||||
|
args.access = {
|
||||||
|
allowed = ["${builtins.unsafeDiscardStringContext cfg.sessionManagerExecutable}"];
|
||||||
|
rejected = [];
|
||||||
|
restricted = [];
|
||||||
|
force = "flatpak";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
libpipewire-module-adapter = "null";
|
||||||
|
libpipewire-module-link-factory = "null";
|
||||||
|
libpipewire-module-session-manager = "null";
|
||||||
|
};
|
||||||
|
|
||||||
|
objects = {
|
||||||
|
## create-object [-nofail] <factory-name> [<key>=<value> ...]
|
||||||
|
#
|
||||||
|
# Creates an object from a PipeWire factory with the given parameters.
|
||||||
|
# If -nofail is given, errors are ignored (and no object is created)
|
||||||
|
#
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
exec = {
|
||||||
|
## exec <program-name>
|
||||||
|
#
|
||||||
|
# Execute the given program. This is usually used to start the
|
||||||
|
# session manager. run the session manager with -h for options
|
||||||
|
#
|
||||||
|
"${builtins.unsafeDiscardStringContext cfg.sessionManagerExecutable}" = { args = "\"${lib.concatStringsSep " " cfg.sessionManagerArguments}\""; };
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
sessionManagerExecutable = mkOption {
|
||||||
|
type = types.str;
|
||||||
|
default = "";
|
||||||
|
example = literalExample ''${pkgs.pipewire.mediaSession}/bin/pipewire-media-session'';
|
||||||
|
description = ''
|
||||||
|
Path to the session manager executable.
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
|
||||||
|
sessionManagerArguments = mkOption {
|
||||||
|
type = types.listOf types.str;
|
||||||
|
default = [];
|
||||||
|
example = literalExample ''["-p" "bluez5.msbc-support=true"]'';
|
||||||
|
description = ''
|
||||||
|
Arguments passed to the pipewire session manager.
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
|
||||||
|
alsa = {
|
||||||
|
enable = mkEnableOption "ALSA support";
|
||||||
|
support32Bit = mkEnableOption "32-bit ALSA support on 64-bit systems";
|
||||||
|
};
|
||||||
|
|
||||||
|
jack = {
|
||||||
|
enable = mkEnableOption "JACK audio emulation";
|
||||||
|
};
|
||||||
|
|
||||||
|
pulse = {
|
||||||
|
enable = mkEnableOption "PulseAudio server emulation";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
###### implementation
|
||||||
|
config = mkIf cfg.enable {
|
||||||
|
assertions = [
|
||||||
|
{
|
||||||
|
assertion = cfg.pulse.enable -> !config.hardware.pulseaudio.enable;
|
||||||
|
message = "PipeWire based PulseAudio server emulation replaces PulseAudio. This option requires `hardware.pulseaudio.enable` to be set to false";
|
||||||
|
}
|
||||||
|
{
|
||||||
|
assertion = cfg.jack.enable -> !config.services.jack.jackd.enable;
|
||||||
|
message = "PipeWire based JACK emulation doesn't use the JACK service. This option requires `services.jack.jackd.enable` to be set to false";
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
environment.systemPackages = [ cfg.package ]
|
||||||
|
++ lib.optional cfg.jack.enable jack-libs;
|
||||||
|
|
||||||
|
systemd.packages = [ cfg.package ]
|
||||||
|
++ lib.optional cfg.pulse.enable cfg.package.pulse;
|
||||||
|
|
||||||
|
# PipeWire depends on DBUS but doesn't list it. Without this booting
|
||||||
|
# into a terminal results in the service crashing with an error.
|
||||||
|
systemd.user.sockets.pipewire.wantedBy = lib.mkIf cfg.socketActivation [ "sockets.target" ];
|
||||||
|
systemd.user.sockets.pipewire-pulse.wantedBy = lib.mkIf (cfg.socketActivation && cfg.pulse.enable) ["sockets.target"];
|
||||||
|
systemd.user.services.pipewire.bindsTo = [ "dbus.service" ];
|
||||||
|
services.udev.packages = [ cfg.package ];
|
||||||
|
|
||||||
|
# If any paths are updated here they must also be updated in the package test.
|
||||||
|
environment.etc."alsa/conf.d/49-pipewire-modules.conf" = mkIf cfg.alsa.enable {
|
||||||
|
text = ''
|
||||||
|
pcm_type.pipewire {
|
||||||
|
libs.native = ${cfg.package.lib}/lib/alsa-lib/libasound_module_pcm_pipewire.so ;
|
||||||
|
${optionalString enable32BitAlsaPlugins
|
||||||
|
"libs.32Bit = ${pkgs.pkgsi686Linux.pipewire.lib}/lib/alsa-lib/libasound_module_pcm_pipewire.so ;"}
|
||||||
|
}
|
||||||
|
ctl_type.pipewire {
|
||||||
|
libs.native = ${cfg.package.lib}/lib/alsa-lib/libasound_module_ctl_pipewire.so ;
|
||||||
|
${optionalString enable32BitAlsaPlugins
|
||||||
|
"libs.32Bit = ${pkgs.pkgsi686Linux.pipewire.lib}/lib/alsa-lib/libasound_module_ctl_pipewire.so ;"}
|
||||||
|
}
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
environment.etc."alsa/conf.d/50-pipewire.conf" = mkIf cfg.alsa.enable {
|
||||||
|
source = "${cfg.package}/share/alsa/alsa.conf.d/50-pipewire.conf";
|
||||||
|
};
|
||||||
|
environment.etc."alsa/conf.d/99-pipewire-default.conf" = mkIf cfg.alsa.enable {
|
||||||
|
source = "${cfg.package}/share/alsa/alsa.conf.d/99-pipewire-default.conf";
|
||||||
|
};
|
||||||
|
|
||||||
|
environment.sessionVariables.LD_LIBRARY_PATH =
|
||||||
|
lib.optional cfg.jack.enable "/run/current-system/sw/lib/pipewire";
|
||||||
|
|
||||||
|
# https://gitlab.freedesktop.org/pipewire/pipewire/-/issues/464#note_723554
|
||||||
|
systemd.user.services.pipewire.environment = {
|
||||||
|
"PIPEWIRE_LINK_PASSIVE" = "1";
|
||||||
|
"PIPEWIRE_CONFIG_FILE" = pkgs.writeText "pipewire.conf" (toSPAJSON cfg.config);
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
@ -19,13 +19,16 @@
|
|||||||
, libsndfile
|
, libsndfile
|
||||||
, vulkan-headers
|
, vulkan-headers
|
||||||
, vulkan-loader
|
, vulkan-loader
|
||||||
|
, ncurses
|
||||||
, makeFontsConf
|
, makeFontsConf
|
||||||
, callPackage
|
, callPackage
|
||||||
, nixosTests
|
, nixosTests
|
||||||
|
, withMediaSession ? true
|
||||||
, gstreamerSupport ? true, gst_all_1 ? null
|
, gstreamerSupport ? true, gst_all_1 ? null
|
||||||
, ffmpegSupport ? true, ffmpeg ? null
|
, ffmpegSupport ? true, ffmpeg ? null
|
||||||
, bluezSupport ? true, bluez ? null, sbc ? null, libopenaptx ? null, ldacbt ? null
|
, bluezSupport ? true, bluez ? null, sbc ? null, libopenaptx ? null, ldacbt ? null, fdk_aac ? null
|
||||||
, nativeHspSupport ? true
|
, nativeHspSupport ? true
|
||||||
|
, nativeHfpSupport ? true
|
||||||
, ofonoSupport ? true
|
, ofonoSupport ? true
|
||||||
, hsphfpdSupport ? true
|
, hsphfpdSupport ? true
|
||||||
}:
|
}:
|
||||||
@ -36,111 +39,117 @@ let
|
|||||||
};
|
};
|
||||||
|
|
||||||
mesonBool = b: if b then "true" else "false";
|
mesonBool = b: if b then "true" else "false";
|
||||||
in
|
|
||||||
stdenv.mkDerivation rec {
|
|
||||||
pname = "pipewire";
|
|
||||||
version = "0.3.18";
|
|
||||||
|
|
||||||
outputs = [
|
self = stdenv.mkDerivation rec {
|
||||||
"out"
|
pname = "pipewire";
|
||||||
"lib"
|
version = "0.3.21";
|
||||||
"pulse"
|
|
||||||
"jack"
|
|
||||||
"dev"
|
|
||||||
"doc"
|
|
||||||
"installedTests"
|
|
||||||
];
|
|
||||||
|
|
||||||
src = fetchFromGitLab {
|
outputs = [
|
||||||
domain = "gitlab.freedesktop.org";
|
"out"
|
||||||
owner = "pipewire";
|
"lib"
|
||||||
repo = "pipewire";
|
"pulse"
|
||||||
rev = version;
|
"jack"
|
||||||
sha256 = "1yghhgs18yqrnd0b2r75l5n8yng962r1wszbsi01v6i9zib3jc9g";
|
"dev"
|
||||||
};
|
"doc"
|
||||||
|
"mediaSession"
|
||||||
|
"installedTests"
|
||||||
|
];
|
||||||
|
|
||||||
patches = [
|
src = fetchFromGitLab {
|
||||||
# Break up a dependency cycle between outputs.
|
domain = "gitlab.freedesktop.org";
|
||||||
./alsa-profiles-use-libdir.patch
|
owner = "pipewire";
|
||||||
# Move installed tests into their own output.
|
repo = "pipewire";
|
||||||
./installed-tests-path.patch
|
rev = version;
|
||||||
# Change the path of the pipewire-pulse binary in the service definition.
|
hash = "sha256:2YJzPTMPIoQQeNja3F53SD4gtpdSlbD/i77hBWiQfuQ=";
|
||||||
./pipewire-pulse-path.patch
|
};
|
||||||
# Add flag to specify configuration directory (different from the installation directory).
|
|
||||||
./pipewire-config-dir.patch
|
|
||||||
];
|
|
||||||
|
|
||||||
nativeBuildInputs = [
|
patches = [
|
||||||
doxygen
|
# Break up a dependency cycle between outputs.
|
||||||
graphviz
|
./alsa-profiles-use-libdir.patch
|
||||||
meson
|
# Move installed tests into their own output.
|
||||||
ninja
|
./installed-tests-path.patch
|
||||||
pkg-config
|
# Change the path of the pipewire-pulse binary in the service definition.
|
||||||
];
|
./pipewire-pulse-path.patch
|
||||||
|
# Add flag to specify configuration directory (different from the installation directory).
|
||||||
|
./pipewire-config-dir.patch
|
||||||
|
];
|
||||||
|
|
||||||
buildInputs = [
|
nativeBuildInputs = [
|
||||||
alsaLib
|
doxygen
|
||||||
dbus
|
graphviz
|
||||||
glib
|
meson
|
||||||
libjack2
|
ninja
|
||||||
libsndfile
|
pkg-config
|
||||||
udev
|
];
|
||||||
vulkan-headers
|
|
||||||
vulkan-loader
|
|
||||||
valgrind
|
|
||||||
systemd
|
|
||||||
] ++ lib.optionals gstreamerSupport [ gst_all_1.gst-plugins-base gst_all_1.gstreamer ]
|
|
||||||
++ lib.optional ffmpegSupport ffmpeg
|
|
||||||
++ lib.optionals bluezSupport [ bluez libopenaptx ldacbt sbc ];
|
|
||||||
|
|
||||||
mesonFlags = [
|
buildInputs = [
|
||||||
"-Ddocs=true"
|
alsaLib
|
||||||
"-Dman=false" # we don't have xmltoman
|
dbus
|
||||||
"-Dexamples=true" # only needed for `pipewire-media-session`
|
glib
|
||||||
"-Dudevrulesdir=lib/udev/rules.d"
|
libjack2
|
||||||
"-Dinstalled_tests=true"
|
libsndfile
|
||||||
"-Dinstalled_test_prefix=${placeholder "installedTests"}"
|
ncurses
|
||||||
"-Dpipewire_pulse_prefix=${placeholder "pulse"}"
|
udev
|
||||||
"-Dlibjack-path=${placeholder "jack"}/lib"
|
vulkan-headers
|
||||||
"-Dgstreamer=${mesonBool gstreamerSupport}"
|
vulkan-loader
|
||||||
"-Dffmpeg=${mesonBool ffmpegSupport}"
|
valgrind
|
||||||
"-Dbluez5=${mesonBool bluezSupport}"
|
systemd
|
||||||
"-Dbluez5-backend-native=${mesonBool nativeHspSupport}"
|
] ++ lib.optionals gstreamerSupport [ gst_all_1.gst-plugins-base gst_all_1.gstreamer ]
|
||||||
"-Dbluez5-backend-ofono=${mesonBool ofonoSupport}"
|
++ lib.optional ffmpegSupport ffmpeg
|
||||||
"-Dbluez5-backend-hsphfpd=${mesonBool hsphfpdSupport}"
|
++ lib.optionals bluezSupport [ bluez libopenaptx ldacbt sbc fdk_aac ];
|
||||||
"-Dpipewire_config_dir=/etc/pipewire"
|
|
||||||
];
|
|
||||||
|
|
||||||
FONTCONFIG_FILE = fontsConf; # Fontconfig error: Cannot load default config file
|
mesonFlags = [
|
||||||
|
"-Ddocs=true"
|
||||||
|
"-Dman=false" # we don't have xmltoman
|
||||||
|
"-Dexamples=${mesonBool withMediaSession}" # only needed for `pipewire-media-session`
|
||||||
|
"-Dudevrulesdir=lib/udev/rules.d"
|
||||||
|
"-Dinstalled_tests=true"
|
||||||
|
"-Dinstalled_test_prefix=${placeholder "installedTests"}"
|
||||||
|
"-Dpipewire_pulse_prefix=${placeholder "pulse"}"
|
||||||
|
"-Dlibjack-path=${placeholder "jack"}/lib"
|
||||||
|
"-Dgstreamer=${mesonBool gstreamerSupport}"
|
||||||
|
"-Dffmpeg=${mesonBool ffmpegSupport}"
|
||||||
|
"-Dbluez5=${mesonBool bluezSupport}"
|
||||||
|
"-Dbluez5-backend-hsp-native=${mesonBool nativeHspSupport}"
|
||||||
|
"-Dbluez5-backend-hfp-native=${mesonBool nativeHfpSupport}"
|
||||||
|
"-Dbluez5-backend-ofono=${mesonBool ofonoSupport}"
|
||||||
|
"-Dbluez5-backend-hsphfpd=${mesonBool hsphfpdSupport}"
|
||||||
|
"-Dpipewire_config_dir=/etc/pipewire"
|
||||||
|
];
|
||||||
|
|
||||||
doCheck = true;
|
FONTCONFIG_FILE = fontsConf; # Fontconfig error: Cannot load default config file
|
||||||
|
|
||||||
postInstall = ''
|
doCheck = true;
|
||||||
moveToOutput "share/systemd/user/pipewire-pulse.*" "$pulse"
|
|
||||||
moveToOutput "lib/systemd/user/pipewire-pulse.*" "$pulse"
|
|
||||||
moveToOutput "bin/pipewire-pulse" "$pulse"
|
|
||||||
'';
|
|
||||||
|
|
||||||
passthru.tests = {
|
postInstall = ''
|
||||||
installedTests = nixosTests.installed-tests.pipewire;
|
moveToOutput "share/systemd/user/pipewire-pulse.*" "$pulse"
|
||||||
|
moveToOutput "lib/systemd/user/pipewire-pulse.*" "$pulse"
|
||||||
|
moveToOutput "bin/pipewire-pulse" "$pulse"
|
||||||
|
moveToOutput "bin/pipewire-media-session" "$mediaSession"
|
||||||
|
'';
|
||||||
|
|
||||||
# This ensures that all the paths used by the NixOS module are found.
|
passthru.tests = {
|
||||||
test-paths = callPackage ./test-paths.nix {
|
installedTests = nixosTests.installed-tests.pipewire;
|
||||||
paths-out = [
|
|
||||||
"share/alsa/alsa.conf.d/50-pipewire.conf"
|
# This ensures that all the paths used by the NixOS module are found.
|
||||||
];
|
test-paths = callPackage ./test-paths.nix {
|
||||||
paths-lib = [
|
paths-out = [
|
||||||
"lib/alsa-lib/libasound_module_pcm_pipewire.so"
|
"share/alsa/alsa.conf.d/50-pipewire.conf"
|
||||||
"share/alsa-card-profile/mixer"
|
];
|
||||||
];
|
paths-lib = [
|
||||||
|
"lib/alsa-lib/libasound_module_pcm_pipewire.so"
|
||||||
|
"share/alsa-card-profile/mixer"
|
||||||
|
];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
meta = with lib; {
|
||||||
|
description = "Server and user space API to deal with multimedia pipelines";
|
||||||
|
homepage = "https://pipewire.org/";
|
||||||
|
license = licenses.mit;
|
||||||
|
platforms = platforms.linux;
|
||||||
|
maintainers = with maintainers; [ jtojnar ];
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
meta = with lib; {
|
in self
|
||||||
description = "Server and user space API to deal with multimedia pipelines";
|
|
||||||
homepage = "https://pipewire.org/";
|
|
||||||
license = licenses.mit;
|
|
||||||
platforms = platforms.linux;
|
|
||||||
maintainers = with maintainers; [ jtojnar ];
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
@ -1,19 +1,22 @@
|
|||||||
diff --git a/meson_options.txt b/meson_options.txt
|
diff --git a/meson_options.txt b/meson_options.txt
|
||||||
index 4b9e46b8..9d73ed06 100644
|
index 050a4c31..c481e76c 100644
|
||||||
--- a/meson_options.txt
|
--- a/meson_options.txt
|
||||||
+++ b/meson_options.txt
|
+++ b/meson_options.txt
|
||||||
@@ -147,3 +147,6 @@ option('pw-cat',
|
@@ -148,6 +148,9 @@ option('udev',
|
||||||
option('udevrulesdir',
|
option('udevrulesdir',
|
||||||
type : 'string',
|
type : 'string',
|
||||||
description : 'Directory for udev rules (defaults to /lib/udev/rules.d)')
|
description : 'Directory for udev rules (defaults to /lib/udev/rules.d)')
|
||||||
+option('pipewire_pulse_prefix',
|
+option('pipewire_pulse_prefix',
|
||||||
+ type : 'string',
|
+ type : 'string',
|
||||||
+ description : 'Install directory for the pipewire-pulse daemon')
|
+ description : 'Install directory for the pipewire-pulse daemon')
|
||||||
|
option('systemd-user-unit-dir',
|
||||||
|
type : 'string',
|
||||||
|
description : 'Directory for user systemd units (defaults to /usr/lib/systemd/user)')
|
||||||
diff --git a/src/daemon/systemd/user/meson.build b/src/daemon/systemd/user/meson.build
|
diff --git a/src/daemon/systemd/user/meson.build b/src/daemon/systemd/user/meson.build
|
||||||
index 29fc93d4..f78946f2 100644
|
index 46dfbbc8..0d975cec 100644
|
||||||
--- a/src/daemon/systemd/user/meson.build
|
--- a/src/daemon/systemd/user/meson.build
|
||||||
+++ b/src/daemon/systemd/user/meson.build
|
+++ b/src/daemon/systemd/user/meson.build
|
||||||
@@ -6,7 +6,7 @@ install_data(
|
@@ -9,7 +9,7 @@ install_data(
|
||||||
|
|
||||||
systemd_config = configuration_data()
|
systemd_config = configuration_data()
|
||||||
systemd_config.set('PW_BINARY', join_paths(pipewire_bindir, 'pipewire'))
|
systemd_config.set('PW_BINARY', join_paths(pipewire_bindir, 'pipewire'))
|
||||||
|
Loading…
Reference in New Issue
Block a user