Reuse buffers for video frames

At least on OS X, allocating the buffers is one of the more expensive
parts of video playback, and on an arbitrary 720p H.264 file with simple
subtitles this cuts CPU usage while playing by about 30%.
This commit is contained in:
Thomas Goyne 2014-06-12 14:34:21 -07:00
parent a574d6ac67
commit 4bdccb889c
12 changed files with 72 additions and 64 deletions

View file

@ -20,6 +20,7 @@
#include "ass_file.h" #include "ass_file.h"
#include "export_fixstyle.h" #include "export_fixstyle.h"
#include "include/aegisub/subtitles_provider.h" #include "include/aegisub/subtitles_provider.h"
#include "video_frame.h"
#include "video_provider_manager.h" #include "video_provider_manager.h"
#include <libaegisub/dispatch.h> #include <libaegisub/dispatch.h>
@ -30,10 +31,22 @@ enum {
}; };
std::shared_ptr<VideoFrame> AsyncVideoProvider::ProcFrame(int frame_number, double time, bool raw) { std::shared_ptr<VideoFrame> AsyncVideoProvider::ProcFrame(int frame_number, double time, bool raw) {
// Find an unused buffer to use or allocate a new one if needed
std::shared_ptr<VideoFrame> frame; std::shared_ptr<VideoFrame> frame;
for (auto& buffer : buffers) {
if (buffer.use_count() == 1) {
frame = buffer;
break;
}
}
if (!frame) {
frame = std::make_shared<VideoFrame>();
buffers.push_back(frame);
}
try { try {
frame = source_provider->GetFrame(frame_number); source_provider->GetFrame(frame_number, *frame);
} }
catch (VideoProviderError const& err) { throw VideoProviderErrorEvent(err); } catch (VideoProviderError const& err) { throw VideoProviderErrorEvent(err); }

View file

@ -76,6 +76,8 @@ class AsyncVideoProvider {
/// they can be rendered /// they can be rendered
std::atomic<uint_fast32_t> version{ 0 }; std::atomic<uint_fast32_t> version{ 0 };
std::vector<std::shared_ptr<VideoFrame>> buffers;
public: public:
/// @brief Load the passed subtitle file /// @brief Load the passed subtitle file
/// @param subs File to load /// @param subs File to load

View file

@ -37,7 +37,6 @@
#include <libaegisub/exception.h> #include <libaegisub/exception.h>
#include <libaegisub/vfr.h> #include <libaegisub/vfr.h>
#include <memory>
#include <string> #include <string>
struct VideoFrame; struct VideoFrame;
@ -47,7 +46,7 @@ public:
virtual ~VideoProvider() = default; virtual ~VideoProvider() = default;
/// Override this method to actually get frames /// Override this method to actually get frames
virtual std::shared_ptr<VideoFrame> GetFrame(int n)=0; virtual void GetFrame(int n, VideoFrame &frame)=0;
/// Set the YCbCr matrix to the specified one /// Set the YCbCr matrix to the specified one
/// ///

View file

@ -101,18 +101,19 @@ void SubtitlesPreview::SetColour(agi::Color col) {
void SubtitlesPreview::UpdateBitmap() { void SubtitlesPreview::UpdateBitmap() {
if (!vid) return; if (!vid) return;
auto frame = vid->GetFrame(0); VideoFrame frame;
vid->GetFrame(0, frame);
if (provider) { if (provider) {
try { try {
provider->LoadSubtitles(sub_file.get()); provider->LoadSubtitles(sub_file.get());
provider->DrawSubtitles(*frame, 0.1); provider->DrawSubtitles(frame, 0.1);
} }
catch (...) { } catch (...) { }
} }
// Convert frame to bitmap // Convert frame to bitmap
*bmp = static_cast<wxBitmap>(GetImage(*frame)); *bmp = static_cast<wxBitmap>(GetImage(frame));
Refresh(); Refresh();
} }

View file

@ -19,24 +19,6 @@
#include <boost/gil/gil_all.hpp> #include <boost/gil/gil_all.hpp>
#include <wx/image.h> #include <wx/image.h>
VideoFrame::VideoFrame(const unsigned char *data, size_t width, size_t height, size_t pitch, bool flipped)
: data(data, data + width * height * 4)
, width(width)
, height(height)
, pitch(pitch)
, flipped(flipped)
{
}
VideoFrame::VideoFrame(std::vector<unsigned char>&& data, size_t width, size_t height, size_t pitch, bool flipped)
: data(std::move(data))
, width(width)
, height(height)
, pitch(pitch)
, flipped(flipped)
{
}
namespace { namespace {
// We actually have bgr_, not bgra, so we need a custom converter which ignores the alpha channel // We actually have bgr_, not bgra, so we need a custom converter which ignores the alpha channel
struct color_converter { struct color_converter {

View file

@ -24,9 +24,6 @@ struct VideoFrame {
size_t height; size_t height;
size_t pitch; size_t pitch;
bool flipped; bool flipped;
VideoFrame(const unsigned char *data, size_t width, size_t height, size_t pitch, bool fipped);
VideoFrame(std::vector<unsigned char>&& data, size_t width, size_t height, size_t pitch, bool fipped);
}; };
wxImage GetImage(VideoFrame const& frame); wxImage GetImage(VideoFrame const& frame);

View file

@ -73,7 +73,7 @@ class AvisynthVideoProvider: public VideoProvider {
public: public:
AvisynthVideoProvider(agi::fs::path const& filename, std::string const& colormatrix); AvisynthVideoProvider(agi::fs::path const& filename, std::string const& colormatrix);
std::shared_ptr<VideoFrame> GetFrame(int n); void GetFrame(int n, VideoFrame &frame) override;
void SetColorSpace(std::string const& matrix) override { void SetColorSpace(std::string const& matrix) override {
// Can't really do anything if this fails // Can't really do anything if this fails
@ -309,11 +309,16 @@ AVSValue AvisynthVideoProvider::Open(agi::fs::path const& filename) {
throw VideoNotSupported("No function suitable for opening the video found"); throw VideoNotSupported("No function suitable for opening the video found");
} }
std::shared_ptr<VideoFrame> AvisynthVideoProvider::GetFrame(int n) { void AvisynthVideoProvider::GetFrame(int n, VideoFrame &out) {
std::lock_guard<std::mutex> lock(avs.GetMutex()); std::lock_guard<std::mutex> lock(avs.GetMutex());
auto frame = RGB32Video->GetFrame(n, avs.GetEnv()); auto frame = RGB32Video->GetFrame(n, avs.GetEnv());
return std::make_shared<VideoFrame>(frame->GetReadPtr(), frame->GetRowSize() / 4, frame->GetHeight(), frame->GetPitch(), true); auto ptr = frame->GetReadPtr();
out.data.assign(ptr, ptr + frame->GetPitch() * frame->GetHeight());
out.flipped = true;
out.height = frame->GetHeight();
out.width = frame->GetRowSize() / 4;
out.pitch = frame->GetPitch();
} }
} }

View file

@ -25,14 +25,14 @@
namespace { namespace {
/// A video frame and its frame number /// A video frame and its frame number
struct CachedFrame final : public VideoFrame { struct CachedFrame {
VideoFrame frame;
int frame_number; int frame_number;
CachedFrame(int frame_number, VideoFrame const& frame) CachedFrame(VideoFrame const& frame, int frame_number)
: VideoFrame(frame.data.data(), frame.width, frame.height, frame.pitch, frame.flipped) : frame(frame), frame_number(frame_number) { }
, frame_number(frame_number)
{ CachedFrame(CachedFrame const&) = delete;
}
}; };
/// @class VideoProviderCache /// @class VideoProviderCache
@ -45,19 +45,15 @@ class VideoProviderCache final : public VideoProvider {
/// ///
/// Note that this is a soft limit. The cache stops allocating new frames /// Note that this is a soft limit. The cache stops allocating new frames
/// once it has exceeded the limit, but it never tries to shrink /// once it has exceeded the limit, but it never tries to shrink
const size_t max_cache_size; const size_t max_cache_size = OPT_GET("Provider/Video/Cache/Size")->GetInt() << 20; // convert MB to bytes
/// Cache of video frames with the most recently used ones at the front /// Cache of video frames with the most recently used ones at the front
std::list<CachedFrame> cache; std::list<CachedFrame> cache;
public: public:
VideoProviderCache(std::unique_ptr<VideoProvider> master) VideoProviderCache(std::unique_ptr<VideoProvider> master) : master(std::move(master)) { }
: master(std::move(master))
, max_cache_size(OPT_GET("Provider/Video/Cache/Size")->GetInt() << 20) // convert MB to bytes
{
}
std::shared_ptr<VideoFrame> GetFrame(int n) override; void GetFrame(int n, VideoFrame &frame) override;
void SetColorSpace(std::string const& m) override { void SetColorSpace(std::string const& m) override {
cache.clear(); cache.clear();
@ -78,25 +74,28 @@ public:
bool HasAudio() const override { return master->HasAudio(); } bool HasAudio() const override { return master->HasAudio(); }
}; };
std::shared_ptr<VideoFrame> VideoProviderCache::GetFrame(int n) { void VideoProviderCache::GetFrame(int n, VideoFrame &out) {
size_t total_size = 0; size_t total_size = 0;
for (auto cur = cache.begin(); cur != cache.end(); ++cur) { for (auto cur = cache.begin(); cur != cache.end(); ++cur) {
if (cur->frame_number == n) { if (cur->frame_number == n) {
cache.splice(cache.begin(), cache, cur); // Move to front cache.splice(cache.begin(), cache, cur); // Move to front
return std::make_shared<VideoFrame>(cache.front()); out = cache.front().frame;
return;
} }
total_size += cur->data.size(); total_size += cur->frame.data.size();
} }
auto frame = master->GetFrame(n); master->GetFrame(n, out);
if (total_size >= max_cache_size) if (total_size >= max_cache_size) {
cache.pop_back(); cache.splice(cache.begin(), cache, --cache.end()); // Move last to front
cache.emplace_front(n, *frame); cache.front().frame_number = n;
cache.front().frame = out;
return frame; }
else
cache.emplace_front(out, n);
} }
} }

View file

@ -92,8 +92,12 @@ std::string DummyVideoProvider::MakeFilename(double fps, int frames, int width,
return agi::format("?dummy:%f:%d:%d:%d:%d:%d:%d:%s", fps, frames, width, height, (int)colour.r, (int)colour.g, (int)colour.b, (pattern ? "c" : "")); return agi::format("?dummy:%f:%d:%d:%d:%d:%d:%d:%s", fps, frames, width, height, (int)colour.r, (int)colour.g, (int)colour.b, (pattern ? "c" : ""));
} }
std::shared_ptr<VideoFrame> DummyVideoProvider::GetFrame(int) { void DummyVideoProvider::GetFrame(int, VideoFrame &frame) {
return std::make_shared<VideoFrame>(data.data(), width, height, width * 4, false); frame.data = data;
frame.width = width;
frame.height = height;
frame.pitch = width * 4;
frame.flipped = false;
} }
namespace agi { class BackgroundRunner; } namespace agi { class BackgroundRunner; }

View file

@ -64,7 +64,7 @@ public:
/// string will result in a video with the given parameters /// string will result in a video with the given parameters
static std::string MakeFilename(double fps, int frames, int width, int height, agi::Color colour, bool pattern); static std::string MakeFilename(double fps, int frames, int width, int height, agi::Color colour, bool pattern);
std::shared_ptr<VideoFrame> GetFrame(int n) override; void GetFrame(int n, VideoFrame &frame) override;
void SetColorSpace(std::string const&) override { } void SetColorSpace(std::string const&) override { }
int GetFrameCount() const override { return framecount; } int GetFrameCount() const override { return framecount; }

View file

@ -70,7 +70,7 @@ class FFmpegSourceVideoProvider final : public VideoProvider, FFmpegSourceProvid
public: public:
FFmpegSourceVideoProvider(agi::fs::path const& filename, std::string const& colormatrix, agi::BackgroundRunner *br); FFmpegSourceVideoProvider(agi::fs::path const& filename, std::string const& colormatrix, agi::BackgroundRunner *br);
std::shared_ptr<VideoFrame> GetFrame(int n) override; void GetFrame(int n, VideoFrame &out) override;
void SetColorSpace(std::string const& matrix) override { void SetColorSpace(std::string const& matrix) override {
#if FFMS_VERSION >= ((2 << 24) | (17 << 16) | (1 << 8) | 0) #if FFMS_VERSION >= ((2 << 24) | (17 << 16) | (1 << 8) | 0)
@ -285,14 +285,18 @@ void FFmpegSourceVideoProvider::LoadVideo(agi::fs::path const& filename, std::st
Timecodes = agi::vfr::Framerate(TimecodesVector); Timecodes = agi::vfr::Framerate(TimecodesVector);
} }
std::shared_ptr<VideoFrame> FFmpegSourceVideoProvider::GetFrame(int n) { void FFmpegSourceVideoProvider::GetFrame(int n, VideoFrame &out) {
n = mid(0, n, GetFrameCount() - 1); n = mid(0, n, GetFrameCount() - 1);
auto frame = FFMS_GetFrame(VideoSource, n, &ErrInfo); auto frame = FFMS_GetFrame(VideoSource, n, &ErrInfo);
if (!frame) if (!frame)
throw VideoDecodeError(std::string("Failed to retrieve frame: ") + ErrInfo.Buffer); throw VideoDecodeError(std::string("Failed to retrieve frame: ") + ErrInfo.Buffer);
return std::make_shared<VideoFrame>(frame->Data[0], Width, Height, frame->Linesize[0], false); out.data.assign(frame->Data[0], frame->Data[0] + frame->Linesize[0] * Height);
out.flipped = false;
out.width = Width;
out.height = Height;
out.pitch = frame->Linesize[0];
} }
} }

View file

@ -143,7 +143,7 @@ class YUV4MPEGVideoProvider final : public VideoProvider {
public: public:
YUV4MPEGVideoProvider(agi::fs::path const& filename); YUV4MPEGVideoProvider(agi::fs::path const& filename);
std::shared_ptr<VideoFrame> GetFrame(int n) override; void GetFrame(int n, VideoFrame &frame) override;
void SetColorSpace(std::string const&) override { } void SetColorSpace(std::string const&) override { }
int GetFrameCount() const override { return num_frames; } int GetFrameCount() const override { return num_frames; }
@ -391,7 +391,7 @@ int YUV4MPEGVideoProvider::IndexFile(uint64_t pos) {
return framecount; return framecount;
} }
std::shared_ptr<VideoFrame> YUV4MPEGVideoProvider::GetFrame(int n) { void YUV4MPEGVideoProvider::GetFrame(int n, VideoFrame &frame) {
n = mid(0, n, num_frames - 1); n = mid(0, n, num_frames - 1);
int uv_width = w / 2; int uv_width = w / 2;
@ -408,9 +408,8 @@ std::shared_ptr<VideoFrame> YUV4MPEGVideoProvider::GetFrame(int n) {
auto src_y = reinterpret_cast<const unsigned char *>(file.read(seek_table[n], luma_sz + chroma_sz * 2)); auto src_y = reinterpret_cast<const unsigned char *>(file.read(seek_table[n], luma_sz + chroma_sz * 2));
auto src_u = src_y + luma_sz; auto src_u = src_y + luma_sz;
auto src_v = src_u + chroma_sz; auto src_v = src_u + chroma_sz;
std::vector<unsigned char> data; frame.data.resize(w * h * 4);
data.resize(w * h * 4); unsigned char *dst = &frame.data[0];
unsigned char *dst = &data[0];
for (int py = 0; py < h; ++py) { for (int py = 0; py < h; ++py) {
for (int px = 0; px < w / 2; ++px) { for (int px = 0; px < w / 2; ++px) {
@ -433,7 +432,10 @@ std::shared_ptr<VideoFrame> YUV4MPEGVideoProvider::GetFrame(int n) {
} }
} }
return std::make_shared<VideoFrame>(std::move(data), w, h, w * 4, false); frame.flipped = false;
frame.width = w;
frame.height = h;
frame.pitch = w * 4;
} }
} }