Skip to content
This repository has been archived by the owner on Apr 13, 2023. It is now read-only.

Commit

Permalink
Fix Audio (previous commit had faults)
Browse files Browse the repository at this point in the history
  • Loading branch information
MathewSachin committed Jul 3, 2018
1 parent 82dd8bf commit 01d0027
Show file tree
Hide file tree
Showing 2 changed files with 64 additions and 21 deletions.
79 changes: 58 additions & 21 deletions src/Captura.Bass/MixedAudioProvider.cs
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
using System;
using System.Collections.Generic;
using System.Linq;
using ManagedBass;
using ManagedBass.Mix;
using System.Runtime.InteropServices;
Expand All @@ -21,11 +20,11 @@ class RecordingItem
public int DeviceId { get; set; }

public int RecordingHandle { get; set; }

public int SilenceHandle { get; set; }
}

readonly Dictionary<int, RecordingItem> _devices = new Dictionary<int, RecordingItem>();
readonly int _filler; // Fill when no audio source is selected
bool _fillerAdded;
readonly int _mixer;
readonly object _syncLock = new object();
bool _running;
Expand All @@ -42,13 +41,14 @@ public MixedAudioProvider(IEnumerable<BassItem> Devices, int FrameRate, bool Mut

Bass.UpdatePeriod = updatePeriod.Clip(5, 100);

Console.WriteLine($"BASS Update Period: {Bass.UpdatePeriod}");

for (var i = 0; i < BufferCount; ++i)
{
_buffers.Add(new byte[0]);
}

_mixer = BassMix.CreateMixerStream(44100, 2, BassFlags.Default);
_filler = Bass.CreateStream(44100, 2, BassFlags.Decode, Extensions.SilenceStreamProcedure);
_mixer = BassMix.CreateMixerStream(44100, 2, BassFlags.MixerNonStop);

foreach (var recordingDevice in Devices)
{
Expand Down Expand Up @@ -99,7 +99,22 @@ void RemoveDevice(BassItem Device)
Bass.StreamFree(handle);

_devices[Device.Id].RecordingHandle = 0;

Bass.StreamFree(_devices[Device.Id].SilenceHandle);

_devices[Device.Id].SilenceHandle = 0;
}
}

int FindPlaybackDevice(DeviceInfo LoopbackDeviceInfo)
{
for (var i = 0; Bass.GetDeviceInfo(i, out var info); ++i)
{
if (info.Driver == LoopbackDeviceInfo.Driver)
return i;
}

return -1;
}

void AddDevice(BassItem Device)
Expand All @@ -110,6 +125,26 @@ void AddDevice(BassItem Device)
return;

Bass.RecordInit(Device.Id);

var devInfo = Bass.RecordGetDeviceInfo(Device.Id);

if (devInfo.IsLoopback)
{
var playbackDevice = FindPlaybackDevice(devInfo);

if (playbackDevice != -1)
{
Bass.Init(playbackDevice);
Bass.CurrentDevice = playbackDevice;

var silence = Bass.CreateStream(44100, 2, BassFlags.Default, Extensions.SilenceStreamProcedure);

Bass.ChannelSetAttribute(silence, ChannelAttribute.Volume, 0);

_devices[Device.Id].SilenceHandle = silence;
}
}

Bass.CurrentRecordingDevice = Device.Id;

var info = Bass.RecordingInfo;
Expand Down Expand Up @@ -175,12 +210,14 @@ public void Dispose()
{
Bass.StreamFree(_mixer);

foreach (var rec in _devices.Values.Where(M => M.RecordingHandle != 0))
foreach (var rec in _devices.Values)
{
Bass.StreamFree(rec.RecordingHandle);
}
if (rec.RecordingHandle != 0)
Bass.StreamFree(rec.RecordingHandle);

Bass.StreamFree(_filler);
if (rec.SilenceHandle != 0)
Bass.StreamFree(rec.SilenceHandle);
}

_running = false;
}
Expand All @@ -193,21 +230,17 @@ public void Start()
{
lock (_syncLock)
{
foreach (var rec in _devices.Values.Where(M => M.RecordingHandle != 0))
foreach (var rec in _devices.Values)
{
Bass.ChannelPlay(rec.RecordingHandle);
if (rec.SilenceHandle != 0)
Bass.ChannelPlay(rec.SilenceHandle);

if (rec.RecordingHandle != 0)
Bass.ChannelPlay(rec.RecordingHandle);
}

Bass.ChannelPlay(_mixer);

if (!_fillerAdded)
{
// Add Filler only after mixer has started
BassMix.MixerAddChannel(_mixer, _filler, BassFlags.Default);

_fillerAdded = true;
}

_running = true;
}
}
Expand All @@ -221,9 +254,13 @@ public void Stop()
{
Bass.ChannelPause(_mixer);

foreach (var rec in _devices.Values.Where(M => M.RecordingHandle != 0))
foreach (var rec in _devices.Values)
{
Bass.ChannelPause(rec.RecordingHandle);
if (rec.RecordingHandle != 0)
Bass.ChannelPause(rec.RecordingHandle);

if (rec.SilenceHandle != 0)
Bass.ChannelPause(rec.SilenceHandle);
}

_running = false;
Expand Down
6 changes: 6 additions & 0 deletions src/Screna/Recorder/Recorder.cs
Original file line number Diff line number Diff line change
Expand Up @@ -221,6 +221,12 @@ void AudioProvider_DataAvailable(object Sender, DataAvailableEventArgs E)
{
try
{
lock (_syncLock)
{
if (_disposed)
return;
}

_videoWriter.WriteAudio(E.Buffer, E.Length);
}
catch (Exception e)
Expand Down

0 comments on commit 01d0027

Please sign in to comment.