3

我正在尝试播放从 ohLibSpotify c# 库 ( https://github.com/openhome/ohLibSpotify ) 提供的原始 pcm 数据。

我在以下回调中获取数据:

public void MusicDeliveryCallback(SpotifySession session, AudioFormat format, IntPtr frames, int num_frames)
{
    //EXAMPLE DATA
    //format.channels = 2, format.samplerate = 44100, format.sample_type = Int16NativeEndian
    //frames = ?
    //num_frames = 2048
}

现在我想用 NAudio ( http://naudio.codeplex.com/ ) 直接播放接收到的数据。使用以下代码片段,我可以从磁盘播放 mp3 文件。是否可以将Spotify接收到的数据直接传递给NAudio并实时播放?

using (var ms = File.OpenRead("test.pcm"))
using (var rdr = new Mp3FileReader(ms))
using (var wavStream = WaveFormatConversionStream.CreatePcmStream(rdr))
using (var baStream = new BlockAlignReductionStream(wavStream))
using (var waveOut = new WaveOut(WaveCallbackInfo.FunctionCallback()))
{
    waveOut.Init(baStream);
    waveOut.Play();
    while (waveOut.PlaybackState == PlaybackState.Playing)
    {
        Thread.Sleep(100);
    }
}

编辑: 我更新了我的代码。该程序不会抛出任何错误,但我也听不到音乐。我的代码有什么问题吗?

这是音乐传递回调:

public void MusicDeliveryCallback(SpotifySession session, AudioFormat format, IntPtr frames, int num_frames)
{
    //format.channels = 2, format.samplerate = 44100, format.sample_type = Int16NativeEndian
    //frames = ?
    //num_frames = 2048

    byte[] frames_copy = new byte[num_frames];
    Marshal.Copy(frames, frames_copy, 0, num_frames);

    bufferedWaveProvider = new BufferedWaveProvider(new WaveFormat(format.sample_rate, format.channels));
    bufferedWaveProvider.BufferDuration = TimeSpan.FromSeconds(40);            
    bufferedWaveProvider.AddSamples(frames_copy, 0, num_frames);
    bufferedWaveProvider.Read(frames_copy, 0, num_frames);

    if (_waveOutDeviceInitialized == false)
    {
        IWavePlayer waveOutDevice = new WaveOut();
        waveOutDevice.Init(bufferedWaveProvider);
        waveOutDevice.Play();
        _waveOutDeviceInitialized = true;
    }
}

这些是 SessionListener 中被覆盖的回调:

public override int MusicDelivery(SpotifySession session, AudioFormat format, IntPtr frames, int num_frames)
{
    _sessionManager.MusicDeliveryCallback(session, format, frames, num_frames);
    return base.MusicDelivery(session, format, frames, num_frames);
}

public override void GetAudioBufferStats(SpotifySession session, out AudioBufferStats stats)
{
    stats.samples = 2048 / 2;   //???
    stats.stutter = 0;          //???
}
4

2 回答 2

4

我认为你可以这样做:

  1. 创建一个 BufferedWaveProvider。
  2. 将此传递给waveOut.Init。
  3. 在您的 MusicDeliveryCallback 中,使用 Marshal.Copy 从本机缓冲区复制到托管字节数组。
  4. 将此托管字节数组传递给 BufferedWaveProvider 上的 AddSamples。
  5. 在您的 GetAudioBufferStats 回调中,将 bufferedWaveProvider.BufferedBytes / 2 用于“samples”并将“stutters”保留为 0。

认为这会奏效。它涉及一些不必要的复制,并且不能准确地跟踪口吃,但这是一个很好的起点。我认为实现 IWaveProvider 并自己管理缓冲可能是一个更好(更有效和更可靠)的解决方案。

我编写了 ohLibSpotify 包装库,但我不再为同一家公司工作,所以我不再参与它的开发。您也许可以从这个论坛上的某个人那里获得更多帮助:http: //forum.openhome.org/forumdisplay.php? fid=6 就音乐交付而言,ohLibSpotify 旨在尽可能减少开销。它根本不复制音乐数据,它只是将 libspotify 库本身提供的相同本机指针传递给您,这样您就可以自己将其复制到其最终目的地,并避免不必要的复制层。不过,对于简单的使用来说,它确实有点笨拙。

祝你好运!

于 2014-01-23T14:22:37.107 回答
3

首先,上面显示的代码片段比它需要的更复杂。您只需要五个,而不是两个using陈述。Mp3FileReader为您解码为 PCM。其次,WaveOutEvent优先WaveOut使用函数回调。它更可靠。

using (var rdr = new Mp3FileReader("test.pcm"))
using (var waveOut = new WaveOutEvent())
{
   //...
}

要回答您的实际问题,您需要使用BufferedWaveProvider. 您创建其中之一,并在 Init 方法中将其传递给您的输出设备。现在,当您接收到音频时,将其解压缩为 PCM(如果已压缩)并将其放入BufferedWaveProvider. NAudioDemo 应用程序包含如何执行此操作的示例,因此请查看 NAudio 源代码以了解其完成方式。

于 2014-01-23T14:16:28.790 回答