Skip to content

Commit

Permalink
Add YUVA support
Browse files Browse the repository at this point in the history
EIREXE committed Oct 26, 2024
1 parent 507507e commit 49efa2a
Showing 7 changed files with 153 additions and 242 deletions.
37 changes: 27 additions & 10 deletions ffmpeg_video_stream.cpp
Original file line number Diff line number Diff line change
@@ -192,10 +192,11 @@ void FFmpegVideoStreamPlayback::update_internal(double p_delta) {
#ifndef FFMPEG_MT_GPU_UPLOAD
if (got_new_frame) {
// YUV conversion
if (last_frame->get_format() == FFmpegFrameFormat::YUV420P) {
if (last_frame->get_format() == FFmpegFrameFormat::YUV420P || last_frame->get_format() == FFmpegFrameFormat::YUVA420P) {
Ref<Image> y_plane = last_frame->get_yuv_image_plane(0);
Ref<Image> u_plane = last_frame->get_yuv_image_plane(1);
Ref<Image> v_plane = last_frame->get_yuv_image_plane(2);
Ref<Image> a_plane = last_frame->get_yuv_image_plane(3);

ERR_FAIL_COND(!y_plane.is_valid());
ERR_FAIL_COND(!u_plane.is_valid());
@@ -204,6 +205,7 @@ void FFmpegVideoStreamPlayback::update_internal(double p_delta) {
yuv_converter->set_plane_image(0, y_plane);
yuv_converter->set_plane_image(1, u_plane);
yuv_converter->set_plane_image(2, v_plane);
yuv_converter->set_plane_image(3, a_plane);
yuv_converter->convert();
// RGBA texture handling
} else if (texture.is_valid()) {
@@ -279,7 +281,7 @@ Error FFmpegVideoStreamPlayback::load(Ref<FileAccess> p_file_access) {
return FAILED;
}

if (decoder->get_frame_format() == FFmpegFrameFormat::YUV420P) {
if (decoder->get_frame_format() == FFmpegFrameFormat::YUV420P || decoder->get_frame_format() == FFmpegFrameFormat::YUVA420P) {
yuv_converter.instantiate();
yuv_converter->set_frame_size(size);
yuv_texture = yuv_converter->get_output_texture();
@@ -433,8 +435,8 @@ Error YUVGPUConverter::_ensure_plane_textures() {
if (yuv_plane_textures[i].is_valid()) {
RDTextureFormatC format = TEXTURE_FORMAT_COMPAT(rd->texture_get_format(yuv_plane_textures[i]));

int desired_frame_width = i == 0 ? frame_size.width : Math::ceil(frame_size.width / 2.0f);
int desired_frame_height = i == 0 ? frame_size.height : Math::ceil(frame_size.height / 2.0f);
int desired_frame_width = i == 0 || i == 3 ? frame_size.width : Math::ceil(frame_size.width / 2.0f);
int desired_frame_height = i == 0 || i == 3 ? frame_size.height : Math::ceil(frame_size.height / 2.0f);

if (static_cast<int>(format.width) == desired_frame_width && static_cast<int>(format.height) == desired_frame_height) {
continue;
@@ -452,8 +454,8 @@ Error YUVGPUConverter::_ensure_plane_textures() {
RDTextureFormatC new_format;
new_format.format = RenderingDevice::DATA_FORMAT_R8_UNORM;
// chroma planes are half the size of the luma plane
new_format.width = i == 0 ? frame_size.width : Math::ceil(frame_size.width / 2.0f);
new_format.height = i == 0 ? frame_size.height : Math::ceil(frame_size.height / 2.0f);
new_format.width = i == 0 || i == 3 ? frame_size.width : Math::ceil(frame_size.width / 2.0f);
new_format.height = i == 0 || i == 3 ? frame_size.height : Math::ceil(frame_size.height / 2.0f);
new_format.depth = 1;
new_format.array_layers = 1;
new_format.mipmaps = 1;
@@ -547,17 +549,24 @@ RID YUVGPUConverter::_create_uniform_set(const RID &p_texture_rd_rid) {

void YUVGPUConverter::_upload_plane_images() {
for (size_t i = 0; i < std::size(yuv_plane_images); i++) {
ERR_CONTINUE_MSG(!yuv_plane_images[i].is_valid(), vformat("YUV plane %d was missing, cannot upload texture data.", (int)i));
ERR_CONTINUE_MSG(!yuv_plane_images[i].is_valid() && i != 3, vformat("YUV plane %d was missing, cannot upload texture data.", (int)i));
if (!yuv_plane_images[i].is_valid()) {
continue;
}
RS::get_singleton()->get_rendering_device()->texture_update(yuv_plane_textures[i], 0, yuv_plane_images[i]->get_data());
}
}

void YUVGPUConverter::set_plane_image(int p_plane_idx, Ref<Image> p_image) {
if (!p_image.is_valid()) {
yuv_plane_images[p_plane_idx] = p_image;
return;
}
ERR_FAIL_COND(!p_image.is_valid());
ERR_FAIL_INDEX((size_t)p_plane_idx, std::size(yuv_plane_images));
// Sanity checks
int desired_frame_width = p_plane_idx == 0 ? frame_size.width : Math::ceil(frame_size.width / 2.0f);
int desired_frame_height = p_plane_idx == 0 ? frame_size.height : Math::ceil(frame_size.height / 2.0f);
int desired_frame_width = p_plane_idx == 0 || p_plane_idx == 3 ? frame_size.width : Math::ceil(frame_size.width / 2.0f);
int desired_frame_height = p_plane_idx == 0 || p_plane_idx == 3 ? frame_size.height : Math::ceil(frame_size.height / 2.0f);
ERR_FAIL_COND_MSG(p_image->get_width() != desired_frame_width, vformat("Wrong YUV plane width for plane %d, expected %d got %d", p_plane_idx, desired_frame_width, p_image->get_width()));
ERR_FAIL_COND_MSG(p_image->get_height() != desired_frame_height, vformat("Wrong YUV plane height for plane %, expected %d got %d", p_plane_idx, desired_frame_height, p_image->get_height()));
ERR_FAIL_COND_MSG(p_image->get_format() != Image::FORMAT_R8, "Wrong image format, expected R8");
@@ -585,12 +594,20 @@ void YUVGPUConverter::convert() {

RD *rd = RS::get_singleton()->get_rendering_device();

push_constant.use_alpha = yuv_plane_images[3].is_valid();

PackedByteArray push_constant_data;
push_constant_data.resize(sizeof(push_constant));
memcpy(push_constant_data.ptrw(), &push_constant, push_constant_data.size());

ComputeListID compute_list = rd->compute_list_begin();
rd->compute_list_bind_compute_pipeline(compute_list, pipeline);
rd->compute_list_set_push_constant(compute_list, push_constant_data, push_constant_data.size());
rd->compute_list_bind_uniform_set(compute_list, yuv_planes_uniform_sets[0], 0);
rd->compute_list_bind_uniform_set(compute_list, yuv_planes_uniform_sets[1], 1);
rd->compute_list_bind_uniform_set(compute_list, yuv_planes_uniform_sets[2], 2);
rd->compute_list_bind_uniform_set(compute_list, out_uniform_set, 3);
rd->compute_list_bind_uniform_set(compute_list, yuv_planes_uniform_sets[3], 3);
rd->compute_list_bind_uniform_set(compute_list, out_uniform_set, 4);
rd->compute_list_dispatch(compute_list, Math::ceil(frame_size.x / 8.0f), Math::ceil(frame_size.y / 8.0f), 1);
rd->compute_list_end();
}
21 changes: 13 additions & 8 deletions ffmpeg_video_stream.h
Original file line number Diff line number Diff line change
@@ -28,8 +28,8 @@
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
/**************************************************************************/

#ifndef ET_VIDEO_STREAM_H
#define ET_VIDEO_STREAM_H
#ifndef FFMPEG_VIDEO_STREAM_H
#define FFMPEG_VIDEO_STREAM_H

#ifdef GDEXTENSION

@@ -56,14 +56,19 @@ using namespace godot;

class YUVGPUConverter : public RefCounted {
RID shader;
Ref<Image> yuv_plane_images[3];
RID yuv_plane_textures[3];
RID yuv_planes_uniform_sets[3];
Ref<Image> yuv_plane_images[4];
RID yuv_plane_textures[4];
RID yuv_planes_uniform_sets[4];
RID pipeline;
Ref<Texture2DRD> out_texture;
RID out_uniform_set;
Vector2i frame_size;

struct PushConstant {
uint8_t use_alpha;
uint8_t padding[15];
} push_constant;

private:
void _ensure_pipeline();
Error _ensure_plane_textures();
@@ -129,7 +134,7 @@ class FFmpegVideoStreamPlayback : public VideoStreamPlayback {

protected:
void clear();
static void _bind_methods(){}; // Required by GDExtension, do not remove
static void _bind_methods() {}; // Required by GDExtension, do not remove

public:
Error load(Ref<FileAccess> p_file_access);
@@ -153,7 +158,7 @@ class FFmpegVideoStream : public VideoStream {
GDCLASS(FFmpegVideoStream, VideoStream);

protected:
static void _bind_methods(){}; // Required by GDExtension, do not remove
static void _bind_methods() {}; // Required by GDExtension, do not remove
Ref<VideoStreamPlayback> instantiate_playback_internal() {
Ref<FileAccess> fa = FileAccess::open(get_file(), FileAccess::READ);
if (!fa.is_valid()) {
@@ -171,4 +176,4 @@ class FFmpegVideoStream : public VideoStream {
STREAM_FUNC_REDIRECT_0(Ref<VideoStreamPlayback>, instantiate_playback);
};

#endif // ET_VIDEO_STREAM_H
#endif // FFMPEG_VIDEO_STREAM_H
5 changes: 4 additions & 1 deletion gdextension_build/SConstruct
Original file line number Diff line number Diff line change
@@ -107,12 +107,15 @@ else:
source=sources,
)


def install_ffmpeg_libname(target, source, env):
methods.osx_rename_libname("^@.*framework/", "@loader_path/", "", [lib.get_path() for lib in target])


def install_gdextension_libname(target, source, env):
methods.osx_rename_libname("^@.*framework/", "@loader_path/../", "", [lib.get_path() for lib in target])


ffmpeg_install_action = ffmpeg_download.ffmpeg_install(env, f"#{addon_platform_dir}", ffmpeg_path)
env.Depends(sources, env.Install(addon_base_dir, "ffmpeg.gdextension"))
env.Depends(sources, ffmpeg_install_action)
@@ -122,7 +125,7 @@ env.Depends(Glob("../*.glsl.gen.h"), ["#glsl_builders.py"])

Default(library)

if env['platform'] == 'macos':
if env["platform"] == "macos":
env.AddPostAction(ffmpeg_install_action, env.Action(install_ffmpeg_libname))
env.AddPostAction(library, env.Action(install_gdextension_libname))

17 changes: 9 additions & 8 deletions gdextension_build/methods.py
Original file line number Diff line number Diff line change
@@ -196,20 +196,21 @@ def get_soname(filename):
else:
return ""


def osx_rename_libname(pattern, replacement, tool_prefix, filenames):
otool = tool_prefix + 'otool'
install_name_tool = tool_prefix + 'install_name_tool'
otool = tool_prefix + "otool"
install_name_tool = tool_prefix + "install_name_tool"
for filename in filenames:
data = str(subprocess.check_output([otool, '-L', filename]).decode('utf-8')).strip()
val = map(lambda x: x[0], map(str.split, map(str.strip, data.strip().split('\n'))))
data = str(subprocess.check_output([otool, "-L", filename]).decode("utf-8")).strip()
val = map(lambda x: x[0], map(str.split, map(str.strip, data.strip().split("\n"))))
val = list(val)[2:]

to_change = {}
for path in val:
if re.findall(pattern, path):
new_path = re.sub(pattern, replacement, path)
to_change[path] = new_path if new_path.endswith('.dylib') else new_path + '.dylib'
to_change[path] = new_path if new_path.endswith(".dylib") else new_path + ".dylib"

for k,v in to_change.items():
print(k, v, sep=' -> ')
subprocess.call([install_name_tool, '-change', k, v, filename])
for k, v in to_change.items():
print(k, v, sep=" -> ")
subprocess.call([install_name_tool, "-change", k, v, filename])
Loading

0 comments on commit 49efa2a

Please sign in to comment.