| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | /* Spa
 | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-11-05 17:48:52 +01:00
										 |  |  |  * Copyright © 2018 Wim Taymans | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-11-05 17:48:52 +01:00
										 |  |  |  * Permission is hereby granted, free of charge, to any person obtaining a | 
					
						
							|  |  |  |  * copy of this software and associated documentation files (the "Software"), | 
					
						
							|  |  |  |  * to deal in the Software without restriction, including without limitation | 
					
						
							|  |  |  |  * the rights to use, copy, modify, merge, publish, distribute, sublicense, | 
					
						
							|  |  |  |  * and/or sell copies of the Software, and to permit persons to whom the | 
					
						
							|  |  |  |  * Software is furnished to do so, subject to the following conditions: | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-11-05 17:48:52 +01:00
										 |  |  |  * The above copyright notice and this permission notice (including the next | 
					
						
							|  |  |  |  * paragraph) shall be included in all copies or substantial portions of the | 
					
						
							|  |  |  |  * Software. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
					
						
							|  |  |  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
					
						
							|  |  |  |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL | 
					
						
							|  |  |  |  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
					
						
							|  |  |  |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING | 
					
						
							|  |  |  |  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER | 
					
						
							|  |  |  |  * DEALINGS IN THE SOFTWARE. | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <errno.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <spa/support/log.h>
 | 
					
						
							|  |  |  | #include <spa/utils/list.h>
 | 
					
						
							|  |  |  | #include <spa/node/node.h>
 | 
					
						
							|  |  |  | #include <spa/node/io.h>
 | 
					
						
							|  |  |  | #include <spa/param/audio/format-utils.h>
 | 
					
						
							| 
									
										
										
										
											2018-08-25 12:08:29 +02:00
										 |  |  | #include <spa/param/param.h>
 | 
					
						
							| 
									
										
										
										
											2018-08-13 17:17:23 +02:00
										 |  |  | #include <spa/pod/filter.h>
 | 
					
						
							| 
									
										
										
										
											2018-09-14 15:23:12 +02:00
										 |  |  | #include <spa/debug/types.h>
 | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 15:58:55 +01:00
										 |  |  | #include "resample.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "resample-speex.h"
 | 
					
						
							|  |  |  | #include "resample-peaks.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | #define NAME "resample"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-25 14:34:36 +02:00
										 |  |  | #define DEFAULT_RATE		44100
 | 
					
						
							|  |  |  | #define DEFAULT_CHANNELS	2
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | #define MAX_BUFFERS     32
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct impl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct props { | 
					
						
							| 
									
										
										
										
											2018-11-16 16:54:56 +01:00
										 |  |  | 	double rate; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void props_reset(struct props *props) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-11-16 16:54:56 +01:00
										 |  |  | 	props->rate = 1.0; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct buffer { | 
					
						
							| 
									
										
										
										
											2019-01-07 17:57:03 +01:00
										 |  |  | 	uint32_t id; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | #define BUFFER_FLAG_OUT		(1 << 0)
 | 
					
						
							|  |  |  | 	uint32_t flags; | 
					
						
							| 
									
										
										
										
											2019-01-07 17:57:03 +01:00
										 |  |  | 	struct spa_list link; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 	struct spa_buffer *outbuf; | 
					
						
							|  |  |  | 	struct spa_meta_header *h; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct port { | 
					
						
							|  |  |  | 	uint32_t id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct spa_io_buffers *io; | 
					
						
							| 
									
										
										
										
											2018-11-16 16:54:56 +01:00
										 |  |  | 	struct spa_io_range *io_range; | 
					
						
							|  |  |  | 	struct spa_io_sequence *io_control; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 	struct spa_port_info info; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bool have_format; | 
					
						
							|  |  |  | 	struct spa_audio_info format; | 
					
						
							|  |  |  | 	uint32_t stride; | 
					
						
							|  |  |  | 	uint32_t blocks; | 
					
						
							|  |  |  | 	uint32_t size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct buffer buffers[MAX_BUFFERS]; | 
					
						
							|  |  |  | 	uint32_t n_buffers; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct spa_list queue; | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 	uint32_t offset; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct impl { | 
					
						
							|  |  |  | 	struct spa_handle handle; | 
					
						
							|  |  |  | 	struct spa_node node; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct spa_log *log; | 
					
						
							| 
									
										
										
										
											2018-12-19 16:47:20 +01:00
										 |  |  | 	struct spa_cpu *cpu; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	struct props props; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const struct spa_node_callbacks *callbacks; | 
					
						
							|  |  |  | 	void *user_data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct port in_port; | 
					
						
							|  |  |  | 	struct port out_port; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bool started; | 
					
						
							| 
									
										
										
										
											2018-12-05 15:58:55 +01:00
										 |  |  | 	bool monitor; | 
					
						
							| 
									
										
										
										
											2018-04-10 16:54:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 15:58:55 +01:00
										 |  |  | 	struct resample resample; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define CHECK_PORT(this,d,id)		(id == 0)
 | 
					
						
							|  |  |  | #define GET_IN_PORT(this,id)		(&this->in_port)
 | 
					
						
							|  |  |  | #define GET_OUT_PORT(this,id)		(&this->out_port)
 | 
					
						
							|  |  |  | #define GET_PORT(this,d,id)		(d == SPA_DIRECTION_INPUT ? GET_IN_PORT(this,id) : GET_OUT_PORT(this,id))
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int setup_convert(struct impl *this, | 
					
						
							|  |  |  | 		enum spa_direction direction, | 
					
						
							|  |  |  | 		const struct spa_audio_info *info) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const struct spa_audio_info *src_info, *dst_info; | 
					
						
							| 
									
										
										
										
											2018-04-10 16:54:15 +02:00
										 |  |  | 	int err; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (direction == SPA_DIRECTION_INPUT) { | 
					
						
							|  |  |  | 		src_info = info; | 
					
						
							|  |  |  | 		dst_info = &GET_OUT_PORT(this, 0)->format; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		src_info = &GET_IN_PORT(this, 0)->format; | 
					
						
							|  |  |  | 		dst_info = info; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-14 15:23:12 +02:00
										 |  |  | 	spa_log_info(this->log, NAME " %p: %s/%d@%d->%s/%d@%d", this, | 
					
						
							|  |  |  | 			spa_debug_type_find_name(spa_type_audio_format, src_info->info.raw.format), | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 			src_info->info.raw.channels, | 
					
						
							|  |  |  | 			src_info->info.raw.rate, | 
					
						
							| 
									
										
										
										
											2018-09-14 15:23:12 +02:00
										 |  |  | 			spa_debug_type_find_name(spa_type_audio_format, dst_info->info.raw.format), | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 			dst_info->info.raw.channels, | 
					
						
							| 
									
										
										
										
											2018-09-13 17:03:56 +02:00
										 |  |  | 			dst_info->info.raw.rate); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (src_info->info.raw.channels != dst_info->info.raw.channels) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 16:05:40 +01:00
										 |  |  | 	if (this->resample.free) | 
					
						
							|  |  |  | 		resample_free(&this->resample); | 
					
						
							| 
									
										
										
										
											2018-04-10 16:54:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 15:58:55 +01:00
										 |  |  | 	this->resample.channels = src_info->info.raw.channels; | 
					
						
							|  |  |  | 	this->resample.i_rate = src_info->info.raw.rate; | 
					
						
							|  |  |  | 	this->resample.o_rate = dst_info->info.raw.rate; | 
					
						
							| 
									
										
										
										
											2018-04-10 16:54:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 15:58:55 +01:00
										 |  |  | 	if (this->monitor) | 
					
						
							|  |  |  | 		err = impl_peaks_init(&this->resample); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		err = impl_speex_init(&this->resample); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return err; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int impl_node_enum_params(struct spa_node *node, | 
					
						
							|  |  |  | 				 uint32_t id, uint32_t *index, | 
					
						
							|  |  |  | 				 const struct spa_pod *filter, | 
					
						
							|  |  |  | 				 struct spa_pod **param, | 
					
						
							|  |  |  | 				 struct spa_pod_builder *builder) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return -ENOTSUP; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 16:54:56 +01:00
										 |  |  | static int apply_props(struct impl *this, const struct spa_pod *param) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spa_pod_prop *prop; | 
					
						
							|  |  |  | 	struct spa_pod_object *obj = (struct spa_pod_object *) param; | 
					
						
							|  |  |  | 	struct props *p = &this->props; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	SPA_POD_OBJECT_FOREACH(obj, prop) { | 
					
						
							|  |  |  | 		switch (prop->key) { | 
					
						
							|  |  |  | 		case SPA_PROP_rate: | 
					
						
							|  |  |  | 			spa_pod_get_double(&prop->value, &p->rate); | 
					
						
							| 
									
										
										
										
											2018-12-05 15:58:55 +01:00
										 |  |  | 			resample_update_rate(&this->resample, p->rate); | 
					
						
							| 
									
										
										
										
											2018-11-16 16:54:56 +01:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | static int impl_node_set_param(struct spa_node *node, uint32_t id, uint32_t flags, | 
					
						
							|  |  |  | 			       const struct spa_pod *param) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-11-16 16:54:56 +01:00
										 |  |  | 	struct impl *this; | 
					
						
							|  |  |  | 	int res = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(node != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	this = SPA_CONTAINER_OF(node, struct impl, node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (id) { | 
					
						
							|  |  |  | 	case SPA_PARAM_Props: | 
					
						
							|  |  |  | 		apply_props(this, param); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return -ENOTSUP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-23 12:31:41 +02:00
										 |  |  | static int impl_node_set_io(struct spa_node *node, uint32_t id, void *data, size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return -ENOTSUP; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | static int impl_node_send_command(struct spa_node *node, const struct spa_command *command) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct impl *this; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(node != NULL, -EINVAL); | 
					
						
							|  |  |  | 	spa_return_val_if_fail(command != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	this = SPA_CONTAINER_OF(node, struct impl, node); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-24 10:53:09 +02:00
										 |  |  | 	switch (SPA_NODE_COMMAND_ID(command)) { | 
					
						
							|  |  |  | 	case SPA_NODE_COMMAND_Start: | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		this->started = true; | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 		break; | 
					
						
							| 
									
										
										
										
											2018-08-24 10:53:09 +02:00
										 |  |  | 	case SPA_NODE_COMMAND_Pause: | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		this->started = false; | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		return -ENOTSUP; | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | impl_node_set_callbacks(struct spa_node *node, | 
					
						
							|  |  |  | 			const struct spa_node_callbacks *callbacks, | 
					
						
							|  |  |  | 			void *user_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct impl *this; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(node != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	this = SPA_CONTAINER_OF(node, struct impl, node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	this->callbacks = callbacks; | 
					
						
							|  |  |  | 	this->user_data = user_data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | impl_node_get_n_ports(struct spa_node *node, | 
					
						
							|  |  |  | 		      uint32_t *n_input_ports, | 
					
						
							|  |  |  | 		      uint32_t *max_input_ports, | 
					
						
							|  |  |  | 		      uint32_t *n_output_ports, | 
					
						
							|  |  |  | 		      uint32_t *max_output_ports) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	spa_return_val_if_fail(node != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (n_input_ports) | 
					
						
							|  |  |  | 		*n_input_ports = 1; | 
					
						
							|  |  |  | 	if (max_input_ports) | 
					
						
							|  |  |  | 		*max_input_ports = 1; | 
					
						
							|  |  |  | 	if (n_output_ports) | 
					
						
							|  |  |  | 		*n_output_ports = 1; | 
					
						
							|  |  |  | 	if (max_output_ports) | 
					
						
							|  |  |  | 		*max_output_ports = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | impl_node_get_port_ids(struct spa_node *node, | 
					
						
							|  |  |  | 		       uint32_t *input_ids, | 
					
						
							|  |  |  | 		       uint32_t n_input_ids, | 
					
						
							|  |  |  | 		       uint32_t *output_ids, | 
					
						
							|  |  |  | 		       uint32_t n_output_ids) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	spa_return_val_if_fail(node != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (n_input_ids && input_ids) | 
					
						
							|  |  |  | 		input_ids[0] = 0; | 
					
						
							|  |  |  | 	if (n_output_ids > 0 && output_ids) | 
					
						
							|  |  |  | 		output_ids[0] = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int impl_node_add_port(struct spa_node *node, enum spa_direction direction, uint32_t port_id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return -ENOTSUP; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | impl_node_remove_port(struct spa_node *node, enum spa_direction direction, uint32_t port_id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return -ENOTSUP; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | impl_node_port_get_info(struct spa_node *node, | 
					
						
							|  |  |  | 			enum spa_direction direction, | 
					
						
							|  |  |  | 			uint32_t port_id, | 
					
						
							|  |  |  | 			const struct spa_port_info **info) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct impl *this; | 
					
						
							|  |  |  | 	struct port *port; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(node != NULL, -EINVAL); | 
					
						
							|  |  |  | 	spa_return_val_if_fail(info != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	this = SPA_CONTAINER_OF(node, struct impl, node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(CHECK_PORT(this, direction, port_id), -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	port = GET_PORT(this, direction, port_id); | 
					
						
							|  |  |  | 	*info = &port->info; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int port_enum_formats(struct spa_node *node, | 
					
						
							|  |  |  | 			     enum spa_direction direction, uint32_t port_id, | 
					
						
							|  |  |  | 			     uint32_t *index, | 
					
						
							|  |  |  | 			     struct spa_pod **param, | 
					
						
							|  |  |  | 			     struct spa_pod_builder *builder) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct impl *this = SPA_CONTAINER_OF(node, struct impl, node); | 
					
						
							|  |  |  | 	struct port *other; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	other = GET_PORT(this, SPA_DIRECTION_REVERSE(direction), 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (*index) { | 
					
						
							|  |  |  | 	case 0: | 
					
						
							|  |  |  | 		if (other->have_format) { | 
					
						
							| 
									
										
										
										
											2018-09-17 09:41:41 +02:00
										 |  |  | 			spa_pod_builder_push_object(builder, | 
					
						
							|  |  |  | 				SPA_TYPE_OBJECT_Format, SPA_PARAM_EnumFormat); | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 			spa_pod_builder_add(builder, | 
					
						
							|  |  |  | 				SPA_FORMAT_mediaType,      SPA_POD_Id(SPA_MEDIA_TYPE_audio), | 
					
						
							|  |  |  | 				SPA_FORMAT_mediaSubtype,   SPA_POD_Id(SPA_MEDIA_SUBTYPE_raw), | 
					
						
							|  |  |  | 				SPA_FORMAT_AUDIO_format,   SPA_POD_Id(SPA_AUDIO_FORMAT_F32P), | 
					
						
							|  |  |  | 				SPA_FORMAT_AUDIO_rate,     SPA_POD_CHOICE_RANGE_Int( | 
					
						
							| 
									
										
										
										
											2018-09-05 16:41:07 +02:00
										 |  |  | 								other->format.info.raw.rate, 1, INT32_MAX), | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 				SPA_FORMAT_AUDIO_channels, SPA_POD_Int(other->format.info.raw.channels), | 
					
						
							| 
									
										
										
										
											2018-09-05 16:41:07 +02:00
										 |  |  | 				0); | 
					
						
							| 
									
										
										
										
											2018-09-17 09:41:41 +02:00
										 |  |  | 			spa_pod_builder_prop(builder, SPA_FORMAT_AUDIO_position, 0); | 
					
						
							|  |  |  | 			spa_pod_builder_array(builder, sizeof(uint32_t), SPA_TYPE_Id, | 
					
						
							|  |  |  | 					other->format.info.raw.channels, other->format.info.raw.position); | 
					
						
							|  |  |  | 			*param = spa_pod_builder_pop(builder); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 			*param = spa_pod_builder_add_object(builder, | 
					
						
							| 
									
										
										
										
											2018-08-27 15:03:11 +02:00
										 |  |  | 				SPA_TYPE_OBJECT_Format, SPA_PARAM_EnumFormat, | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 				SPA_FORMAT_mediaType,      SPA_POD_Id(SPA_MEDIA_TYPE_audio), | 
					
						
							|  |  |  | 				SPA_FORMAT_mediaSubtype,   SPA_POD_Id(SPA_MEDIA_SUBTYPE_raw), | 
					
						
							|  |  |  | 				SPA_FORMAT_AUDIO_format,   SPA_POD_Id(SPA_AUDIO_FORMAT_F32P), | 
					
						
							|  |  |  | 				SPA_FORMAT_AUDIO_rate,     SPA_POD_CHOICE_RANGE_Int(DEFAULT_RATE, 1, INT32_MAX), | 
					
						
							|  |  |  | 				SPA_FORMAT_AUDIO_channels, SPA_POD_CHOICE_RANGE_Int(DEFAULT_CHANNELS, 1, INT32_MAX)); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | impl_node_port_enum_params(struct spa_node *node, | 
					
						
							|  |  |  | 			   enum spa_direction direction, uint32_t port_id, | 
					
						
							|  |  |  | 			   uint32_t id, uint32_t *index, | 
					
						
							|  |  |  | 			   const struct spa_pod *filter, | 
					
						
							|  |  |  | 			   struct spa_pod **result, | 
					
						
							|  |  |  | 			   struct spa_pod_builder *builder) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct impl *this; | 
					
						
							|  |  |  | 	struct port *port, *other; | 
					
						
							|  |  |  | 	struct spa_pod *param; | 
					
						
							|  |  |  | 	struct spa_pod_builder b = { 0 }; | 
					
						
							|  |  |  | 	uint8_t buffer[1024]; | 
					
						
							|  |  |  | 	int res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(node != NULL, -EINVAL); | 
					
						
							|  |  |  | 	spa_return_val_if_fail(index != NULL, -EINVAL); | 
					
						
							|  |  |  | 	spa_return_val_if_fail(builder != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	this = SPA_CONTAINER_OF(node, struct impl, node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(CHECK_PORT(this, direction, port_id), -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	port = GET_PORT(this, direction, port_id); | 
					
						
							|  |  |  | 	other = GET_PORT(this, SPA_DIRECTION_REVERSE(direction), port_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       next: | 
					
						
							|  |  |  | 	spa_pod_builder_init(&b, buffer, sizeof(buffer)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 	switch (id) { | 
					
						
							| 
									
										
										
										
											2018-08-25 12:08:29 +02:00
										 |  |  | 	case  SPA_PARAM_List: | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2018-08-25 12:08:29 +02:00
										 |  |  | 		uint32_t list[] = { SPA_PARAM_EnumFormat, | 
					
						
							|  |  |  | 				    SPA_PARAM_Format, | 
					
						
							|  |  |  | 				    SPA_PARAM_Buffers, | 
					
						
							|  |  |  | 				    SPA_PARAM_Meta, | 
					
						
							|  |  |  | 				    SPA_PARAM_IO }; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (*index < SPA_N_ELEMENTS(list)) | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 			param = spa_pod_builder_add_object(&b, | 
					
						
							| 
									
										
										
										
											2018-09-05 16:41:07 +02:00
										 |  |  | 					SPA_TYPE_OBJECT_ParamList, id, | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 					SPA_PARAM_LIST_id, SPA_POD_Id(list[*index])); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		else | 
					
						
							|  |  |  | 			return 0; | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 		break; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-25 12:08:29 +02:00
										 |  |  | 	case SPA_PARAM_EnumFormat: | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		if ((res = port_enum_formats(node, direction, port_id, index, ¶m, &b)) <= 0) | 
					
						
							|  |  |  | 			return res; | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 		break; | 
					
						
							| 
									
										
										
										
											2018-08-25 12:08:29 +02:00
										 |  |  | 	case SPA_PARAM_Format: | 
					
						
							| 
									
										
										
										
											2018-08-29 15:53:26 +02:00
										 |  |  | 		if (!port->have_format) | 
					
						
							|  |  |  | 			return -EIO; | 
					
						
							|  |  |  | 		if (*index > 0) | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		param = spa_format_audio_raw_build(&b, id, &port->format.info.raw); | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 		break; | 
					
						
							| 
									
										
										
										
											2018-08-25 12:08:29 +02:00
										 |  |  | 	case SPA_PARAM_Buffers: | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2018-06-25 14:34:36 +02:00
										 |  |  | 		uint32_t buffers, size; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		if (!port->have_format) | 
					
						
							|  |  |  | 			return -EIO; | 
					
						
							|  |  |  | 		if (*index > 0) | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (other->n_buffers > 0) { | 
					
						
							| 
									
										
										
										
											2018-06-25 14:34:36 +02:00
										 |  |  | 			buffers = other->n_buffers; | 
					
						
							|  |  |  | 			size = other->size / other->stride; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			buffers = 1; | 
					
						
							| 
									
										
										
										
											2018-08-20 17:56:59 +02:00
										 |  |  | 			size = 1024; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-06-25 14:34:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 		param = spa_pod_builder_add_object(&b, | 
					
						
							| 
									
										
										
										
											2018-08-27 15:03:11 +02:00
										 |  |  | 			SPA_TYPE_OBJECT_ParamBuffers, id, | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 			SPA_PARAM_BUFFERS_buffers, SPA_POD_CHOICE_RANGE_Int(buffers, 1, MAX_BUFFERS), | 
					
						
							|  |  |  | 			SPA_PARAM_BUFFERS_blocks,  SPA_POD_Int(port->blocks), | 
					
						
							|  |  |  | 			SPA_PARAM_BUFFERS_size,    SPA_POD_CHOICE_RANGE_Int( | 
					
						
							| 
									
										
										
										
											2018-09-05 16:41:07 +02:00
										 |  |  | 							size * port->stride, | 
					
						
							|  |  |  | 							16 * port->stride, | 
					
						
							|  |  |  | 							INT32_MAX / port->stride), | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 			SPA_PARAM_BUFFERS_stride,  SPA_POD_Int(port->stride), | 
					
						
							|  |  |  | 			SPA_PARAM_BUFFERS_align,   SPA_POD_Int(16)); | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 		break; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-25 12:08:29 +02:00
										 |  |  | 	case SPA_PARAM_Meta: | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		if (!port->have_format) | 
					
						
							|  |  |  | 			return -EIO; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		switch (*index) { | 
					
						
							|  |  |  | 		case 0: | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 			param = spa_pod_builder_add_object(&b, | 
					
						
							| 
									
										
										
										
											2018-08-27 15:03:11 +02:00
										 |  |  | 				SPA_TYPE_OBJECT_ParamMeta, id, | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 				SPA_PARAM_META_type, SPA_POD_Id(SPA_META_Header), | 
					
						
							|  |  |  | 				SPA_PARAM_META_size, SPA_POD_Int(sizeof(struct spa_meta_header))); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 		break; | 
					
						
							| 
									
										
										
										
											2018-08-25 12:08:29 +02:00
										 |  |  | 	case SPA_PARAM_IO: | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		switch (*index) { | 
					
						
							|  |  |  | 		case 0: | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 			param = spa_pod_builder_add_object(&b, | 
					
						
							| 
									
										
										
										
											2018-08-27 15:03:11 +02:00
										 |  |  | 				SPA_TYPE_OBJECT_ParamIO, id, | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 				SPA_PARAM_IO_id,   SPA_POD_Id(SPA_IO_Buffers), | 
					
						
							|  |  |  | 				SPA_PARAM_IO_size, SPA_POD_Int(sizeof(struct spa_io_buffers))); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2018-08-29 12:43:31 +02:00
										 |  |  | 		case 1: | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 			param = spa_pod_builder_add_object(&b, | 
					
						
							| 
									
										
										
										
											2018-08-29 12:43:31 +02:00
										 |  |  | 				SPA_TYPE_OBJECT_ParamIO, id, | 
					
						
							| 
									
										
										
										
											2019-01-16 11:05:12 +01:00
										 |  |  | 				SPA_PARAM_IO_id,   SPA_POD_Id(SPA_IO_Range), | 
					
						
							|  |  |  | 				SPA_PARAM_IO_size, SPA_POD_Int(sizeof(struct spa_io_range))); | 
					
						
							| 
									
										
										
										
											2018-08-29 12:43:31 +02:00
										 |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		default: | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		return -ENOENT; | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	(*index)++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (spa_pod_filter(builder, result, param, filter) < 0) | 
					
						
							|  |  |  | 		goto next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int clear_buffers(struct impl *this, struct port *port) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (port->n_buffers > 0) { | 
					
						
							| 
									
										
										
										
											2018-05-11 09:56:46 +02:00
										 |  |  | 		spa_log_debug(this->log, NAME " %p: clear buffers %p", this, port); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		port->n_buffers = 0; | 
					
						
							|  |  |  | 		spa_list_init(&port->queue); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int port_set_format(struct spa_node *node, | 
					
						
							|  |  |  | 			   enum spa_direction direction, | 
					
						
							|  |  |  | 			   uint32_t port_id, | 
					
						
							|  |  |  | 			   uint32_t flags, | 
					
						
							|  |  |  | 			   const struct spa_pod *format) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct impl *this = SPA_CONTAINER_OF(node, struct impl, node); | 
					
						
							|  |  |  | 	struct port *port, *other; | 
					
						
							|  |  |  | 	int res = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	port = GET_PORT(this, direction, port_id); | 
					
						
							|  |  |  | 	other = GET_PORT(this, SPA_DIRECTION_REVERSE(direction), port_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (format == NULL) { | 
					
						
							|  |  |  | 		if (port->have_format) { | 
					
						
							|  |  |  | 			port->have_format = false; | 
					
						
							|  |  |  | 			clear_buffers(this, port); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		struct spa_audio_info info = { 0 }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-29 14:50:58 +02:00
										 |  |  | 		if ((res = spa_format_parse(format, &info.media_type, &info.media_subtype)) < 0) | 
					
						
							|  |  |  | 			return res; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 		if (info.media_type != SPA_MEDIA_TYPE_audio || | 
					
						
							|  |  |  | 		    info.media_subtype != SPA_MEDIA_SUBTYPE_raw) | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 			return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 		if (spa_format_audio_raw_parse(format, &info.info.raw) < 0) | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 			return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 17:03:56 +02:00
										 |  |  | 		if (info.info.raw.format != SPA_AUDIO_FORMAT_F32P) | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 			return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		port->stride = sizeof(float); | 
					
						
							|  |  |  | 		port->blocks = info.info.raw.channels; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (other->have_format) { | 
					
						
							|  |  |  | 			if ((res = setup_convert(this, direction, &info)) < 0) | 
					
						
							|  |  |  | 				return res; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		port->format = info; | 
					
						
							|  |  |  | 		port->have_format = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-11 09:56:46 +02:00
										 |  |  | 		spa_log_debug(this->log, NAME " %p: set format on port %d %d", this, port_id, res); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | impl_node_port_set_param(struct spa_node *node, | 
					
						
							|  |  |  | 			 enum spa_direction direction, uint32_t port_id, | 
					
						
							|  |  |  | 			 uint32_t id, uint32_t flags, | 
					
						
							|  |  |  | 			 const struct spa_pod *param) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	spa_return_val_if_fail(node != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 	spa_return_val_if_fail(CHECK_PORT(node, direction, port_id), -EINVAL); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-25 12:08:29 +02:00
										 |  |  | 	if (id == SPA_PARAM_Format) { | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		return port_set_format(node, direction, port_id, flags, param); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return -ENOENT; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | impl_node_port_use_buffers(struct spa_node *node, | 
					
						
							|  |  |  | 			   enum spa_direction direction, | 
					
						
							|  |  |  | 			   uint32_t port_id, | 
					
						
							|  |  |  | 			   struct spa_buffer **buffers, | 
					
						
							|  |  |  | 			   uint32_t n_buffers) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct impl *this; | 
					
						
							|  |  |  | 	struct port *port; | 
					
						
							|  |  |  | 	uint32_t i, size = SPA_ID_INVALID; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(node != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	this = SPA_CONTAINER_OF(node, struct impl, node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(CHECK_PORT(this, direction, port_id), -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	port = GET_PORT(this, direction, port_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(port->have_format, -EIO); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-11 09:56:46 +02:00
										 |  |  | 	spa_log_debug(this->log, NAME " %p: use buffers %d on port %d", this, n_buffers, port_id); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	clear_buffers(this, port); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < n_buffers; i++) { | 
					
						
							|  |  |  | 		struct buffer *b; | 
					
						
							|  |  |  | 		struct spa_data *d = buffers[i]->datas; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		b = &port->buffers[i]; | 
					
						
							| 
									
										
										
										
											2019-01-07 17:57:03 +01:00
										 |  |  | 		b->id = i; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		b->flags = 0; | 
					
						
							|  |  |  | 		b->outbuf = buffers[i]; | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 		b->h = spa_buffer_find_meta_data(buffers[i], SPA_META_Header, sizeof(*b->h)); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (size == SPA_ID_INVALID) | 
					
						
							|  |  |  | 			size = d[0].maxsize; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			if (size != d[0].maxsize) | 
					
						
							|  |  |  | 				return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-23 17:47:57 +02:00
										 |  |  | 		if (!((d[0].type == SPA_DATA_MemPtr || | 
					
						
							|  |  |  | 		       d[0].type == SPA_DATA_MemFd || | 
					
						
							|  |  |  | 		       d[0].type == SPA_DATA_DmaBuf) && d[0].data != NULL)) { | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 			spa_log_error(this->log, NAME " %p: invalid memory on buffer %p", this, | 
					
						
							|  |  |  | 				      buffers[i]); | 
					
						
							|  |  |  | 			return -EINVAL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (direction == SPA_DIRECTION_OUTPUT) | 
					
						
							|  |  |  | 			spa_list_append(&port->queue, &b->link); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			SPA_FLAG_SET(b->flags, BUFFER_FLAG_OUT); | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		port->offset = 0; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	port->n_buffers = n_buffers; | 
					
						
							|  |  |  | 	port->size = size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | impl_node_port_alloc_buffers(struct spa_node *node, | 
					
						
							|  |  |  | 			     enum spa_direction direction, | 
					
						
							|  |  |  | 			     uint32_t port_id, | 
					
						
							|  |  |  | 			     struct spa_pod **params, | 
					
						
							|  |  |  | 			     uint32_t n_params, | 
					
						
							|  |  |  | 			     struct spa_buffer **buffers, | 
					
						
							|  |  |  | 			     uint32_t *n_buffers) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return -ENOTSUP; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | impl_node_port_set_io(struct spa_node *node, | 
					
						
							|  |  |  | 		      enum spa_direction direction, uint32_t port_id, | 
					
						
							|  |  |  | 		      uint32_t id, void *data, size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct impl *this; | 
					
						
							|  |  |  | 	struct port *port; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(node != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	this = SPA_CONTAINER_OF(node, struct impl, node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(CHECK_PORT(this, direction, port_id), -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	port = GET_PORT(this, direction, port_id); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 18:16:41 +02:00
										 |  |  | 	switch (id) { | 
					
						
							|  |  |  | 	case SPA_IO_Buffers: | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		port->io = data; | 
					
						
							| 
									
										
										
										
											2018-08-28 18:16:41 +02:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 	case SPA_IO_Range: | 
					
						
							| 
									
										
										
										
											2018-11-16 16:54:56 +01:00
										 |  |  | 		port->io_range = data; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case SPA_IO_Control: | 
					
						
							|  |  |  | 		port->io_control = data; | 
					
						
							| 
									
										
										
										
											2018-08-28 18:16:41 +02:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		return -ENOENT; | 
					
						
							| 
									
										
										
										
											2018-08-28 18:16:41 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void recycle_buffer(struct impl *this, uint32_t id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct port *port = GET_OUT_PORT(this, 0); | 
					
						
							|  |  |  | 	struct buffer *b = &port->buffers[id]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (SPA_FLAG_CHECK(b->flags, BUFFER_FLAG_OUT)) { | 
					
						
							|  |  |  | 		spa_list_append(&port->queue, &b->link); | 
					
						
							|  |  |  | 		SPA_FLAG_UNSET(b->flags, BUFFER_FLAG_OUT); | 
					
						
							|  |  |  | 		spa_log_trace(this->log, NAME " %p: recycle buffer %d", this, id); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | static struct buffer *peek_buffer(struct impl *this, struct port *port) | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct buffer *b; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (spa_list_is_empty(&port->queue)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	b = spa_list_first(&port->queue, struct buffer, link); | 
					
						
							|  |  |  | 	return b; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | static void dequeue_buffer(struct impl *this, struct buffer *b) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	spa_list_remove(&b->link); | 
					
						
							|  |  |  | 	SPA_FLAG_SET(b->flags, BUFFER_FLAG_OUT); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int impl_node_port_reuse_buffer(struct spa_node *node, uint32_t port_id, uint32_t buffer_id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct impl *this; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(node != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	this = SPA_CONTAINER_OF(node, struct impl, node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(CHECK_PORT(this, SPA_DIRECTION_OUTPUT, port_id), -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	recycle_buffer(this, buffer_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | impl_node_port_send_command(struct spa_node *node, | 
					
						
							|  |  |  | 			    enum spa_direction direction, | 
					
						
							|  |  |  | 			    uint32_t port_id, | 
					
						
							|  |  |  | 			    const struct spa_command *command) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return -ENOTSUP; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 16:54:56 +01:00
										 |  |  | static int process_control(struct impl *this, struct port *port, struct spa_pod_sequence *sequence) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spa_pod_control *c; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	SPA_POD_SEQUENCE_FOREACH(sequence, c) { | 
					
						
							|  |  |  | 		switch (c->type) { | 
					
						
							|  |  |  | 		case SPA_CONTROL_Properties: | 
					
						
							|  |  |  | 			apply_props(this, (const struct spa_pod *) &c->value); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | static int impl_node_process(struct spa_node *node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct impl *this; | 
					
						
							|  |  |  | 	struct port *outport, *inport; | 
					
						
							|  |  |  | 	struct spa_io_buffers *outio, *inio; | 
					
						
							|  |  |  | 	struct buffer *sbuf, *dbuf; | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 	struct spa_buffer *sb, *db; | 
					
						
							| 
									
										
										
										
											2018-06-08 12:27:19 +02:00
										 |  |  | 	uint32_t i, size, in_len, out_len, pin_len, pout_len, maxsize; | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 	int res = 0; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(node != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	this = SPA_CONTAINER_OF(node, struct impl, node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	outport = GET_OUT_PORT(this, 0); | 
					
						
							|  |  |  | 	inport = GET_IN_PORT(this, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	outio = outport->io; | 
					
						
							|  |  |  | 	inio = inport->io; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(outio != NULL, -EIO); | 
					
						
							|  |  |  | 	spa_return_val_if_fail(inio != NULL, -EIO); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 16:54:56 +01:00
										 |  |  | 	spa_log_trace(this->log, NAME " %p: status %d %d %p", this, inio->status, outio->status, outport->io_control); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (outport->io_control) | 
					
						
							|  |  |  | 		process_control(this, outport, &outport->io_control->sequence); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (outio->status == SPA_STATUS_HAVE_BUFFER) | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 		return SPA_STATUS_HAVE_BUFFER; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (inio->status != SPA_STATUS_HAVE_BUFFER) | 
					
						
							|  |  |  | 		return SPA_STATUS_NEED_BUFFER; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* recycle */ | 
					
						
							|  |  |  | 	if (outio->buffer_id < outport->n_buffers) { | 
					
						
							|  |  |  | 		recycle_buffer(this, outio->buffer_id); | 
					
						
							|  |  |  | 		outio->buffer_id = SPA_ID_INVALID; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (inio->buffer_id >= inport->n_buffers) | 
					
						
							|  |  |  | 		return inio->status = -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 	if ((dbuf = peek_buffer(this, outport)) == NULL) | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		return outio->status = -EPIPE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sbuf = &inport->buffers[inio->buffer_id]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 	sb = sbuf->outbuf; | 
					
						
							|  |  |  | 	db = dbuf->outbuf; | 
					
						
							| 
									
										
										
										
											2018-04-10 16:54:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 	size = sb->datas[0].chunk->size; | 
					
						
							|  |  |  | 	maxsize = db->datas[0].maxsize; | 
					
						
							| 
									
										
										
										
											2018-11-16 16:54:56 +01:00
										 |  |  | 	if (outport->io_range) | 
					
						
							|  |  |  | 		maxsize = SPA_MIN(outport->io_range->max_size, maxsize); | 
					
						
							| 
									
										
										
										
											2018-04-10 16:54:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-08 12:27:19 +02:00
										 |  |  | 	pin_len = in_len = (size - inport->offset) / sizeof(float); | 
					
						
							|  |  |  | 	pout_len = out_len = (maxsize - outport->offset) / sizeof(float); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 	for (i = 0; i < sb->n_datas; i++) { | 
					
						
							| 
									
										
										
										
											2018-12-05 15:58:55 +01:00
										 |  |  | 		void *src, *dst; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-08 12:27:19 +02:00
										 |  |  | 		in_len = pin_len; | 
					
						
							|  |  |  | 		out_len = pout_len; | 
					
						
							| 
									
										
										
										
											2018-12-05 15:58:55 +01:00
										 |  |  | 		src = SPA_MEMBER(sb->datas[i].data, inport->offset, void); | 
					
						
							|  |  |  | 		dst = SPA_MEMBER(db->datas[i].data, outport->offset, void); | 
					
						
							| 
									
										
										
										
											2018-06-08 12:27:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 15:58:55 +01:00
										 |  |  | 		resample_process(&this->resample, i, src, &in_len, dst, &out_len); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-13 16:23:02 +01:00
										 |  |  | 		spa_log_trace(this->log, NAME " %p: in %d/%d %ld %d out %d/%d %ld %d", | 
					
						
							|  |  |  | 				this, pin_len, in_len, size / sizeof(float), inport->offset, | 
					
						
							|  |  |  | 				pout_len, out_len, maxsize / sizeof(float), outport->offset); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-25 17:09:45 +02:00
										 |  |  | 		db->datas[i].chunk->offset = 0; | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 		db->datas[i].chunk->size = outport->offset + (out_len * sizeof(float)); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 	inport->offset += in_len * sizeof(float); | 
					
						
							|  |  |  | 	if (inport->offset >= size) { | 
					
						
							|  |  |  | 		inio->status = SPA_STATUS_NEED_BUFFER; | 
					
						
							|  |  |  | 		inport->offset = 0; | 
					
						
							|  |  |  | 		SPA_FLAG_SET(res, SPA_STATUS_NEED_BUFFER); | 
					
						
							| 
									
										
										
										
											2018-11-16 16:54:56 +01:00
										 |  |  | 		if (outport->io_range == NULL) | 
					
						
							| 
									
										
										
										
											2018-06-25 17:09:45 +02:00
										 |  |  | 			maxsize = 0; | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	outport->offset += out_len * sizeof(float); | 
					
						
							| 
									
										
										
										
											2018-10-22 11:32:53 +02:00
										 |  |  | 	if (outport->offset > 0 && outport->offset >= maxsize) { | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 		outio->status = SPA_STATUS_HAVE_BUFFER; | 
					
						
							| 
									
										
										
										
											2019-01-07 17:57:03 +01:00
										 |  |  | 		outio->buffer_id = dbuf->id; | 
					
						
							| 
									
										
										
										
											2018-04-19 20:12:24 +02:00
										 |  |  | 		dequeue_buffer(this, dbuf); | 
					
						
							|  |  |  | 		outport->offset = 0; | 
					
						
							|  |  |  | 		SPA_FLAG_SET(res, SPA_STATUS_HAVE_BUFFER); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct spa_node impl_node = { | 
					
						
							|  |  |  | 	SPA_VERSION_NODE, | 
					
						
							|  |  |  | 	impl_node_enum_params, | 
					
						
							|  |  |  | 	impl_node_set_param, | 
					
						
							| 
									
										
										
										
											2018-10-23 12:31:41 +02:00
										 |  |  | 	impl_node_set_io, | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 	impl_node_send_command, | 
					
						
							|  |  |  | 	impl_node_set_callbacks, | 
					
						
							|  |  |  | 	impl_node_get_n_ports, | 
					
						
							|  |  |  | 	impl_node_get_port_ids, | 
					
						
							|  |  |  | 	impl_node_add_port, | 
					
						
							|  |  |  | 	impl_node_remove_port, | 
					
						
							|  |  |  | 	impl_node_port_get_info, | 
					
						
							|  |  |  | 	impl_node_port_enum_params, | 
					
						
							|  |  |  | 	impl_node_port_set_param, | 
					
						
							|  |  |  | 	impl_node_port_use_buffers, | 
					
						
							|  |  |  | 	impl_node_port_alloc_buffers, | 
					
						
							|  |  |  | 	impl_node_port_set_io, | 
					
						
							|  |  |  | 	impl_node_port_reuse_buffer, | 
					
						
							|  |  |  | 	impl_node_port_send_command, | 
					
						
							|  |  |  | 	impl_node_process, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-27 15:03:11 +02:00
										 |  |  | static int impl_get_interface(struct spa_handle *handle, uint32_t type, void **interface) | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct impl *this; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(handle != NULL, -EINVAL); | 
					
						
							|  |  |  | 	spa_return_val_if_fail(interface != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	this = (struct impl *) handle; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-27 15:03:11 +02:00
										 |  |  | 	if (type == SPA_TYPE_INTERFACE_Node) | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 		*interface = &this->node; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return -ENOENT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int impl_clear(struct spa_handle *handle) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-04-10 16:54:15 +02:00
										 |  |  | 	struct impl *this; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(handle != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	this = (struct impl *) handle; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 16:05:40 +01:00
										 |  |  | 	if (this->resample.free) | 
					
						
							|  |  |  | 		resample_free(&this->resample); | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static size_t | 
					
						
							|  |  |  | impl_get_size(const struct spa_handle_factory *factory, | 
					
						
							|  |  |  | 	      const struct spa_dict *params) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return sizeof(struct impl); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | impl_init(const struct spa_handle_factory *factory, | 
					
						
							|  |  |  | 	  struct spa_handle *handle, | 
					
						
							|  |  |  | 	  const struct spa_dict *info, | 
					
						
							|  |  |  | 	  const struct spa_support *support, | 
					
						
							|  |  |  | 	  uint32_t n_support) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct impl *this; | 
					
						
							|  |  |  | 	struct port *port; | 
					
						
							|  |  |  | 	uint32_t i; | 
					
						
							| 
									
										
										
										
											2018-12-05 15:58:55 +01:00
										 |  |  | 	const char *str; | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	spa_return_val_if_fail(factory != NULL, -EINVAL); | 
					
						
							|  |  |  | 	spa_return_val_if_fail(handle != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	handle->get_interface = impl_get_interface; | 
					
						
							|  |  |  | 	handle->clear = impl_clear; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	this = (struct impl *) handle; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < n_support; i++) { | 
					
						
							| 
									
										
										
										
											2018-12-19 16:47:20 +01:00
										 |  |  | 		switch (support[i].type) { | 
					
						
							|  |  |  | 		case SPA_TYPE_INTERFACE_Log: | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 			this->log = support[i].data; | 
					
						
							| 
									
										
										
										
											2018-12-19 16:47:20 +01:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		case SPA_TYPE_INTERFACE_CPU: | 
					
						
							|  |  |  | 			this->cpu = support[i].data; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-19 16:47:20 +01:00
										 |  |  | 	if (this->cpu) | 
					
						
							|  |  |  | 		this->resample.cpu_flags = spa_cpu_get_flags(this->cpu); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 15:58:55 +01:00
										 |  |  | 	if (info != NULL && (str = spa_dict_lookup(info, "resample.peaks")) != NULL) | 
					
						
							|  |  |  | 		this->monitor = atoi(str); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | 	this->node = impl_node; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	port = GET_OUT_PORT(this, 0); | 
					
						
							|  |  |  | 	port->id = 0; | 
					
						
							|  |  |  | 	port->info.flags = SPA_PORT_INFO_FLAG_CAN_USE_BUFFERS; | 
					
						
							|  |  |  | 	spa_list_init(&port->queue); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	port = GET_IN_PORT(this, 0); | 
					
						
							|  |  |  | 	port->id = 0; | 
					
						
							|  |  |  | 	port->info.flags = SPA_PORT_INFO_FLAG_CAN_USE_BUFFERS; | 
					
						
							|  |  |  | 	spa_list_init(&port->queue); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	props_reset(&this->props); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct spa_interface_info impl_interfaces[] = { | 
					
						
							| 
									
										
										
										
											2018-08-27 15:03:11 +02:00
										 |  |  | 	{SPA_TYPE_INTERFACE_Node,}, | 
					
						
							| 
									
										
										
										
											2018-04-10 15:54:29 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | impl_enum_interface_info(const struct spa_handle_factory *factory, | 
					
						
							|  |  |  | 			 const struct spa_interface_info **info, | 
					
						
							|  |  |  | 			 uint32_t *index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	spa_return_val_if_fail(factory != NULL, -EINVAL); | 
					
						
							|  |  |  | 	spa_return_val_if_fail(info != NULL, -EINVAL); | 
					
						
							|  |  |  | 	spa_return_val_if_fail(index != NULL, -EINVAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (*index) { | 
					
						
							|  |  |  | 	case 0: | 
					
						
							|  |  |  | 		*info = &impl_interfaces[*index]; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	(*index)++; | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const struct spa_handle_factory spa_resample_factory = { | 
					
						
							|  |  |  | 	SPA_VERSION_HANDLE_FACTORY, | 
					
						
							|  |  |  | 	NAME, | 
					
						
							|  |  |  | 	NULL, | 
					
						
							|  |  |  | 	impl_get_size, | 
					
						
							|  |  |  | 	impl_init, | 
					
						
							|  |  |  | 	impl_enum_interface_info, | 
					
						
							|  |  |  | }; |