gstreamer/subprojects/gst-plugins-bad/gst-libs/gst/analytics/gsttensor.c
Daniel Morin 0a343fa738 analytics: remove batch-size
- Batch-size will be the outer-most dimension. Presence of batch dimension can
  be identified using `dims` and `id`.

Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/8191>
2025-01-03 23:14:39 +00:00

212 lines
5.4 KiB
C

/* GStreamer
* Copyright (C) 2023 Collabora Ltd
*
* gstanalyticsmeta.c
*
* 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.
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include "gsttensor.h"
#define GST_TENSOR_SIZE(num_dims) \
(sizeof (GstTensor) + (sizeof (GstTensorDim) * num_dims))
G_DEFINE_BOXED_TYPE (GstTensor, gst_tensor,
(GBoxedCopyFunc) gst_tensor_copy, (GBoxedFreeFunc) gst_tensor_free);
/**
* gst_tensor_alloc: (constructor)
* @num_dims: Number of dimension of the tensors
*
* Allocate a tensor with @num_dims dimensions.
*
* Returns: (transfer full) (not nullable): tensor allocated
*
* Since: 1.26
*/
GstTensor *
gst_tensor_alloc (gsize num_dims)
{
GstTensor *tensor = g_malloc0 (GST_TENSOR_SIZE (num_dims));
tensor->num_dims = num_dims;
return tensor;
}
static gsize
size_for_elements (GstTensorDataType data_type, gsize elements)
{
switch (data_type) {
case GST_TENSOR_DATA_TYPE_INT4:
case GST_TENSOR_DATA_TYPE_UINT4:
return (elements / 2) + (elements % 2);
case GST_TENSOR_DATA_TYPE_INT8:
case GST_TENSOR_DATA_TYPE_UINT8:
return elements;
case GST_TENSOR_DATA_TYPE_INT16:
case GST_TENSOR_DATA_TYPE_UINT16:
case GST_TENSOR_DATA_TYPE_FLOAT16:
case GST_TENSOR_DATA_TYPE_BFLOAT16:
return elements * 2;
case GST_TENSOR_DATA_TYPE_INT32:
case GST_TENSOR_DATA_TYPE_UINT32:
case GST_TENSOR_DATA_TYPE_FLOAT32:
return elements * 4;
case GST_TENSOR_DATA_TYPE_INT64:
case GST_TENSOR_DATA_TYPE_UINT64:
case GST_TENSOR_DATA_TYPE_FLOAT64:
return elements * 8;
default:
g_assert_not_reached ();
return 0;
}
}
/**
* gst_tensor_new_simple:
* @id: semantically identify the contents of the tensor
* @data_type: #GstTensorDataType of tensor data
* @data: (transfer full): #GstBuffer holding tensor data
* @dims_order: Indicate tensor dimension indexing order
* @num_dims: number of tensor dimensions
* @dims: (array length=num_dims): tensor dimensions. Value of 0 mean the
* dimension is dynamic.
*
* Allocates a new #GstTensor of @dims_order ROW_MAJOR or COLUMN_MAJOR and
* with an interleaved layout
*
* Returns: A newly allocated #GstTensor
*
* Since: 1.26
*/
GstTensor *
gst_tensor_new_simple (GQuark id, GstTensorDataType data_type, GstBuffer * data,
GstTensorDimOrder dims_order, gsize num_dims, gsize * dims)
{
GstTensor *tensor;
gsize num_elements = 1;
gsize i;
gboolean dynamic_tensor_size = FALSE;
/* Update this if adding more to GstTensorDataType */
g_return_val_if_fail (data_type <= GST_TENSOR_DATA_TYPE_BFLOAT16, NULL);
g_return_val_if_fail (GST_IS_BUFFER (data), NULL);
g_return_val_if_fail (dims_order == GST_TENSOR_DIM_ORDER_ROW_MAJOR ||
dims_order == GST_TENSOR_DIM_ORDER_COL_MAJOR, NULL);
g_return_val_if_fail (num_dims > 0, NULL);
for (i = 0; i < num_dims; i++) {
dynamic_tensor_size = dims[i] == 0;
if (dynamic_tensor_size == FALSE)
num_elements *= dims[i];
else
break;
}
/* We can't do this validation if the tensor size is dynamic */
if (dynamic_tensor_size == FALSE &&
gst_buffer_get_size (data) !=
size_for_elements (data_type, num_elements)) {
g_critical ("Expected buffer of size %zu (%zu elements),"
" but buffer has size %zu",
size_for_elements (data_type, num_elements), num_elements,
gst_buffer_get_size (data));
return NULL;
}
tensor = gst_tensor_alloc (num_dims);
tensor->id = id;
tensor->layout = GST_TENSOR_LAYOUT_CONTIGUOUS;
tensor->data_type = data_type;
tensor->data = data;
tensor->dims_order = dims_order;
tensor->num_dims = num_dims;
for (i = 0; i < num_dims; i++) {
tensor->dims[i].size = dims[i];
}
return tensor;
}
/**
* gst_tensor_free:
* @tensor: (in) (transfer full): pointer to tensor to free
*
* Free tensor
*
* Since: 1.26
*/
void
gst_tensor_free (GstTensor * tensor)
{
if (tensor->data != NULL) {
gst_buffer_unref (tensor->data);
}
g_free (tensor);
}
/**
* gst_tensor_copy:
* @tensor: (transfer none) (nullable): a #GstTensor to be copied
*
* Create a copy of @tensor.
*
* Returns: (transfer full) (nullable): a new #GstTensor
*
* Since: 1.26
*/
GstTensor *
gst_tensor_copy (const GstTensor * tensor)
{
GstTensor *copy = NULL;
if (tensor) {
copy = (GstTensor *) g_memdup2 (tensor, GST_TENSOR_SIZE (tensor->num_dims));
if (copy->data)
gst_buffer_ref (copy->data);
}
return copy;
}
/**
* gst_tensor_get_dims:
* @tensor: a #GstTensor
* @num_dims: (out): The number of dimensions
*
* Gets the dimensions of the tensor.
*
* Returns: (array length=num_dims) (transfer none): The dims array form the tensor
*
* Since: 1.26
*/
GstTensorDim *
gst_tensor_get_dims (GstTensor * tensor, gsize * num_dims)
{
if (num_dims)
*num_dims = tensor->num_dims;
return tensor->dims;
}