PipeWire: Difference between revisions
| imported>Thiagokokada  Add SBC-XQ configuration. |  Add section "System-wide PipeWire" | ||
| (69 intermediate revisions by 42 users not shown) | |||
| Line 1: | Line 1: | ||
| PipeWire is a new low-level multimedia framework. It aims to offer capture and playback for both audio and video with minimal latency and support for PulseAudio-, JACK-, ALSA- and GStreamer-based applications. PipeWire has a great bluetooth support: because Pulseaudio was [https://github.com/NixOS/nixpkgs/issues/123784 reported to have troubles with bluetooth], PipeWire can be a good alternative.   | |||
| [https://www.pipewire.org/ PipeWire] is a relatively new (first release in 2017) low-level multimedia framework. It aims to offer capture and playback for both audio and video with minimal latency and support for [[PulseAudio]]-, [[JACK]]-, [[ALSA]]- and [[GStreamer]]-based applications. PipeWire has a great [[Bluetooth|bluetooth]] support: because Pulseaudio was [https://github.com/NixOS/nixpkgs/issues/123784 reported to have troubles with bluetooth], PipeWire can be a good alternative. | |||
| The daemon based on the framework can be configured to be both an audio server (with PulseAudio and JACK features) and a video capture server. | The daemon based on the framework can be configured to be both an audio server (with PulseAudio and JACK features) and a video capture server. | ||
| PipeWire also supports containers like Flatpak and does not rely on audio and video user groups but rather it uses a Polkit-like security model asking Flatpak or Wayland for permission to record screen or audio.   | PipeWire also supports containers like [[Flatpak]] and does not rely on <code>audio</code> and <code>video</code> user groups, but rather it uses a [[Polkit]]-like security model, asking Flatpak or [[Wayland]] for permission to record screen or audio. | ||
| As of [https://nixos.org/manual/nixos/stable/release-notes#sec-release-24.11 NixOS 24.11], PipeWire is the default sound server for most graphical sessions. | |||
| ==Configuring PipeWire== | |||
| <syntaxhighlight lang="nix"> | |||
|   # rtkit (optional, recommended) allows Pipewire to use the realtime scheduler for increased performance. | |||
|   security.rtkit.enable = true; | |||
|   services.pipewire = { | |||
|     enable = true; # if not already enabled | |||
|     alsa.enable = true; | |||
|     alsa.support32Bit = true; | |||
|     pulse.enable = true; | |||
|     # If you want to use JACK applications, uncomment the following | |||
|     #jack.enable = true; | |||
|   }; | |||
| </syntaxhighlight> | |||
| It is possible to use the {{nixos:option|services.pipewire.extraConfig}} option hierarchy in NixOS to create drop-in configuration files, if needed. For example, to disable the [https://docs.pipewire.org/page_module_x11_bell.html PipeWire x11-bell module] (which plays a sound on every X11 urgency hint), use: | |||
| <syntaxhighlight lang="nix"> | |||
|   services.pipewire = { | |||
|     enable = true; | |||
|     # Disable X11 bell module, which plays a sound on urgency hint | |||
|     # (my prompt includes an urgency hint, so I want no sounds). | |||
|     extraConfig = { | |||
|       pipewire."99-silent-bell.conf" = { | |||
|         "context.properties" = { | |||
|           "module.x11.bell" = false; | |||
|         }; | |||
|       }; | |||
|     }; | |||
|   }; | |||
| </syntaxhighlight> | |||
| ==Bluetooth Configuration== | |||
| {{main|Bluetooth}}  | |||
| PipeWire can be configured to use specific codecs, by default all codecs and most connection modes are enabled, see [https://pipewire.pages.freedesktop.org/wireplumber/daemon/configuration/bluetooth.html#monitor-properties this link] for precise details of which connections modes are enabled by default. The mSBC codec provides slightly better sound quality in calls than regular HFP/HSP, while the SBC-XQ provides better sound quality for audio listening. For more information [https://www.guyrutenberg.com/2021/03/11/replacing-pulseaudio-with-pipewire/ see this link]. | |||
| Wireplumber ({{nixos:option|services.pipewire.wireplumber}}) is the default modular session / policy manager for PipeWire in unstable. To add custom configuration you can use {{nixos:option|services.pipewire.wireplumber.extraConfig}} directly. For example: | |||
| < | <syntaxhighlight lang="nix"> | ||
|   services.pipewire.wireplumber.extraConfig."10-bluez" = { | |||
|     "monitor.bluez.properties" = { | |||
|       "bluez5.enable-sbc-xq" = true; | |||
|       "bluez5.enable-msbc" = true; | |||
|       "bluez5.enable-hw-volume" = true; | |||
|       "bluez5.roles" = [ | |||
|         "hsp_hs" | |||
|         "hsp_ag" | |||
|         "hfp_hf" | |||
|         "hfp_ag" | |||
|       ]; | |||
|     }; | |||
|   }; | |||
| </syntaxhighlight>{{Note|This config may break wireplumber, so that the following options were no longer available for Bluetooth headsets | |||
| - High Fidelity Playback (A2DP Sink, codec SBC) | |||
| - High Fidelity Playback (A2DP Sink, codec SBC-XQ) | |||
| - High Fidelity Playback (A2DP Sink, codec AAC)}} | |||
| < | Or, to disable automatic HSP/HFP and A2DP mode switching, which is part of the <code>11-bluetooth-policy</code> configuration:<syntaxhighlight lang="nix"> | ||
| services.pipewire  |   services.pipewire.wireplumber.extraConfig."11-bluetooth-policy" = { | ||
|      "wireplumber.settings" = { | |||
|      " |        "bluetooth.autoswitch-to-headset-profile" = false; | ||
|     }; | |||
|    }; |    }; | ||
| }; | </syntaxhighlight>Alternatively you can set {{nixos:option|services.pipewire.wireplumber.configPackages}} as well, adding derivations that output wireplumber config files in <code>$out/share/wireplumber/wireplumber.conf.d/*.conf</code>: | ||
| </ | |||
| <syntaxhighlight lang="nix"> | |||
|   services.pipewire.wireplumber.configPackages = [ | |||
|     (pkgs.writeTextDir "share/wireplumber/wireplumber.conf.d/10-bluez.conf" '' | |||
|       monitor.bluez.properties = { | |||
|         bluez5.enable-sbc-xq = true | |||
|         bluez5.enable-msbc = true | |||
|         bluez5.enable-hw-volume = true | |||
|         bluez5.roles = [hsp_hs hsp_ag hfp_hf hfp_ag] | |||
|       } | |||
|     '') | |||
|   ]; | |||
| </syntaxhighlight> | |||
| It is possible change a particular user instead of system-wide, with adding this to <code>~/.config/wireplumber/bluetooth.conf.d</code> (<code>~/.config/wireplumber/bluetooth.lua.d</code> for wireplumber 4.X and below) instead, manually or using [[Home Manager]]. Refer to [https://wiki.archlinux.org/title/PipeWire ArchWiki] for possible configurations, as well as the [https://docs.pipewire.org/ Full Documentation]. | |||
| ==AirPlay/RAOP configuration== | |||
| Remote Audio Output Protocol, branded as AirPlay, is the apple-developed wireless audio stack used in apple devices, many "smart speakers" and similar appliances as well as several open source implementations. It's based on RTSP, streams in PCM and is supported natively by PipeWire. With the following configuration AirPlay servers on your local network should be automatically added as output devices: | |||
| < | <syntaxhighlight lang="nix"> | ||
| # avahi required for service discovery | |||
| services.avahi.enable = true; | |||
| == | services.pipewire = { | ||
|   # opens UDP ports 6001-6002 | |||
|   raopOpenFirewall = true; | |||
|   extraConfig.pipewire = { | |||
|     "10-airplay" = { | |||
|       "context.modules" = [ | |||
|         { | |||
|           name = "libpipewire-module-raop-discover"; | |||
|           # increase the buffer size if you get dropouts/glitches | |||
|           # args = { | |||
|            #   "raop.latency.ms" = 500; | |||
|            # }; | |||
|          } | |||
|            " | |||
|            #  | |||
|        ]; |        ]; | ||
|     }; | |||
|   }; | |||
| }; | }; | ||
| </ | </syntaxhighlight> | ||
| Note that to set up an airplay server as opposed to a client, separate software is required. | |||
| ==Graphical tools== | ==Graphical tools== | ||
| Line 85: | Line 127: | ||
| All protocols (Pulseaudio/JACK) are now talking to the PipeWire protocol and are managed by the PipeWire daemon (therefore, applications can be managed by both Pulseaudio and JACK tools). For that reason, all graphical tools used for these protocols can be used: | All protocols (Pulseaudio/JACK) are now talking to the PipeWire protocol and are managed by the PipeWire daemon (therefore, applications can be managed by both Pulseaudio and JACK tools). For that reason, all graphical tools used for these protocols can be used: | ||
| * pavucontrol: controls the volume (per-sink and per-app basis), the default outputs/inputs, the different profiles (for HDMI outputs/bluetooth devices), routes each application to a different input/output, etc. | * {{nixos:package|pavucontrol}}: controls the volume (per-sink and per-app basis), the default outputs/inputs, the different profiles (for HDMI outputs/bluetooth devices), routes each application to a different input/output, etc. | ||
| * plasma-pa: a Plasma applet to change volume directly from the systray. Also deals with volume keys. | * {{nixos:package|kdePackages.plasma-pa}}: a [[KDE|Plasma]] applet to change volume directly from the systray. Also deals with volume keys. | ||
| * qjackctl: with JACK emulation, provides a patchbay (to connect applications together). Note that JACK does not provide any way to change the volume of a single application; use Pulseaudio tools for that purpose. | * {{nixos:package|qjackctl}}: with JACK emulation, provides a patchbay (to connect applications together). Note that JACK does not provide any way to change the volume of a single application; use Pulseaudio tools for that purpose. | ||
| * carla: with JACK emulation, provides a patchbay (make sure to go to "Patchbay" tab and check "Canvas > Show External"). | * {{nixos:package|carla}}: with JACK emulation, provides a patchbay (make sure to go to "Patchbay" tab and check "Canvas > Show External"). | ||
| * catia/patchage: similar to qjackctl and carla. | * catia/{{nixos:package|patchage}}: similar to qjackctl and carla. | ||
| *  | * {{nixos:package|helvum}}: GTK-based patchbay for PipeWire (uses the PipeWire protocol). Volume control is planned for later. | ||
| ==Advanced Configuration== | ==Advanced Configuration== | ||
| PipeWire can be extensively configured to fit the users' needs. Should the user want to do some fancy routing with null sinks, these can be defined directly in the config as shown below. | PipeWire can be extensively configured to fit the users' needs.   | ||
| === Null sinks === | |||
| Should the user want to do some fancy routing with null sinks, these can be defined directly in the config as shown below. | |||
| This is especially convenient if the user has a multi-channel (8+, or something "weird" like 2x2, 3x2) soundcard that keeps confusing applications with too many channels or a bad channel layout. | This is especially convenient if the user has a multi-channel (8+, or something "weird" like 2x2, 3x2) soundcard that keeps confusing applications with too many channels or a bad channel layout. | ||
| Line 99: | Line 144: | ||
| <strong>Note</strong>: those cards can be set to the "Pro Audio" profile with <code>pavucontrol</code> so PipeWire doesn't try to guess a wrong channel layout for them. | <strong>Note</strong>: those cards can be set to the "Pro Audio" profile with <code>pavucontrol</code> so PipeWire doesn't try to guess a wrong channel layout for them. | ||
| <syntaxHighlight lang= | <syntaxHighlight lang=nix> | ||
| services.pipewire  |   services.pipewire.extraConfig.pipewire."91-null-sinks" = { | ||
|      "context.objects" = [ |      "context.objects" = [ | ||
|        { |        { | ||
| Line 108: | Line 152: | ||
|          factory = "spa-node-factory"; |          factory = "spa-node-factory"; | ||
|          args = { |          args = { | ||
|            "factory.name"  |            "factory.name" = "support.node.driver"; | ||
|            "node.name"  |            "node.name" = "Dummy-Driver"; | ||
|            "priority.driver"  |            "priority.driver" = 8000; | ||
|          }; |          }; | ||
|        } |        } | ||
| Line 116: | Line 160: | ||
|          factory = "adapter"; |          factory = "adapter"; | ||
|          args = { |          args = { | ||
|            "factory.name"  |            "factory.name" = "support.null-audio-sink"; | ||
|            "node.name"  |            "node.name" = "Microphone-Proxy"; | ||
|            "node.description" = "Microphone"; |            "node.description" = "Microphone"; | ||
|            "media.class"  |            "media.class" = "Audio/Source/Virtual"; | ||
|            "audio.position"  |            "audio.position" = "MONO"; | ||
|          }; |          }; | ||
|        } |        } | ||
| Line 126: | Line 170: | ||
|          factory = "adapter"; |          factory = "adapter"; | ||
|          args = { |          args = { | ||
|            "factory.name"  |            "factory.name" = "support.null-audio-sink"; | ||
|            "node.name"  |            "node.name" = "Main-Output-Proxy"; | ||
|            "node.description" = "Main Output"; |            "node.description" = "Main Output"; | ||
|            "media.class"  |            "media.class" = "Audio/Sink"; | ||
|            "audio.position"  |            "audio.position" = "FL,FR"; | ||
|          }; |          }; | ||
|        } |        } | ||
|      ]; |      ]; | ||
|    }; |    }; | ||
| </syntaxHighlight> | </syntaxHighlight> | ||
| ===Linking nodes=== | ===Linking nodes=== | ||
| The config does not currently cover linking nodes together, but this can be fixed with a script. Soundcard names and ports should be replaced with the ones from the user's configuration: | The config does not currently cover linking nodes together, but this can be fixed with a script. Soundcard names and ports should be replaced with the ones from the user's configuration: | ||
| <syntaxHighlight lang= | <syntaxHighlight lang=bash> | ||
| #!/usr/bin/env bash | #!/usr/bin/env bash | ||
| Line 153: | Line 196: | ||
| In order to load the script on startup, it can be added to <code>~/.xprofile</code> or the specific DE/WM autostart config. Similarly, a one-shot user service can be created that runs the script. | In order to load the script on startup, it can be added to <code>~/.xprofile</code> or the specific DE/WM autostart config. Similarly, a one-shot user service can be created that runs the script. | ||
| ==Low-latency setup== | === Low-latency setup === | ||
| Audio production and rhythm games require lower latency audio than general applications. PipeWire can achieve the required latency with much less CPU usage compared to PulseAudio, with the appropriate configuration. | Audio production and rhythm games require lower latency audio than general applications. PipeWire can achieve the required latency with much less CPU usage compared to PulseAudio, with the appropriate configuration. | ||
| The minimum period size controls how small a buffer can be. The lower it is, the less latency there is. PipeWire has a value of 32/48000 by default, which amounts to 0.667ms. It can be brought lower if needed: | The minimum period size controls how small a buffer can be. The lower it is, the less latency there is. PipeWire has a value of 32/48000 by default, which amounts to 0.667ms. It can be brought lower if needed: | ||
| <syntaxHighlight lang= | |||
| services.pipewire  | <syntaxHighlight lang=nix> | ||
|   services.pipewire.extraConfig.pipewire."92-low-latency" = { | |||
|      "context.properties" = { |      "context.properties" = { | ||
|        "default.clock.rate" = 48000; |        "default.clock.rate" = 48000; | ||
|        "default.clock.quantum" = 32; |        "default.clock.quantum" = 32; | ||
|        "default.clock.min-quantum" = 32; |        "default.clock.min-quantum" = 32; | ||
|        "default.clock.max-quantum" = 32; |        "default.clock.max-quantum" = 32; | ||
|      }; |      }; | ||
|    }; |    }; | ||
| </syntaxHighlight> | </syntaxHighlight> | ||
| <strong>NOTE</strong>: Every setup is different, and a lot of factors determine your final latency, like CPU speed, RT/PREEMPTIVE kernels and soundcards supporting different audio formats. That's why 32/48000 isn't always a value that's going to work for everyone. The best way to get everything working is to keep increasing the quant value until you get no crackles (underruns) or until you get audio again (in case there wasn't any). This won't guarantee the lowest possible latency, but will provide a decent one paired with stable audio. | <strong>NOTE</strong>: Every setup is different, and a lot of factors determine your final latency, like CPU speed, RT/PREEMPTIVE kernels and soundcards supporting different audio formats. That's why 32/48000 isn't always a value that's going to work for everyone. The best way to get everything working is to keep increasing the quant value until you get no crackles (underruns) or until you get audio again (in case there wasn't any). This won't guarantee the lowest possible latency, but will provide a decent one paired with stable audio. | ||
| ===PulseAudio backend=== | ==== PulseAudio backend ==== | ||
| Applications using the Pulse backend have a separate configuration. The default minimum value is 1024, so it needs to be tweaked if low-latency audio is desired. | Applications using the Pulse backend have a separate configuration. The default minimum value is 1024, so it needs to be tweaked if low-latency audio is desired. | ||
| < | |||
| services.pipewire  | <syntaxhighlight lang="nix"> | ||
|   services.pipewire.extraConfig.pipewire-pulse."92-low-latency" = { | |||
|     "context.properties" = [ | |||
|        { |        { | ||
|          name = "libpipewire-module-protocol-pulse"; |          name = "libpipewire-module-protocol-pulse"; | ||
|          args = { |          args = { }; | ||
|        } |        } | ||
|      ]; |      ]; | ||
|     "pulse.properties" = { | |||
|       "pulse.min.req" = "32/48000"; | |||
|       "pulse.default.req" = "32/48000"; | |||
|       "pulse.max.req" = "32/48000"; | |||
|       "pulse.min.quantum" = "32/48000"; | |||
|       "pulse.max.quantum" = "32/48000"; | |||
|     }; | |||
|      "stream.properties" = { |      "stream.properties" = { | ||
|        "node.latency" = "32/48000"; |        "node.latency" = "32/48000"; | ||
| Line 245: | Line 237: | ||
|      }; |      }; | ||
|    }; |    }; | ||
| </syntaxhighlight> | |||
| </ | |||
| As a general rule, the values in <code>pipewire-pulse</code> should not be lower than the ones in <code>pipewire</code>. | As a general rule, the values in <code>pipewire-pulse</code> should not be lower than the ones in <code>pipewire</code>. | ||
| ===Controlling the ALSA devices=== | ===Controlling the ALSA devices=== | ||
| It is possible to configure various aspects of soundcards through PipeWire, including format, period size and batch mode: | It is possible to configure various aspects of soundcards through PipeWire, including format, period size and batch mode: | ||
| <syntaxHighlight lang="nix"> | |||
| services.pipewire =  | <syntaxHighlight lang=nix> | ||
|    media- |   services.pipewire.wireplumber.configPackages = [ | ||
|      rules = [ |     (pkgs.writeTextDir "share/wireplumber/main.lua.d/99-alsa-lowlatency.lua" '' | ||
|       alsa_monitor.rules = { | |||
|         { | |||
|           matches = {{{ "node.name", "matches", "alsa_output.*" }}}; | |||
|           apply_properties = { | |||
|             ["audio.format"] = "S32LE", | |||
|             ["audio.rate"] = "96000", -- for USB soundcards it should be twice your desired rate | |||
|             ["api.alsa.period-size"] = 2, -- defaults to 1024, tweak by trial-and-error | |||
|             -- ["api.alsa.disable-batch"] = true, -- generally, USB soundcards use the batch mode | |||
|           }, | |||
|         }, | |||
|       } | |||
|     '') | |||
|   ]; | |||
| </syntaxHighlight> | |||
| The <code>matches</code> attribute applies the <code>actions</code> to the devices/properties listed there. It is usually used with soundcard names, like shown in the config above. <code><matches></code> can match any of the outputs of | |||
| <syntaxHighlight lang=console> | |||
| $ pw-dump | grep node.name | grep alsa | |||
| </syntaxHighlight> | |||
| === Headless operation === | |||
| PipeWire can run on a headless device (without a GUI) such as a Raspberry Pi connected to a speaker. In that case, it may be preferable to start PipeWire on boot and keep it running rather than only running during an interactive login session. Among other things, this helps prevent a race condition that may occur when socket activation fails to initialize  audio devices in time for their first use, leading to one-time errors after reboots. The following additional configuration facilitates this: | |||
| <syntaxhighlight lang="nix"> | |||
|   # Socket activation too slow for headless; start at boot instead. | |||
|   services.pipewire.socketActivation = false;  | |||
|   # Start WirePlumber (with PipeWire) at boot. | |||
|   systemd.user.services.wireplumber.wantedBy = [ "default.target" ]; | |||
|   users.users.<name>.linger = true; # keep user services running | |||
|    users.users.<name>.extraGroups = [ ... "audio" ]; | |||
| </syntaxhighlight> | |||
| Despite early activation, you may still experience a race condition that prevents audio from working if you play media immediately after a new login such as running an SSH command. If this occurs, try introducing a short delay (e.g. <code>sleep 5</code>) before invoking the media player application. | |||
| ==== System-wide PipeWire ==== | |||
| As an alternative to having lingering systemd user services, PipeWire can also run as a system-wide Systemd service. See {{nixos:option|services.pipewire.systemWide}} for more. | |||
| {{file|/etc/nixos/configuration.nix|nix|3= | |||
| services.pipewire.systemWide = true; | |||
| services.pipewire.pulse.enable = true; # pipewire-pulse also supports running system-wide | |||
| # PipeWire users must be in the `pipewire` group | |||
| users.users.myservice1.extraGroups = [ "pipewire" ]; | |||
| systemd.services.myservice2.serviceConfig.SupplementaryGroups = [ "pipewire" ]; | |||
| }} | |||
| ==Troubleshooting== | |||
| ===pactl not found=== | |||
| The <code>pactl</code> functionality is superseded in PipeWire with the native <code>pw-cli</code>, <code>pw-mon</code> and <code>pw-top</code> CLI tools. | |||
| When using WirePlumber (which is enabled by default), you can also use <code>wpctl</code> as a <code>pactl</code> alternative with similar high level subcommands. | |||
| === No signal detected in audio application from external audio interface === | |||
| This issue was seen when attempting to use a Roland STUDIO-CAPTURE 16x10 Audio Interface to record audio in Ardour8 on NixOS. One possible solution: | |||
| * Install {{nixos:package|pavucontrol}}. | |||
| * Run {{ic|pavucontrol}}, navigate to the configuration tab, and set the Profile for STUDIO-CAPTURE (or your audio interface) to "Pro Audio". | |||
| * Ensure the audio interface and Ardour are set to the same sample rate. Ardour8 will accept a configuration where its sample rate does not match the audio interface's sample rate, when using PipeWire, but will not actually be able to record audio unless they match. If you are unable to successfully activate a track's record button, this may be the issue. | |||
| === Sound pops a few seconds after playback stops OR audio takes a long time to start playing after a couple of seconds === | |||
| By default Wireplumber suspends a sink after 5 seconds of no playback which can create a sound pop on sensitive audio equipment, or a noticeable delay on equipment that takes a moment to start back up. You can disable this by providing extra configuration to wireplumber but first you need to find out the name of the problematic sink in the wireplumber namespace (or blanket disable the functionality). | |||
| <ol> | |||
| <li>'''Find out the name of the sink (Optional)''' | |||
| <ol> | |||
| <li>Run <code>pw-top</code> to monitor pipewire processes</li> | |||
| <li>Play and pause sound on the problematic device. After 5 seconds the numeric columns of at least one process will disappear. This is the suspension in action. You need the value of the NAME column to refer to this sink in the configuration below. Since the display updates each second, you can't copy the value easily. Rerun <code>pw-top -b</code> and abort with <kbd>CTRL</kbd>+<kbd>C</kbd> to get persistent output.</li> | |||
| </ol> | |||
| </li> | |||
| <li>'''Add configuration''': | |||
| If you want to blanket do this for all devices, you can use <code>~alsa_input.*</code> and <code>~alsa_output.*</code> to match all input and output devices. You can also use (from the step above) an exact device if you only want to change the setting for that single device. In that case you can match only a single device, and replace the node name with EG: <code>alsa_output.usb-ASUSTeK_Xonar_SoundCard-00.iec958-stereo</code>, and it will target that specific device. | |||
| <syntaxhighlight lang="nix"> | |||
|  # Disable suspend of Toslink output to prevent audio popping. | |||
|   services.pipewire.wireplumber.extraConfig."99-disable-suspend" = { | |||
|      "monitor.alsa.rules" = [ | |||
|        { |        { | ||
|          matches = [ { "node.name" = "alsa_output.*" } ]; |          matches = [ | ||
|           { | |||
|             "node.name" = "~alsa_input.*"; | |||
|           } | |||
|           { | |||
|             "node.name" = "~alsa_output.*"; | |||
|           } | |||
|         ]; | |||
|          actions = { |          actions = { | ||
|            update-props = { |            update-props = { | ||
|              " |              "session.suspend-timeout-seconds" = 0; | ||
|            }; |            }; | ||
|          }; |          }; | ||
|        } |        } | ||
|      ]; |      ]; | ||
|    }; |    }; | ||
| </syntaxhighlight> | |||
| </ | The value 0 disables suspension entirely. You could also set it to a higher value. 5 (seconds) is the default value. | ||
| </li> | |||
| < | |||
| <li>'''Verify resolution''': | |||
| </ | After switching to the new configuration you need to also run <code>systemctl --user restart wireplumber</code> as your non-root user to apply the new wireplumber configuration, since wireplumber runs as your non-root user - <code>nixos-rebuild switch</code> is not sufficient. Play and pause sound again and verify that no sound pops occur anymore and observe in <code>pw-top</code> that the numeric columns for that sink do not disappear after 5 seconds. | ||
| </li> | |||
| </ol> | |||
| === Eliminating audio startup delay after suspend or boot === | |||
| In addition to audio popping, some HTPC systems may experience a delay when audio playback starts after resuming from suspend or system boot. This occurs even if suspension is disabled as described above. | |||
| <ol> | |||
| <li>'''Identify your sink''' using the method in the previous section.</li> | |||
| == | <li>'''Enhance configuration''': | ||
| <syntaxhighlight lang="nix"> | |||
| # Disable node suspend on NVIDIA graphics | |||
| services.pipewire.wireplumber.extraConfig."99-disable-suspend"."monitor.alsa.rules" = [ | |||
|   { | |||
|     matches = [ { "node.name" = "alsa_output.pci-0000_01_00.1.hdmi-stereo-extra1"; } ]; | |||
|     actions.update-props = { | |||
|       "session.suspend-timeout-seconds" = 0; | |||
|       "node.always-process" = true; | |||
|       "dither.method" = "wannamaker3"; | |||
|       "dither.noise" = 1; | |||
|     }; | |||
|   } | |||
| ]; | |||
| </syntaxhighlight> | |||
| This configuration prevents audio popping and keeps the audio pipeline active, effectively reducing startup delay from ~5s to ~1s. Adding the dither settings further eliminates the remaining delay. | |||
| </li> | |||
| ==See also== | ==See also== | ||
| * https://github.com/NixOS/nixpkgs/issues/102547 | * https://github.com/NixOS/nixpkgs/issues/102547 | ||
| * https://gitlab.freedesktop.org/pipewire/pipewire/-/issues/3858 | |||
| [[Category:Audio]] | [[Category:Audio]] | ||
Latest revision as of 15:41, 5 September 2025
PipeWire is a relatively new (first release in 2017) low-level multimedia framework. It aims to offer capture and playback for both audio and video with minimal latency and support for PulseAudio-, JACK-, ALSA- and GStreamer-based applications. PipeWire has a great bluetooth support: because Pulseaudio was reported to have troubles with bluetooth, PipeWire can be a good alternative.
The daemon based on the framework can be configured to be both an audio server (with PulseAudio and JACK features) and a video capture server.
PipeWire also supports containers like Flatpak and does not rely on audio and video user groups, but rather it uses a Polkit-like security model, asking Flatpak or Wayland for permission to record screen or audio.
As of NixOS 24.11, PipeWire is the default sound server for most graphical sessions.
Configuring PipeWire
  # rtkit (optional, recommended) allows Pipewire to use the realtime scheduler for increased performance.
  security.rtkit.enable = true;
  services.pipewire = {
    enable = true; # if not already enabled
    alsa.enable = true;
    alsa.support32Bit = true;
    pulse.enable = true;
    # If you want to use JACK applications, uncomment the following
    #jack.enable = true;
  };
It is possible to use the services.pipewire.extraConfig option hierarchy in NixOS to create drop-in configuration files, if needed. For example, to disable the PipeWire x11-bell module (which plays a sound on every X11 urgency hint), use:
  services.pipewire = {
    enable = true;
    # Disable X11 bell module, which plays a sound on urgency hint
    # (my prompt includes an urgency hint, so I want no sounds).
    extraConfig = {
      pipewire."99-silent-bell.conf" = {
        "context.properties" = {
          "module.x11.bell" = false;
        };
      };
    };
  };
Bluetooth Configuration
- Main article: Bluetooth
PipeWire can be configured to use specific codecs, by default all codecs and most connection modes are enabled, see this link for precise details of which connections modes are enabled by default. The mSBC codec provides slightly better sound quality in calls than regular HFP/HSP, while the SBC-XQ provides better sound quality for audio listening. For more information see this link.
Wireplumber (services.pipewire.wireplumber) is the default modular session / policy manager for PipeWire in unstable. To add custom configuration you can use services.pipewire.wireplumber.extraConfig directly. For example:
  services.pipewire.wireplumber.extraConfig."10-bluez" = {
    "monitor.bluez.properties" = {
      "bluez5.enable-sbc-xq" = true;
      "bluez5.enable-msbc" = true;
      "bluez5.enable-hw-volume" = true;
      "bluez5.roles" = [
        "hsp_hs"
        "hsp_ag"
        "hfp_hf"
        "hfp_ag"
      ];
    };
  };
- High Fidelity Playback (A2DP Sink, codec SBC)
- High Fidelity Playback (A2DP Sink, codec SBC-XQ)
- High Fidelity Playback (A2DP Sink, codec AAC)
Or, to disable automatic HSP/HFP and A2DP mode switching, which is part of the 11-bluetooth-policy configuration:
  services.pipewire.wireplumber.extraConfig."11-bluetooth-policy" = {
    "wireplumber.settings" = {
      "bluetooth.autoswitch-to-headset-profile" = false;
    };
  };
Alternatively you can set services.pipewire.wireplumber.configPackages as well, adding derivations that output wireplumber config files in $out/share/wireplumber/wireplumber.conf.d/*.conf:
  services.pipewire.wireplumber.configPackages = [
    (pkgs.writeTextDir "share/wireplumber/wireplumber.conf.d/10-bluez.conf" ''
      monitor.bluez.properties = {
        bluez5.enable-sbc-xq = true
        bluez5.enable-msbc = true
        bluez5.enable-hw-volume = true
        bluez5.roles = [hsp_hs hsp_ag hfp_hf hfp_ag]
      }
    '')
  ];
It is possible change a particular user instead of system-wide, with adding this to ~/.config/wireplumber/bluetooth.conf.d (~/.config/wireplumber/bluetooth.lua.d for wireplumber 4.X and below) instead, manually or using Home Manager. Refer to ArchWiki for possible configurations, as well as the Full Documentation.
AirPlay/RAOP configuration
Remote Audio Output Protocol, branded as AirPlay, is the apple-developed wireless audio stack used in apple devices, many "smart speakers" and similar appliances as well as several open source implementations. It's based on RTSP, streams in PCM and is supported natively by PipeWire. With the following configuration AirPlay servers on your local network should be automatically added as output devices:
# avahi required for service discovery
services.avahi.enable = true;
services.pipewire = {
  # opens UDP ports 6001-6002
  raopOpenFirewall = true;
  extraConfig.pipewire = {
    "10-airplay" = {
      "context.modules" = [
        {
          name = "libpipewire-module-raop-discover";
          # increase the buffer size if you get dropouts/glitches
          # args = {
          #   "raop.latency.ms" = 500;
          # };
        }
      ];
    };
  };
};
Note that to set up an airplay server as opposed to a client, separate software is required.
Graphical tools
All protocols (Pulseaudio/JACK) are now talking to the PipeWire protocol and are managed by the PipeWire daemon (therefore, applications can be managed by both Pulseaudio and JACK tools). For that reason, all graphical tools used for these protocols can be used:
- pavucontrol: controls the volume (per-sink and per-app basis), the default outputs/inputs, the different profiles (for HDMI outputs/bluetooth devices), routes each application to a different input/output, etc.
- kdePackages.plasma-pa: a Plasma applet to change volume directly from the systray. Also deals with volume keys.
- qjackctl: with JACK emulation, provides a patchbay (to connect applications together). Note that JACK does not provide any way to change the volume of a single application; use Pulseaudio tools for that purpose.
- carla: with JACK emulation, provides a patchbay (make sure to go to "Patchbay" tab and check "Canvas > Show External").
- catia/patchage: similar to qjackctl and carla.
- helvum: GTK-based patchbay for PipeWire (uses the PipeWire protocol). Volume control is planned for later.
Advanced Configuration
PipeWire can be extensively configured to fit the users' needs.
Null sinks
Should the user want to do some fancy routing with null sinks, these can be defined directly in the config as shown below.
This is especially convenient if the user has a multi-channel (8+, or something "weird" like 2x2, 3x2) soundcard that keeps confusing applications with too many channels or a bad channel layout.
Note: those cards can be set to the "Pro Audio" profile with pavucontrol so PipeWire doesn't try to guess a wrong channel layout for them.
  services.pipewire.extraConfig.pipewire."91-null-sinks" = {
    "context.objects" = [
      {
        # A default dummy driver. This handles nodes marked with the "node.always-driver"
        # properyty when no other driver is currently active. JACK clients need this.
        factory = "spa-node-factory";
        args = {
          "factory.name" = "support.node.driver";
          "node.name" = "Dummy-Driver";
          "priority.driver" = 8000;
        };
      }
      {
        factory = "adapter";
        args = {
          "factory.name" = "support.null-audio-sink";
          "node.name" = "Microphone-Proxy";
          "node.description" = "Microphone";
          "media.class" = "Audio/Source/Virtual";
          "audio.position" = "MONO";
        };
      }
      {
        factory = "adapter";
        args = {
          "factory.name" = "support.null-audio-sink";
          "node.name" = "Main-Output-Proxy";
          "node.description" = "Main Output";
          "media.class" = "Audio/Sink";
          "audio.position" = "FL,FR";
        };
      }
    ];
  };
Linking nodes
The config does not currently cover linking nodes together, but this can be fixed with a script. Soundcard names and ports should be replaced with the ones from the user's configuration:
#!/usr/bin/env bash
# ports obtained from `pw-link -io`
pw-link "Main-Output-Proxy:monitor_FL" "alsa_output.usb-Native_Instruments_Komplete_Audio_6_69BC86B9-00.pro-audio:playback_1"
pw-link "Main-Output-Proxy:monitor_FR" "alsa_output.usb-Native_Instruments_Komplete_Audio_6_69BC86B9-00.pro-audio:playback_2"
pw-link "alsa_input.usb-M-Audio_Fast_Track-00.pro-audio:capture_1" "Microphone-Proxy:input_MONO"
In order to load the script on startup, it can be added to ~/.xprofile or the specific DE/WM autostart config. Similarly, a one-shot user service can be created that runs the script.
Low-latency setup
Audio production and rhythm games require lower latency audio than general applications. PipeWire can achieve the required latency with much less CPU usage compared to PulseAudio, with the appropriate configuration. The minimum period size controls how small a buffer can be. The lower it is, the less latency there is. PipeWire has a value of 32/48000 by default, which amounts to 0.667ms. It can be brought lower if needed:
  services.pipewire.extraConfig.pipewire."92-low-latency" = {
    "context.properties" = {
      "default.clock.rate" = 48000;
      "default.clock.quantum" = 32;
      "default.clock.min-quantum" = 32;
      "default.clock.max-quantum" = 32;
    };
  };
NOTE: Every setup is different, and a lot of factors determine your final latency, like CPU speed, RT/PREEMPTIVE kernels and soundcards supporting different audio formats. That's why 32/48000 isn't always a value that's going to work for everyone. The best way to get everything working is to keep increasing the quant value until you get no crackles (underruns) or until you get audio again (in case there wasn't any). This won't guarantee the lowest possible latency, but will provide a decent one paired with stable audio.
PulseAudio backend
Applications using the Pulse backend have a separate configuration. The default minimum value is 1024, so it needs to be tweaked if low-latency audio is desired.
  services.pipewire.extraConfig.pipewire-pulse."92-low-latency" = {
    "context.properties" = [
      {
        name = "libpipewire-module-protocol-pulse";
        args = { };
      }
    ];
    "pulse.properties" = {
      "pulse.min.req" = "32/48000";
      "pulse.default.req" = "32/48000";
      "pulse.max.req" = "32/48000";
      "pulse.min.quantum" = "32/48000";
      "pulse.max.quantum" = "32/48000";
    };
    "stream.properties" = {
      "node.latency" = "32/48000";
      "resample.quality" = 1;
    };
  };
As a general rule, the values in pipewire-pulse should not be lower than the ones in pipewire.
Controlling the ALSA devices
It is possible to configure various aspects of soundcards through PipeWire, including format, period size and batch mode:
  services.pipewire.wireplumber.configPackages = [
    (pkgs.writeTextDir "share/wireplumber/main.lua.d/99-alsa-lowlatency.lua" ''
      alsa_monitor.rules = {
        {
          matches = {{{ "node.name", "matches", "alsa_output.*" }}};
          apply_properties = {
            ["audio.format"] = "S32LE",
            ["audio.rate"] = "96000", -- for USB soundcards it should be twice your desired rate
            ["api.alsa.period-size"] = 2, -- defaults to 1024, tweak by trial-and-error
            -- ["api.alsa.disable-batch"] = true, -- generally, USB soundcards use the batch mode
          },
        },
      }
    '')
  ];
The matches attribute applies the actions to the devices/properties listed there. It is usually used with soundcard names, like shown in the config above. <matches> can match any of the outputs of
$ pw-dump | grep node.name | grep alsa
Headless operation
PipeWire can run on a headless device (without a GUI) such as a Raspberry Pi connected to a speaker. In that case, it may be preferable to start PipeWire on boot and keep it running rather than only running during an interactive login session. Among other things, this helps prevent a race condition that may occur when socket activation fails to initialize audio devices in time for their first use, leading to one-time errors after reboots. The following additional configuration facilitates this:
  # Socket activation too slow for headless; start at boot instead.
  services.pipewire.socketActivation = false; 
  # Start WirePlumber (with PipeWire) at boot.
  systemd.user.services.wireplumber.wantedBy = [ "default.target" ];
  users.users.<name>.linger = true; # keep user services running
  users.users.<name>.extraGroups = [ ... "audio" ];
Despite early activation, you may still experience a race condition that prevents audio from working if you play media immediately after a new login such as running an SSH command. If this occurs, try introducing a short delay (e.g. sleep 5) before invoking the media player application.
System-wide PipeWire
As an alternative to having lingering systemd user services, PipeWire can also run as a system-wide Systemd service. See services.pipewire.systemWide for more.
services.pipewire.systemWide = true;
services.pipewire.pulse.enable = true; # pipewire-pulse also supports running system-wide
# PipeWire users must be in the `pipewire` group
users.users.myservice1.extraGroups = [ "pipewire" ];
systemd.services.myservice2.serviceConfig.SupplementaryGroups = [ "pipewire" ];
Troubleshooting
pactl not found
The pactl functionality is superseded in PipeWire with the native pw-cli, pw-mon and pw-top CLI tools.
When using WirePlumber (which is enabled by default), you can also use wpctl as a pactl alternative with similar high level subcommands.
No signal detected in audio application from external audio interface
This issue was seen when attempting to use a Roland STUDIO-CAPTURE 16x10 Audio Interface to record audio in Ardour8 on NixOS. One possible solution:
- Install pavucontrol.
- Run pavucontrol, navigate to the configuration tab, and set the Profile for STUDIO-CAPTURE (or your audio interface) to "Pro Audio".
- Ensure the audio interface and Ardour are set to the same sample rate. Ardour8 will accept a configuration where its sample rate does not match the audio interface's sample rate, when using PipeWire, but will not actually be able to record audio unless they match. If you are unable to successfully activate a track's record button, this may be the issue.
Sound pops a few seconds after playback stops OR audio takes a long time to start playing after a couple of seconds
By default Wireplumber suspends a sink after 5 seconds of no playback which can create a sound pop on sensitive audio equipment, or a noticeable delay on equipment that takes a moment to start back up. You can disable this by providing extra configuration to wireplumber but first you need to find out the name of the problematic sink in the wireplumber namespace (or blanket disable the functionality).
- Find out the name of the sink (Optional)
- Run pw-topto monitor pipewire processes
- Play and pause sound on the problematic device. After 5 seconds the numeric columns of at least one process will disappear. This is the suspension in action. You need the value of the NAME column to refer to this sink in the configuration below. Since the display updates each second, you can't copy the value easily. Rerun pw-top -band abort with CTRL+C to get persistent output.
 
- Run 
- Add configuration:
If you want to blanket do this for all devices, you can use ~alsa_input.*and~alsa_output.*to match all input and output devices. You can also use (from the step above) an exact device if you only want to change the setting for that single device. In that case you can match only a single device, and replace the node name with EG:alsa_output.usb-ASUSTeK_Xonar_SoundCard-00.iec958-stereo, and it will target that specific device.# Disable suspend of Toslink output to prevent audio popping. services.pipewire.wireplumber.extraConfig."99-disable-suspend" = { "monitor.alsa.rules" = [ { matches = [ { "node.name" = "~alsa_input.*"; } { "node.name" = "~alsa_output.*"; } ]; actions = { update-props = { "session.suspend-timeout-seconds" = 0; }; }; } ]; }; The value 0 disables suspension entirely. You could also set it to a higher value. 5 (seconds) is the default value. 
- Verify resolution:
After switching to the new configuration you need to also run systemctl --user restart wireplumberas your non-root user to apply the new wireplumber configuration, since wireplumber runs as your non-root user -nixos-rebuild switchis not sufficient. Play and pause sound again and verify that no sound pops occur anymore and observe inpw-topthat the numeric columns for that sink do not disappear after 5 seconds.
Eliminating audio startup delay after suspend or boot
In addition to audio popping, some HTPC systems may experience a delay when audio playback starts after resuming from suspend or system boot. This occurs even if suspension is disabled as described above.
- Identify your sink using the method in the previous section.
- Enhance configuration:
# Disable node suspend on NVIDIA graphics services.pipewire.wireplumber.extraConfig."99-disable-suspend"."monitor.alsa.rules" = [ { matches = [ { "node.name" = "alsa_output.pci-0000_01_00.1.hdmi-stereo-extra1"; } ]; actions.update-props = { "session.suspend-timeout-seconds" = 0; "node.always-process" = true; "dither.method" = "wannamaker3"; "dither.noise" = 1; }; } ]; This configuration prevents audio popping and keeps the audio pipeline active, effectively reducing startup delay from ~5s to ~1s. Adding the dither settings further eliminates the remaining delay. 
