ALSA configuration
One of the components of WirePlumber is the ALSA monitor. This monitor is responsible for creating PipeWire devices and nodes for all the ALSA cards that are available on the system. It also manages the configuration of these devices.
The ALSA monitor is enabled by default and can be disabled using the
monitor.alsa
feature in the configuration file.
The monitor, as with all device monitors, is implemented as a SPA plugin and is part of PipeWire. WirePlumber merely loads the plugin and lets it do its work. The plugin then monitors UDev and creates device and node objects for all the ALSA cards that are available on the system.
Note
One thing worth remembering here is that in ALSA, a “card” represents a physical sound controller device, and a “device” is a logical access point that represents a set of inputs and/or outputs that are part of the card. In PipeWire, a “device” is the direct equivalent of an ALSA “card” and a “node” is almost equivalent (close, but not quite) of an ALSA “device”.
Properties
The ALSA monitor SPA plugin (api.alsa.enum.udev
) supports properties that
can be used to configure it when it is loaded. These properties can be set in
the monitor.alsa.properties
section of the WirePlumber configuration file.
Example:
monitor.alsa.properties = {
alsa.use-acp = true
}
- alsa.use-acp
A boolean that controls whether the ACP (alsa card profile) code is to be the default manager of the device. This will probe the device and configure the available profiles, ports and mixer settings. The code to do this is taken directly from PulseAudio and provides devices that look and feel exactly like the PulseAudio devices.
Rules
When device and node objects are created by the ALSA monitor, they can be configured using rules. These rules allow matching the existing properties of these objects and updating them with new values. This is the main way of configuring ALSA device settings.
These rules can be set in the monitor.alsa.rules
section of the WirePlumber
configuration file.
Example:
monitor.alsa.rules = [
{
matches = [
{
# This matches the value of the 'device.name' property of the device.
device.name = "~alsa_card.*"
}
]
actions = {
update-props = {
# Apply all the desired device settings here.
api.alsa.use-acp = true
}
}
}
{
matches = [
# This matches the value of the 'node.name' property of the node.
{
node.name = "~alsa_output.*"
}
]
actions = {
# Apply all the desired node specific settings here.
update-props = {
node.nick = "My Node"
priority.driver = 100
session.suspend-timeout-seconds = 5
}
}
}
]
Device properties
The following properties can be configured on devices created by the monitor:
- api.alsa.use-acp
Use the ACP (alsa card profile) code to manage this device. This will probe the device and configure the available profiles, ports and mixer settings. The code to do this is taken directly from PulseAudio and provides devices that look and feel exactly like the PulseAudio devices.
- Default value:
true
- Type:
boolean
- api.alsa.use-ucm
When ACP is enabled and a UCM configuration is available for a device, by default it is used instead of the ACP profiles. This option allows you to disable this and use the ACP profiles instead.
This option does nothing if
api.alsa.use-acp
is set tofalse
.- Default value:
true
- Type:
boolean
- api.alsa.soft-mixer
Setting this option to
true
will disable the hardware mixer for volume control and mute. All volume handling will then use software volume and mute, leaving the hardware mixer untouched. The hardware mixer will still be used to mute unused audio paths in the device.- Type:
boolean
- api.alsa.ignore-dB
Setting this option to
true
will ignore the decibel setting configured by the driver. Use this when the driver reports wrong settings.- Type:
boolean
- device.profile-set
This option can be used to select a custom ACP profile-set name for the device. This can be configured in UDev rules, but it can also be specified here. The default is to use “default.conf”.
- Type:
string
- device.profile
The initial active profile name. The default is to start from the “Off” profile and then let WirePlumber select the best profile based on its policy.
- Type:
string
- api.acp.auto-profile
Automatically select the best profile for the device. Normally this option is disabled because WirePlumber will manage the profile of the device. WirePlumber can save and load previously selected profiles. Enable this in custom configurations where the relevant WirePlumber components are disabled.
- Type:
boolean
- api.acp.auto-port
Automatically select the highest priority port that is available (“port” is a PulseAudio/ACP term, the equivalent of a “Route” in PipeWire). This is by default disabled because WirePlumber handles the task of selecting and restoring Routes. Enable this in custom configurations where the relevant WirePlumber components are disabled.
- Type:
boolean
- api.acp.probe-rate
Sets the samplerate used for probing the ALSA devices and collecting the profiles and ports.
- Type:
integer
- api.acp.pro-channels
Sets the number of channels to use when probing the “Pro Audio” profile. Normally, the maximum amount of channels will be used but with this setting this can be reduced, which can make it possible to use other samplerates on some devices.
- Type:
integer
Some of the other properties that can be configured on devices:
- device.nick
A short name for the device.
- device.description
A longer, user-friendly name of the device. This will show up in most user interfaces as the device’s name.
- device.disabled
Disables the device. PipeWire will remove it from the list of cards or devices.
- Type:
boolean
Node properties
The following properties can be configured on nodes created by the monitor:
- priority.driver
This configures the node driver priority. Nodes with higher priority will be used as a driver in the graph. Other nodes with lower priority will have to resample to the driver node when they are joined in the same graph. The default value is set based on some heuristics.
- Type:
integer
- priority.session
This configures the priority of the node when selecting a default node (default sink/source as a link target for streams). Higher priority nodes will be more likely candidates for becoming the default node.
- Type:
integer
Note
By default, sources have a
priority.session
value around 1600-2000 and sinks have a value around 600-1000. If you are increasing the priority of a sink, it is not advised to use a value higher than 1500, as it may cause a sink’s monitor to be selected as the default source.
- node.pause-on-idle
Pause the node when nothing is linked to it anymore. This is by default false because some devices make a “pop” sound when they are opened/closed. The node will normally pause and suspend after a timeout (see below).
- Type:
boolean
- session.suspend-timeout-seconds
This option configures a different suspend timeout on the node. By default this is
5
seconds. For some devices (HiFi amplifiers, for example) it might make sense to set a higher timeout because they might require some time to restart after being idle.A value of
0
disables suspend for a node and will leave the ALSA device busy. The device can then be manually suspended withpactl suspend-sink|source
.- Type:
integer
- audio.format
The sample format of the device. By default, PipeWire will use a 32 bits sample format but a different format can be set here.
- Type:
string (
"S16LE"
,"S32LE"
,"F32LE"
, …)
- audio.rate
The sample rate of the device. By default, the ALSA device will be configured with the same samplerate as the global graph. If this is not supported, or a custom value is set here, resampling will be used to match the graph rate.
- Type:
integer
- audio.channels
The number of channels of the device. By default the channels and their position are determined by the selected device profile. You can override this setting here.
- Type:
integer
- audio.position
The position of the channels. By default the number of channels and their position are determined by the selected device profile. You can override this setting here and optionally swap or reconfigure the channel positions.
- Type:
array of strings (example:
["FL", "FR", "LFE", "FC", "RL", "RR"]
)
- api.alsa.use-chmap
Use the channel map as reported by the driver. This is disabled by default because it is often wrong and the ACP code handles this better.
- Type:
boolean
- api.alsa.disable-mmap
Disable the use of mmap for the ALSA device. By default, PipeWire will access the memory of the device using mmap. This can be disabled and force the usage of the slower read and write access modes, in case the mmap support of the device is not working properly.
- Type:
boolean
- channelmix.normalize
Normalize the channel volumes when mixing & resampling, making sure that the original 0 dB level is preserved so that nothing sounds wildly quieter/louder. This is disabled by default.
- Type:
boolean
- channelmix.mix-lfe
Creates a “center” channel for X.0 recordings from the front stereo on X.1 setups and pushes some low-frequency/bass from the “center” of X.1 recordings into the front stereo on X.0 setups. This is disabled by default.
- Type:
boolean
- monitor.channel-volumes
By default, the volume of the sink/source does not influence the volume on the monitor ports. Set this option to true to change this. PulseAudio has inconsistent behaviour regarding this option, it applies channel-volumes only when the sink/source is using software volumes.
- Type:
boolean
- node.disabled
Disables the node. Pipewire will remove it from the list of the nodes.
- Type:
boolean
ALSA buffer properties
PipeWire by default uses a timer to consume and produce samples to/from ALSA devices. After every timeout, it queries the hardware pointers of the device and uses this information to set a new timeout. This works well for most devices, but there is a class of devices, so called “batch” devices, that need extra buffering and timing tweaks to work properly. This is because batch devices only get their hardware pointers updated after each hardware interrupt. When the hardware interrupt frequency and the timer frequency are aligned, it is possible for the hardware pointers to be updated just after the timer has expired, resulting in sometimes wrong timing information being returned by the query. In contrast, non-batch devices get pointer updates independent of the interrupt.
This means that for batch devices we need to set the interrupt at a sufficiently high frequency, at the cost of CPU usage, while for non-batch devices we want to set the interrupt frequency as low as possible to save CPU. For batch devices we also need to take the extra buffering into account caused by the delayed updates of the hardware pointers.
Note
Most USB devices are batch devices and will be handled as such by PipeWire by default.
There are 2 tunable parameters to control the buffering and timeouts in a device:
- api.alsa.period-size
This sets the device interrupt to every period-size samples for non-batch devices and to half of this for batch devices. For batch devices, the other half of the period-size is used as extra buffering to compensate for the delayed update. So, for batch devices, there is an additional period-size/2 delay. It makes sense to lower the period-size for batch devices to reduce this delay.
- Type:
integer (samples)
- api.alsa.headroom
This adds extra delay between the hardware pointers and software pointers. In most cases this can be set to 0. For very bad devices or emulated devices (like in a VM) it might be necessary to increase the headroom value.
- Type:
integer (samples)
- api.alsa.period-num
This configures the number of periods in the hardware buffer, which controls its size. Note that this is multiplied by the period of the device to determine the size, so for batch devices, the total buffer size is effectively period-num * period-size/2.
- Type:
integer
In summary, this is the overview of buffering and timings:
Property |
Batch |
Non-Batch |
---|---|---|
IRQ Frequency |
api.alsa.period-size/2 |
api.alsa.period-size |
Extra Delay |
api.alsa.headroom + api.alsa.period-size/2 |
api.alsa.headroom |
Buffer Size |
api.alsa.period-num * api.alsa.period-size/2 |
api.alsa.period-num * api.alsa.period-size |
Finally, it is possible to disable the batch device tweaks with:
- api.alsa.disable-batch
This disables the batch device tweaks. It removes the extra delay added of period-size/2 if the device can support this. For batch devices it is also a good idea to lower the period-size (and increase the IRQ frequency) to get smaller batch updates and lower latency.
- Type:
boolean
ALSA extra latency properties
Extra internal delay in the DAC and ADC converters of the device itself can be
set with the latency.internal.*
properties:
latency.internal.rate = 256
latency.internal.ns = 0
You can configure a latency in samples (relative to rate with
latency.internal.rate
) or in nanoseconds (latency.internal.ns
).
This value will be added to the total reported latency by the node of the device.
You can use a tool like jack_iodelay
to get the number of samples of
internal latency of your device.
This property is also adjustable at runtime with the ProcessLatency
param.
You will need to find the id of the Node you want to change. For example:
Query the current internal latency of an ALSA node with id 58:
$ pw-cli e 58 ProcessLatency
Object: size 80, type Spa:Pod:Object:Param:ProcessLatency (262156), id Spa:Enum:ParamId:ProcessLatency (16)
Prop: key Spa:Pod:Object:Param:ProcessLatency:quantum (1), flags 00000000
Float 0.000000
Prop: key Spa:Pod:Object:Param:ProcessLatency:rate (2), flags 00000000
Int 0
Prop: key Spa:Pod:Object:Param:ProcessLatency:ns (3), flags 00000000
Long 0
Set the internal latency to 256 samples:
$ pw-cli s 58 ProcessLatency '{ rate = 256 }'
Object: size 32, type Spa:Pod:Object:Param:ProcessLatency (262156), id Spa:Enum:ParamId:ProcessLatency (16)
Prop: key Spa:Pod:Object:Param:ProcessLatency:rate (2), flags 00000000
Int 256
remote 0 node 58 changed
remote 0 port 70 changed
remote 0 port 72 changed
remote 0 port 74 changed
remote 0 port 76 changed
Startup tweaks
- api.alsa.start-delay
Some devices need some time before they can report accurate hardware pointer positions. In those cases, an extra start delay can be added to compensate for this startup delay. This sets the startup delay in samples. The default is 0.
- Type:
integer (samples)
IEC958 (S/PDIF) passthrough
- iec958.codecs
S/PDIF passthrough will only be enabled when the accepted codecs are configured on the ALSA device. This can be done by setting the list of supported codecs on this property.
Note that it is possible to also configure this property at runtime, either with tools like pavucontrol or with the
pw-cli
tool, like this:pw-cli s <node-id> Props '{ iec958Codecs : [ PCM ] }'
- Type:
array of strings (example:
[ "PCM", "DTS", "AC3", "EAC3", "TrueHD", "DTS-HD" ]
)