Используйте NAudio, чтобы получить образцы Ulaw для RTP - PullRequest
2 голосов
/ 29 февраля 2012

Я просматривал примеры NAudio, пытаясь понять, как получить образцы ulaw, подходящие для упаковки в качестве полезной нагрузки RTP.Я пытаюсь сгенерировать образцы из mp3-файла, используя код ниже.Не удивительно, так как я не имею понятия, что я делаю с NAudio, когда я передаю сэмплы по сети на программный телефон, все, что я получаю, - это статический сигнал.

Может ли кто-нибудь дать какое-либо указание оЯ должен получить 160 байтов (8 кГц при 20 мс) сэмплов ULAW из файла MP3 с помощью NAudio?

private void GetAudioSamples()
{
    var pcmStream = WaveFormatConversionStream.CreatePcmStream(new Mp3FileReader("whitelight.mp3"));
    byte[] buffer = new byte[2];
    byte[] sampleBuffer = new byte[160];
    int sampleIndex = 0;
    int bytesRead = pcmStream.Read(buffer, 0, 2);

    while (bytesRead > 0)
    {
        var ulawByte = MuLawEncoder.LinearToMuLawSample(BitConverter.ToInt16(buffer, 0));
        sampleBuffer[sampleIndex++] = ulawByte;

        if (sampleIndex == 160)
        {
            m_rtpChannel.AddSample(sampleBuffer);
            sampleBuffer = new byte[160];
            sampleIndex = 0;
        }

        bytesRead = pcmStream.Read(buffer, 0, 2);
    }

    logger.Debug("Finished adding audio samples.");
}

Ответы [ 2 ]

3 голосов
/ 03 марта 2012

Ниже я понял, как это работает.Я теряю один из каналов из mp3, и я думаю, что есть какой-то способ объединить каналы как часть преобразования, но это не имеет значения для моей ситуации.

Размер буфера 160 байт дает мнеОбразцы Ulaw 20 мс, которые отлично работают с программным телефоном SIP, с которым я тестирую.

var pcmFormat = new WaveFormat(8000, 16, 1);
var ulawFormat = WaveFormat.CreateMuLawFormat(8000, 1);

using (WaveFormatConversionStream pcmStm = new WaveFormatConversionStream(pcmFormat, new Mp3FileReader("whitelight.mp3")))
{
    using (WaveFormatConversionStream ulawStm = new WaveFormatConversionStream(ulawFormat, pcmStm))
    {
        byte[] buffer = new byte[160];
        int bytesRead = ulawStm.Read(buffer, 0, 160);

        while (bytesRead > 0)
        {
            byte[] sample = new byte[bytesRead];
            Array.Copy(buffer, sample, bytesRead);
            m_rtpChannel.AddSample(sample);

            bytesRead = ulawStm.Read(buffer, 0, 160);
        }
    }
}
2 голосов
/ 01 марта 2012

Вот несколько указателей. Прежде всего, пока вы используете NAudio 1.5, нет необходимости в дополнительном WaveFormatConversionStream - метод Read Mp3FileReader возвращает PCM.

Однако вы не получите 8 кГц, поэтому вам нужно сначала пересчитать его. WaveFormatConversionStream может сделать это, хотя он использует встроенное преобразование частоты дискретизации Windows ACM, которое, похоже, плохо фильтрует входящий звук, поэтому могут возникнуть артефакты сглаживания.

Кроме того, вы обычно читаете больше блоков, чем два байта за раз, поскольку декодеру MP3 необходимо декодировать кадры по одному (ресемплер также захочет работать с блоками большего размера). Я бы попробовал читать байты не менее 20 мс за раз.

Ваше использование BitConverter.ToInt16 подходит для получения значения 16-битного семпла, но имейте в виду, что MP3, скорее всего, стереофонический, с левым и правым семплами. Вы уверены, что ваш телефон ожидает стерео.

Наконец, я рекомендую сделать WAV-файл мю-закона в качестве первого шага, используя WaveFileWriter. Затем вы можете легко прослушать его в проигрывателе Windows Media и проверить, что вы отправляете на свой программный телефон именно то, что и хотели.

...