pipewire/src/gst/gstpipewireformat.c

853 lines
24 KiB
C
Raw Normal View History

/* GStreamer
* Copyright (C) 2016 Wim Taymans <wim.taymans@gmail.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.
*/
2017-03-21 13:36:22 +01:00
#define _GNU_SOURCE
#include <stdio.h>
#include <gst/gst.h>
#include <gst/video/video.h>
#include <gst/audio/audio.h>
#include <spa/param/video/format-utils.h>
#include <spa/param/audio/format-utils.h>
#include <spa/pod/builder.h>
2017-05-23 19:15:33 +02:00
#include "gstpipewireformat.h"
struct media_type {
2017-02-24 16:27:36 +01:00
const char *name;
uint32_t *media_type;
uint32_t *media_subtype;
};
2017-02-24 16:27:36 +01:00
2017-03-24 11:40:58 +01:00
static struct {
struct spa_type_map *map;
2017-03-24 11:40:58 +01:00
uint32_t format;
struct spa_type_media_type media_type;
struct spa_type_media_subtype media_subtype;
struct spa_type_media_subtype_video media_subtype_video;
struct spa_type_media_subtype_audio media_subtype_audio;
struct spa_type_format_video format_video;
struct spa_type_format_audio format_audio;
struct spa_type_video_format video_format;
struct spa_type_audio_format audio_format;
2017-03-24 18:11:11 +01:00
} type = { NULL, };
static void
ensure_types (struct spa_type_map *map)
{
type.map = map;
2017-03-24 11:40:58 +01:00
type.format = spa_type_map_get_id (map, SPA_TYPE__Format);
spa_type_media_type_map (map, &type.media_type);
spa_type_media_subtype_map (map, &type.media_subtype);
spa_type_media_subtype_video_map (map, &type.media_subtype_video);
spa_type_media_subtype_audio_map (map, &type.media_subtype_audio);
2017-03-24 13:11:57 +01:00
spa_type_format_video_map (map, &type.format_video);
spa_type_format_audio_map (map, &type.format_audio);
2017-03-24 11:40:58 +01:00
spa_type_video_format_map (map, &type.video_format);
spa_type_audio_format_map (map, &type.audio_format);
}
static const struct media_type media_type_map[] = {
2017-03-24 11:40:58 +01:00
{ "video/x-raw", &type.media_type.video, &type.media_subtype.raw },
{ "audio/x-raw", &type.media_type.audio, &type.media_subtype.raw },
{ "image/jpeg", &type.media_type.video, &type.media_subtype_video.mjpg },
{ "video/x-h264", &type.media_type.video, &type.media_subtype_video.h264 },
2017-02-24 16:27:36 +01:00
{ NULL, }
};
2017-03-21 13:36:22 +01:00
static const uint32_t *video_format_map[] = {
2017-03-24 11:40:58 +01:00
&type.video_format.UNKNOWN,
&type.video_format.ENCODED,
&type.video_format.I420,
&type.video_format.YV12,
&type.video_format.YUY2,
&type.video_format.UYVY,
&type.video_format.AYUV,
&type.video_format.RGBx,
&type.video_format.BGRx,
&type.video_format.xRGB,
&type.video_format.xBGR,
&type.video_format.RGBA,
&type.video_format.BGRA,
&type.video_format.ARGB,
&type.video_format.ABGR,
&type.video_format.RGB,
&type.video_format.BGR,
&type.video_format.Y41B,
&type.video_format.Y42B,
&type.video_format.YVYU,
&type.video_format.Y444,
&type.video_format.v210,
&type.video_format.v216,
&type.video_format.NV12,
&type.video_format.NV21,
&type.video_format.GRAY8,
&type.video_format.GRAY16_BE,
&type.video_format.GRAY16_LE,
&type.video_format.v308,
&type.video_format.RGB16,
&type.video_format.BGR16,
&type.video_format.RGB15,
&type.video_format.BGR15,
&type.video_format.UYVP,
&type.video_format.A420,
&type.video_format.RGB8P,
&type.video_format.YUV9,
&type.video_format.YVU9,
&type.video_format.IYU1,
&type.video_format.ARGB64,
&type.video_format.AYUV64,
&type.video_format.r210,
&type.video_format.I420_10BE,
&type.video_format.I420_10LE,
&type.video_format.I422_10BE,
&type.video_format.I422_10LE,
&type.video_format.Y444_10BE,
&type.video_format.Y444_10LE,
&type.video_format.GBR,
&type.video_format.GBR_10BE,
&type.video_format.GBR_10LE,
&type.video_format.NV16,
&type.video_format.NV24,
&type.video_format.NV12_64Z32,
&type.video_format.A420_10BE,
&type.video_format.A420_10LE,
&type.video_format.A422_10BE,
&type.video_format.A422_10LE,
&type.video_format.A444_10BE,
&type.video_format.A444_10LE,
&type.video_format.NV61,
&type.video_format.P010_10BE,
&type.video_format.P010_10LE,
&type.video_format.IYU2,
&type.video_format.VYUY,
&type.video_format.GBRA,
&type.video_format.GBRA_10BE,
&type.video_format.GBRA_10LE,
&type.video_format.GBR_12BE,
&type.video_format.GBR_12LE,
&type.video_format.GBRA_12BE,
&type.video_format.GBRA_12LE,
&type.video_format.I420_12BE,
&type.video_format.I420_12LE,
&type.video_format.I422_12BE,
&type.video_format.I422_12LE,
&type.video_format.Y444_12BE,
&type.video_format.Y444_12LE,
2017-03-21 13:36:22 +01:00
};
2017-03-21 16:50:44 +01:00
#if __BYTE_ORDER == __BIG_ENDIAN
2017-03-24 11:40:58 +01:00
#define _FORMAT_LE(fmt) &type.audio_format. fmt ## _OE
#define _FORMAT_BE(fmt) &type.audio_format. fmt
2017-03-21 16:50:44 +01:00
#elif __BYTE_ORDER == __LITTLE_ENDIAN
2017-03-24 11:40:58 +01:00
#define _FORMAT_LE(fmt) &type.audio_format. fmt
#define _FORMAT_BE(fmt) &type.audio_format. fmt ## _OE
2017-03-21 16:50:44 +01:00
#endif
static const uint32_t *audio_format_map[] = {
2017-03-24 11:40:58 +01:00
&type.audio_format.UNKNOWN,
&type.audio_format.ENCODED,
&type.audio_format.S8,
&type.audio_format.U8,
2017-03-21 16:50:44 +01:00
_FORMAT_LE (S16),
_FORMAT_BE (S16),
_FORMAT_LE (U16),
_FORMAT_BE (U16),
_FORMAT_LE (S24_32),
_FORMAT_BE (S24_32),
_FORMAT_LE (U24_32),
_FORMAT_BE (U24_32),
_FORMAT_LE (S32),
_FORMAT_BE (S32),
_FORMAT_LE (U32),
_FORMAT_BE (U32),
_FORMAT_LE (S24),
_FORMAT_BE (S24),
_FORMAT_LE (U24),
_FORMAT_BE (U24),
_FORMAT_LE (S20),
_FORMAT_BE (S20),
_FORMAT_LE (U20),
_FORMAT_BE (U20),
_FORMAT_LE (S18),
_FORMAT_BE (S18),
_FORMAT_LE (U18),
_FORMAT_BE (U18),
_FORMAT_LE (F32),
_FORMAT_BE (F32),
_FORMAT_LE (F64),
_FORMAT_BE (F64),
};
2017-02-24 16:27:36 +01:00
typedef struct {
struct spa_pod_builder b;
const struct media_type *type;
2017-02-24 16:27:36 +01:00
const GstCapsFeatures *cf;
const GstStructure *cs;
} ConvertData;
static const struct media_type *
2017-02-24 16:27:36 +01:00
find_media_types (const char *name)
{
int i;
for (i = 0; media_type_map[i].name; i++) {
if (!strcmp (media_type_map[i].name, name))
return &media_type_map[i];
2017-02-24 16:27:36 +01:00
}
return NULL;
}
2017-02-24 16:27:36 +01:00
static const char *
get_nth_string (const GValue *val, int idx)
{
2017-02-24 16:27:36 +01:00
const GValue *v = NULL;
GType type = G_VALUE_TYPE (val);
if (type == G_TYPE_STRING && idx == 0)
v = val;
else if (type == GST_TYPE_LIST) {
GArray *array = g_value_peek_pointer (val);
if (idx < array->len + 1) {
v = &g_array_index (array, GValue, SPA_MAX (idx - 1, 0));
}
2017-02-24 16:27:36 +01:00
}
if (v)
return g_value_get_string (v);
return NULL;
}
static bool
get_nth_int (const GValue *val, int idx, int *res)
{
const GValue *v = NULL;
GType type = G_VALUE_TYPE (val);
if (type == G_TYPE_INT && idx == 0) {
v = val;
} else if (type == GST_TYPE_INT_RANGE) {
if (idx == 0 || idx == 1) {
*res = gst_value_get_int_range_min (val);
return true;
} else if (idx == 2) {
*res = gst_value_get_int_range_max (val);
return true;
}
2017-02-24 16:27:36 +01:00
} else if (type == GST_TYPE_LIST) {
GArray *array = g_value_peek_pointer (val);
if (idx < array->len + 1) {
v = &g_array_index (array, GValue, SPA_MAX (idx - 1, 0));
}
2017-02-24 16:27:36 +01:00
}
if (v) {
*res = g_value_get_int (v);
return true;
}
return false;
}
static gboolean
get_nth_fraction (const GValue *val, int idx, struct spa_fraction *f)
2017-02-24 16:27:36 +01:00
{
const GValue *v = NULL;
GType type = G_VALUE_TYPE (val);
if (type == GST_TYPE_FRACTION && idx == 0) {
v = val;
} else if (type == GST_TYPE_FRACTION_RANGE) {
if (idx == 0 || idx == 1) {
v = gst_value_get_fraction_range_min (val);
} else if (idx == 2) {
v = gst_value_get_fraction_range_max (val);
}
2017-02-24 16:27:36 +01:00
} else if (type == GST_TYPE_LIST) {
GArray *array = g_value_peek_pointer (val);
if (idx < array->len + 1) {
v = &g_array_index (array, GValue, SPA_MAX (idx-1, 0));
2016-09-07 16:57:25 +02:00
}
2017-02-24 16:27:36 +01:00
}
if (v) {
f->num = gst_value_get_fraction_numerator (v);
f->denom = gst_value_get_fraction_denominator (v);
return true;
}
return false;
}
static gboolean
get_nth_rectangle (const GValue *width, const GValue *height, int idx, struct spa_rectangle *r)
2017-02-24 16:27:36 +01:00
{
const GValue *w = NULL, *h = NULL;
GType wt = G_VALUE_TYPE (width);
GType ht = G_VALUE_TYPE (height);
if (wt == G_TYPE_INT && ht == G_TYPE_INT && idx == 0) {
w = width;
h = height;
} else if (wt == GST_TYPE_INT_RANGE && ht == GST_TYPE_INT_RANGE) {
if (idx == 0 || idx == 1) {
r->width = gst_value_get_int_range_min (width);
r->height = gst_value_get_int_range_min (height);
return true;
} else if (idx == 2) {
r->width = gst_value_get_int_range_max (width);
r->height = gst_value_get_int_range_max (height);
return true;
2016-09-07 16:57:25 +02:00
}
2017-02-24 16:27:36 +01:00
} else if (wt == GST_TYPE_LIST && ht == GST_TYPE_LIST) {
GArray *wa = g_value_peek_pointer (width);
GArray *ha = g_value_peek_pointer (height);
if (idx < wa->len + 1)
w = &g_array_index (wa, GValue, SPA_MAX (idx-1, 0));
if (idx < ha->len + 1)
h = &g_array_index (ha, GValue, SPA_MAX (idx-1, 0));
}
if (w && h) {
r->width = g_value_get_int (w);
r->height = g_value_get_int (h);
return true;
}
2017-02-24 16:27:36 +01:00
return false;
}
2017-02-24 16:27:36 +01:00
static const uint32_t
get_range_type (const GValue *val)
{
GType type = G_VALUE_TYPE (val);
if (type == GST_TYPE_LIST)
return SPA_POD_PROP_RANGE_ENUM;
if (type == GST_TYPE_DOUBLE_RANGE || type == GST_TYPE_FRACTION_RANGE)
return SPA_POD_PROP_RANGE_MIN_MAX;
if (type == GST_TYPE_INT_RANGE) {
if (gst_value_get_int_range_step (val) == 1)
return SPA_POD_PROP_RANGE_MIN_MAX;
else
return SPA_POD_PROP_RANGE_STEP;
}
if (type == GST_TYPE_INT64_RANGE) {
if (gst_value_get_int64_range_step (val) == 1)
return SPA_POD_PROP_RANGE_MIN_MAX;
else
return SPA_POD_PROP_RANGE_STEP;
}
return SPA_POD_PROP_RANGE_NONE;
}
2017-02-24 16:27:36 +01:00
static const uint32_t
get_range_type2 (const GValue *v1, const GValue *v2)
2016-09-07 16:57:25 +02:00
{
2017-02-24 16:27:36 +01:00
uint32_t r1, r2;
r1 = get_range_type (v1);
r2 = get_range_type (v2);
if (r1 == r2)
return r1;
if (r1 == SPA_POD_PROP_RANGE_STEP || r2 == SPA_POD_PROP_RANGE_STEP)
return SPA_POD_PROP_RANGE_STEP;
if (r1 == SPA_POD_PROP_RANGE_MIN_MAX || r2 == SPA_POD_PROP_RANGE_MIN_MAX)
return SPA_POD_PROP_RANGE_MIN_MAX;
return SPA_POD_PROP_RANGE_MIN_MAX;
}
static gboolean
handle_video_fields (ConvertData *d)
{
const GValue *value, *value2;
int i;
struct spa_pod_prop *prop;
2017-02-24 16:27:36 +01:00
value = gst_structure_get_value (d->cs, "format");
if (value) {
const char *v;
int idx;
2017-02-24 16:27:36 +01:00
for (i = 0; (v = get_nth_string (value, i)); i++) {
if (i == 0)
spa_pod_builder_push_prop (&d->b,
2017-03-24 13:11:57 +01:00
type.format_video.format,
get_range_type (value));
2017-02-24 16:27:36 +01:00
idx = gst_video_format_from_string (v);
if (idx < SPA_N_ELEMENTS (video_format_map))
spa_pod_builder_id (&d->b, *video_format_map[idx]);
2017-02-24 16:27:36 +01:00
}
prop = spa_pod_builder_pop(&d->b);
2017-02-24 16:27:36 +01:00
if (i > 1)
prop->body.flags |= SPA_POD_PROP_FLAG_UNSET;
2017-02-24 16:27:36 +01:00
}
value = gst_structure_get_value (d->cs, "width");
value2 = gst_structure_get_value (d->cs, "height");
if (value || value2) {
struct spa_rectangle v;
2017-02-24 16:27:36 +01:00
for (i = 0; get_nth_rectangle (value, value2, i, &v); i++) {
if (i == 0)
spa_pod_builder_push_prop (&d->b,
2017-03-24 13:11:57 +01:00
type.format_video.size,
get_range_type2 (value, value2));
2017-02-24 16:27:36 +01:00
spa_pod_builder_rectangle (&d->b, v.width, v.height);
}
prop = spa_pod_builder_pop(&d->b);
2017-02-24 16:27:36 +01:00
if (i > 1)
prop->body.flags |= SPA_POD_PROP_FLAG_UNSET;
2017-02-24 16:27:36 +01:00
}
value = gst_structure_get_value (d->cs, "framerate");
if (value) {
struct spa_fraction v;
2017-02-24 16:27:36 +01:00
for (i = 0; get_nth_fraction (value, i, &v); i++) {
if (i == 0)
spa_pod_builder_push_prop (&d->b,
2017-03-24 13:11:57 +01:00
type.format_video.framerate,
get_range_type (value));
2017-02-24 16:27:36 +01:00
spa_pod_builder_fraction (&d->b, v.num, v.denom);
2016-09-07 16:57:25 +02:00
}
prop = spa_pod_builder_pop(&d->b);
2017-02-24 16:27:36 +01:00
if (i > 1)
prop->body.flags |= SPA_POD_PROP_FLAG_UNSET;
2016-09-07 16:57:25 +02:00
}
2017-05-17 10:10:27 +02:00
value = gst_structure_get_value (d->cs, "max-framerate");
if (value) {
struct spa_fraction v;
2017-05-17 10:10:27 +02:00
for (i = 0; get_nth_fraction (value, i, &v); i++) {
if (i == 0)
spa_pod_builder_push_prop (&d->b,
2017-05-17 10:10:27 +02:00
type.format_video.max_framerate,
get_range_type (value));
spa_pod_builder_fraction (&d->b, v.num, v.denom);
}
prop = spa_pod_builder_pop(&d->b);
2017-05-17 10:10:27 +02:00
if (i > 1)
prop->body.flags |= SPA_POD_PROP_FLAG_UNSET;
2017-05-17 10:10:27 +02:00
}
2017-02-24 16:27:36 +01:00
return TRUE;
2016-09-07 16:57:25 +02:00
}
2017-02-24 16:27:36 +01:00
static gboolean
handle_audio_fields (ConvertData *d)
2016-09-07 16:57:25 +02:00
{
const GValue *value;
struct spa_pod_prop *prop;
2017-02-24 16:27:36 +01:00
int i = 0;
value = gst_structure_get_value (d->cs, "format");
if (value) {
const char *v;
int idx;
2017-02-24 16:27:36 +01:00
for (i = 0; (v = get_nth_string (value, i)); i++) {
if (i == 0)
spa_pod_builder_push_prop (&d->b,
2017-03-24 13:11:57 +01:00
type.format_audio.format,
get_range_type (value));
2017-02-24 16:27:36 +01:00
idx = gst_audio_format_from_string (v);
if (idx < SPA_N_ELEMENTS (audio_format_map))
spa_pod_builder_id (&d->b, *audio_format_map[idx]);
}
prop = spa_pod_builder_pop(&d->b);
2017-02-24 16:27:36 +01:00
if (i > 1)
prop->body.flags |= SPA_POD_PROP_FLAG_UNSET;
2016-09-07 16:57:25 +02:00
}
2017-02-24 16:27:36 +01:00
value = gst_structure_get_value (d->cs, "layout");
if (value) {
const char *v;
for (i = 0; (v = get_nth_string (value, i)); i++) {
enum spa_audio_layout layout;
2017-02-24 16:27:36 +01:00
if (!strcmp (v, "interleaved"))
layout = SPA_AUDIO_LAYOUT_INTERLEAVED;
else if (!strcmp (v, "non-interleaved"))
layout = SPA_AUDIO_LAYOUT_NON_INTERLEAVED;
else
break;
if (i == 0)
spa_pod_builder_push_prop (&d->b,
2017-03-24 13:11:57 +01:00
type.format_audio.layout,
get_range_type (value));
2017-02-24 16:27:36 +01:00
spa_pod_builder_int (&d->b, layout);
}
prop = spa_pod_builder_pop(&d->b);
2017-02-24 16:27:36 +01:00
if (i > 1)
prop->body.flags |= SPA_POD_PROP_FLAG_UNSET;
2017-02-24 16:27:36 +01:00
}
value = gst_structure_get_value (d->cs, "rate");
if (value) {
int v;
for (i = 0; get_nth_int (value, i, &v); i++) {
if (i == 0)
spa_pod_builder_push_prop (&d->b,
2017-03-24 13:11:57 +01:00
type.format_audio.rate,
get_range_type (value));
2017-02-24 16:27:36 +01:00
spa_pod_builder_int (&d->b, v);
}
prop = spa_pod_builder_pop(&d->b);
2017-02-24 16:27:36 +01:00
if (i > 1)
prop->body.flags |= SPA_POD_PROP_FLAG_UNSET;
2017-02-24 16:27:36 +01:00
}
value = gst_structure_get_value (d->cs, "channels");
if (value) {
int v;
for (i = 0; get_nth_int (value, i, &v); i++) {
if (i == 0)
spa_pod_builder_push_prop (&d->b,
2017-03-24 13:11:57 +01:00
type.format_audio.channels,
get_range_type (value));
2017-02-24 16:27:36 +01:00
spa_pod_builder_int (&d->b, v);
}
prop = spa_pod_builder_pop(&d->b);
2017-02-24 16:27:36 +01:00
if (i > 1)
prop->body.flags |= SPA_POD_PROP_FLAG_UNSET;
2017-02-24 16:27:36 +01:00
}
return TRUE;
2016-09-07 16:57:25 +02:00
}
static uint32_t
write_pod (struct spa_pod_builder *b, const void *data, uint32_t size)
2016-09-07 16:57:25 +02:00
{
uint32_t ref = b->state.offset;
2017-02-24 16:27:36 +01:00
if (b->size <= ref) {
b->size = SPA_ROUND_UP_N (ref + size, 512);
2017-02-24 16:27:36 +01:00
b->data = realloc (b->data, b->size);
if (b->data == NULL)
return -1;
2016-09-07 16:57:25 +02:00
}
2017-02-24 16:27:36 +01:00
memcpy (b->data + ref, data, size);
return ref;
2016-09-07 16:57:25 +02:00
}
static struct spa_pod_object *
2016-09-07 16:57:25 +02:00
convert_1 (GstCapsFeatures *cf, GstStructure *cs)
{
ConvertData d;
2017-02-24 16:27:36 +01:00
spa_zero (d);
2016-09-07 16:57:25 +02:00
d.cf = cf;
d.cs = cs;
2017-02-24 16:27:36 +01:00
if (!(d.type = find_media_types (gst_structure_get_name (cs))))
return NULL;
d.b.write = write_pod;
spa_pod_builder_push_object (&d.b, 0, type.format);
spa_pod_builder_id(&d.b, *d.type->media_type);
spa_pod_builder_id(&d.b, *d.type->media_subtype);
2017-02-24 16:27:36 +01:00
2017-03-24 11:40:58 +01:00
if (*d.type->media_type == type.media_type.video)
2017-02-24 16:27:36 +01:00
handle_video_fields (&d);
2017-03-24 11:40:58 +01:00
else if (*d.type->media_type == type.media_type.audio)
2017-02-24 16:27:36 +01:00
handle_audio_fields (&d);
spa_pod_builder_pop (&d.b);
2017-02-24 16:27:36 +01:00
return SPA_MEMBER (d.b.data, 0, struct spa_pod_object);
}
struct spa_pod_object *
gst_caps_to_format (GstCaps *caps, guint index, struct spa_type_map *map)
{
GstCapsFeatures *f;
GstStructure *s;
struct spa_pod_object *res;
g_return_val_if_fail (GST_IS_CAPS (caps), NULL);
g_return_val_if_fail (gst_caps_is_fixed (caps), NULL);
ensure_types(map);
f = gst_caps_get_features (caps, index);
s = gst_caps_get_structure (caps, index);
res = convert_1 (f, s);
return res;
}
static gboolean
foreach_func (GstCapsFeatures *features,
GstStructure *structure,
GPtrArray *array)
{
struct spa_pod_object *fmt;
if ((fmt = convert_1 (features, structure)))
g_ptr_array_insert (array, -1, fmt);
return TRUE;
}
GPtrArray *
gst_caps_to_format_all (GstCaps *caps, struct spa_type_map *map)
{
GPtrArray *res;
ensure_types(map);
res = g_ptr_array_new_full (gst_caps_get_size (caps), (GDestroyNotify)g_free);
gst_caps_foreach (caps, (GstCapsForeachFunc) foreach_func, res);
return res;
}
2017-03-27 13:00:04 +02:00
static void
handle_id_prop (struct spa_pod_prop *prop, const char *key, GstCaps *res)
2017-03-27 13:00:04 +02:00
{
const char * str;
uint32_t *id = SPA_POD_CONTENTS (struct spa_pod_prop, prop);
2017-03-27 13:00:04 +02:00
uint32_t i, n_items = SPA_POD_PROP_N_VALUES (prop);
uint32_t flags;
2017-03-27 13:00:04 +02:00
flags = prop->body.flags;
if (!(flags & SPA_POD_PROP_FLAG_UNSET))
flags &= ~SPA_POD_PROP_RANGE_MASK;
2017-04-19 12:33:42 +02:00
switch (flags & SPA_POD_PROP_RANGE_MASK) {
2017-03-27 13:00:04 +02:00
case SPA_POD_PROP_RANGE_NONE:
if (!(str = spa_type_map_get_type (type.map, id[0])))
return;
gst_caps_set_simple (res, key, G_TYPE_STRING, rindex (str, ':') + 1, NULL);
break;
case SPA_POD_PROP_RANGE_ENUM:
{
GValue list = { 0 }, v = { 0 };
g_value_init (&list, GST_TYPE_LIST);
for (i = 1; i < n_items; i++) {
if (!(str = spa_type_map_get_type (type.map, id[i])))
continue;
g_value_init (&v, G_TYPE_STRING);
g_value_set_string (&v, rindex (str, ':') + 1);
gst_value_list_append_and_take_value (&list, &v);
}
gst_caps_set_value (res, key, &list);
g_value_unset (&list);
break;
}
default:
break;
}
}
static void
handle_int_prop (struct spa_pod_prop *prop, const char *key, GstCaps *res)
2017-03-27 13:00:04 +02:00
{
uint32_t *val = SPA_POD_CONTENTS (struct spa_pod_prop, prop);
2017-03-27 13:00:04 +02:00
uint32_t i, n_items = SPA_POD_PROP_N_VALUES (prop);
uint32_t flags;
flags = prop->body.flags;
if (!(flags & SPA_POD_PROP_FLAG_UNSET))
flags &= ~SPA_POD_PROP_RANGE_MASK;
2017-03-27 13:00:04 +02:00
2017-04-19 12:33:42 +02:00
switch (flags & SPA_POD_PROP_RANGE_MASK) {
2017-03-27 13:00:04 +02:00
case SPA_POD_PROP_RANGE_NONE:
gst_caps_set_simple (res, key, G_TYPE_INT, val[0], NULL);
break;
case SPA_POD_PROP_RANGE_MIN_MAX:
case SPA_POD_PROP_RANGE_STEP:
{
if (n_items < 3)
return;
gst_caps_set_simple (res, key, GST_TYPE_INT_RANGE, val[1], val[2], NULL);
break;
}
case SPA_POD_PROP_RANGE_ENUM:
{
GValue list = { 0 }, v = { 0 };
g_value_init (&list, GST_TYPE_LIST);
for (i = 1; i < n_items; i++) {
g_value_init (&v, G_TYPE_INT);
g_value_set_int (&v, val[i]);
gst_value_list_append_and_take_value (&list, &v);
}
gst_caps_set_value (res, key, &list);
g_value_unset (&list);
break;
}
default:
break;
}
}
static void
handle_rect_prop (struct spa_pod_prop *prop, const char *width, const char *height, GstCaps *res)
2017-03-27 13:00:04 +02:00
{
struct spa_rectangle *rect = SPA_POD_CONTENTS (struct spa_pod_prop, prop);
2017-03-27 13:00:04 +02:00
uint32_t i, n_items = SPA_POD_PROP_N_VALUES (prop);
uint32_t flags;
2017-03-27 13:00:04 +02:00
flags = prop->body.flags;
if (!(flags & SPA_POD_PROP_FLAG_UNSET))
flags &= ~SPA_POD_PROP_RANGE_MASK;
2017-04-19 12:33:42 +02:00
switch (flags & SPA_POD_PROP_RANGE_MASK) {
2017-03-27 13:00:04 +02:00
case SPA_POD_PROP_RANGE_NONE:
gst_caps_set_simple (res, width, G_TYPE_INT, rect[0].width,
height, G_TYPE_INT, rect[0].height, NULL);
break;
case SPA_POD_PROP_RANGE_MIN_MAX:
case SPA_POD_PROP_RANGE_STEP:
{
if (n_items < 3)
return;
gst_caps_set_simple (res, width, GST_TYPE_INT_RANGE, rect[1].width, rect[2].width,
height, GST_TYPE_INT_RANGE, rect[1].height, rect[2].height, NULL);
break;
}
case SPA_POD_PROP_RANGE_ENUM:
{
GValue l1 = { 0 }, l2 = { 0 }, v1 = { 0 }, v2 = { 0 };
g_value_init (&l1, GST_TYPE_LIST);
g_value_init (&l2, GST_TYPE_LIST);
for (i = 1; i < n_items; i++) {
g_value_init (&v1, G_TYPE_INT);
g_value_set_int (&v1, rect[i].width);
gst_value_list_append_and_take_value (&l1, &v1);
g_value_init (&v2, G_TYPE_INT);
g_value_set_int (&v2, rect[i].height);
gst_value_list_append_and_take_value (&l2, &v2);
}
gst_caps_set_value (res, width, &l1);
gst_caps_set_value (res, height, &l2);
g_value_unset (&l1);
g_value_unset (&l2);
break;
}
default:
break;
}
}
static void
handle_fraction_prop (struct spa_pod_prop *prop, const char *key, GstCaps *res)
2017-03-27 13:00:04 +02:00
{
struct spa_fraction *fract = SPA_POD_CONTENTS (struct spa_pod_prop, prop);
2017-03-27 13:00:04 +02:00
uint32_t i, n_items = SPA_POD_PROP_N_VALUES (prop);
uint32_t flags;
flags = prop->body.flags;
if (!(flags & SPA_POD_PROP_FLAG_UNSET))
flags &= ~SPA_POD_PROP_RANGE_MASK;
2017-03-27 13:00:04 +02:00
2017-04-19 12:33:42 +02:00
switch (flags & SPA_POD_PROP_RANGE_MASK) {
2017-03-27 13:00:04 +02:00
case SPA_POD_PROP_RANGE_NONE:
gst_caps_set_simple (res, key, GST_TYPE_FRACTION, fract[0].num, fract[0].denom, NULL);
break;
case SPA_POD_PROP_RANGE_MIN_MAX:
case SPA_POD_PROP_RANGE_STEP:
{
if (n_items < 3)
return;
gst_caps_set_simple (res, key, GST_TYPE_FRACTION_RANGE, fract[1].num, fract[1].denom,
fract[2].num, fract[2].denom, NULL);
break;
}
case SPA_POD_PROP_RANGE_ENUM:
{
GValue l1 = { 0 }, v1 = { 0 };
g_value_init (&l1, GST_TYPE_LIST);
for (i = 1; i < n_items; i++) {
g_value_init (&v1, GST_TYPE_FRACTION);
gst_value_set_fraction (&v1, fract[i].num, fract[i].denom);
gst_value_list_append_and_take_value (&l1, &v1);
}
gst_caps_set_value (res, key, &l1);
g_value_unset (&l1);
break;
}
default:
break;
}
}
GstCaps *
gst_caps_from_format (const struct spa_pod_object *format, struct spa_type_map *map)
{
GstCaps *res = NULL;
2017-02-27 21:25:33 +01:00
uint32_t media_type, media_subtype;
struct spa_pod_prop *prop;
ensure_types(map);
spa_pod_object_parse(format, "I", &media_type,
"I", &media_subtype);
2017-02-27 21:25:33 +01:00
2017-03-24 11:40:58 +01:00
if (media_type == type.media_type.video) {
if (media_subtype == type.media_subtype.raw) {
2017-03-27 13:00:04 +02:00
res = gst_caps_new_empty_simple ("video/x-raw");
if ((prop = spa_pod_object_find_prop (format, type.format_video.format))) {
2017-03-27 13:00:04 +02:00
handle_id_prop (prop, "format", res);
}
}
2017-03-24 11:40:58 +01:00
else if (media_subtype == type.media_subtype_video.mjpg) {
2017-03-27 13:00:04 +02:00
res = gst_caps_new_empty_simple ("image/jpeg");
}
2017-03-24 11:40:58 +01:00
else if (media_subtype == type.media_subtype_video.h264) {
res = gst_caps_new_simple ("video/x-h264",
"stream-format", G_TYPE_STRING, "byte-stream",
"alignment", G_TYPE_STRING, "au",
NULL);
}
if ((prop = spa_pod_object_find_prop (format, type.format_video.size))) {
2017-03-27 13:00:04 +02:00
handle_rect_prop (prop, "width", "height", res);
}
if ((prop = spa_pod_object_find_prop (format, type.format_video.framerate))) {
2017-03-27 13:00:04 +02:00
handle_fraction_prop (prop, "framerate", res);
}
if ((prop = spa_pod_object_find_prop (format, type.format_video.max_framerate))) {
2017-05-17 10:10:27 +02:00
handle_fraction_prop (prop, "max-framerate", res);
}
2017-03-24 11:40:58 +01:00
} else if (media_type == type.media_type.audio) {
if (media_subtype == type.media_subtype.raw) {
res = gst_caps_new_simple ("audio/x-raw",
"layout", G_TYPE_STRING, "interleaved",
NULL);
if ((prop = spa_pod_object_find_prop (format, type.format_audio.format))) {
2017-03-27 13:00:04 +02:00
handle_id_prop (prop, "format", res);
}
if ((prop = spa_pod_object_find_prop (format, type.format_audio.rate))) {
2017-03-27 13:00:04 +02:00
handle_int_prop (prop, "rate", res);
}
if ((prop = spa_pod_object_find_prop (format, type.format_audio.channels))) {
2017-03-27 13:00:04 +02:00
handle_int_prop (prop, "channels", res);
}
}
else if (media_subtype == type.media_subtype_audio.aac) {
}
}
return res;
}