-
Notifications
You must be signed in to change notification settings - Fork 114
Improving Videoencoder #395
base: master
Are you sure you want to change the base?
Changes from 8 commits
c970780
4432731
b54c641
9cc0fbb
9104cf4
09f62de
c76e00f
cae0f9f
83afc70
fc8a8ea
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -6,6 +6,8 @@ | |
|
||
#include "codecapi.h" | ||
|
||
#define NUM_VIDEO_BUFFERS 10 | ||
|
||
VideoEncoder::VideoEncoder(UINT frameWidth, UINT frameHeight, UINT frameStride, UINT fps, | ||
UINT32 audioSampleRate, UINT32 audioChannels, UINT32 audioBPS, UINT32 videoBitrate, UINT32 videoMpegLevel) : | ||
frameWidth(frameWidth), | ||
|
@@ -21,10 +23,11 @@ VideoEncoder::VideoEncoder(UINT frameWidth, UINT frameHeight, UINT frameStride, | |
isRecording(false) | ||
{ | ||
#if HARDWARE_ENCODE_VIDEO | ||
inputFormat = MFVideoFormat_NV12; | ||
inputFormat = MFVideoFormat_NV12; | ||
#else | ||
inputFormat = MFVideoFormat_RGB32; | ||
inputFormat = MFVideoFormat_RGB32; | ||
#endif | ||
inputFormat = MFVideoFormat_RGB32; | ||
} | ||
|
||
VideoEncoder::~VideoEncoder() | ||
|
@@ -41,11 +44,26 @@ bool VideoEncoder::Initialize(ID3D11Device* device) | |
|
||
#if HARDWARE_ENCODE_VIDEO | ||
MFCreateDXGIDeviceManager(&resetToken, &deviceManager); | ||
this->device = device; | ||
|
||
if (deviceManager != nullptr) | ||
{ | ||
OutputDebugString(L"Resetting device manager with graphics device.\n"); | ||
deviceManager->ResetDevice(device, resetToken); | ||
hr = deviceManager->ResetDevice(device, resetToken); | ||
} | ||
for (int i = 0; i < NUM_VIDEO_BUFFERS; i++) | ||
{ | ||
videoInputPool.push(std::make_unique<VideoInput>(device)); | ||
} | ||
|
||
ID3D10Multithread* multithread; | ||
device->QueryInterface(&multithread); | ||
multithread->SetMultithreadProtected(TRUE); | ||
|
||
#else | ||
for (int i = 0; i < NUM_VIDEO_BUFFERS; i++) | ||
{ | ||
videoInputPool.push(std::make_unique<VideoInput>(frameHeight * frameStride)); | ||
} | ||
#endif | ||
|
||
|
@@ -72,7 +90,7 @@ void VideoEncoder::StartRecording(LPCWSTR videoPath, bool encodeAudio) | |
prevVideoTime = INVALID_TIMESTAMP; | ||
prevAudioTime = INVALID_TIMESTAMP; | ||
|
||
HRESULT hr = E_PENDING; | ||
HRESULT hr = S_OK; | ||
|
||
sinkWriter = NULL; | ||
videoStreamIndex = MAXDWORD; | ||
|
@@ -87,13 +105,14 @@ void VideoEncoder::StartRecording(LPCWSTR videoPath, bool encodeAudio) | |
#endif | ||
|
||
IMFAttributes *attr = nullptr; | ||
MFCreateAttributes(&attr, 3); | ||
MFCreateAttributes(&attr, 4); | ||
|
||
if (SUCCEEDED(hr)) { hr = attr->SetUINT32(MF_SINK_WRITER_DISABLE_THROTTLING, TRUE); } | ||
|
||
#if HARDWARE_ENCODE_VIDEO | ||
if (SUCCEEDED(hr)) { hr = attr->SetUINT32(MF_READWRITE_ENABLE_HARDWARE_TRANSFORMS, true); } | ||
if (SUCCEEDED(hr)) { hr = attr->SetUINT32(MF_READWRITE_DISABLE_CONVERTERS, false); } | ||
if (SUCCEEDED(hr)) { hr = attr->SetUnknown(MF_SINK_WRITER_D3D_MANAGER, deviceManager); } | ||
#endif | ||
|
||
hr = MFCreateSinkWriterFromURL(videoPath, NULL, attr, &sinkWriter); | ||
|
@@ -138,6 +157,10 @@ void VideoEncoder::StartRecording(LPCWSTR videoPath, bool encodeAudio) | |
if (SUCCEEDED(hr)) { hr = MFSetAttributeSize(pVideoTypeIn, MF_MT_FRAME_SIZE, frameWidth, frameHeight); } | ||
if (SUCCEEDED(hr)) { hr = MFSetAttributeRatio(pVideoTypeIn, MF_MT_FRAME_RATE, fps, 1); } | ||
if (SUCCEEDED(hr)) { hr = MFSetAttributeRatio(pVideoTypeIn, MF_MT_PIXEL_ASPECT_RATIO, 1, 1); } | ||
if (SUCCEEDED(hr)) { hr = pVideoTypeIn->SetUINT32(MF_MT_ALL_SAMPLES_INDEPENDENT, TRUE); } | ||
if (SUCCEEDED(hr)) { hr = pVideoTypeIn->SetUINT32(MF_MT_DEFAULT_STRIDE, frameStride); } | ||
if (SUCCEEDED(hr)) { hr = pVideoTypeIn->SetUINT32(MF_MT_FIXED_SIZE_SAMPLES, TRUE); } | ||
if (SUCCEEDED(hr)) { hr = pVideoTypeIn->SetUINT32(MF_MT_SAMPLE_SIZE, frameStride * frameHeight); } | ||
if (SUCCEEDED(hr)) { hr = sinkWriter->SetInputMediaType(videoStreamIndex, pVideoTypeIn, NULL); } | ||
|
||
if (encodeAudio) | ||
|
@@ -284,12 +307,12 @@ void VideoEncoder::WriteAudio(byte* buffer, int bufferSize, LONGLONG timestamp) | |
#endif | ||
} | ||
|
||
void VideoEncoder::WriteVideo(byte* buffer, LONGLONG timestamp, LONGLONG duration) | ||
void VideoEncoder::WriteVideo(std::unique_ptr<VideoEncoder::VideoInput> frame) | ||
{ | ||
std::shared_lock<std::shared_mutex> lock(videoStateLock); | ||
#if _DEBUG | ||
{ | ||
std::wstring debugString = L"Writing Video, Timestamp:" + std::to_wstring(timestamp) + L"\n"; | ||
std::wstring debugString = L"Writing Video, Timestamp:" + std::to_wstring(frame->timestamp) + L"\n"; | ||
OutputDebugString(debugString.data()); | ||
} | ||
#endif | ||
|
@@ -302,126 +325,93 @@ void VideoEncoder::WriteVideo(byte* buffer, LONGLONG timestamp, LONGLONG duratio | |
|
||
if (startTime == INVALID_TIMESTAMP) | ||
{ | ||
startTime = timestamp; | ||
startTime = frame->timestamp; | ||
#if _DEBUG | ||
std::wstring debugString = L"Start time set from video, Timestamp:" + std::to_wstring(timestamp) + L", StartTime:" + std::to_wstring(startTime) + L"\n"; | ||
std::wstring debugString = L"Start time set from video, Timestamp:" + std::to_wstring(frame->timestamp) + L", StartTime:" + std::to_wstring(startTime) + L"\n"; | ||
OutputDebugString(debugString.data()); | ||
#endif | ||
} | ||
else if (timestamp < startTime) | ||
else if (frame->timestamp < startTime) | ||
{ | ||
#if _DEBUG | ||
std::wstring debugString = L"Video not recorded, Timestamp less than start time. Timestamp:" + std::to_wstring(timestamp) + L", StartTime:" + std::to_wstring(startTime) + L"\n"; | ||
std::wstring debugString = L"Video not recorded, Timestamp less than start time. Timestamp:" + std::to_wstring(frame->timestamp) + L", StartTime:" + std::to_wstring(startTime) + L"\n"; | ||
OutputDebugString(debugString.data()); | ||
#endif | ||
return; | ||
} | ||
|
||
if (timestamp == prevVideoTime) | ||
if (frame->timestamp == prevVideoTime) | ||
{ | ||
#if _DEBUG | ||
std::wstring debugString = L"Video not recorded, Timestamp equals prevVideoTime. Timestamp:" + std::to_wstring(timestamp) + L", StartTime:" + std::to_wstring(prevVideoTime) + L"\n"; | ||
std::wstring debugString = L"Video not recorded, Timestamp equals prevVideoTime. Timestamp:" + std::to_wstring(frame->timestamp) + L", StartTime:" + std::to_wstring(prevVideoTime) + L"\n"; | ||
OutputDebugString(debugString.data()); | ||
#endif | ||
return; | ||
} | ||
|
||
LONGLONG sampleTimeNow = timestamp; | ||
LONGLONG sampleTimeNow = frame->timestamp; | ||
LONGLONG sampleTimeStart = startTime; | ||
|
||
LONGLONG sampleTime = sampleTimeNow - sampleTimeStart; | ||
|
||
if (prevVideoTime != INVALID_TIMESTAMP) | ||
{ | ||
duration = sampleTime - prevVideoTime; | ||
frame->duration = sampleTime - prevVideoTime; | ||
#if _DEBUG | ||
std::wstring debugString = L"Updated write video duration:" + std::to_wstring(duration) + L", SampleTime:" + std::to_wstring(sampleTime) + L", PrevVideoTime:" + std::to_wstring(prevVideoTime) + L"\n"; | ||
std::wstring debugString = L"Updated write video duration:" + std::to_wstring(frame->duration) + L", SampleTime:" + std::to_wstring(sampleTime) + L", PrevVideoTime:" + std::to_wstring(prevVideoTime) + L"\n"; | ||
OutputDebugString(debugString.data()); | ||
#endif | ||
} | ||
|
||
// Copy frame to a temporary buffer and process on a background thread. | ||
#if HARDWARE_ENCODE_VIDEO | ||
BYTE* tmpVideoBuffer = new BYTE[(int)(FRAME_BPP_NV12 * frameHeight * frameWidth)]; | ||
memcpy(tmpVideoBuffer, buffer, (int)(FRAME_BPP_NV12 * frameHeight * frameWidth)); | ||
#else | ||
BYTE* tmpVideoBuffer = new BYTE[frameHeight * frameStride]; | ||
memcpy(tmpVideoBuffer, buffer, frameHeight * frameStride); | ||
#endif | ||
|
||
concurrency::create_task([=]() | ||
videoWriteFuture = std::async(std::launch::async, [=, frame{ std::move(frame) }, previousWriteFuture{ std::move(videoWriteFuture) }]() mutable | ||
{ | ||
if (previousWriteFuture.valid()) | ||
{ | ||
previousWriteFuture.wait(); | ||
previousWriteFuture = {}; | ||
} | ||
std::shared_lock<std::shared_mutex> lock(videoStateLock); | ||
|
||
HRESULT hr = E_PENDING; | ||
HRESULT hr = S_OK; | ||
if (sinkWriter == NULL || !isRecording) | ||
{ | ||
OutputDebugString(L"Must start recording before writing video frames.\n"); | ||
delete[] tmpVideoBuffer; | ||
return; | ||
} | ||
|
||
LONG cbWidth = frameStride; | ||
DWORD cbBuffer = cbWidth * frameHeight; | ||
DWORD imageHeight = frameHeight; | ||
|
||
#if HARDWARE_ENCODE_VIDEO | ||
cbWidth = frameWidth; | ||
cbBuffer = (int)(FRAME_BPP_NV12 * frameWidth * frameHeight); | ||
imageHeight = (int)(FRAME_BPP_NV12 * frameHeight); | ||
#endif | ||
|
||
DWORD cbBuffer = frameStride * frameHeight; | ||
IMFSample* pVideoSample = NULL; | ||
IMFMediaBuffer* pVideoBuffer = NULL; | ||
BYTE* pData = NULL; | ||
|
||
// Create a new memory buffer. | ||
hr = MFCreateMemoryBuffer(cbBuffer, &pVideoBuffer); | ||
|
||
// Lock the buffer and copy the video frame to the buffer. | ||
if (SUCCEEDED(hr)) { hr = pVideoBuffer->Lock(&pData, NULL, NULL); } | ||
|
||
if (SUCCEEDED(hr)) | ||
{ | ||
//TODO: Can pVideoBuffer be created from an ID3D11Texture2D*? | ||
hr = MFCopyImage( | ||
pData, // Destination buffer. | ||
cbWidth, // Destination stride. | ||
tmpVideoBuffer, | ||
cbWidth, // Source stride. | ||
cbWidth, // Image width in bytes. | ||
imageHeight // Image height in pixels. | ||
); | ||
} | ||
|
||
if (pVideoBuffer) | ||
#if _DEBUG | ||
{ | ||
pVideoBuffer->Unlock(); | ||
std::wstring debugString = L"Writing Video Sample, SampleTime:" + std::to_wstring(sampleTime) + L", SampleDuration:" + std::to_wstring(frame->duration) + L", BufferLength:" + std::to_wstring(cbBuffer) + L"\n"; | ||
OutputDebugString(debugString.data()); | ||
} | ||
#endif | ||
|
||
#if _DEBUG | ||
{ | ||
std::wstring debugString = L"Writing Video Sample, SampleTime:" + std::to_wstring(sampleTime) + L", SampleDuration:" + std::to_wstring(duration) + L", BufferLength:" + std::to_wstring(cbBuffer) + L"\n"; | ||
OutputDebugString(debugString.data()); | ||
} | ||
#if !HARDWARE_ENCODE_VIDEO | ||
// In case the user locks the frame but forgets to unlock | ||
frame->Unlock(); | ||
#endif | ||
hermann-noll marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
// Set the data length of the buffer. | ||
if (SUCCEEDED(hr)) { hr = pVideoBuffer->SetCurrentLength(cbBuffer); } | ||
if (SUCCEEDED(hr)) { hr = frame->mediaBuffer->SetCurrentLength(frameHeight * frameStride); } | ||
|
||
// Create a media sample and add the buffer to the sample. | ||
if (SUCCEEDED(hr)) { hr = MFCreateSample(&pVideoSample); } | ||
if (SUCCEEDED(hr)) { hr = pVideoSample->AddBuffer(pVideoBuffer); } | ||
if (SUCCEEDED(hr)) { hr = pVideoSample->AddBuffer(frame->mediaBuffer); } | ||
|
||
if (SUCCEEDED(hr)) { hr = pVideoSample->SetSampleTime(sampleTime); } //100-nanosecond units | ||
if (SUCCEEDED(hr)) { hr = pVideoSample->SetSampleDuration(duration); } //100-nanosecond units | ||
if (SUCCEEDED(hr)) { hr = pVideoSample->SetSampleDuration(frame->duration); } //100-nanosecond units | ||
|
||
// Send the sample to the Sink Writer. | ||
if (SUCCEEDED(hr)) { hr = sinkWriter->WriteSample(videoStreamIndex, pVideoSample); } | ||
|
||
SafeRelease(pVideoSample); | ||
SafeRelease(pVideoBuffer); | ||
delete[] tmpVideoBuffer; | ||
|
||
{ | ||
std::shared_lock<std::shared_mutex>(videoInputPoolLock); | ||
videoInputPool.push(std::move(frame)); | ||
} | ||
|
||
if (FAILED(hr)) | ||
{ | ||
|
@@ -456,6 +446,11 @@ void VideoEncoder::StopRecording() | |
|
||
concurrency::create_task([&] | ||
{ | ||
if (videoWriteFuture.valid()) | ||
{ | ||
videoWriteFuture.wait(); | ||
videoWriteFuture = {}; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Its inconsistent, but i have seen an exception throw during this cleanup process for debug flavors of the compositor dll when running locally. I have been using the split channels recording option There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. specifically what looks like a stack overflow when setting this videoWriteFuture to an empty struct In reply to: 443834338 [](ancestors = 443834338) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'll look into reproducing that exception, I haven't experienced any stack overflows before with the current changes. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. A very nice little bug, caused by There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. that does sound quite nice. :) In general, would you like me to test this out again locally so we can look into getting these changes in? Or would you prefer to continue looking into the TODO items? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I would prefer a bit more work done to it, there are still some unclean parts to it. Nevertheless it would be great to know whether you guys have a dependency on the RGB->NV12 color conversion? (e.g. idk whether it works on Intel/AMD hardware encoder) EDIT: Testing-wise with the current state I could no longer reproduce any stuttering caused by recording, although definitely old frame insertions can be observed on very high load (e.g. I used artificial load on the CPU/GPU for repro) but this is caused by the unguarded ring buffers in both the k4a and Decklink frame providers and should be dealt with in separate issues (workaround is setting the ring size to 100). |
||
} | ||
while (!videoQueue.empty()) | ||
{ | ||
videoQueue.pop(); | ||
|
@@ -508,17 +503,36 @@ void VideoEncoder::StopRecording() | |
SafeRelease(sinkWriter); | ||
} | ||
|
||
void VideoEncoder::QueueVideoFrame(byte* buffer, LONGLONG timestamp, LONGLONG duration) | ||
std::unique_ptr<VideoEncoder::VideoInput> VideoEncoder::GetAvailableVideoFrame() | ||
{ | ||
std::shared_lock<std::shared_mutex> lock(videoInputPoolLock); | ||
if (videoInputPool.empty()) | ||
{ | ||
#if HARDWARE_ENCODE_VIDEO | ||
return std::make_unique<VideoInput>(device); | ||
#else | ||
return std::make_unique<VideoInput>(frameStride * frameHeight); | ||
#endif | ||
} | ||
else | ||
{ | ||
auto result = std::move(videoInputPool.front()); | ||
videoInputPool.pop(); | ||
return result; | ||
} | ||
} | ||
|
||
void VideoEncoder::QueueVideoFrame(std::unique_ptr<VideoEncoder::VideoInput> frame) | ||
{ | ||
std::shared_lock<std::shared_mutex> lock(videoStateLock); | ||
|
||
if (acceptQueuedFrames) | ||
{ | ||
videoQueue.push(VideoInput(buffer, timestamp, duration)); | ||
#if _DEBUG | ||
std::wstring debugString = L"Pushed Video Input, Timestamp:" + std::to_wstring(timestamp) + L"\n"; | ||
OutputDebugString(debugString.data()); | ||
std::wstring debugString = L"Pushed Video Input, Timestamp:" + std::to_wstring(frame->timestamp) + L"\n"; | ||
OutputDebugString(debugString.data()); | ||
#endif | ||
videoQueue.push(std::move(frame)); | ||
} | ||
} | ||
|
||
|
@@ -548,8 +562,7 @@ void VideoEncoder::Update() | |
{ | ||
if (isRecording) | ||
{ | ||
VideoInput input = videoQueue.front(); | ||
WriteVideo(input.sharedBuffer, input.timestamp, input.duration); | ||
WriteVideo(std::move(videoQueue.front())); | ||
videoQueue.pop(); | ||
} | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Does this change mean that hardware encoding succeeds using RGB32?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For me it only worked with RGB32, I guess this is because the converted texture is still reported to D3D as being RGB. While it gets the information that the input is supposed to be NV12 with a therefore smaller buffer.
MF seems to be confused and writing samples failed with MF_E_BUFFERTOOSMALL.