共用方式為


教學課程:使用 WMContainer 物件撰寫 WMA 檔案

本教學課程示範如何從未壓縮的音訊檔案(.wav)擷取媒體內容,然後以 ASF 格式壓縮它,來示範如何撰寫新的音訊檔案(.wma)。 用於轉換的編碼模式是 常數比特率編碼 (CBR)。 在此模式中,在編碼會話之前,應用程式會指定編碼器必須達到的目標比特率。

在本教學課程中,您將建立主控台應用程式,以輸入和輸出檔名作為自變數。 應用程式會從波浪檔案剖析應用程式取得未壓縮的媒體範例,本教學課程會提供此範例。 這些範例會傳送至編碼器,以轉換成 Windows Media Audio 9 格式。 編碼器已針對 CBR 編碼進行設定,並使用媒體類型交涉期間提供的第一個比特率作為目標比特率。 編碼的樣本會傳送至多任務器,以使用 ASF 數據格式進行封包處理。 這些封包會寫入代表 ASF 數據物件的位元組數據流。 數據區段準備就緒之後,您將建立 ASF 音訊檔案,並寫入新的 ASF 標頭物件,以合併所有標頭資訊,然後附加 ASF 數據物件位元組數據流。

本教學課程包含下列各節:

先決條件

本教程假設以下事項:

  • 您熟悉 ASF 檔案的結構,以及媒體基礎所提供的元件,以使用 ASF 物件。 這些元件包括 ContentInfo、分割器、多工器和設定檔物件。 如需詳細資訊,請參閱 WMContainer ASF 元件
  • 您熟悉 Windows Media 編碼器,以及各種編碼類型,特別是 CBR。 如需詳細資訊,請參閱 Windows Media Encoders
  • 您熟悉媒體緩衝區 和位元組數據流:具體而言,使用位元組數據流的檔案作業,以及將媒體緩衝區的內容寫入位元組數據流。

術語

本教學課程使用下列詞彙:

  • 來源媒體類型:媒體類型物件,會公開 IMFMediaType 介面,其中描述輸入檔的內容。
  • 音訊配置檔:配置檔對象,公開 IMFASFProfile 介面,其中只包含輸出檔案的音訊數據流。
  • 串流樣本:媒體樣本會公開 IMFSample 介面,代表從編碼器以壓縮狀態取得的輸入檔的媒體數據。
  • ContentInfo 物件:ASF ContentInfo 物件,會公開 IMFASFContentInfo 介面,代表輸出檔的 ASF 標頭物件。
  • 數據位元組數據流:位元組數據流物件會公開 IMFByteStream 介面,代表輸出檔案的整個 ASF 資料物件部分。
  • 數據封包:媒體範例會公開由 ASF Multiplexer所產生的 IMFSample 介面,表示這是將寫入數據位元組流的 ASF 數據封包。
  • 輸出位元組數據流:位元組數據流物件,會公開 IMFByteStream 介面,其中包含輸出檔的內容。

1.設定專案

  1. 在您的原始程式檔中包含下列標頭:

    #include <new>
    #include <stdio.h>       // Standard I/O
    #include <windows.h>     // Windows headers
    #include <mfapi.h>       // Media Foundation platform
    #include <wmcontainer.h> // ASF interfaces
    #include <mferror.h>     // Media Foundation error codes
    
  2. 請將下列程式庫檔案連結起來:

    • mfplat.lib
    • mf.lib
    • mfuuid.lib
  3. 宣告 SafeRelease 函式。

  4. 在您的專案中包含 CWmaEncoder 類別。 如您要此類別的完整原始碼,請參閱 編碼器範例程式代碼

2. 宣告輔助函數

本教學課程會使用下列協助程式函式,從位元組數據流讀取和寫入。

  • AppendToByteStream:將一個字節數據流的內容附加至另一個字節數據流。
  • WriteBufferToByteStream:將數據從媒體緩衝區寫入位元組數據流。

如需詳細資訊,請參閱 IMFByteStream::Write。 下列程式代碼顯示這些協助程式函式:

//-------------------------------------------------------------------
// AppendToByteStream
//
// Reads the contents of pSrc and writes them to pDest.
//-------------------------------------------------------------------

HRESULT AppendToByteStream(IMFByteStream *pSrc, IMFByteStream *pDest)
{
    HRESULT hr = S_OK;

    const DWORD READ_SIZE = 1024;

    BYTE buffer[READ_SIZE];

    while (1)
    {
        ULONG cbRead;

        hr = pSrc->Read(buffer, READ_SIZE, &cbRead);

        if (FAILED(hr)) { break; }

        if (cbRead == 0)
        {
            break;
        }

        hr = pDest->Write(buffer, cbRead, &cbRead);

        if (FAILED(hr)) { break; }
    }

    return hr;
}
//-------------------------------------------------------------------
// WriteBufferToByteStream
//
// Writes data from a media buffer to a byte stream.
//-------------------------------------------------------------------

HRESULT WriteBufferToByteStream(
    IMFByteStream *pStream,   // Pointer to the byte stream.
    IMFMediaBuffer *pBuffer,  // Pointer to the media buffer.
    DWORD *pcbWritten         // Receives the number of bytes written.
    )
{
    HRESULT hr = S_OK;
    DWORD cbData = 0;
    DWORD cbWritten = 0;
    BYTE *pMem = NULL;

    hr = pBuffer->Lock(&pMem, NULL, &cbData);

    if (SUCCEEDED(hr))
    {
        hr = pStream->Write(pMem, cbData, &cbWritten);
    }

    if (SUCCEEDED(hr))
    {
        if (pcbWritten)
        {
            *pcbWritten = cbWritten;
        }
    }

    if (pMem)
    {
        pBuffer->Unlock();
    }
    return hr;
}

3.開啟音訊檔案

本教學課程假設您的應用程式會產生未壓縮的音訊以進行編碼。 為此,本教程會宣告兩個函式:

HRESULT OpenAudioFile(PCWSTR pszURL, IMFMediaType **ppAudioFormat);
HRESULT GetNextAudioSample(BOOL *pbEOS, IMFSample **ppSample);

這些函式的實作會留給讀取器。

  • OpenAudioFile 函式應該會開啟 pszURL 所指定的媒體檔案,並傳回描述音訊數據流之媒體類型的指標。
  • GetNextAudioSample 函式應該從由 OpenAudioFile開啟的檔案讀取未壓縮的 PCM 音訊。 到達檔尾時,pbEOS 會接收到 TRUE的值。 否則,ppSample 將接收包含音訊緩衝的媒體樣本。

4.設定編碼器

接下來,建立編碼器、設定它以產生 CBR 編碼的數據流範例,以及交涉輸入和輸出媒體類型。

在媒體基礎中,編碼器(公開 IMFTransform 介面)會實作為媒體基礎轉換 (MFT)

在本教學課程中,編碼器會在提供 MFT 包裝函式的 CWmaEncoder 類別中實作。 如您要此類別的完整原始碼,請參閱 編碼器範例程式代碼

注意

您可以選擇性地將編碼類型指定為 CBR。 根據預設,編碼器會設定為使用 CBR 編碼。 如需詳細資訊,請參閱 常數比特率編碼。 您可以根據編碼類型來設定其他屬性,如需編碼模式特定屬性的詳細資訊,請參閱 Quality-Based 變數比特率編碼不受限制的變數比特率編碼,以及 Peak-Constrained 變數比特率編碼

 

    CWmaEncoder* pEncoder = NULL; //Pointer to the Encoder object.

    hr = OpenAudioFile(sInputFileName, &pInputType);
    if (FAILED(hr))
    {
        goto done;
    }

    // Initialize the WMA encoder wrapper.

    pEncoder = new (std::nothrow) CWmaEncoder();
    if (pEncoder == NULL)
    {
        hr = E_OUTOFMEMORY;
        goto done;
    }

    hr = pEncoder->Initialize();
    if (FAILED(hr))
    {
        goto done;
    }

    hr = pEncoder->SetEncodingType(EncodeMode_CBR);
    if (FAILED(hr))
    {
        goto done;
    }

    hr = pEncoder->SetInputType(pInputType);
    if (FAILED(hr))
    {
        goto done;
    }

5.建立 ASF ContentInfo 物件。

ASF ContentInfo 物件 包含輸出檔之各種標頭對象的相關信息。

首先,建立音訊數據流的 ASF 配置檔:

  1. 呼叫 MFCreateASFProfile 來建立空的 ASF 配置文件物件。 ASF 配置檔會公開 IMFASFProfile 介面。 如需詳細資訊,請參閱 建立和設定 ASF 資料流
  2. CWmaEncoder 物件取得編碼的音訊格式。
  3. 呼叫 IMFASFProfile::CreateStream,為 ASF 配置檔建立新的數據流。 傳入代表數據流格式之 IMFMediaType 介面的指標。
  4. 呼叫 IMFASFStreamConfig::SetStreamNumber 來指派數據流標識符。
  5. 在流對象上設定 MF_ASFSTREAMCONFIG_LEAKYBUCKET1 屬性,以設定「漏桶」參數。
  6. 呼叫 IMFASFProfile::SetStream,將新的數據流新增至配置檔。

現在建立 ASF ContentInfo 物件,如下所示:

  1. 呼叫 MFCreateASFContentInfo 來建立空的 ContentInfo 物件。
  2. 呼叫 IMFASFContentInfo::SetProfile 來設定 ASF 配置檔。

下列程式代碼顯示下列步驟:

HRESULT CreateASFContentInfo(
    CWmaEncoder* pEncoder,
    IMFASFContentInfo** ppContentInfo
    )
{
    HRESULT hr = S_OK;
    
    IMFASFProfile* pProfile = NULL;
    IMFMediaType* pMediaType = NULL;
    IMFASFStreamConfig* pStream = NULL;
    IMFASFContentInfo* pContentInfo = NULL;

    // Create the ASF profile object.

    hr = MFCreateASFProfile(&pProfile); 
    if (FAILED(hr))
    {
        goto done;
    }

    // Create a stream description for the encoded audio.

    hr = pEncoder->GetOutputType(&pMediaType); 
    if (FAILED(hr))
    {
        goto done;
    }

    hr = pProfile->CreateStream(pMediaType, &pStream); 
    if (FAILED(hr))
    {
        goto done;
    }

    hr = pStream->SetStreamNumber(DEFAULT_STREAM_NUMBER); 
    if (FAILED(hr))
    {
        goto done;
    }

    // Set "leaky bucket" values.

    LeakyBucket bucket;

    hr = pEncoder->GetLeakyBucket1(&bucket);
    if (FAILED(hr))
    {
        goto done;
    }

    hr = pStream->SetBlob(
        MF_ASFSTREAMCONFIG_LEAKYBUCKET1, 
        (UINT8*)&bucket, 
        sizeof(bucket)
        );

    if (FAILED(hr))
    {
        goto done;
    }

    //Add the stream to the profile

    hr = pProfile->SetStream(pStream);
    if (FAILED(hr))
    {
        goto done;
    }

    // Create the ASF ContentInfo object.

    hr = MFCreateASFContentInfo(&pContentInfo); 
    if (FAILED(hr))
    {
        goto done;
    }

    hr = pContentInfo->SetProfile(pProfile); 
    if (FAILED(hr))
    {
        goto done;
    }

    // Return the pointer to the caller.

    *ppContentInfo = pContentInfo;
    (*ppContentInfo)->AddRef();

done:
    SafeRelease(&pProfile);
    SafeRelease(&pStream);
    SafeRelease(&pMediaType);
    SafeRelease(&pContentInfo);
    return hr;
}

6. 建立 ASF 多工器

ASF 多工器 會產生 ASF 數據封包。

  1. 呼叫 MFCreateASFMultiplexer 以建立 ASF 多工器。
  2. 呼叫 IMFASFMultiplexer::Initialize 來初始化多任務器。 傳入在上一節中建立的 ASF 內容資訊物件的指標。
  3. 呼叫 IMFASFMultiplexer::SetFlags 來設定 MFASF_MULTIPLEXER_AUTOADJUST_BITRATE 旗標。 使用此設定時,多任務器會自動調整 ASF 內容的比特率,以符合多任務處理的數據流特性。
HRESULT CreateASFMux( 
    IMFASFContentInfo* pContentInfo,
    IMFASFMultiplexer** ppMultiplexer
    )
{
    HRESULT hr = S_OK;
    
    IMFMediaType* pMediaType = NULL;
    IMFASFMultiplexer *pMultiplexer = NULL;

    // Create and initialize the ASF Multiplexer object.

    hr = MFCreateASFMultiplexer(&pMultiplexer);
    if (FAILED(hr))
    {
        goto done;
    }

    hr = pMultiplexer->Initialize(pContentInfo);
    if (FAILED(hr))
    {
        goto done;
    }

    // Enable automatic bit-rate adjustment.

    hr = pMultiplexer->SetFlags(MFASF_MULTIPLEXER_AUTOADJUST_BITRATE);
    if (FAILED(hr))
    {
        goto done;
    }

    *ppMultiplexer = pMultiplexer;
    (*ppMultiplexer)->AddRef();

done:
    SafeRelease(&pMultiplexer);
    return hr;
}

7.產生新的 ASF 數據封包

接下來,產生新檔案的 ASF 數據封包。 這些數據封包將構成新檔案的最終 ASF 資料物件。 若要產生新的 ASF 資料封包:

  1. 呼叫 MFCreateTempFile 建立暫存位元組數據流來保存 ASF 數據封包。
  2. 呼叫應用程式定義的 GetNextAudioSample 函式,以取得編碼器的未壓縮音訊數據。
  3. 將未壓縮的音訊傳遞至編碼器以進行壓縮。 如需詳細資訊,請參閱編碼器 處理數據
  4. 呼叫 IMFASFMultiplexer::P rocessSample,將壓縮的音頻樣本傳送至 ASF 多任務器進行封包化。
  5. 從多任務器取得 ASF 封包,並將其寫入暫存位元組數據流。 如需詳細資訊,請參閱 產生新的 ASF 資料封包
  6. 當您到達來源數據流的結尾時,請清空編碼器,並從編碼器提取剩餘的壓縮樣本。 如需清空 MFT 的詳細資訊,請參閱 基本 MFT 處理模型
  7. 將所有樣本傳送至多任務器之後,請呼叫 IMFASFMultiplexer::Flush,並從多任務器提取剩餘的 ASF 封包。
  8. 呼叫 IMFASFMultiplexer::End

下列程式代碼會產生 ASF 數據封包。 函式會傳回指向包含 ASF 資料物件的位元組資料流的指標。

HRESULT EncodeData(
    CWmaEncoder* pEncoder, 
    IMFASFContentInfo* pContentInfo,
    IMFASFMultiplexer* pMux, 
    IMFByteStream** ppDataStream) 
{
    HRESULT hr = S_OK;

    IMFByteStream* pStream = NULL;
    IMFSample* pInputSample = NULL;
    IMFSample* pWmaSample = NULL;

    BOOL bEOF = FALSE;

   // Create a temporary file to hold the data stream.
   hr = MFCreateTempFile(
        MF_ACCESSMODE_READWRITE, 
        MF_OPENMODE_DELETE_IF_EXIST,
        MF_FILEFLAGS_NONE,
        &pStream
        );

   if (FAILED(hr))
   {
       goto done;
   }

    BOOL bNeedInput = TRUE;

    while (TRUE)
    {
        if (bNeedInput)
        {
            hr = GetNextAudioSample(&bEOF, &pInputSample);
            if (FAILED(hr))
            {
                goto done;
            }

            if (bEOF)
            {
                // Reached the end of the input file.
                break;
            }

            // Encode the uncompressed audio sample.
            hr = pEncoder->ProcessInput(pInputSample);
            if (FAILED(hr))
            {
                goto done;
            }

            bNeedInput = FALSE;
        }

        if (bNeedInput == FALSE)
        {
            // Get data from the encoder.

            hr = pEncoder->ProcessOutput(&pWmaSample);
            if (FAILED(hr))
            {
                goto done;
            }

            // pWmaSample can be NULL if the encoder needs more input.

            if (pWmaSample)
            {
                hr = pMux->ProcessSample(DEFAULT_STREAM_NUMBER, pWmaSample, 0);
                if (FAILED(hr))
                {
                    goto done;
                }

                //Collect the data packets and write them to a stream
                hr = GenerateASFDataPackets(pMux, pStream);
                if (FAILED(hr))
                {
                    goto done;
                }
            }
            else
            {
                bNeedInput = TRUE;
            }
        }
        
        SafeRelease(&pInputSample);
        SafeRelease(&pWmaSample);
    }

    // Drain the MFT and pull any remaining samples from the encoder.

    hr = pEncoder->Drain();
    if (FAILED(hr))
    {
        goto done;
    }

    while (TRUE)
    {
        hr = pEncoder->ProcessOutput(&pWmaSample);
        if (FAILED(hr))
        {
            goto done;
        }

        if (pWmaSample == NULL)
        {
            break;
        }

        hr = pMux->ProcessSample(DEFAULT_STREAM_NUMBER, pWmaSample, 0);
        if (FAILED(hr))
        {
            goto done;
        }

        //Collect the data packets and write them to a stream
        hr = GenerateASFDataPackets(pMux, pStream);
        if (FAILED(hr))
        {
            goto done;
        }

        SafeRelease(&pWmaSample);
    }

    // Flush the mux and get any pending ASF data packets.
    hr = pMux->Flush();
    if (FAILED(hr))
    {
        goto done;
    }

    hr = GenerateASFDataPackets(pMux, pStream);
    if (FAILED(hr))
    {
        goto done;
    }
    
    // Update the ContentInfo object
    hr = pMux->End(pContentInfo);
    if (FAILED(hr))
    {
        goto done;
    }

    //Return stream to the caller that contains the ASF encoded data.
    *ppDataStream = pStream;
    (*ppDataStream)->AddRef();

done:
    SafeRelease(&pStream);
    SafeRelease(&pInputSample);
    SafeRelease(&pWmaSample);
    return hr;
}

在主題 產生新的 ASF 數據封包中顯示了 GenerateASFDataPackets 函式的程式代碼。

8.寫入 ASF 檔案

接下來,呼叫 IMFASFContentInfo::GenerateHeader,將 ASF 標頭寫入媒體緩衝區。 這個方法會將儲存在 ContentInfo 物件中的數據轉換成 ASF 標頭物件格式的二進位數據。 如需詳細資訊,請參閱 產生新的 ASF 標頭物件

產生新的 ASF 標頭對象之後,請建立輸出檔案的位元組數據流。 首先,將 Header 物件寫入輸出位元組數據流。 按順序在資料位元組流中,將 Header 物件後接著數據物件。

HRESULT WriteASFFile( 
     IMFASFContentInfo *pContentInfo, 
     IMFByteStream *pDataStream,
     PCWSTR pszFile
     )
{
    HRESULT hr = S_OK;
    
    IMFMediaBuffer* pHeaderBuffer = NULL;
    IMFByteStream* pWmaStream = NULL;

    DWORD cbHeaderSize = 0;
    DWORD cbWritten = 0;

    //Create output file
    hr = MFCreateFile(MF_ACCESSMODE_WRITE, MF_OPENMODE_DELETE_IF_EXIST,
        MF_FILEFLAGS_NONE, pszFile, &pWmaStream);

    if (FAILED(hr))
    {
        goto done;
    }


    // Get the size of the ASF Header Object.
    hr = pContentInfo->GenerateHeader (NULL, &cbHeaderSize);
    if (FAILED(hr))
    {
        goto done;
    }

    // Create a media buffer.
    hr = MFCreateMemoryBuffer(cbHeaderSize, &pHeaderBuffer);
    if (FAILED(hr))
    {
        goto done;
    }

    // Populate the media buffer with the ASF Header Object.
    hr = pContentInfo->GenerateHeader(pHeaderBuffer, &cbHeaderSize);
    if (FAILED(hr))
    {
        goto done;
    }

    // Write the ASF header to the output file.
    hr = WriteBufferToByteStream(pWmaStream, pHeaderBuffer, &cbWritten);
    if (FAILED(hr))
    {
        goto done;
    }

    // Append the data stream to the file.

    hr = pDataStream->SetCurrentPosition(0);
    if (FAILED(hr))
    {
        goto done;
    }

    hr = AppendToByteStream(pDataStream, pWmaStream);

done:
    SafeRelease(&pHeaderBuffer);
    SafeRelease(&pWmaStream);
    return hr;
}

9.定義 Entry-Point 函式

現在您可以將先前的步驟放在一起,並放入完整的應用程式。 使用任何 Media Foundation 物件之前,請先呼叫 MFStartup來初始化 Media Foundation 平臺。 完成時,請呼叫 MFShutdown。 如需詳細資訊,請參閱 初始化媒體基礎

下列程式代碼顯示完整的控制台應用程式。 命令行自變數會指定要轉換的檔名,以及新音訊檔的名稱。

int wmain(int argc, WCHAR* argv[])
{
    HeapSetInformation(NULL, HeapEnableTerminationOnCorruption, NULL, 0);

    if (argc != 3)
    {
        wprintf_s(L"Usage: %s input.wmv, %s output.wma");
        return 0;
    }

    const WCHAR* sInputFileName = argv[1];    // Source file name
    const WCHAR* sOutputFileName = argv[2];  // Output file name
    
    IMFMediaType* pInputType = NULL;
    IMFASFContentInfo* pContentInfo = NULL;
    IMFASFMultiplexer* pMux = NULL;
    IMFByteStream* pDataStream = NULL;

    CWmaEncoder* pEncoder = NULL; //Pointer to the Encoder object.

    HRESULT hr = CoInitializeEx(
        NULL, COINIT_APARTMENTTHREADED | COINIT_DISABLE_OLE1DDE);

    if (FAILED(hr))
    {
        goto done;
    }

    hr = MFStartup(MF_VERSION);
    if (FAILED(hr))
    {
        goto done;
    }

    CWmaEncoder* pEncoder = NULL; //Pointer to the Encoder object.

    hr = OpenAudioFile(sInputFileName, &pInputType);
    if (FAILED(hr))
    {
        goto done;
    }

    // Initialize the WMA encoder wrapper.

    pEncoder = new (std::nothrow) CWmaEncoder();
    if (pEncoder == NULL)
    {
        hr = E_OUTOFMEMORY;
        goto done;
    }

    hr = pEncoder->Initialize();
    if (FAILED(hr))
    {
        goto done;
    }

    hr = pEncoder->SetEncodingType(EncodeMode_CBR);
    if (FAILED(hr))
    {
        goto done;
    }

    hr = pEncoder->SetInputType(pInputType);
    if (FAILED(hr))
    {
        goto done;
    }

    // Create the WMContainer objects.
    hr = CreateASFContentInfo(pEncoder, &pContentInfo);
    if (FAILED(hr))
    {
        goto done;
    }

    hr = CreateASFMux(pContentInfo, &pMux);
    if (FAILED(hr))
    {
        goto done;
    }

    // Convert uncompressed data to ASF format.
    hr = EncodeData(pEncoder, pContentInfo, pMux, &pDataStream);
    if (FAILED(hr))
    {
        goto done;
    }

    // Write the ASF objects to the output file.
    hr = WriteASFFile(pContentInfo, pDataStream, sOutputFileName);

done:
    SafeRelease(&pInputType);
    SafeRelease(&pContentInfo);
    SafeRelease(&pMux);
    SafeRelease(&pDataStream);
    delete pEncoder;

    MFShutdown();
    CoUninitialize();

    if (FAILED(hr))
    {
        wprintf_s(L"Error: 0x%X\n", hr);
    }

    return 0;
} 

WMContainer ASF 元件

媒體基礎中的 ASF 支援