move apm (webrtc audio processor) to audio crate
This commit is contained in:
parent
5a4f284e45
commit
cef721465f
7 changed files with 205 additions and 121 deletions
40
Cargo.lock
generated
40
Cargo.lock
generated
|
@ -1381,6 +1381,8 @@ dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"collections",
|
"collections",
|
||||||
"gpui",
|
"gpui",
|
||||||
|
"libwebrtc",
|
||||||
|
"parking_lot",
|
||||||
"rodio",
|
"rodio",
|
||||||
"schemars",
|
"schemars",
|
||||||
"serde",
|
"serde",
|
||||||
|
@ -2681,7 +2683,7 @@ dependencies = [
|
||||||
"cap-primitives",
|
"cap-primitives",
|
||||||
"cap-std",
|
"cap-std",
|
||||||
"io-lifetimes",
|
"io-lifetimes",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -2710,7 +2712,7 @@ dependencies = [
|
||||||
"maybe-owned",
|
"maybe-owned",
|
||||||
"rustix 1.0.7",
|
"rustix 1.0.7",
|
||||||
"rustix-linux-procfs",
|
"rustix-linux-procfs",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
"winx",
|
"winx",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -5327,7 +5329,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "976dd42dc7e85965fe702eb8164f21f450704bdde31faefd6471dba214cb594e"
|
checksum = "976dd42dc7e85965fe702eb8164f21f450704bdde31faefd6471dba214cb594e"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"libc",
|
"libc",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -5714,7 +5716,7 @@ checksum = "0ce92ff622d6dadf7349484f42c93271a0d49b7cc4d466a936405bacbe10aa78"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
"rustix 1.0.7",
|
"rustix 1.0.7",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -6092,7 +6094,7 @@ checksum = "94e7099f6313ecacbe1256e8ff9d617b75d1bcb16a6fddef94866d225a01a14a"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"io-lifetimes",
|
"io-lifetimes",
|
||||||
"rustix 1.0.7",
|
"rustix 1.0.7",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -8551,7 +8553,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "2285ddfe3054097ef4b2fe909ef8c3bcd1ea52a8f0d274416caebeef39f04a65"
|
checksum = "2285ddfe3054097ef4b2fe909ef8c3bcd1ea52a8f0d274416caebeef39f04a65"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"io-lifetimes",
|
"io-lifetimes",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -8624,7 +8626,7 @@ checksum = "e04d7f318608d35d4b61ddd75cbdaee86b023ebe2bd5a66ee0915f0bf93095a9"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"hermit-abi 0.5.0",
|
"hermit-abi 0.5.0",
|
||||||
"libc",
|
"libc",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -8706,7 +8708,7 @@ dependencies = [
|
||||||
"portable-atomic",
|
"portable-atomic",
|
||||||
"portable-atomic-util",
|
"portable-atomic-util",
|
||||||
"serde",
|
"serde",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -9406,7 +9408,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "fc2f4eb4bc735547cfed7c0a4922cbd04a4655978c09b54f1f7b228750664c34"
|
checksum = "fc2f4eb4bc735547cfed7c0a4922cbd04a4655978c09b54f1f7b228750664c34"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
"windows-targets 0.52.6",
|
"windows-targets 0.48.5",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -13062,7 +13064,7 @@ dependencies = [
|
||||||
"once_cell",
|
"once_cell",
|
||||||
"socket2",
|
"socket2",
|
||||||
"tracing",
|
"tracing",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -13874,7 +13876,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "rodio"
|
name = "rodio"
|
||||||
version = "0.21.1"
|
version = "0.21.1"
|
||||||
source = "git+https://github.com/RustAudio/rodio?branch=microphone#bb560f30b17d330459b81afc918f2a4a123c41aa"
|
source = "git+https://github.com/RustAudio/rodio?branch=microphone#cad73716a363a5ba92fcb73ec37a4b98a7d7de5f"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cpal",
|
"cpal",
|
||||||
"dasp_sample",
|
"dasp_sample",
|
||||||
|
@ -14104,7 +14106,7 @@ dependencies = [
|
||||||
"itoa",
|
"itoa",
|
||||||
"libc",
|
"libc",
|
||||||
"linux-raw-sys 0.4.15",
|
"linux-raw-sys 0.4.15",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -14117,7 +14119,7 @@ dependencies = [
|
||||||
"errno 0.3.11",
|
"errno 0.3.11",
|
||||||
"libc",
|
"libc",
|
||||||
"linux-raw-sys 0.9.4",
|
"linux-raw-sys 0.9.4",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -14239,7 +14241,7 @@ dependencies = [
|
||||||
"security-framework 3.2.0",
|
"security-framework 3.2.0",
|
||||||
"security-framework-sys",
|
"security-framework-sys",
|
||||||
"webpki-root-certs",
|
"webpki-root-certs",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -15591,7 +15593,7 @@ dependencies = [
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
"libc",
|
"libc",
|
||||||
"psm",
|
"psm",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -16249,7 +16251,7 @@ dependencies = [
|
||||||
"fd-lock",
|
"fd-lock",
|
||||||
"io-lifetimes",
|
"io-lifetimes",
|
||||||
"rustix 0.38.44",
|
"rustix 0.38.44",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
"winx",
|
"winx",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -16431,7 +16433,7 @@ dependencies = [
|
||||||
"getrandom 0.3.2",
|
"getrandom 0.3.2",
|
||||||
"once_cell",
|
"once_cell",
|
||||||
"rustix 1.0.7",
|
"rustix 1.0.7",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -18993,7 +18995,7 @@ version = "0.1.9"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb"
|
checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.48.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -19652,7 +19654,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "3f3fd376f71958b862e7afb20cfe5a22830e1963462f3a17f49d82a6c1d1f42d"
|
checksum = "3f3fd376f71958b862e7afb20cfe5a22830e1963462f3a17f49d82a6c1d1f42d"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"bitflags 2.9.0",
|
"bitflags 2.9.0",
|
||||||
"windows-sys 0.59.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
|
|
@ -19,6 +19,10 @@ gpui.workspace = true
|
||||||
settings.workspace = true
|
settings.workspace = true
|
||||||
schemars.workspace = true
|
schemars.workspace = true
|
||||||
serde.workspace = true
|
serde.workspace = true
|
||||||
|
parking_lot.workspace = true
|
||||||
rodio = { workspace = true, features = [ "wav", "playback", "tracing" ] }
|
rodio = { workspace = true, features = [ "wav", "playback", "tracing" ] }
|
||||||
util.workspace = true
|
util.workspace = true
|
||||||
workspace-hack.workspace = true
|
workspace-hack.workspace = true
|
||||||
|
|
||||||
|
[target.'cfg(not(any(all(target_os = "windows", target_env = "gnu"), target_os = "freebsd")))'.dependencies]
|
||||||
|
libwebrtc = { rev = "5f04705ac3f356350ae31534ffbc476abc9ea83d", git = "https://github.com/zed-industries/livekit-rust-sdks" }
|
||||||
|
|
|
@ -1,12 +1,15 @@
|
||||||
use anyhow::{Context as _, Result, anyhow};
|
use anyhow::{Context as _, Result, anyhow};
|
||||||
use collections::HashMap;
|
use collections::HashMap;
|
||||||
use gpui::{App, BorrowAppContext, Global};
|
use gpui::{App, BorrowAppContext, Global};
|
||||||
use rodio::{Decoder, OutputStream, OutputStreamBuilder, Source, source::Buffered};
|
use libwebrtc::native::apm;
|
||||||
|
use parking_lot::Mutex;
|
||||||
|
use rodio::{Decoder, OutputStream, OutputStreamBuilder, Source, mixer::Mixer, source::Buffered};
|
||||||
use settings::Settings;
|
use settings::Settings;
|
||||||
use std::io::Cursor;
|
use std::{io::Cursor, sync::Arc};
|
||||||
use util::ResultExt;
|
use util::ResultExt;
|
||||||
|
|
||||||
mod audio_settings;
|
mod audio_settings;
|
||||||
|
mod rodio_ext;
|
||||||
pub use audio_settings::AudioSettings;
|
pub use audio_settings::AudioSettings;
|
||||||
|
|
||||||
pub fn init(cx: &mut App) {
|
pub fn init(cx: &mut App) {
|
||||||
|
@ -38,18 +41,46 @@ impl Sound {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Default)]
|
|
||||||
pub struct Audio {
|
pub struct Audio {
|
||||||
output_handle: Option<OutputStream>,
|
output_handle: Option<OutputStream>,
|
||||||
|
output_mixer: Option<Mixer>,
|
||||||
|
echo_canceller: Arc<Mutex<apm::AudioProcessingModule>>,
|
||||||
source_cache: HashMap<Sound, Buffered<Decoder<Cursor<Vec<u8>>>>>,
|
source_cache: HashMap<Sound, Buffered<Decoder<Cursor<Vec<u8>>>>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl Default for Audio {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self {
|
||||||
|
output_handle: Default::default(),
|
||||||
|
output_mixer: Default::default(),
|
||||||
|
echo_canceller: Arc::new(Mutex::new(apm::AudioProcessingModule::new(
|
||||||
|
true, false, false, false,
|
||||||
|
))),
|
||||||
|
source_cache: Default::default(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Global for Audio {}
|
impl Global for Audio {}
|
||||||
|
|
||||||
impl Audio {
|
impl Audio {
|
||||||
fn ensure_output_exists(&mut self) -> Option<&OutputStream> {
|
fn ensure_output_exists(&mut self) -> Option<&OutputStream> {
|
||||||
if self.output_handle.is_none() {
|
if self.output_handle.is_none() {
|
||||||
self.output_handle = OutputStreamBuilder::open_default_stream().log_err();
|
self.output_handle = OutputStreamBuilder::open_default_stream().log_err();
|
||||||
|
if let Some(output_handle) = self.output_handle {
|
||||||
|
let config = output_handle.config();
|
||||||
|
let (mixer, source) =
|
||||||
|
rodio::mixer::mixer(config.channel_count(), config.sample_rate());
|
||||||
|
self.output_mixer = Some(mixer);
|
||||||
|
|
||||||
|
let echo_canceller = Arc::clone(&self.echo_canceller);
|
||||||
|
let source = source.inspect_buffered(
|
||||||
|
|buffer| echo_canceller.lock().process_reverse_stream(&mut buf),
|
||||||
|
config.sample_rate().get() as i32,
|
||||||
|
config.channel_count().get().into(),
|
||||||
|
);
|
||||||
|
output_handle.mixer().add(source);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
self.output_handle.as_ref()
|
self.output_handle.as_ref()
|
||||||
|
@ -72,6 +103,7 @@ impl Audio {
|
||||||
cx.update_default_global(|this: &mut Self, cx| {
|
cx.update_default_global(|this: &mut Self, cx| {
|
||||||
let source = this.sound_source(sound, cx).log_err()?;
|
let source = this.sound_source(sound, cx).log_err()?;
|
||||||
let output_handle = this.ensure_output_exists()?;
|
let output_handle = this.ensure_output_exists()?;
|
||||||
|
|
||||||
output_handle.mixer().add(source);
|
output_handle.mixer().add(source);
|
||||||
Some(())
|
Some(())
|
||||||
});
|
});
|
||||||
|
|
93
crates/audio/src/rodio_ext.rs
Normal file
93
crates/audio/src/rodio_ext.rs
Normal file
|
@ -0,0 +1,93 @@
|
||||||
|
use rodio::Source;
|
||||||
|
|
||||||
|
pub trait RodioExt: Source + Sized {
|
||||||
|
fn process_buffer<const N: usize, F>(self, callback: F) -> ProcessBuffer<N, Self, F>
|
||||||
|
where
|
||||||
|
F: FnMut(&mut [rodio::Sample; N]);
|
||||||
|
fn inspect_buffer<const N: usize, F>(self, callback: F) -> ProcessBuffer<N, Self, F>
|
||||||
|
where
|
||||||
|
F: FnMut(&[rodio::Sample; N]);
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: Source> RodioExt for S {
|
||||||
|
fn process_buffer<const N: usize, F>(self, callback: F) -> ProcessBuffer<N, Self, F>
|
||||||
|
where
|
||||||
|
F: FnMut(&mut [rodio::Sample; N]),
|
||||||
|
{
|
||||||
|
ProcessBuffer {
|
||||||
|
inner: self,
|
||||||
|
callback,
|
||||||
|
buffer: [0.0; N],
|
||||||
|
next: N,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn inspect_buffer<const N: usize, F>(self, callback: F) -> ProcessBuffer<N, Self, F>
|
||||||
|
where
|
||||||
|
F: FnMut(&[rodio::Sample; N]),
|
||||||
|
{
|
||||||
|
InspectBuffer {
|
||||||
|
inner: self,
|
||||||
|
callback,
|
||||||
|
buffer: [0.0; N],
|
||||||
|
next: N,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ProcessBuffer<const N: usize, S, F>
|
||||||
|
where
|
||||||
|
S: Source + Sized,
|
||||||
|
F: FnMut(&mut [rodio::Sample; N]),
|
||||||
|
{
|
||||||
|
inner: S,
|
||||||
|
callback: F,
|
||||||
|
buffer: [rodio::Sample; N],
|
||||||
|
next: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<const N: usize, S, F> Iterator for ProcessBuffer<N, S, F>
|
||||||
|
where
|
||||||
|
S: Source + Sized,
|
||||||
|
F: FnMut(&mut [rodio::Sample; N]),
|
||||||
|
{
|
||||||
|
type Item = rodio::Sample;
|
||||||
|
|
||||||
|
fn next(&mut self) -> Option<Self::Item> {
|
||||||
|
self.next += 1;
|
||||||
|
if self.next < self.buffer.len() {
|
||||||
|
let sample = self.buffer[self.next];
|
||||||
|
return Some(sample);
|
||||||
|
}
|
||||||
|
|
||||||
|
for sample in &mut self.buffer {
|
||||||
|
*sample = self.inner.next()?
|
||||||
|
}
|
||||||
|
(self.callback)(&mut self.buffer);
|
||||||
|
|
||||||
|
self.next = 0;
|
||||||
|
Some(self.buffer[0])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<const N: usize, S, F> Source for ProcessBuffer<N, S, F>
|
||||||
|
where
|
||||||
|
S: Source + Sized,
|
||||||
|
F: FnMut(&mut [rodio::Sample; N]),
|
||||||
|
{
|
||||||
|
fn current_span_len(&self) -> Option<usize> {
|
||||||
|
// TODO dvdsk this should be a spanless Source
|
||||||
|
None
|
||||||
|
}
|
||||||
|
|
||||||
|
fn channels(&self) -> rodio::ChannelCount {
|
||||||
|
self.inner.channels()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn sample_rate(&self) -> rodio::SampleRate {
|
||||||
|
self.inner.sample_rate()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn total_duration(&self) -> Option<std::time::Duration> {
|
||||||
|
self.inner.total_duration()
|
||||||
|
}
|
||||||
|
}
|
|
@ -99,7 +99,7 @@ impl Room {
|
||||||
&self,
|
&self,
|
||||||
cx: &mut AsyncApp,
|
cx: &mut AsyncApp,
|
||||||
) -> Result<(LocalTrackPublication, playback::AudioStream)> {
|
) -> Result<(LocalTrackPublication, playback::AudioStream)> {
|
||||||
let (track, stream) = self.playback.capture_local_microphone_track()?;
|
let (track, stream) = self.playback.capture_local_microphone_track(&cx)?;
|
||||||
let publication = self
|
let publication = self
|
||||||
.local_participant()
|
.local_participant()
|
||||||
.publish_track(
|
.publish_track(
|
||||||
|
|
|
@ -1,12 +1,14 @@
|
||||||
use anyhow::{Context as _, Result};
|
use anyhow::{Context as _, Result};
|
||||||
|
|
||||||
|
use audio::AudioSettings;
|
||||||
use cpal::Sample;
|
use cpal::Sample;
|
||||||
use cpal::traits::{DeviceTrait, StreamTrait as _};
|
use cpal::traits::{DeviceTrait, StreamTrait as _};
|
||||||
use dasp_sample::ToSample;
|
use dasp_sample::ToSample;
|
||||||
use futures::channel::mpsc::UnboundedSender;
|
use futures::channel::mpsc::UnboundedSender;
|
||||||
use futures::{Stream, StreamExt as _};
|
use futures::{Stream, StreamExt as _};
|
||||||
use gpui::{
|
use gpui::{
|
||||||
BackgroundExecutor, ScreenCaptureFrame, ScreenCaptureSource, ScreenCaptureStream, Task,
|
AsyncApp, BackgroundExecutor, ScreenCaptureFrame, ScreenCaptureSource, ScreenCaptureStream,
|
||||||
|
Task,
|
||||||
};
|
};
|
||||||
use libwebrtc::native::{apm, audio_mixer, audio_resampler};
|
use libwebrtc::native::{apm, audio_mixer, audio_resampler};
|
||||||
use livekit::track;
|
use livekit::track;
|
||||||
|
@ -19,9 +21,11 @@ use livekit::webrtc::{
|
||||||
video_source::{RtcVideoSource, VideoResolution, native::NativeVideoSource},
|
video_source::{RtcVideoSource, VideoResolution, native::NativeVideoSource},
|
||||||
video_stream::native::NativeVideoStream,
|
video_stream::native::NativeVideoStream,
|
||||||
};
|
};
|
||||||
|
use log::info;
|
||||||
use parking_lot::Mutex;
|
use parking_lot::Mutex;
|
||||||
use rodio::Source;
|
use rodio::Source;
|
||||||
use rodio::source::{LimitSettings, UniformSourceIterator};
|
use rodio::source::{LimitSettings, UniformSourceIterator};
|
||||||
|
use settings::Settings;
|
||||||
use std::cell::RefCell;
|
use std::cell::RefCell;
|
||||||
use std::num::NonZero;
|
use std::num::NonZero;
|
||||||
use std::sync::Weak;
|
use std::sync::Weak;
|
||||||
|
@ -45,8 +49,8 @@ pub(crate) struct AudioStack {
|
||||||
// 16kHz, 32kHz and 48kHz. As 48 is the most common "next step up"
|
// 16kHz, 32kHz and 48kHz. As 48 is the most common "next step up"
|
||||||
// for audio output devices like speakers/bluetooth, we just hard-code
|
// for audio output devices like speakers/bluetooth, we just hard-code
|
||||||
// this; and downsample when we need to.
|
// this; and downsample when we need to.
|
||||||
const SAMPLE_RATE: u32 = 48000;
|
const SAMPLE_RATE: NonZero<u32> = NonZero::new(48000).expect("not zero");
|
||||||
const NUM_CHANNELS: u32 = 2;
|
const NUM_CHANNELS: NonZero<u16> = NonZero::new(2).expect("not zero");
|
||||||
|
|
||||||
pub(crate) fn play_remote_audio_track(
|
pub(crate) fn play_remote_audio_track(
|
||||||
track: &livekit::track::RemoteAudioTrack,
|
track: &livekit::track::RemoteAudioTrack,
|
||||||
|
@ -55,6 +59,7 @@ pub(crate) fn play_remote_audio_track(
|
||||||
let stop_handle = Arc::new(AtomicBool::new(false));
|
let stop_handle = Arc::new(AtomicBool::new(false));
|
||||||
let stop_handle_clone = stop_handle.clone();
|
let stop_handle_clone = stop_handle.clone();
|
||||||
let stream = source::LiveKitStream::new(cx.background_executor(), track)
|
let stream = source::LiveKitStream::new(cx.background_executor(), track)
|
||||||
|
.process_buffer(|| apm)
|
||||||
.stoppable()
|
.stoppable()
|
||||||
.periodic_access(Duration::from_millis(50), move |s| {
|
.periodic_access(Duration::from_millis(50), move |s| {
|
||||||
if stop_handle.load(Ordering::Relaxed) {
|
if stop_handle.load(Ordering::Relaxed) {
|
||||||
|
@ -95,8 +100,8 @@ impl AudioStack {
|
||||||
let next_ssrc = self.next_ssrc.fetch_add(1, Ordering::Relaxed);
|
let next_ssrc = self.next_ssrc.fetch_add(1, Ordering::Relaxed);
|
||||||
let source = AudioMixerSource {
|
let source = AudioMixerSource {
|
||||||
ssrc: next_ssrc,
|
ssrc: next_ssrc,
|
||||||
sample_rate: SAMPLE_RATE,
|
sample_rate: SAMPLE_RATE.get(),
|
||||||
num_channels: NUM_CHANNELS,
|
num_channels: NUM_CHANNELS.get() as u32,
|
||||||
buffer: Arc::default(),
|
buffer: Arc::default(),
|
||||||
};
|
};
|
||||||
self.mixer.lock().add_source(source.clone());
|
self.mixer.lock().add_source(source.clone());
|
||||||
|
@ -136,7 +141,7 @@ impl AudioStack {
|
||||||
let apm = self.apm.clone();
|
let apm = self.apm.clone();
|
||||||
let mixer = self.mixer.clone();
|
let mixer = self.mixer.clone();
|
||||||
async move {
|
async move {
|
||||||
Self::play_output(apm, mixer, SAMPLE_RATE, NUM_CHANNELS)
|
Self::play_output(apm, mixer, SAMPLE_RATE.get(), NUM_CHANNELS.get().into())
|
||||||
.await
|
.await
|
||||||
.log_err();
|
.log_err();
|
||||||
}
|
}
|
||||||
|
@ -147,12 +152,13 @@ impl AudioStack {
|
||||||
|
|
||||||
pub(crate) fn capture_local_microphone_track(
|
pub(crate) fn capture_local_microphone_track(
|
||||||
&self,
|
&self,
|
||||||
|
cx: &AsyncApp,
|
||||||
) -> Result<(crate::LocalAudioTrack, AudioStream)> {
|
) -> Result<(crate::LocalAudioTrack, AudioStream)> {
|
||||||
let source = NativeAudioSource::new(
|
let source = NativeAudioSource::new(
|
||||||
// n.b. this struct's options are always ignored, noise cancellation is provided by apm.
|
// n.b. this struct's options are always ignored, noise cancellation is provided by apm.
|
||||||
AudioSourceOptions::default(),
|
AudioSourceOptions::default(),
|
||||||
SAMPLE_RATE,
|
SAMPLE_RATE.get(),
|
||||||
NUM_CHANNELS,
|
NUM_CHANNELS.get().into(),
|
||||||
10,
|
10,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -171,8 +177,16 @@ impl AudioStack {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
let rodio_pipeline =
|
||||||
|
AudioSettings::try_read_global(cx, |setting| setting.rodio_audio).unwrap_or(false);
|
||||||
let capture_task = self.executor.spawn(async move {
|
let capture_task = self.executor.spawn(async move {
|
||||||
Self::capture_input(apm, frame_tx, SAMPLE_RATE, NUM_CHANNELS).await
|
if rodio_pipeline {
|
||||||
|
info!("Using experimental.rodio_audio audio pipeline");
|
||||||
|
Self::capture_input_rodio(apm, frame_tx).await
|
||||||
|
} else {
|
||||||
|
Self::capture_input(apm, frame_tx, SAMPLE_RATE.get(), NUM_CHANNELS.get().into())
|
||||||
|
.await
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
let on_drop = util::defer(|| {
|
let on_drop = util::defer(|| {
|
||||||
|
@ -262,12 +276,11 @@ impl AudioStack {
|
||||||
async fn capture_input_rodio(
|
async fn capture_input_rodio(
|
||||||
apm: Arc<Mutex<apm::AudioProcessingModule>>,
|
apm: Arc<Mutex<apm::AudioProcessingModule>>,
|
||||||
frame_tx: UnboundedSender<AudioFrame<'static>>,
|
frame_tx: UnboundedSender<AudioFrame<'static>>,
|
||||||
sample_rate: u32,
|
|
||||||
num_channels: u32,
|
|
||||||
) -> Result<()> {
|
) -> Result<()> {
|
||||||
use crate::livekit_client::playback::source::RodioExt;
|
use crate::livekit_client::playback::source::RodioExt;
|
||||||
const NUM_CHANNELS: usize = 1;
|
const NUM_CHANNELS: usize = 1;
|
||||||
const LIVEKIT_BUFFER_SIZE: usize = (SAMPLE_RATE as usize / 100) * NUM_CHANNELS as usize;
|
const LIVEKIT_BUFFER_SIZE: usize =
|
||||||
|
(SAMPLE_RATE.get() as usize / 100) * NUM_CHANNELS as usize;
|
||||||
|
|
||||||
let (stream_error_tx, stream_error_rx) = channel();
|
let (stream_error_tx, stream_error_rx) = channel();
|
||||||
|
|
||||||
|
@ -275,18 +288,31 @@ impl AudioStack {
|
||||||
let stream = rodio::microphone::MicrophoneBuilder::new()
|
let stream = rodio::microphone::MicrophoneBuilder::new()
|
||||||
.default_device()?
|
.default_device()?
|
||||||
.default_config()?
|
.default_config()?
|
||||||
|
.prefer_sample_rates([
|
||||||
|
SAMPLE_RATE,
|
||||||
|
SAMPLE_RATE.saturating_mul(NonZero::new(2).expect("not zero")),
|
||||||
|
])
|
||||||
|
.prefer_channel_counts([
|
||||||
|
NonZero::new(1).expect("not zero"),
|
||||||
|
NonZero::new(2).expect("not zero"),
|
||||||
|
])
|
||||||
|
.prefer_buffer_sizes(512..)
|
||||||
.open_stream()?;
|
.open_stream()?;
|
||||||
let mut stream = UniformSourceIterator::new(
|
let mut stream = UniformSourceIterator::new(
|
||||||
stream,
|
stream,
|
||||||
NonZero::new(1).expect("1 is not zero"),
|
NonZero::new(1).expect("1 is not zero"),
|
||||||
NonZero::new(SAMPLE_RATE).expect("constant is not zero"),
|
SAMPLE_RATE,
|
||||||
)
|
)
|
||||||
.limit(LimitSettings::live_performance())
|
.limit(LimitSettings::live_performance())
|
||||||
.process_buffer::<LIVEKIT_BUFFER_SIZE, _>(|buffer| {
|
.process_buffer::<LIVEKIT_BUFFER_SIZE, _>(|buffer| {
|
||||||
let mut int_buffer: [i16; _] = buffer.map(|s| s.to_sample());
|
let mut int_buffer: [i16; _] = buffer.map(|s| s.to_sample());
|
||||||
if let Err(e) = apm
|
if let Err(e) = apm
|
||||||
.lock()
|
.lock()
|
||||||
.process_stream(&mut int_buffer, SAMPLE_RATE as i32, NUM_CHANNELS as i32)
|
.process_stream(
|
||||||
|
&mut int_buffer,
|
||||||
|
SAMPLE_RATE.get() as i32,
|
||||||
|
NUM_CHANNELS as i32,
|
||||||
|
)
|
||||||
.context("livekit audio processor error")
|
.context("livekit audio processor error")
|
||||||
{
|
{
|
||||||
let _ = stream_error_tx.send(e);
|
let _ = stream_error_tx.send(e);
|
||||||
|
@ -299,7 +325,7 @@ impl AudioStack {
|
||||||
.automatic_gain_control(1.0, 4.0, 0.0, 5.0);
|
.automatic_gain_control(1.0, 4.0, 0.0, 5.0);
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
let sampled = stream
|
let sampled: Vec<_> = stream
|
||||||
.by_ref()
|
.by_ref()
|
||||||
.take(LIVEKIT_BUFFER_SIZE)
|
.take(LIVEKIT_BUFFER_SIZE)
|
||||||
.map(|s| s.to_sample())
|
.map(|s| s.to_sample())
|
||||||
|
@ -315,10 +341,10 @@ impl AudioStack {
|
||||||
|
|
||||||
frame_tx
|
frame_tx
|
||||||
.unbounded_send(AudioFrame {
|
.unbounded_send(AudioFrame {
|
||||||
|
sample_rate: SAMPLE_RATE.get(),
|
||||||
|
num_channels: NUM_CHANNELS as u32,
|
||||||
|
samples_per_channel: sampled.len() as u32 / NUM_CHANNELS as u32,
|
||||||
data: Cow::Owned(sampled),
|
data: Cow::Owned(sampled),
|
||||||
sample_rate,
|
|
||||||
num_channels,
|
|
||||||
samples_per_channel: sample_rate / 100,
|
|
||||||
})
|
})
|
||||||
.context("Failed to send audio frame")?
|
.context("Failed to send audio frame")?
|
||||||
}
|
}
|
||||||
|
@ -419,6 +445,8 @@ impl AudioStack {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
use crate::livekit_client::playback::source::RodioExt;
|
||||||
|
|
||||||
use super::LocalVideoTrack;
|
use super::LocalVideoTrack;
|
||||||
|
|
||||||
pub enum AudioStream {
|
pub enum AudioStream {
|
||||||
|
|
|
@ -26,8 +26,11 @@ pub struct LiveKitStream {
|
||||||
|
|
||||||
impl LiveKitStream {
|
impl LiveKitStream {
|
||||||
pub fn new(executor: &gpui::BackgroundExecutor, track: &RemoteAudioTrack) -> Self {
|
pub fn new(executor: &gpui::BackgroundExecutor, track: &RemoteAudioTrack) -> Self {
|
||||||
let mut stream =
|
let mut stream = NativeAudioStream::new(
|
||||||
NativeAudioStream::new(track.rtc_track(), SAMPLE_RATE as i32, NUM_CHANNELS as i32);
|
track.rtc_track(),
|
||||||
|
SAMPLE_RATE.get() as i32,
|
||||||
|
NUM_CHANNELS.get().into(),
|
||||||
|
);
|
||||||
let (queue_input, queue_output) = rodio::queue::queue(true);
|
let (queue_input, queue_output) = rodio::queue::queue(true);
|
||||||
// spawn rtc stream
|
// spawn rtc stream
|
||||||
let receiver_task = executor.spawn({
|
let receiver_task = executor.spawn({
|
||||||
|
@ -71,81 +74,3 @@ impl Source for LiveKitStream {
|
||||||
self.inner.total_duration()
|
self.inner.total_duration()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait RodioExt: Source + Sized {
|
|
||||||
fn process_buffer<const N: usize, F>(self, callback: F) -> ProcessBuffer<N, Self, F>
|
|
||||||
where
|
|
||||||
F: FnMut(&mut [rodio::Sample; N]);
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: Source> RodioExt for S {
|
|
||||||
fn process_buffer<const N: usize, F>(self, callback: F) -> ProcessBuffer<N, Self, F>
|
|
||||||
where
|
|
||||||
F: FnMut(&mut [rodio::Sample; N]),
|
|
||||||
{
|
|
||||||
ProcessBuffer {
|
|
||||||
inner: self,
|
|
||||||
callback,
|
|
||||||
buffer: [0.0; N],
|
|
||||||
next: N,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct ProcessBuffer<const N: usize, S, F>
|
|
||||||
where
|
|
||||||
S: Source + Sized,
|
|
||||||
F: FnMut(&mut [rodio::Sample; N]),
|
|
||||||
{
|
|
||||||
inner: S,
|
|
||||||
callback: F,
|
|
||||||
buffer: [rodio::Sample; N],
|
|
||||||
next: usize,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<const N: usize, S, F> Iterator for ProcessBuffer<N, S, F>
|
|
||||||
where
|
|
||||||
S: Source + Sized,
|
|
||||||
F: FnMut(&mut [rodio::Sample; N]),
|
|
||||||
{
|
|
||||||
type Item = rodio::Sample;
|
|
||||||
|
|
||||||
fn next(&mut self) -> Option<Self::Item> {
|
|
||||||
self.next += 1;
|
|
||||||
if self.next < self.buffer.len() {
|
|
||||||
let sample = self.buffer[self.next];
|
|
||||||
return Some(sample);
|
|
||||||
}
|
|
||||||
|
|
||||||
for sample in &mut self.buffer {
|
|
||||||
*sample = self.inner.next()?
|
|
||||||
}
|
|
||||||
(self.callback)(&mut self.buffer);
|
|
||||||
|
|
||||||
self.next = 0;
|
|
||||||
Some(self.buffer[0])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<const N: usize, S, F> Source for ProcessBuffer<N, S, F>
|
|
||||||
where
|
|
||||||
S: Source + Sized,
|
|
||||||
F: FnMut(&mut [rodio::Sample; N]),
|
|
||||||
{
|
|
||||||
fn current_span_len(&self) -> Option<usize> {
|
|
||||||
// TODO dvdsk this should be a spanless Source
|
|
||||||
None
|
|
||||||
}
|
|
||||||
|
|
||||||
fn channels(&self) -> rodio::ChannelCount {
|
|
||||||
self.inner.channels()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn sample_rate(&self) -> rodio::SampleRate {
|
|
||||||
self.inner.sample_rate()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn total_duration(&self) -> Option<std::time::Duration> {
|
|
||||||
self.inner.total_duration()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue