mirror of
				https://gitlab.freedesktop.org/pipewire/pipewire.git
				synced 2025-11-03 09:01:54 -05:00 
			
		
		
		
	module-rtp: Add source/destination.ip to props
Make it possible to change source.ip in module-rtp-source and destination.ip in module-rtp-sink.
This commit is contained in:
		
							parent
							
								
									e84184f28f
								
							
						
					
					
						commit
						4715fa1775
					
				
					 4 changed files with 187 additions and 50 deletions
				
			
		| 
						 | 
				
			
			@ -186,47 +186,6 @@ struct impl {
 | 
			
		|||
	int rtp_fd;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static void stream_destroy(void *d)
 | 
			
		||||
{
 | 
			
		||||
	struct impl *impl = d;
 | 
			
		||||
	impl->stream = NULL;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void stream_send_packet(void *data, struct iovec *iov, size_t iovlen)
 | 
			
		||||
{
 | 
			
		||||
	struct impl *impl = data;
 | 
			
		||||
	struct msghdr msg;
 | 
			
		||||
	ssize_t n;
 | 
			
		||||
 | 
			
		||||
	spa_zero(msg);
 | 
			
		||||
	msg.msg_iov = iov;
 | 
			
		||||
	msg.msg_iovlen = iovlen;
 | 
			
		||||
	msg.msg_control = NULL;
 | 
			
		||||
	msg.msg_controllen = 0;
 | 
			
		||||
	msg.msg_flags = 0;
 | 
			
		||||
 | 
			
		||||
	n = sendmsg(impl->rtp_fd, &msg, MSG_NOSIGNAL);
 | 
			
		||||
	if (n < 0)
 | 
			
		||||
		pw_log_warn("sendmsg() failed: %m");
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void stream_state_changed(void *data, bool started, const char *error)
 | 
			
		||||
{
 | 
			
		||||
	struct impl *impl = data;
 | 
			
		||||
 | 
			
		||||
	if (error) {
 | 
			
		||||
		pw_log_error("stream error: %s", error);
 | 
			
		||||
		pw_impl_module_schedule_destroy(impl->module);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static const struct rtp_stream_events stream_events = {
 | 
			
		||||
	RTP_VERSION_STREAM_EVENTS,
 | 
			
		||||
	.destroy = stream_destroy,
 | 
			
		||||
	.state_changed = stream_state_changed,
 | 
			
		||||
	.send_packet = stream_send_packet,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static int parse_address(const char *address, uint16_t port,
 | 
			
		||||
		struct sockaddr_storage *addr, socklen_t *len)
 | 
			
		||||
{
 | 
			
		||||
| 
						 | 
				
			
			@ -315,6 +274,123 @@ error:
 | 
			
		|||
	return res;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void stream_destroy(void *d)
 | 
			
		||||
{
 | 
			
		||||
	struct impl *impl = d;
 | 
			
		||||
	impl->stream = NULL;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void stream_send_packet(void *data, struct iovec *iov, size_t iovlen)
 | 
			
		||||
{
 | 
			
		||||
	struct impl *impl = data;
 | 
			
		||||
	struct msghdr msg;
 | 
			
		||||
	ssize_t n;
 | 
			
		||||
 | 
			
		||||
	spa_zero(msg);
 | 
			
		||||
	msg.msg_iov = iov;
 | 
			
		||||
	msg.msg_iovlen = iovlen;
 | 
			
		||||
	msg.msg_control = NULL;
 | 
			
		||||
	msg.msg_controllen = 0;
 | 
			
		||||
	msg.msg_flags = 0;
 | 
			
		||||
 | 
			
		||||
	n = sendmsg(impl->rtp_fd, &msg, MSG_NOSIGNAL);
 | 
			
		||||
	if (n < 0)
 | 
			
		||||
		pw_log_warn("sendmsg() failed: %m");
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void stream_state_changed(void *data, bool started, const char *error)
 | 
			
		||||
{
 | 
			
		||||
	struct impl *impl = data;
 | 
			
		||||
 | 
			
		||||
	if (error) {
 | 
			
		||||
		pw_log_error("stream error: %s", error);
 | 
			
		||||
		pw_impl_module_schedule_destroy(impl->module);
 | 
			
		||||
	} else if (started) {
 | 
			
		||||
		int res;
 | 
			
		||||
 | 
			
		||||
		if ((res = make_socket(&impl->src_addr, impl->src_len,
 | 
			
		||||
					&impl->dst_addr, impl->dst_len,
 | 
			
		||||
					impl->mcast_loop, impl->ttl, impl->dscp,
 | 
			
		||||
					impl->ifname)) < 0) {
 | 
			
		||||
			pw_log_error("can't make socket: %s", spa_strerror(res));
 | 
			
		||||
			return;
 | 
			
		||||
		}
 | 
			
		||||
		impl->rtp_fd = res;
 | 
			
		||||
	} else {
 | 
			
		||||
		close(impl->rtp_fd);
 | 
			
		||||
		impl->rtp_fd = -1;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void stream_props_changed(struct impl *impl, uint32_t id, const struct spa_pod *param)
 | 
			
		||||
{
 | 
			
		||||
	struct spa_pod_object *obj = (struct spa_pod_object *)param;
 | 
			
		||||
	struct spa_pod_prop *prop;
 | 
			
		||||
 | 
			
		||||
	if (param == NULL)
 | 
			
		||||
		return;
 | 
			
		||||
 | 
			
		||||
	SPA_POD_OBJECT_FOREACH(obj, prop) {
 | 
			
		||||
		if (prop->key == SPA_PROP_params) {
 | 
			
		||||
			struct spa_pod *params = NULL;
 | 
			
		||||
			struct spa_pod_parser prs;
 | 
			
		||||
			struct spa_pod_frame f;
 | 
			
		||||
			const char *key;
 | 
			
		||||
			struct spa_pod *pod;
 | 
			
		||||
			const char *value;
 | 
			
		||||
 | 
			
		||||
			if (spa_pod_parse_object(param, SPA_TYPE_OBJECT_Props, NULL, SPA_PROP_params,
 | 
			
		||||
					SPA_POD_OPT_Pod(¶ms)) < 0)
 | 
			
		||||
				return;
 | 
			
		||||
			spa_pod_parser_pod(&prs, params);
 | 
			
		||||
			if (spa_pod_parser_push_struct(&prs, &f) < 0)
 | 
			
		||||
				return;
 | 
			
		||||
 | 
			
		||||
			while (true) {
 | 
			
		||||
				if (spa_pod_parser_get_string(&prs, &key) < 0)
 | 
			
		||||
					break;
 | 
			
		||||
				if (spa_pod_parser_get_pod(&prs, &pod) < 0)
 | 
			
		||||
					break;
 | 
			
		||||
				if (spa_pod_get_string(pod, &value) < 0)
 | 
			
		||||
					continue;
 | 
			
		||||
				pw_log_info("key '%s', value '%s'", key, value);
 | 
			
		||||
				if (!spa_streq(key, "destination.ip"))
 | 
			
		||||
					continue;
 | 
			
		||||
				if (parse_address(value, impl->dst_port, &impl->dst_addr,
 | 
			
		||||
						&impl->dst_len) < 0) {
 | 
			
		||||
					pw_log_error("invalid destination.ip: '%s'", value);
 | 
			
		||||
					break;
 | 
			
		||||
				}
 | 
			
		||||
				pw_properties_set(impl->stream_props, "rtp.destination.ip", value);
 | 
			
		||||
				struct spa_dict_item item[1];
 | 
			
		||||
				item[0] = SPA_DICT_ITEM_INIT("rtp.destination.ip", value);
 | 
			
		||||
				rtp_stream_update_properties(impl->stream, &SPA_DICT_INIT(item, 1));
 | 
			
		||||
				break;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void stream_param_changed(void *data, uint32_t id, const struct spa_pod *param)
 | 
			
		||||
{
 | 
			
		||||
	struct impl *impl = data;
 | 
			
		||||
 | 
			
		||||
	switch (id) {
 | 
			
		||||
	case SPA_PARAM_Props:
 | 
			
		||||
		if (param != NULL)
 | 
			
		||||
			stream_props_changed(impl, id, param);
 | 
			
		||||
		break;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static const struct rtp_stream_events stream_events = {
 | 
			
		||||
	RTP_VERSION_STREAM_EVENTS,
 | 
			
		||||
	.destroy = stream_destroy,
 | 
			
		||||
	.state_changed = stream_state_changed,
 | 
			
		||||
	.param_changed = stream_param_changed,
 | 
			
		||||
	.send_packet = stream_send_packet,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static int get_ip(const struct sockaddr_storage *sa, char *ip, size_t len)
 | 
			
		||||
{
 | 
			
		||||
	if (sa->ss_family == AF_INET) {
 | 
			
		||||
| 
						 | 
				
			
			@ -531,15 +607,6 @@ int pipewire__module_init(struct pw_impl_module *module, const char *args)
 | 
			
		|||
			&impl->core_listener,
 | 
			
		||||
			&core_events, impl);
 | 
			
		||||
 | 
			
		||||
	if ((res = make_socket(&impl->src_addr, impl->src_len,
 | 
			
		||||
					&impl->dst_addr, impl->dst_len,
 | 
			
		||||
					impl->mcast_loop, impl->ttl, impl->dscp,
 | 
			
		||||
					impl->ifname)) < 0) {
 | 
			
		||||
		pw_log_error("can't make socket: %s", spa_strerror(res));
 | 
			
		||||
		goto out;
 | 
			
		||||
	}
 | 
			
		||||
	impl->rtp_fd = res;
 | 
			
		||||
 | 
			
		||||
	impl->stream = rtp_stream_new(impl->core,
 | 
			
		||||
			PW_DIRECTION_INPUT, pw_properties_copy(stream_props),
 | 
			
		||||
			&stream_events, impl);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -378,10 +378,72 @@ static void stream_state_changed(void *data, bool started, const char *error)
 | 
			
		|||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void stream_props_changed(struct impl *impl, uint32_t id, const struct spa_pod *param)
 | 
			
		||||
{
 | 
			
		||||
	struct spa_pod_object *obj = (struct spa_pod_object *)param;
 | 
			
		||||
	struct spa_pod_prop *prop;
 | 
			
		||||
 | 
			
		||||
	if (param == NULL)
 | 
			
		||||
		return;
 | 
			
		||||
 | 
			
		||||
	SPA_POD_OBJECT_FOREACH(obj, prop) {
 | 
			
		||||
		if (prop->key == SPA_PROP_params) {
 | 
			
		||||
			struct spa_pod *params = NULL;
 | 
			
		||||
			struct spa_pod_parser prs;
 | 
			
		||||
			struct spa_pod_frame f;
 | 
			
		||||
			const char *key;
 | 
			
		||||
			struct spa_pod *pod;
 | 
			
		||||
			const char *value;
 | 
			
		||||
 | 
			
		||||
			if (spa_pod_parse_object(param, SPA_TYPE_OBJECT_Props, NULL, SPA_PROP_params,
 | 
			
		||||
					SPA_POD_OPT_Pod(¶ms)) < 0)
 | 
			
		||||
				return;
 | 
			
		||||
			spa_pod_parser_pod(&prs, params);
 | 
			
		||||
			if (spa_pod_parser_push_struct(&prs, &f) < 0)
 | 
			
		||||
				return;
 | 
			
		||||
 | 
			
		||||
			while (true) {
 | 
			
		||||
				if (spa_pod_parser_get_string(&prs, &key) < 0)
 | 
			
		||||
					break;
 | 
			
		||||
				if (spa_pod_parser_get_pod(&prs, &pod) < 0)
 | 
			
		||||
					break;
 | 
			
		||||
				if (spa_pod_get_string(pod, &value) < 0)
 | 
			
		||||
					continue;
 | 
			
		||||
				pw_log_info("key '%s', value '%s'", key, value);
 | 
			
		||||
				if (!spa_streq(key, "source.ip"))
 | 
			
		||||
					continue;
 | 
			
		||||
				if (parse_address(value, impl->src_port, &impl->src_addr,
 | 
			
		||||
						&impl->src_len) < 0) {
 | 
			
		||||
					pw_log_error("invalid source.ip: '%s'", value);
 | 
			
		||||
					break;
 | 
			
		||||
				}
 | 
			
		||||
				pw_properties_set(impl->stream_props, "rtp.source.ip", value);
 | 
			
		||||
				struct spa_dict_item item[1];
 | 
			
		||||
				item[0] = SPA_DICT_ITEM_INIT("rtp.source.ip", value);
 | 
			
		||||
				rtp_stream_update_properties(impl->stream, &SPA_DICT_INIT(item, 1));
 | 
			
		||||
				break;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void stream_param_changed(void *data, uint32_t id, const struct spa_pod *param)
 | 
			
		||||
{
 | 
			
		||||
	struct impl *impl = data;
 | 
			
		||||
 | 
			
		||||
	switch (id) {
 | 
			
		||||
	case SPA_PARAM_Props:
 | 
			
		||||
		if (param != NULL)
 | 
			
		||||
			stream_props_changed(impl, id, param);
 | 
			
		||||
		break;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static const struct rtp_stream_events stream_events = {
 | 
			
		||||
	RTP_VERSION_STREAM_EVENTS,
 | 
			
		||||
	.destroy = stream_destroy,
 | 
			
		||||
	.state_changed = stream_state_changed,
 | 
			
		||||
	.param_changed = stream_param_changed,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static void on_timer_event(void *data, uint64_t expirations)
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -602,6 +602,12 @@ void rtp_stream_destroy(struct rtp_stream *s)
 | 
			
		|||
	free(impl);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int rtp_stream_update_properties(struct rtp_stream *s, const struct spa_dict *dict)
 | 
			
		||||
{
 | 
			
		||||
	struct impl *impl = (struct impl*)s;
 | 
			
		||||
	return pw_stream_update_properties(impl->stream, dict);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int rtp_stream_receive_packet(struct rtp_stream *s, uint8_t *buffer, size_t len)
 | 
			
		||||
{
 | 
			
		||||
	struct impl *impl = (struct impl*)s;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -44,6 +44,8 @@ struct rtp_stream *rtp_stream_new(struct pw_core *core,
 | 
			
		|||
 | 
			
		||||
void rtp_stream_destroy(struct rtp_stream *s);
 | 
			
		||||
 | 
			
		||||
int rtp_stream_update_properties(struct rtp_stream *s, const struct spa_dict *dict);
 | 
			
		||||
 | 
			
		||||
int rtp_stream_receive_packet(struct rtp_stream *s, uint8_t *buffer, size_t len);
 | 
			
		||||
 | 
			
		||||
uint64_t rtp_stream_get_time(struct rtp_stream *s, uint64_t *rate);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue