Nota
O acesso a esta página requer autorização. Podes tentar iniciar sessão ou mudar de diretório.
O acesso a esta página requer autorização. Podes tentar mudar de diretório.
A saída compactada produzida pelos codecs do Windows Media Video 9 não pode ser descompactada corretamente sem alguns dados fornecidos pelo codificador. Esses dados, chamados dados privados do codec, devem ser anexados ao tipo de mídia de saída. Você pode obter os dados privados do codec chamando os métodos do IWMCodecPrivateData interface. Passe a estrutura DMO_MEDIA_TYPE completa para IWMCodecPrivateData::SetPartialOutputType. Em seguida, chame IWMCodecPrivateData::GetPrivateData duas vezes, uma para obter o tamanho dos dados e, em seguida, novamente para copiar os dados para um buffer desse tamanho. Crie um novo buffer para conter a estrutura VIDEOINFOHEADER com os dados privados anexados, e copie a estrutura e os dados para esse buffer. Finalmente, defina o membro pbFormat da estrutura DMO_MEDIA_TYPE para o endereço do buffer recém-criado e defina o membro cbFormat para o tamanho combinado, em bytes, do VIDEOINFOHEADER e dos dados privados.
Se você estiver usando o MediaFoundation, poderá construir uma estrutura de DMO_MEDIA_TYPE a partir de uma interfaceIMFMediaTypechamando MFCreateAMMediaTypeFromMFMediaType.
Você deve usar os dados privados do codec obtidos após a primeira configuração das propriedades no codificador. Se alguma propriedade for alterada, você deverá obter novos dados privados. Se não utilizar os dados privados obtidos depois de todas as propriedades estarem definidas para a sessão de codificação, o descodificador poderá não conseguir descomprimir os dados.
O exemplo de código a seguir demonstra como obter os dados privados para um tipo de vídeo:
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;
}
Observação
Não é garantido que os dados privados do codec fornecidos por um codificador de vídeo sejam os mesmos que os dados privados fornecidos por uma implementação diferente do mesmo codec para a mesma configuração. Você sempre deve gerar esse valor usando as etapas neste tópico; Nunca copie os dados privados de outro ficheiro.
Tópicos relacionados