cuda: Rewrite colorspace/rescale object

Rewriting GstCudaConverter object, since the old implementation was not
well organized and it's hard to add new features.
Moreover, the conversion operations were not very optimized.

Major change of this implementation:
* Remove redundant intermediate conversion operations such as
  any RGB -> ARGB(64) conversion or any YUV -> Y444 (or 16bits Y444).
  That's not required most of cases. The only required case is
  converting 24bits (such as RGB/BGR) packed format to 32bits format
  because CUDA texture object does not support sampling 24bits format
* Use normalized sample fetching (i.e., [0, 1] range float value)
  and also normalized coordinates system for CUDA texture.
  It's consistent with the other graphics APIs such as Direct3D
  and OpenGL, that makes sampling operations much easier.
* Support a kind of viewport and adopt math for colorspace conversion
  from GstD3D11 implementation

Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/3389>
This commit is contained in:
Seungha Yang 2022-11-12 04:06:32 +09:00 committed by GStreamer Marge Bot
parent c1efa9ac4b
commit c11f8fa930
6 changed files with 2347 additions and 2142 deletions

File diff suppressed because it is too large Load diff

View file

@ -1,44 +0,0 @@
/* GStreamer
* Copyright (C) 2019 Seungha Yang <seungha.yang@navercorp.com>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#ifndef __GST_CUDA_CONVERTER_H__
#define __GST_CUDA_CONVERTER_H__
#include <gst/video/video.h>
#include <gst/cuda/gstcudacontext.h>
#include <gst/cuda/gstcudamemory.h>
G_BEGIN_DECLS
typedef struct _GstCudaConverter GstCudaConverter;
GstCudaConverter * gst_cuda_converter_new (GstVideoInfo * in_info,
GstVideoInfo * out_info,
GstCudaContext * cuda_ctx);
void gst_cuda_converter_free (GstCudaConverter * convert);
gboolean gst_cuda_converter_convert_frame (GstCudaConverter * convert,
GstVideoFrame * src_frame,
GstVideoFrame * dst_frame,
CUstream cuda_stream);
G_END_DECLS
#endif /* __GST_CUDA_CONVERTER_H__ */

File diff suppressed because it is too large Load diff

View file

@ -0,0 +1,99 @@
/* GStreamer
* Copyright (C) 2019 Seungha Yang <seungha.yang@navercorp.com>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#pragma once
#include <gst/video/video.h>
#include <gst/cuda/gstcudacontext.h>
G_BEGIN_DECLS
#define GST_TYPE_CUDA_CONVERTER (gst_cuda_converter_get_type())
#define GST_CUDA_CONVERTER(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_CUDA_CONVERTER,GstCudaConverter))
#define GST_CUDA_CONVERTER_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_CUDA_CONVERTER,GstCudaConverterClass))
#define GST_CUDA_CONVERTER_GET_CLASS(obj) (GST_CUDA_CONVERTER_CLASS(G_OBJECT_GET_CLASS(obj)))
#define GST_IS_CUDA_CONVERTER(obj) (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_CUDA_CONVERTER))
#define GST_IS_CUDA_CONVERTER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_CUDA_CONVERTER))
#define GST_CUDA_CONVERTER_CAST(obj) ((GstCudaConverter*)(obj))
typedef struct _GstCudaConverter GstCudaConverter;
typedef struct _GstCudaConverterClass GstCudaConverterClass;
typedef struct _GstCudaConverterPrivate GstCudaConverterPrivate;
/**
* GST_CUDA_CONVERTER_OPT_DEST_X:
*
* #G_TYPE_INT, x position in the destination frame, default 0
*/
#define GST_CUDA_CONVERTER_OPT_DEST_X "GstCudaConverter.dest-x"
/**
* GST_CUDA_CONVERTER_OPT_DEST_Y:
*
* #G_TYPE_INT, y position in the destination frame, default 0
*/
#define GST_CUDA_CONVERTER_OPT_DEST_Y "GstCudaConverter.dest-y"
/**
* GST_CUDA_CONVERTER_OPT_DEST_WIDTH:
*
* #G_TYPE_INT, width in the destination frame, default destination width
*/
#define GST_CUDA_CONVERTER_OPT_DEST_WIDTH "GstCudaConverter.dest-width"
/**
* GST_CUDA_CONVERTER_OPT_DEST_HEIGHT:
*
* #G_TYPE_INT, height in the destination frame, default destination height
*/
#define GST_CUDA_CONVERTER_OPT_DEST_HEIGHT "GstCudaConverter.dest-height"
struct _GstCudaConverter
{
GstObject parent;
GstCudaContext *context;
/*< private >*/
GstCudaConverterPrivate *priv;
gpointer _gst_reserved[GST_PADDING];
};
struct _GstCudaConverterClass
{
GstObjectClass parent_class;
/*< private >*/
gpointer _gst_reserved[GST_PADDING];
};
GType gst_cuda_converter_get_type (void);
GstCudaConverter * gst_cuda_converter_new (const GstVideoInfo * in_info,
const GstVideoInfo * out_info,
GstCudaContext * context,
GstStructure * config);
gboolean gst_cuda_converter_convert_frame (GstCudaConverter * converter,
GstVideoFrame * src_frame,
GstVideoFrame * dst_frame,
CUstream cuda_stream);
G_END_DECLS

View file

@ -24,7 +24,7 @@
#include <gst/cuda/gstcudautils.h>
#include "gstcudaconvertscale.h"
#include "cuda-converter.h"
#include "gstcudaconverter.h"
GST_DEBUG_CATEGORY_STATIC (gst_cuda_base_convert_debug);
#define GST_CAT_DEFAULT gst_cuda_base_convert_debug
@ -133,10 +133,7 @@ gst_cuda_base_convert_dispose (GObject * object)
{
GstCudaBaseConvert *self = GST_CUDA_BASE_CONVERT (object);
if (self->converter) {
gst_cuda_converter_free (self->converter);
self->converter = NULL;
}
gst_clear_object (&self->converter);
G_OBJECT_CLASS (parent_class)->dispose (object);
}
@ -1229,7 +1226,7 @@ gst_cuda_base_convert_set_info (GstCudaBaseTransform * btrans,
gint from_dar_n, from_dar_d, to_dar_n, to_dar_d;
GstVideoInfo tmp_info;
g_clear_pointer (&self->converter, gst_cuda_converter_free);
gst_clear_object (&self->converter);
if (!gst_util_fraction_multiply (in_info->width,
in_info->height, in_info->par_n, in_info->par_d, &from_dar_n,
@ -1288,7 +1285,7 @@ gst_cuda_base_convert_set_info (GstCudaBaseTransform * btrans,
gst_base_transform_set_passthrough (GST_BASE_TRANSFORM (self), FALSE);
self->converter = gst_cuda_converter_new (in_info,
out_info, btrans->context);
out_info, btrans->context, NULL);
if (!self->converter) {
GST_ERROR_OBJECT (self, "Couldn't create converter");
return FALSE;

View file

@ -1,6 +1,6 @@
nvcodec_sources = [
'cuda-converter.c',
'gstcudabasetransform.c',
'gstcudaconverter.c',
'gstcudaconvertscale.c',
'gstcudafilter.c',
'gstcudamemorycopy.c',