Skip to content

Commit 3cd87e7

Browse files
anyj0527myungjoo
authored andcommitted
[tensor_sparse] Implement util functions for sparse tensors
- Implement util functions for enc/decoding sparse tensors - Allow sparse format caps in tensor_sink Signed-off-by: Yongjoo Ahn <yongjoo1.ahn@samsung.com>
1 parent 4fb5fd5 commit 3cd87e7

File tree

6 files changed

+266
-16
lines changed

6 files changed

+266
-16
lines changed

gst/nnstreamer/include/nnstreamer_plugin_api.h

+5
Original file line numberDiff line numberDiff line change
@@ -287,6 +287,11 @@ gst_tensors_config_copy (GstTensorsConfig * dest, const GstTensorsConfig * src);
287287
*/
288288
#define gst_tensors_config_is_flexible(c) ((c)->format == _NNS_TENSOR_FORMAT_FLEXIBLE)
289289

290+
/**
291+
* @brief Macro to check stream format (sparse tensors for caps negotiation)
292+
*/
293+
#define gst_tensors_config_is_sparse(c) ((c)->format == _NNS_TENSOR_FORMAT_SPARSE)
294+
290295
/**
291296
* @brief Get tensor caps from tensors config (for other/tensor)
292297
* @param config tensors config info

gst/nnstreamer/include/tensor_typedef.h

+8
Original file line numberDiff line numberDiff line change
@@ -117,6 +117,14 @@
117117
#define GST_TENSORS_FLEX_CAP_DEFAULT \
118118
GST_TENSORS_CAP_MAKE ("flexible")
119119

120+
/**
121+
* @brief Caps string for the caps template of sparse tensors.
122+
* This mimetype handles non-static, sparse tensor stream without specifying the data type and shape of the tensor.
123+
* The maximum number of tensors in a buffer is 16 (NNS_TENSOR_SIZE_LIMIT).
124+
*/
125+
#define GST_TENSORS_SPARSE_CAP_DEFAULT \
126+
GST_TENSORS_CAP_MAKE ("sparse")
127+
120128
/**
121129
* @brief Default static capability for Protocol Buffers
122130
* protobuf converter will convert this capability to other/tensor(s)

gst/nnstreamer/tensor_common.c

+17-4
Original file line numberDiff line numberDiff line change
@@ -802,8 +802,8 @@ gst_tensors_config_validate (const GstTensorsConfig * config)
802802
return FALSE;
803803
}
804804

805-
/* cannot check tensor info when tensor is flexible */
806-
if (gst_tensors_config_is_flexible (config)) {
805+
/* cannot check tensor info when tensor is not static */
806+
if (!gst_tensors_config_is_static (config)) {
807807
return TRUE;
808808
}
809809

@@ -834,8 +834,8 @@ gst_tensors_config_is_equal (const GstTensorsConfig * c1,
834834
return FALSE;
835835
}
836836

837-
/* cannot compare tensor info when tensor is flexible */
838-
if (gst_tensors_config_is_flexible (c1)) {
837+
/* cannot compare tensor info when tensor is not static */
838+
if (!gst_tensors_config_is_static (c1)) {
839839
return TRUE;
840840
}
841841

@@ -1592,6 +1592,10 @@ gst_tensor_meta_info_get_data_size (GstTensorMetaInfo * meta)
15921592

15931593
dsize = gst_tensor_get_element_size (meta->type);
15941594

1595+
if (meta->format == _NNS_TENSOR_FORMAT_SPARSE) {
1596+
return meta->sparse_info.nnz * (dsize + sizeof (guint));
1597+
}
1598+
15951599
for (i = 0; i < NNS_TENSOR_META_RANK_LIMIT; i++) {
15961600
if (meta->dimension[i] == 0)
15971601
break;
@@ -1648,6 +1652,15 @@ gst_tensor_meta_info_parse_header (GstTensorMetaInfo * meta, gpointer header)
16481652
meta->format = val[18];
16491653
meta->media_type = val[19];
16501654

1655+
switch ((tensor_format) meta->format) {
1656+
case _NNS_TENSOR_FORMAT_SPARSE:
1657+
meta->sparse_info.nnz = val[20];
1658+
break;
1659+
default:
1660+
break;
1661+
}
1662+
1663+
16511664
/** @todo update meta info for each version */
16521665
return gst_tensor_meta_info_validate (meta);
16531666
}

gst/nnstreamer/tensor_sink/tensor_sink.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -237,7 +237,7 @@ gst_tensor_sink_class_init (GstTensorSinkClass * klass)
237237

238238
/** pad template */
239239
pad_caps = gst_caps_from_string (GST_TENSOR_CAP_DEFAULT ";"
240-
GST_TENSORS_CAP_MAKE ("{ static, flexible }"));
240+
GST_TENSORS_CAP_MAKE ("{ static, flexible, sparse }"));
241241
pad_template = gst_pad_template_new ("sink", GST_PAD_SINK, GST_PAD_ALWAYS,
242242
pad_caps);
243243
gst_element_class_add_pad_template (element_class, pad_template);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,226 @@
1+
/* SPDX-License-Identifier: LGPL-2.1-only */
2+
/**
3+
* GStreamer / NNStreamer Sparse Tensor support
4+
* Copyright (C) 2021 Yongjoo Ahn <yongjoo1.ahn@samsung.com>
5+
*/
6+
/**
7+
* @file tensor_sparse_util.c
8+
* @date 27 Jul 2021
9+
* @brief Util functions for tensor_sparse encoder and decoder.
10+
* @see https://github.com/nnstreamer/nnstreamer
11+
* @author Yongjoo Ahn <yongjoo1.ahn@samsung.com>
12+
* @bug No known bugs except for NYI items
13+
*/
14+
15+
#include <tensor_common.h>
16+
#include <tensor_data.h>
17+
#include "tensor_sparse_util.h"
18+
19+
/**
20+
* @brief Make dense tensor with input sparse tensor.
21+
* @param[in,out] meta tensor meta structure to be updated
22+
* @param[in] in pointer of input sparse tensor data
23+
* @return pointer of GstMemory with dense tensor data or NULL on error. Caller should handle this newly allocated memory.
24+
*/
25+
GstMemory *
26+
gst_tensor_sparse_to_dense (GstTensorMetaInfo * meta, gpointer in)
27+
{
28+
guint i, nnz;
29+
guint8 *output, *input;
30+
guint *indices;
31+
gsize output_size, element_size;
32+
33+
meta->format = _NNS_TENSOR_FORMAT_STATIC;
34+
35+
element_size = gst_tensor_get_element_size (meta->type);
36+
output_size = gst_tensor_meta_info_get_data_size (meta);
37+
38+
if (element_size == 0 || output_size == 0) {
39+
nns_loge ("Got invalid meta info");
40+
return NULL;
41+
}
42+
43+
output = (guint8 *) g_malloc0 (output_size);
44+
45+
nnz = meta->sparse_info.nnz;
46+
input = (guint8 *) in + gst_tensor_meta_info_get_header_size (meta);
47+
indices = ((guint *) ((guint8 *) input + element_size * nnz));
48+
49+
for (i = 0; i < nnz; ++i) {
50+
switch (meta->type) {
51+
case _NNS_INT32:
52+
((int32_t *) output)[indices[i]] = ((int32_t *) input)[i];
53+
break;
54+
case _NNS_UINT32:
55+
((uint32_t *) output)[indices[i]] = ((uint32_t *) input)[i];
56+
break;
57+
case _NNS_INT16:
58+
((int16_t *) output)[indices[i]] = ((int16_t *) input)[i];
59+
break;
60+
case _NNS_UINT16:
61+
((uint16_t *) output)[indices[i]] = ((uint16_t *) input)[i];
62+
break;
63+
case _NNS_INT8:
64+
((int8_t *) output)[indices[i]] = ((int8_t *) input)[i];
65+
break;
66+
case _NNS_UINT8:
67+
((uint8_t *) output)[indices[i]] = ((uint8_t *) input)[i];
68+
break;
69+
case _NNS_FLOAT64:
70+
((double *) output)[indices[i]] = ((double *) input)[i];
71+
break;
72+
case _NNS_FLOAT32:
73+
((float *) output)[indices[i]] = ((float *) input)[i];
74+
break;
75+
case _NNS_INT64:
76+
((int64_t *) output)[indices[i]] = ((int64_t *) input)[i];
77+
break;
78+
case _NNS_UINT64:
79+
((uint64_t *) output)[indices[i]] = ((uint64_t *) input)[i];
80+
break;
81+
default:
82+
nns_loge ("Error occured during get tensor value");
83+
g_free (output);
84+
85+
return NULL;
86+
}
87+
}
88+
89+
return gst_memory_new_wrapped (0, output, output_size, 0,
90+
output_size, output, g_free);
91+
}
92+
93+
/**
94+
* @brief Make sparse tensor with input dense tensor.
95+
* @param[in,out] meta tensor meta structure to be updated
96+
* @param[in] in pointer of input dense tensor data
97+
* @return pointer of GstMemory with sparse tensor data or NULL on error. Caller should handle this newly allocated memory.
98+
*/
99+
GstMemory *
100+
gst_tensor_sparse_from_dense (GstTensorMetaInfo * meta, gpointer in)
101+
{
102+
guint i, nnz = 0;
103+
guint8 *output;
104+
tensor_type data_type;
105+
void *values;
106+
guint *indices;
107+
gsize output_size;
108+
gsize header_size = gst_tensor_meta_info_get_header_size (meta);
109+
gsize element_size = gst_tensor_get_element_size (meta->type);
110+
gulong element_count = gst_tensor_get_element_count (meta->dimension);
111+
112+
if (element_size == 0 || element_count == 0) {
113+
nns_loge ("Got invalid meta info");
114+
return NULL;
115+
}
116+
117+
/** alloc maximum possible size of memory */
118+
values = g_malloc0 (element_size * element_count);
119+
indices = g_malloc0 (sizeof (guint) * element_count);
120+
121+
data_type = meta->type;
122+
123+
/** Consider using macro to reduce loc and readability */
124+
for (i = 0; i < element_count; ++i) {
125+
switch (data_type) {
126+
case _NNS_INT32:
127+
if (((int32_t *) in)[i] != 0) {
128+
((int32_t *) values)[nnz] = ((int32_t *) in)[i];
129+
indices[nnz] = i;
130+
nnz += 1;
131+
}
132+
break;
133+
case _NNS_UINT32:
134+
if (((uint32_t *) in)[i] != 0) {
135+
((uint32_t *) values)[nnz] = ((uint32_t *) in)[i];
136+
indices[nnz] = i;
137+
nnz += 1;
138+
}
139+
break;
140+
case _NNS_INT16:
141+
if (((int16_t *) in)[i] != 0) {
142+
((int16_t *) values)[nnz] = ((int16_t *) in)[i];
143+
indices[nnz] = i;
144+
nnz += 1;
145+
}
146+
break;
147+
case _NNS_UINT16:
148+
if (((uint16_t *) in)[i] != 0) {
149+
((uint16_t *) values)[nnz] = ((uint16_t *) in)[i];
150+
indices[nnz] = i;
151+
nnz += 1;
152+
}
153+
break;
154+
case _NNS_INT8:
155+
if (((int8_t *) in)[i] != 0) {
156+
((int8_t *) values)[nnz] = ((int8_t *) in)[i];
157+
indices[nnz] = i;
158+
nnz += 1;
159+
}
160+
break;
161+
case _NNS_UINT8:
162+
if (((uint8_t *) in)[i] != 0) {
163+
((uint8_t *) values)[nnz] = ((uint8_t *) in)[i];
164+
indices[nnz] = i;
165+
nnz += 1;
166+
}
167+
break;
168+
case _NNS_FLOAT64:
169+
if (((double *) in)[i] != 0) {
170+
((double *) values)[nnz] = ((double *) in)[i];
171+
indices[nnz] = i;
172+
nnz += 1;
173+
}
174+
break;
175+
case _NNS_FLOAT32:
176+
if (((float *) in)[i] != 0) {
177+
((float *) values)[nnz] = ((float *) in)[i];
178+
indices[nnz] = i;
179+
nnz += 1;
180+
}
181+
break;
182+
case _NNS_INT64:
183+
if (((int64_t *) in)[i] != 0) {
184+
((int64_t *) values)[nnz] = ((int64_t *) in)[i];
185+
indices[nnz] = i;
186+
nnz += 1;
187+
}
188+
break;
189+
case _NNS_UINT64:
190+
if (((uint64_t *) in)[i] != 0) {
191+
((uint64_t *) values)[nnz] = ((uint64_t *) in)[i];
192+
indices[nnz] = i;
193+
nnz += 1;
194+
}
195+
break;
196+
default:
197+
nns_loge ("Error occured during get tensor value");
198+
g_free (values);
199+
g_free (indices);
200+
201+
return NULL;
202+
}
203+
}
204+
205+
/** update meta nnz info */
206+
meta->sparse_info.nnz = nnz;
207+
208+
/** write to output buffer */
209+
output_size = element_size * nnz + sizeof (guint) * nnz;
210+
211+
/** add meta info header */
212+
output_size += header_size;
213+
output = g_malloc0 (output_size);
214+
215+
gst_tensor_meta_info_update_header (meta, output);
216+
217+
memcpy (output + header_size, values, element_size * nnz);
218+
memcpy (output + header_size + (element_size * nnz),
219+
indices, sizeof (guint) * nnz);
220+
221+
g_free (values);
222+
g_free (indices);
223+
224+
return gst_memory_new_wrapped (0, output, output_size, 0,
225+
output_size, output, g_free);
226+
}

gst/nnstreamer/tensor_sparse/tensor_sparse_util.h

+9-11
Original file line numberDiff line numberDiff line change
@@ -16,26 +16,24 @@
1616
#define __GST_TENSOR_SPARSE_UTIL_H__
1717

1818
#include <gst/gst.h>
19-
#include <tensor_common.h>
19+
#include <tensor_typedef.h>
2020

2121
/**
22-
* @brief NYI. Make dense tensor with input sparse tensor.
22+
* @brief Make dense tensor with input sparse tensor.
2323
* @param[in,out] meta tensor meta structure to be updated
2424
* @param[in] in pointer of input sparse tensor data
25-
* @param[out] out pointer of output dense tensor data. Assume that it's already allocated.
26-
* @return TRUE if no error
25+
* @return pointer of GstMemory with dense tensor data or NULL on error. Caller should handle this newly allocated memory.
2726
*/
28-
extern gboolean
29-
gst_tensor_sparse_to_dense (GstTensorMetaInfo * meta, gpointer in, gpointer out);
27+
extern GstMemory *
28+
gst_tensor_sparse_to_dense (GstTensorMetaInfo * meta, gpointer in);
3029

3130
/**
32-
* @brief NYI. Make sparse tensor with input dense tensor.
31+
* @brief Make sparse tensor with input dense tensor.
3332
* @param[in,out] meta tensor meta structure to be updated
3433
* @param[in] in pointer of input dense tensor data
35-
* @param[out] out pointer of output sparse tensor data. Assume that it's already allocated.
36-
* @param TRUE if no error
34+
* @return pointer of GstMemory with sparse tensor data or NULL on error. Caller should handle this newly allocated memory.
3735
*/
38-
extern gboolean
39-
gst_tensor_sparse_from_dense (GstTensorMetaInfo * meta, gpointer in, gpointer out);
36+
extern GstMemory *
37+
gst_tensor_sparse_from_dense (GstTensorMetaInfo * meta, gpointer in);
4038

4139
#endif /* __GST_TENSOR_SPARSE_UTIL_H__ */

0 commit comments

Comments
 (0)