mirror of
https://gitlab.freedesktop.org/pipewire/pipewire.git
synced 2025-11-02 09:01:50 -05:00
Reorganise the SPA includes to make it more extensible later Simplify the naming of the buffer and meta params
381 lines
11 KiB
C
381 lines
11 KiB
C
/* Spa
|
|
* Copyright (C) 2017 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.
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
|
|
#include <spa/support/log-impl.h>
|
|
#include <spa/support/type-map-impl.h>
|
|
#include <spa/pod/pod.h>
|
|
#include <spa/pod/builder.h>
|
|
#include <spa/pod/parser.h>
|
|
#include <spa/param/video/format-utils.h>
|
|
|
|
#include <lib/debug.h>
|
|
|
|
#if 0
|
|
/* { video/raw,
|
|
* format: (int 1) { 1 2 },
|
|
* size: (rect (320 240)) { (1 1) (MAX MAX) },
|
|
* framerate: (frac (25 1)) { (0 1) (MAX 1) },
|
|
*
|
|
* "(object $format
|
|
* (id $video) (id $raw)
|
|
* (prop $format (int $I420) { $I420 $YUY2 } )
|
|
* (prop $size (rect (320 240)) [ (1 1) (MAX MAX) ] )
|
|
* (prop $framerate (frac (25 1))))
|
|
* "
|
|
*
|
|
*/
|
|
|
|
bool: true | false
|
|
int: <num>
|
|
float: <num.frac>
|
|
string: "<string>"
|
|
bytes: "$<base64>"
|
|
pointer: "@<pointer>"
|
|
rectangle: "<width>x<height>"
|
|
fraction: "<num>/<denom>"
|
|
bitmask: "|<bits>"
|
|
array: "[<values>,...]"
|
|
object: "{<key>: <value>,...}"
|
|
key: "<name>"
|
|
|
|
|
|
"format": "S16LE",
|
|
"format": [ "S16LE", "S24LE" ],
|
|
"size": "320x240",
|
|
"+size": [ "320x240", { "min": "320x240", "max": "640x480", "step": "8x8" ],
|
|
"-size": [ "320x240", ["320x240", "640x480", "512x400"] ],
|
|
|
|
"schema.size": { "default": "320x240",
|
|
"flags": ["range", "optional"],
|
|
"values": ["320x240", "640x480", "8x8"] }
|
|
|
|
|
|
"size-schema":
|
|
{ "default": "320x240",
|
|
"values": ["320x240", "640x480"]
|
|
},
|
|
"size-schema":
|
|
{ "default": "320x240",
|
|
"values": { "min": "320x240", "max": "640x480", "step": "8x8" }
|
|
},
|
|
|
|
|
|
"size-alt": { "def": "320x240", "type": "range", "vals": ["320x240", "640x480", "8x8"] },
|
|
"size-alt": { "def": "320x240", "type": "enum", "vals": ["320x240", "640x480"],
|
|
|
|
{
|
|
"type": "audio/raw",
|
|
"format": ["S16LE", "enum", "S16LE", "S24LE" ],
|
|
"size": [ "320x240", "range", "320x240", "640x480" ],
|
|
"framerate": [ "25/1", "range", "25/1", "30/1" ]
|
|
}
|
|
|
|
{ "type": "audio", "subtype": "raw",
|
|
"format": "S16LE",
|
|
"size": { "min": "320x240", "max": "640x480", "step": "8x8" },
|
|
|
|
"framerate": [ "25/1", "30/1" ] }
|
|
|
|
* JSON based format description
|
|
|
|
[ <type>,
|
|
[ <media-type>, <media-subtype> ],
|
|
{
|
|
<key> : [ <type>, <value>, [ <value>, ... ] ],
|
|
...
|
|
}
|
|
]
|
|
|
|
<type> = "123.."
|
|
|
|
1: s = string : "value"
|
|
i = int : <number>
|
|
f = float : <float>
|
|
b = bool : true | false
|
|
R = rectangle : [ <width>, <height> ]
|
|
F = fraction : [ <num>, <denom> ]
|
|
|
|
2: - = default (only default value present)
|
|
e = enum : [ <value>, ... ]
|
|
f = flags : [ <number> ]
|
|
m = min/max : [ <min>, <max> ]
|
|
s = min/max/step : [ <min>, <max>, <step> ]
|
|
|
|
3: u = unset : value is unset, choose from options or default
|
|
o = optional : value does not need to be set
|
|
r = readonly : value is read only
|
|
d = deprecated : value is deprecated
|
|
|
|
[ "Format",
|
|
[ "video", "raw"],
|
|
{
|
|
"format" : [ "se", "I420", [ "I420", "YUY2" ] ],
|
|
"size" : [ "Rmu", [320, 240], [[ 640, 480], [1024,786]]],
|
|
"framerate" : [ "Fsu", [25, 1], [[ 0, 1], [65536,1]]]
|
|
}
|
|
]
|
|
|
|
[ "Format",
|
|
[ "audio", "raw"],
|
|
{
|
|
"format" : [ "se", "S16LE", [ "F32LE", "S16LE" ] ],
|
|
"rate" : [ "imu", 44100, [8000, 96000]],
|
|
"channels" : [ "imu", 1, [1, 4096]]
|
|
"interleaved" : [ "beo", true ]
|
|
}
|
|
]
|
|
|
|
spa_build(SPA_MEDIA_TYPE_VIDEO, SPA_MEDIA_SUBTYPE_RAW,
|
|
type.format_video.format, SPA_PROP_TYPE_ID,
|
|
video_format.I420
|
|
SPA_POD_PROP_FLAG_UNSET |
|
|
SPA_PROP_RANGE_ENUM, 2,
|
|
video_format.I420,
|
|
video_format.YUY2,
|
|
type.format_video.size, SPA_PROP_TYPE_RECTANGLE,
|
|
320, 240,
|
|
SPA_POD_PROP_FLAG_UNSET |
|
|
SPA_PROP_RANGE_MIN_MAX,
|
|
1, 1,
|
|
INT32_MAX, INT32_MAX,
|
|
type.format_video.framerate, SPA_PROP_TYPE_FRACTION, 25, 1,
|
|
SPA_POD_PROP_FLAG_UNSET | SPA_PROP_RANGE_MIN_MAX, 0, 1, INT32_MAX, 1, 0);
|
|
#endif
|
|
|
|
static SPA_TYPE_MAP_IMPL(default_map, 4096);
|
|
|
|
static struct {
|
|
uint32_t format;
|
|
struct spa_type_media_type media_type;
|
|
struct spa_type_media_subtype media_subtype;
|
|
struct spa_type_format_video format_video;
|
|
struct spa_type_video_format video_format;
|
|
} type = { 0,};
|
|
|
|
static inline void type_init(struct spa_type_map *map)
|
|
{
|
|
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_format_video_map(map, &type.format_video);
|
|
spa_type_video_format_map(map, &type.video_format);
|
|
}
|
|
|
|
static void do_static_struct(struct spa_type_map *map)
|
|
{
|
|
struct _test_format {
|
|
struct spa_pod_object fmt;
|
|
|
|
struct {
|
|
struct spa_pod_id media_type SPA_ALIGNED(8);
|
|
struct spa_pod_id media_subtype SPA_ALIGNED(8);
|
|
|
|
struct spa_pod_prop prop_format SPA_ALIGNED(8);
|
|
struct {
|
|
uint32_t def_format;
|
|
uint32_t enum_format[2];
|
|
} format_vals;
|
|
|
|
struct spa_pod_prop prop_size SPA_ALIGNED(8);
|
|
struct {
|
|
struct spa_rectangle def_size;
|
|
struct spa_rectangle min_size;
|
|
struct spa_rectangle max_size;
|
|
} size_vals;
|
|
|
|
struct spa_pod_prop prop_framerate SPA_ALIGNED(8);
|
|
struct {
|
|
struct spa_fraction def_framerate;
|
|
struct spa_fraction min_framerate;
|
|
struct spa_fraction max_framerate;
|
|
} framerate_vals;
|
|
} props;
|
|
} test_format = {
|
|
SPA_POD_OBJECT_INIT(sizeof(test_format.props) + sizeof(struct spa_pod_object_body),
|
|
0, type.format),
|
|
{
|
|
SPA_POD_ID_INIT(type.media_type.video),
|
|
SPA_POD_ID_INIT(type.media_subtype.raw),
|
|
|
|
SPA_POD_PROP_INIT(sizeof(test_format.props.format_vals) +
|
|
sizeof(struct spa_pod_prop_body),
|
|
type.format_video.format,
|
|
SPA_POD_PROP_RANGE_ENUM | SPA_POD_PROP_FLAG_UNSET,
|
|
sizeof(uint32_t), SPA_POD_TYPE_ID),
|
|
{
|
|
type.video_format.I420,
|
|
{ type.video_format.I420, type.video_format.YUY2 }
|
|
},
|
|
SPA_POD_PROP_INIT(sizeof(test_format.props.size_vals) +
|
|
sizeof(struct spa_pod_prop_body),
|
|
type.format_video.size,
|
|
SPA_POD_PROP_RANGE_MIN_MAX | SPA_POD_PROP_FLAG_UNSET,
|
|
sizeof(struct spa_rectangle), SPA_POD_TYPE_RECTANGLE),
|
|
|
|
{
|
|
SPA_RECTANGLE(320,243),
|
|
SPA_RECTANGLE(1,1), SPA_RECTANGLE(INT32_MAX, INT32_MAX)
|
|
},
|
|
SPA_POD_PROP_INIT(sizeof(test_format.props.framerate_vals) +
|
|
sizeof(struct spa_pod_prop_body),
|
|
type.format_video.framerate,
|
|
SPA_POD_PROP_RANGE_MIN_MAX | SPA_POD_PROP_FLAG_UNSET,
|
|
sizeof(struct spa_fraction), SPA_POD_TYPE_FRACTION),
|
|
{
|
|
SPA_FRACTION(25,1),
|
|
SPA_FRACTION(0,1), SPA_FRACTION(INT32_MAX,1)
|
|
}
|
|
}
|
|
};
|
|
|
|
spa_debug_pod(&test_format.fmt.pod, 0);
|
|
spa_debug_pod(&test_format.fmt.pod, SPA_DEBUG_FLAG_FORMAT);
|
|
|
|
{
|
|
uint32_t format = -1;
|
|
int res;
|
|
struct spa_fraction frac = { -1, -1 };
|
|
|
|
res = spa_pod_object_parse(&test_format.fmt,
|
|
":",type.format_video.format, "I", &format,
|
|
":",type.format_video.framerate, "F", &frac, NULL);
|
|
|
|
printf("%d format %d num %d denom %d\n", res, format, frac.num, frac.denom);
|
|
|
|
spa_pod_object_fixate(&test_format.fmt);
|
|
|
|
res = spa_pod_object_parse(&test_format.fmt,
|
|
":",type.format_video.format, "I", &format,
|
|
":",type.format_video.framerate, "F", &frac, NULL);
|
|
|
|
printf("%d format %d num %d denom %d\n", res, format, frac.num, frac.denom);
|
|
}
|
|
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
struct spa_pod_builder b = { NULL, };
|
|
uint8_t buffer[1024];
|
|
struct spa_pod_object *fmt;
|
|
struct spa_type_map *map = &default_map.map;
|
|
|
|
type_init(map);
|
|
spa_debug_set_type_map(map);
|
|
|
|
spa_pod_builder_init(&b, buffer, sizeof(buffer));
|
|
|
|
spa_pod_builder_push_object(&b, 0, type.format);
|
|
|
|
spa_pod_builder_id(&b, type.media_type.video);
|
|
spa_pod_builder_id(&b, type.media_subtype.raw);
|
|
|
|
spa_pod_builder_push_prop(&b, type.format_video.format,
|
|
SPA_POD_PROP_RANGE_ENUM | SPA_POD_PROP_FLAG_UNSET);
|
|
spa_pod_builder_id(&b, type.video_format.I420);
|
|
spa_pod_builder_id(&b, type.video_format.I420);
|
|
spa_pod_builder_id(&b, type.video_format.YUY2);
|
|
spa_pod_builder_pop(&b);
|
|
|
|
struct spa_rectangle size_min_max[] = { {1, 1}, {INT32_MAX, INT32_MAX} };
|
|
spa_pod_builder_push_prop(&b,
|
|
type.format_video.size,
|
|
SPA_POD_PROP_RANGE_MIN_MAX | SPA_POD_PROP_FLAG_UNSET);
|
|
spa_pod_builder_rectangle(&b, 320, 240);
|
|
spa_pod_builder_raw(&b, size_min_max, sizeof(size_min_max));
|
|
spa_pod_builder_pop(&b);
|
|
|
|
struct spa_fraction rate_min_max[] = { {0, 1}, {INT32_MAX, 1} };
|
|
spa_pod_builder_push_prop(&b,
|
|
type.format_video.framerate,
|
|
SPA_POD_PROP_RANGE_MIN_MAX | SPA_POD_PROP_FLAG_UNSET);
|
|
spa_pod_builder_fraction(&b, 25, 1);
|
|
spa_pod_builder_raw(&b, rate_min_max, sizeof(rate_min_max));
|
|
spa_pod_builder_pop(&b);
|
|
|
|
fmt = spa_pod_builder_pop_deref(&b);
|
|
|
|
spa_debug_pod(&fmt->pod, 0);
|
|
|
|
spa_pod_builder_init(&b, buffer, sizeof(buffer));
|
|
|
|
fmt = spa_pod_builder_object(&b,
|
|
0, type.format,
|
|
"I", type.media_type.video,
|
|
"I", type.media_subtype.raw,
|
|
":", type.format_video.format, "Ieu", type.video_format.I420,
|
|
2, type.video_format.I420,
|
|
type.video_format.YUY2,
|
|
":", type.format_video.size, "Rru", &SPA_RECTANGLE(320,241),
|
|
2, &SPA_RECTANGLE(1,1),
|
|
&SPA_RECTANGLE(INT32_MAX,INT32_MAX),
|
|
":", type.format_video.framerate, "Fru", &SPA_FRACTION(25,1),
|
|
2, &SPA_FRACTION(0,1),
|
|
&SPA_FRACTION(INT32_MAX,1));
|
|
|
|
spa_debug_pod(&fmt->pod, 0);
|
|
spa_debug_pod(&fmt->pod, SPA_DEBUG_FLAG_FORMAT);
|
|
|
|
spa_pod_builder_init(&b, buffer, sizeof(buffer));
|
|
|
|
/*
|
|
* ( "Format",
|
|
* ("video", "raw" ),
|
|
* {
|
|
* "format": ( "seu", "I420", ( "I420","YUY2" ) ),
|
|
* "size": ( "Rru", (320, 242), ( (1,1), (MAX, MAX)) ),
|
|
* "framerate": ( "Fru", (25, 1), ( (0,1), (MAX, 1)) )
|
|
* }
|
|
* )
|
|
*/
|
|
fmt = spa_pod_builder_add(&b,
|
|
"<", 0, type.format,
|
|
"I", type.media_type.video,
|
|
"I", type.media_subtype.raw,
|
|
":", type.format_video.format, "Ieu", type.video_format.I420,
|
|
2, type.video_format.I420,
|
|
type.video_format.YUY2,
|
|
":", type.format_video.size, "Rru", &SPA_RECTANGLE(320,242),
|
|
2, &SPA_RECTANGLE(1,1),
|
|
&SPA_RECTANGLE(INT32_MAX,INT32_MAX),
|
|
":", type.format_video.framerate, "Fru", &SPA_FRACTION(25,1),
|
|
2, &SPA_FRACTION(0,1),
|
|
&SPA_FRACTION(INT32_MAX,1),
|
|
">", NULL);
|
|
|
|
spa_debug_pod(&fmt->pod, 0);
|
|
spa_debug_pod(&fmt->pod, SPA_DEBUG_FLAG_FORMAT);
|
|
|
|
do_static_struct(map);
|
|
|
|
printf("media type is enum %d\n", spa_type_is_a(SPA_TYPE__MediaType, SPA_TYPE_ENUM_BASE));
|
|
printf("media sybtype is enum %d\n",
|
|
spa_type_is_a(SPA_TYPE__MediaSubtype, SPA_TYPE_ENUM_BASE));
|
|
printf("format is enum %d\n", spa_type_is_a(SPA_TYPE__Format, SPA_TYPE_ENUM_BASE));
|
|
printf("format is pod %d\n", spa_type_is_a(SPA_TYPE__Format, SPA_TYPE_POD_BASE));
|
|
printf("format is object %d\n", spa_type_is_a(SPA_TYPE__Format, SPA_TYPE_POD_OBJECT_BASE));
|
|
|
|
return 0;
|
|
}
|