Cocos2D-X在WP8上播放mp3

起初竟然不知道用Cocos2D-X在wp8上只能播放wav文件,不能放mp3。要是把mp3全转成wav,太费时费力了。发现不少人也遇到这问题,也有人尝试用Lame做encode/decode代码解决的,比如Lame_WP8,还有mp3之痛 这篇文章都给出了代码。可问题是,这代码莫名其妙之处太多,比如只有一个声道,比如写死的16-bit per sample。最关键是,试了试代码后,发现根本放不了。

开始自己琢磨,还去把Lame原始的例子找来看了看,七拼八凑的做出来了。代码算是self-documented吧,若有不明,欢迎指教。

先找到头文件Cocos2D-X找到CocosDenshion/wp8/MediaStreamer.h,声明两个方法:

...
void Initialize(_In_ const WCHAR* url);  
+ void InitializeMp3(_In_ const WCHAR* mp3FilePath);
+ void GetLengthOfId3v2Tag(_In_ const unsigned char * buff, _Outptr_ int * pLength);
void ReadAll(uint8* buffer, uint32 maxBufferSize, uint32* bufferLength);  
...

然后编辑CocosDenshion/wp8/MediaStreamer.cpp,include头文件和指明链接库,然后实现这两个方法。

...
#include <lame.h>
#pragma comment(lib, "LibMpgHip")
#pragma comment(lib, "LibMp3Lame")
...
void MediaStreamer::InitializeMp3(__in const WCHAR* mp3FilePath)  
{
    unsigned int        decodedLength = 0;
    std::vector<std::unique_ptr<unsigned char[]>> wavBuffer;
    std::vector<int>    mp3BufferSize;
    mp3data_struct      mp3Header = {};

    struct hip_closer
    {
        void operator()(hip_t hip) { ::hip_decode_exit(hip); }
    };
    std::unique_ptr<std::remove_pointer<hip_t>::type, hip_closer> hip(::hip_decode_init());

    FILE * pMp3File = nullptr;
    struct file_closer
    {
        void operator()(FILE* file) { ::fclose(file); }
    };
    ThrowIfFailed(::_wfopen_s(&pMp3File, mp3FilePath, L"rb") == 0);
    std::unique_ptr<FILE, file_closer> mp3File(pMp3File);
    pMp3File = nullptr;

    const int frameBufferSize = 128;
    static unsigned char frameBuffer[frameBufferSize];

    // If there is ID3 tags start skipping ID3 tags to locate where the MP3 data starts.
    size_t readLength = 4;
    ThrowIfFailed(::fread_s(frameBuffer, frameBufferSize, 1, readLength, mp3File.get()) == readLength);
    if (frameBuffer[0] == 'I' && frameBuffer[1] == 'D' && frameBuffer[2] == '3')
    {
        readLength = 6;
        ThrowIfFailed(::fread_s(&frameBuffer[4], frameBufferSize - 4, 1, readLength, mp3File.get()) == readLength);
        GetLengthOfId3v2Tag(&frameBuffer[6], (int *)&readLength);
        ThrowIfFailed(::fseek(mp3File.get(), readLength, SEEK_CUR) == 0);
    }
    else
    {
        // Reset where we start parsing the MP3 data to the file beginning.
        ThrowIfFailed(::fseek(mp3File.get(), 0, SEEK_SET) == 0);
    }

    int skipSamples = 528 + 1;          // There are 529 samples we need to skip. Why 529? Just a MAGIC number from the lame library!
    while (::fread_s(frameBuffer, frameBufferSize, 1, frameBufferSize, mp3File.get()) > 0)
    {
        const int INBUF_SIZE = 1152;
        static short pcmL[INBUF_SIZE];
        static short pcmR[INBUF_SIZE];
        int samples = 0;

        // Start looking for MP3 headers
        if (!mp3Header.header_parsed)
        {
            samples = ::hip_decode_headers(hip.get(), frameBuffer, frameBufferSize, pcmL, pcmR, &mp3Header);
            continue;
        }

        samples = ::hip_decode(hip.get(), frameBuffer, frameBufferSize, pcmL, pcmR);
        if (samples == -1)
        {
            break;
        }
        else if (samples == 0)
        {
            continue;
        }

        if (skipSamples > 0
            && skipSamples > samples)
        {
            skipSamples -= samples;
            continue;
        }

        if (samples > skipSamples)
        {
            static char decodingBuffer[2 * INBUF_SIZE * 2];
            int decodingLength = 0;
            int startIndex = skipSamples;
            skipSamples = 0;

            typedef void(*LeftRightChannelsToBuffer) (const short pcmL, const short pcmR, char * buffer, int & pos);
            LeftRightChannelsToBuffer bufferDataMerger = nullptr;
            if (mp3Header.stereo == 1)
            {
                bufferDataMerger = [](const short pcmL, const short pcmR, char * buffer, int & pos)
                {
                    UNREFERENCED_PARAMETER(pcmR);
                    buffer[pos++] = LO_BYTE(pcmL);
                    buffer[pos++] = HI_BYTE(pcmL);
                };
            }
            else
            {
                bufferDataMerger = [](const short pcmL, const short pcmR, char * buffer, int & pos)
                {
                    buffer[pos++] = LO_BYTE(pcmL);
                    buffer[pos++] = HI_BYTE(pcmL);
                    buffer[pos++] = LO_BYTE(pcmR);
                    buffer[pos++] = HI_BYTE(pcmR);
                };
            }

            for (int i = startIndex; i < samples; ++i)
            {
                bufferDataMerger(pcmL[i], pcmR[i], decodingBuffer, decodingLength);
            }

            std::unique_ptr<unsigned char[]> decodedBuffer(new (std::nothrow) unsigned char[decodingLength]);
            ThrowIfFailed(decodedBuffer);
            ThrowIfFailed(::memcpy_s(decodedBuffer.get(), decodingLength, decodingBuffer, decodingLength) == 0);
            wavBuffer.emplace_back(decodedBuffer.release());
            mp3BufferSize.push_back(decodingLength);
            decodedLength += decodingLength;
        }
    }
    ThrowIfFailed(decodedLength > 0);

    m_data.resize(decodedLength);
    int dataIndex = 0;
    for (unsigned int i = 0; i < wavBuffer.size(); ++i)
    {
        ThrowIfFailed(::memcpy_s((void*)&m_data[dataIndex], mp3BufferSize[i], wavBuffer[i].get(), mp3BufferSize[i]) == 0);
        dataIndex += mp3BufferSize[i];
    }

    m_waveFormat.wFormatTag = WAVE_FORMAT_PCM;
    m_waveFormat.nChannels = (WORD)mp3Header.stereo;
    m_waveFormat.nSamplesPerSec = mp3Header.samplerate;
    m_waveFormat.wBitsPerSample = (mp3Header.samplerate == 96,000) ? 24 : 16;           // Sample rate 44,100 or 48,000 is 16 bits/sample but 96,000 is 24 bits/sample.
    m_waveFormat.nBlockAlign = m_waveFormat.nChannels * m_waveFormat.wBitsPerSample / 8;
    m_waveFormat.nAvgBytesPerSec = m_waveFormat.nSamplesPerSec * m_waveFormat.nBlockAlign;
    m_waveFormat.cbSize = 0;
}

用到的inline function for error handling和宏实现在这里:

inline void ThrowIfFailed(bool succeded) {  
    if (!succeded) {
        throw Platform::FailureException::CreateException(E_FAIL, "Unexpected error happened!");
    }
}
#define LO_BYTE(x) (x & 0x00ff)
#define HI_BYTE(x) ((x >> 8) & 0x00ff)

然后在原始的MediaStreamer::Initialize(...)方法里,发现是mp3文件时直接调用新方法InitializeMp3(...)。

void MediaStreamer::Initialize(__in const WCHAR* url) {  
    WCHAR filePath[MAX_PATH] = {0};
...
    wcscat_s(filePath, url);
}
+ if (::_wcsicmp(url + urlLength - 4, L".mp3") == 0)
+ {
+     return this->InitializeMp3(filePath);
+ }
Platform::Array<byte>^ data = ReadData(ref new Platform::String(filePath));  
...

Okay,把lame_wp8引入项目,如果编译链接无误,应该是可以成功播放mp3了。