Поделиться через


Использование частных данных видеокодека (Microsoft Media Foundation)

Сжатые выходные данные, созданные кодеками Windows Media Video 9, не могут быть распакованы должным образом без некоторых данных, предоставленных кодировщиком. Эти данные, называемые частными данными кодека, должны быть добавлены к типу выходного носителя. Вы можете получить частные данные кодека, вызвав методы интерфейса IWMCodecPrivateData . Передайте полную структуру DMO_MEDIA_TYPE в IWMCodecPrivateData::SetPartialOutputType. Затем вызовите IWMCodecPrivateData::GetPrivateData дважды, один раз, чтобы получить размер данных, а затем снова скопировать данные в буфер такого размера. Создайте новый буфер для хранения структуры VIDEOINFOHEADER с добавленными частными данными и скопируйте структуру и данные в этот буфер. Наконец, задайте для элемента pbFormatструктуры DMO_MEDIA_TYPE адрес только что созданного буфера и задайте для элемента cbFormat объединенный размер VIDEOINFOHEADER и частных данных в байтах.

При использовании MediaFoundation можно создать структуру DMO_MEDIA_TYPE из интерфейса IMFMediaType , вызвав MFCreateAMMediaTypeFromMFMediaType.

Необходимо использовать личные данные кодека, полученные после первой настройки свойств кодировщика. При изменении каких-либо свойств необходимо получить новые личные данные. Если вы не используете частные данные, полученные после установки всех свойств для сеанса кодирования, декодер может не распаковать данные.

В следующем примере кода показано, как получить частные данные для типа видео:

HRESULT GetFinalOutputType(DMO_MEDIA_TYPE* pMedia, IMediaObject* pDMO)
{
    // WARNING //
    // This function does not deallocate the memory pointed to by 
    // pMedia->pbFormat. If the VIDEOINFOHEADER referenced by pbFormat
    // was dynamically allocated, a reference to it must be kept before
    //  calling this function so that it can be freed.

    // Perform simple parameter checks.
    if(pMedia == NULL || pDMO == NULL)
        return E_POINTER;
    if(pMedia->formattype != MEDIATYPE_VideoInfo)
        return E_INVALIDARG;

    HRESULT hr = S_OK;

    IWMCodecPrivateData* pPrivData = NULL;
    BYTE* pbData = NULL;
    DWORD cbData = 0;

    BYTE* pbNewVidInf  = NULL;
    DWORD cbNewVidInf  = 0;
    BYTE* pbNewPriv    = NULL;

    // Get the private data interface.
    hr = pDMO->QueryInterface(IID_IWMCodecPrivateData,
                              (void**)&pPrivData);
    GOTO_EXIT_IF_FAILED(hr);

    // Set the partial media type.
    hr = pPrivData->SetPartialOutputType(pMedia);
    GOTO_EXIT_IF_FAILED(hr);

    // Get the size of the private data.
    hr = pPrivData->GetPrivateData(NULL, &cbData);
    GOTO_EXIT_IF_FAILED(hr);

    // Allocate memory for the private data.
    pbData = new BYTE[cbData];
    if(pbData == NULL)
    {
        hr = E_OUTOFMEMORY;
        goto Exit:
    }

    // Get the private data.
    hr = pPrivData->GetPrivateData(pbData, &cbData);

    // Allocate memory for the new VIDEOINFOHEADER.
    cbNewVidInf = pMedia->cbFormat + cbData;
    pbNewVidInf = new BYTE[cbNewVidInf];

    // Copy the VIDEOINFOHEADER to the new buffer.
    memcpy((void*)pbNewVidInf, (void*)pMedia->pbFormat, pMedia->cbFormat);

    // Get the address of the first byte following the VIDEOINFOHEADER.
    pbNewPriv = pbNewVidInf + pMedia->cbFormat;

    // Copy the private data to the new buffer.
    memcpy((void*)pbNewPriv, (void*)pbData, cbData);

    // Set the new VIDEOINFOHEADER in the DMO_MEDIA_TYPE.
    pMedia->pbFormat = pbNewVidInf;
    pMedia->cbFormat = cbNewVidInf;

Exit:
    SAFE_RELEASE(pPrivData);
    SAFE_ARRAY_DELETE(pbData);
    pbNewPriv = NULL;
    return hr;
}

Примечание

Частные данные кодека, предоставляемые кодировщиком видео, не обязательно будут совпадать с частными данными, предоставляемыми другой реализацией одного и того же кодека для той же конфигурации. Это значение всегда необходимо создавать, используя действия, описанные в этом разделе. никогда не копируйте личные данные из другого файла.

 

Настройка кодирования видео

Работа с видео