mirror of
				https://gitlab.freedesktop.org/pipewire/pipewire.git
				synced 2025-11-03 09:01:54 -05:00 
			
		
		
		
	SPDX tags make the licensing information easy to understand and clear, and they are machine parseable. See https://spdx.dev for more information.
		
			
				
	
	
		
			140 lines
		
	
	
	
		
			2.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			140 lines
		
	
	
	
		
			2.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* Spa FFmpeg support */
 | 
						|
/* SPDX-FileCopyrightText: Copyright © 2018 Wim Taymans */
 | 
						|
/* SPDX-License-Identifier: MIT */
 | 
						|
 | 
						|
#include <errno.h>
 | 
						|
#include <stdio.h>
 | 
						|
 | 
						|
#include <spa/support/plugin.h>
 | 
						|
#include <spa/node/node.h>
 | 
						|
 | 
						|
#include <libavcodec/avcodec.h>
 | 
						|
 | 
						|
#include "ffmpeg.h"
 | 
						|
 | 
						|
static int
 | 
						|
ffmpeg_dec_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)
 | 
						|
{
 | 
						|
	if (factory == NULL || handle == NULL)
 | 
						|
		return -EINVAL;
 | 
						|
 | 
						|
	return spa_ffmpeg_dec_init(handle, info, support, n_support);
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
ffmpeg_enc_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)
 | 
						|
{
 | 
						|
	if (factory == NULL || handle == NULL)
 | 
						|
		return -EINVAL;
 | 
						|
 | 
						|
	return spa_ffmpeg_enc_init(handle, info, support, n_support);
 | 
						|
}
 | 
						|
 | 
						|
static const struct spa_interface_info ffmpeg_interfaces[] = {
 | 
						|
	{SPA_TYPE_INTERFACE_Node, },
 | 
						|
};
 | 
						|
 | 
						|
static int
 | 
						|
ffmpeg_enum_interface_info(const struct spa_handle_factory *factory,
 | 
						|
			   const struct spa_interface_info **info,
 | 
						|
			   uint32_t *index)
 | 
						|
{
 | 
						|
	if (factory == NULL || info == NULL || index == NULL)
 | 
						|
		return -EINVAL;
 | 
						|
 | 
						|
	if (*index < SPA_N_ELEMENTS(ffmpeg_interfaces))
 | 
						|
		*info = &ffmpeg_interfaces[(*index)++];
 | 
						|
	else
 | 
						|
		return 0;
 | 
						|
 | 
						|
	return 1;
 | 
						|
}
 | 
						|
 | 
						|
#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(58, 10, 100)
 | 
						|
static const AVCodec *find_codec_by_index(uint32_t index)
 | 
						|
{
 | 
						|
	static void *av_iter_data;
 | 
						|
	static uint32_t next_index;
 | 
						|
 | 
						|
	const AVCodec *c = NULL;
 | 
						|
 | 
						|
	if (index == 0) {
 | 
						|
		av_iter_data = NULL;
 | 
						|
		next_index = 0;
 | 
						|
	}
 | 
						|
 | 
						|
	while (next_index <= index) {
 | 
						|
		c = av_codec_iterate(&av_iter_data);
 | 
						|
		next_index += 1;
 | 
						|
 | 
						|
		if (!c)
 | 
						|
			break;
 | 
						|
	}
 | 
						|
 | 
						|
	return c;
 | 
						|
}
 | 
						|
#else
 | 
						|
static const AVCodec *find_codec_by_index(uint32_t index)
 | 
						|
{
 | 
						|
	static const AVCodec *last_codec;
 | 
						|
	static uint32_t next_index;
 | 
						|
 | 
						|
	if (index == 0) {
 | 
						|
		last_codec = NULL;
 | 
						|
		next_index = 0;
 | 
						|
	}
 | 
						|
 | 
						|
	while (next_index <= index) {
 | 
						|
		last_codec = av_codec_next(last_codec);
 | 
						|
		next_index += 1;
 | 
						|
 | 
						|
		if (!last_codec)
 | 
						|
			break;
 | 
						|
	}
 | 
						|
 | 
						|
	return last_codec;
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
SPA_EXPORT
 | 
						|
int spa_handle_factory_enum(const struct spa_handle_factory **factory, uint32_t *index)
 | 
						|
{
 | 
						|
	static char name[128];
 | 
						|
	static struct spa_handle_factory f = {
 | 
						|
		SPA_VERSION_HANDLE_FACTORY,
 | 
						|
		.name = name,
 | 
						|
		.enum_interface_info = ffmpeg_enum_interface_info,
 | 
						|
	};
 | 
						|
 | 
						|
#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(58, 10, 100)
 | 
						|
	avcodec_register_all();
 | 
						|
#endif
 | 
						|
 | 
						|
	const AVCodec *c = find_codec_by_index(*index);
 | 
						|
 | 
						|
	if (c == NULL)
 | 
						|
		return 0;
 | 
						|
 | 
						|
	if (av_codec_is_encoder(c)) {
 | 
						|
		snprintf(name, sizeof(name), "encoder.%s", c->name);
 | 
						|
		f.get_size = spa_ffmpeg_enc_get_size;
 | 
						|
		f.init = ffmpeg_enc_init;
 | 
						|
	} else {
 | 
						|
		snprintf(name, sizeof(name), "decoder.%s", c->name);
 | 
						|
		f.get_size = spa_ffmpeg_dec_get_size;
 | 
						|
		f.init = ffmpeg_dec_init;
 | 
						|
	}
 | 
						|
 | 
						|
	*factory = &f;
 | 
						|
	(*index)++;
 | 
						|
 | 
						|
	return 1;
 | 
						|
}
 |