1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
// Copyright 2023 The ChromiumOS Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
use anyhow::Context;
use argh::FromArgs;
use base::RawDescriptor;
use cros_async::Executor;
use crate::virtio::snd::parameters::Parameters;
use crate::virtio::vhost::user::device::snd::SndBackend;
use crate::virtio::vhost::user::device::BackendConnection;
#[derive(FromArgs)]
#[argh(subcommand, name = "snd")]
/// Snd device
pub struct Options {
#[argh(option, arg_name = "PATH", hidden_help)]
/// deprecated - please use --socket-path instead
socket: Option<String>,
#[argh(option, arg_name = "PATH")]
/// path to the vhost-user socket to bind to.
/// If this flag is set, --fd cannot be specified.
socket_path: Option<String>,
#[argh(option, arg_name = "FD")]
/// file descriptor of a connected vhost-user socket.
/// If this flag is set, --socket-path cannot be specified.
fd: Option<RawDescriptor>,
#[argh(
option,
arg_name = "CONFIG",
from_str_fn(snd_parameters_from_str),
default = "Default::default()",
long = "config"
)]
/// comma separated key=value pairs for setting up cras snd devices.
/// Possible key values:
/// capture - Enable audio capture. Default to false.
/// backend - Which backend to use for vhost-snd (null|cras).
/// client_type - Set specific client type for cras backend.
/// socket_type - Set socket type for cras backend.
/// num_output_devices - Set number of output PCM devices.
/// num_input_devices - Set number of input PCM devices.
/// num_output_streams - Set number of output PCM streams per device.
/// num_input_streams - Set number of input PCM streams per device.
/// Example: [capture=true,backend=BACKEND,
/// num_output_devices=1,num_input_devices=1,num_output_streams=1,num_input_streams=1]
params: Parameters,
}
fn snd_parameters_from_str(input: &str) -> Result<Parameters, String> {
serde_keyvalue::from_key_values(input).map_err(|e| e.to_string())
}
/// Starts a vhost-user snd device.
/// Returns an error if the given `args` is invalid or the device fails to run.
pub fn run_snd_device(opts: Options) -> anyhow::Result<()> {
let ex = Executor::new().context("Failed to create executor")?;
let snd_device = Box::new(SndBackend::new(&ex, opts.params, 0)?);
let conn =
BackendConnection::from_opts(opts.socket.as_deref(), opts.socket_path.as_deref(), opts.fd)?;
conn.run_device(ex, snd_device)
}