Blame libfreerdp/codec/dsp_ffmpeg.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit 1fb8d4
 * Digital Sound Processing - FFMPEG backend
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2018 Armin Novak <armin.novak@thincast.com>
Packit 1fb8d4
 * Copyright 2018 Thincast Technologies GmbH
Packit 1fb8d4
 *
Packit 1fb8d4
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit 1fb8d4
 * you may not use this file except in compliance with the License.
Packit 1fb8d4
 * You may obtain a copy of the License at
Packit 1fb8d4
 *
Packit 1fb8d4
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit 1fb8d4
 *
Packit 1fb8d4
 * Unless required by applicable law or agreed to in writing, software
Packit 1fb8d4
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 1fb8d4
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 1fb8d4
 * See the License for the specific language governing permissions and
Packit 1fb8d4
 * limitations under the License.
Packit 1fb8d4
 */
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_CONFIG_H
Packit 1fb8d4
#include "config.h"
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#include <freerdp/log.h>
Packit 1fb8d4
Packit 1fb8d4
#include <libavcodec/avcodec.h>
Packit 1fb8d4
#include <libavutil/avutil.h>
Packit 1fb8d4
#include <libavutil/opt.h>
Packit 1fb8d4
#if defined(SWRESAMPLE_FOUND)
Packit 1fb8d4
#include <libswresample/swresample.h>
Packit 1fb8d4
#elif defined(AVRESAMPLE_FOUND)
Packit 1fb8d4
#include <libavresample/avresample.h>
Packit 1fb8d4
#else
Packit 1fb8d4
#error "libswresample or libavresample required"
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#include "dsp.h"
Packit 1fb8d4
#include "dsp_ffmpeg.h"
Packit 1fb8d4
Packit 1fb8d4
#define TAG FREERDP_TAG("dsp.ffmpeg")
Packit 1fb8d4
Packit 1fb8d4
struct _FREERDP_DSP_CONTEXT
Packit 1fb8d4
{
Packit 1fb8d4
	AUDIO_FORMAT format;
Packit 1fb8d4
Packit 1fb8d4
	BOOL isOpen;
Packit 1fb8d4
	BOOL encoder;
Packit 1fb8d4
Packit 1fb8d4
	UINT32 bufferedSamples;
Packit 1fb8d4
Packit 1fb8d4
	enum AVCodecID id;
Packit 1fb8d4
	AVCodec* codec;
Packit 1fb8d4
	AVCodecContext* context;
Packit 1fb8d4
	AVFrame* frame;
Packit 1fb8d4
	AVFrame* resampled;
Packit 1fb8d4
	AVFrame* buffered;
Packit 1fb8d4
	AVPacket* packet;
Packit 1fb8d4
#if defined(SWRESAMPLE_FOUND)
Packit 1fb8d4
	SwrContext* rcontext;
Packit 1fb8d4
#else
Packit 1fb8d4
	AVAudioResampleContext* rcontext;
Packit 1fb8d4
#endif
Packit 1fb8d4
};
Packit 1fb8d4
Packit 1fb8d4
static BOOL ffmpeg_codec_is_filtered(enum AVCodecID id, BOOL encoder)
Packit 1fb8d4
{
Packit 1fb8d4
	switch (id)
Packit 1fb8d4
	{
Packit 1fb8d4
#if !defined(WITH_DSP_EXPERIMENTAL)
Packit 1fb8d4
Packit 1fb8d4
		case AV_CODEC_ID_ADPCM_IMA_OKI:
Packit 1fb8d4
		case AV_CODEC_ID_MP3:
Packit 1fb8d4
		case AV_CODEC_ID_ADPCM_MS:
Packit 1fb8d4
		case AV_CODEC_ID_G723_1:
Packit 1fb8d4
			return TRUE;
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
		case AV_CODEC_ID_NONE:
Packit 1fb8d4
			return TRUE;
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static enum AVCodecID ffmpeg_get_avcodec(const AUDIO_FORMAT* format)
Packit 1fb8d4
{
Packit 1fb8d4
	const char* id;
Packit 1fb8d4
Packit 1fb8d4
	if (!format)
Packit 1fb8d4
		return AV_CODEC_ID_NONE;
Packit 1fb8d4
Packit 1fb8d4
	id = audio_format_get_tag_string(format->wFormatTag);
Packit 1fb8d4
Packit 1fb8d4
	switch (format->wFormatTag)
Packit 1fb8d4
	{
Packit 1fb8d4
		case WAVE_FORMAT_UNKNOWN:
Packit 1fb8d4
			return AV_CODEC_ID_NONE;
Packit 1fb8d4
Packit 1fb8d4
		case WAVE_FORMAT_PCM:
Packit 1fb8d4
			switch (format->wBitsPerSample)
Packit 1fb8d4
			{
Packit 1fb8d4
				case 16:
Packit 1fb8d4
					return AV_CODEC_ID_PCM_U16LE;
Packit 1fb8d4
Packit 1fb8d4
				case 8:
Packit 1fb8d4
					return AV_CODEC_ID_PCM_U8;
Packit 1fb8d4
Packit 1fb8d4
				default:
Packit 1fb8d4
					return AV_CODEC_ID_NONE;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
		case WAVE_FORMAT_DVI_ADPCM:
Packit 1fb8d4
			return AV_CODEC_ID_ADPCM_IMA_OKI;
Packit 1fb8d4
Packit 1fb8d4
		case WAVE_FORMAT_ADPCM:
Packit 1fb8d4
			return AV_CODEC_ID_ADPCM_MS;
Packit 1fb8d4
Packit 1fb8d4
		case WAVE_FORMAT_ALAW:
Packit 1fb8d4
			return AV_CODEC_ID_PCM_ALAW;
Packit 1fb8d4
Packit 1fb8d4
		case WAVE_FORMAT_MULAW:
Packit 1fb8d4
			return AV_CODEC_ID_PCM_MULAW;
Packit 1fb8d4
Packit 1fb8d4
		case WAVE_FORMAT_GSM610:
Packit 1fb8d4
			return AV_CODEC_ID_GSM_MS;
Packit 1fb8d4
Packit 1fb8d4
		case WAVE_FORMAT_MSG723:
Packit 1fb8d4
			return AV_CODEC_ID_G723_1;
Packit 1fb8d4
Packit 1fb8d4
		case WAVE_FORMAT_AAC_MS:
Packit 1fb8d4
			return AV_CODEC_ID_AAC;
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			return AV_CODEC_ID_NONE;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
Packit 1fb8d4
static int ffmpeg_sample_format(const AUDIO_FORMAT* format)
Packit 1fb8d4
{
Packit 1fb8d4
	switch (format->wFormatTag)
Packit 1fb8d4
	{
Packit 1fb8d4
		case WAVE_FORMAT_PCM:
Packit 1fb8d4
			switch (format->wBitsPerSample)
Packit 1fb8d4
			{
Packit 1fb8d4
				case 8:
Packit 1fb8d4
					return AV_SAMPLE_FMT_U8;
Packit 1fb8d4
Packit 1fb8d4
				case 16:
Packit 1fb8d4
					return AV_SAMPLE_FMT_S16;
Packit 1fb8d4
Packit 1fb8d4
				default:
Packit 1fb8d4
					return FALSE;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
		case WAVE_FORMAT_DVI_ADPCM:
Packit 1fb8d4
		case WAVE_FORMAT_ADPCM:
Packit 1fb8d4
			return AV_SAMPLE_FMT_S16P;
Packit 1fb8d4
Packit 1fb8d4
		case WAVE_FORMAT_MPEGLAYER3:
Packit 1fb8d4
		case WAVE_FORMAT_AAC_MS:
Packit 1fb8d4
			return AV_SAMPLE_FMT_FLTP;
Packit 1fb8d4
Packit 1fb8d4
		case WAVE_FORMAT_MSG723:
Packit 1fb8d4
		case WAVE_FORMAT_GSM610:
Packit 1fb8d4
			return AV_SAMPLE_FMT_S16P;
Packit 1fb8d4
Packit 1fb8d4
		case WAVE_FORMAT_ALAW:
Packit 1fb8d4
			return AV_SAMPLE_FMT_S16;
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static void ffmpeg_close_context(FREERDP_DSP_CONTEXT* context)
Packit 1fb8d4
{
Packit 1fb8d4
	if (context)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (context->context)
Packit 1fb8d4
			avcodec_free_context(&context->context);
Packit 1fb8d4
Packit 1fb8d4
		if (context->frame)
Packit 1fb8d4
			av_frame_free(&context->frame);
Packit 1fb8d4
Packit 1fb8d4
		if (context->resampled)
Packit 1fb8d4
			av_frame_free(&context->resampled);
Packit 1fb8d4
Packit 1fb8d4
		if (context->buffered)
Packit 1fb8d4
			av_frame_free(&context->buffered);
Packit 1fb8d4
Packit 1fb8d4
		if (context->packet)
Packit 1fb8d4
			av_packet_free(&context->packet);
Packit 1fb8d4
Packit 1fb8d4
		if (context->rcontext)
Packit 1fb8d4
		{
Packit 1fb8d4
#if defined(SWRESAMPLE_FOUND)
Packit 1fb8d4
			swr_free(&context->rcontext);
Packit 1fb8d4
#else
Packit 1fb8d4
			avresample_free(&context->rcontext);
Packit 1fb8d4
#endif
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		context->id = AV_CODEC_ID_NONE;
Packit 1fb8d4
		context->codec = NULL;
Packit 1fb8d4
		context->isOpen = FALSE;
Packit 1fb8d4
		context->context = NULL;
Packit 1fb8d4
		context->frame = NULL;
Packit 1fb8d4
		context->resampled = NULL;
Packit 1fb8d4
		context->packet = NULL;
Packit 1fb8d4
		context->rcontext = NULL;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL ffmpeg_open_context(FREERDP_DSP_CONTEXT* context)
Packit 1fb8d4
{
Packit 1fb8d4
	int ret;
Packit 1fb8d4
	int layout;
Packit 1fb8d4
	const AUDIO_FORMAT* format;
Packit 1fb8d4
Packit 1fb8d4
	if (!context || context->isOpen)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	format = &context->format;
Packit 1fb8d4
Packit 1fb8d4
	if (!format)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	layout = av_get_default_channel_layout(format->nChannels);
Packit 1fb8d4
	context->id = ffmpeg_get_avcodec(format);
Packit 1fb8d4
Packit 1fb8d4
	if (ffmpeg_codec_is_filtered(context->id, context->encoder))
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	if (context->encoder)
Packit 1fb8d4
		context->codec = avcodec_find_encoder(context->id);
Packit 1fb8d4
	else
Packit 1fb8d4
		context->codec = avcodec_find_decoder(context->id);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->codec)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	context->context = avcodec_alloc_context3(context->codec);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->context)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	switch (context->id)
Packit 1fb8d4
	{
Packit 1fb8d4
		/* We need support for multichannel and sample rates != 8000 */
Packit 1fb8d4
		case AV_CODEC_ID_GSM_MS:
Packit 1fb8d4
			context->context->strict_std_compliance = FF_COMPLIANCE_UNOFFICIAL;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case AV_CODEC_ID_AAC:
Packit 1fb8d4
			context->context->profile = FF_PROFILE_AAC_MAIN;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			break;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	context->context->channels = format->nChannels;
Packit 1fb8d4
	context->context->channel_layout = layout;
Packit 1fb8d4
	context->context->sample_rate = format->nSamplesPerSec;
Packit 1fb8d4
	context->context->block_align = format->nBlockAlign;
Packit 1fb8d4
	context->context->bit_rate = format->nAvgBytesPerSec * 8;
Packit 1fb8d4
	context->context->sample_fmt = ffmpeg_sample_format(format);
Packit 1fb8d4
	context->context->time_base = av_make_q(1, context->context->sample_rate);
Packit 1fb8d4
Packit 1fb8d4
	if ((ret = avcodec_open2(context->context, context->codec, NULL)) < 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		const char* err = av_err2str(ret);
Packit 1fb8d4
		WLog_ERR(TAG, "Error avcodec_open2 %s [%d]", err, ret);
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	context->packet = av_packet_alloc();
Packit 1fb8d4
Packit 1fb8d4
	if (!context->packet)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	context->frame = av_frame_alloc();
Packit 1fb8d4
Packit 1fb8d4
	if (!context->frame)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	context->resampled = av_frame_alloc();
Packit 1fb8d4
Packit 1fb8d4
	if (!context->resampled)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	context->buffered = av_frame_alloc();
Packit 1fb8d4
Packit 1fb8d4
	if (!context->buffered)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
#if defined(SWRESAMPLE_FOUND)
Packit 1fb8d4
	context->rcontext = swr_alloc();
Packit 1fb8d4
#else
Packit 1fb8d4
	context->rcontext = avresample_alloc_context();
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
	if (!context->rcontext)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	context->frame->channel_layout = layout;
Packit 1fb8d4
	context->frame->channels = format->nChannels;
Packit 1fb8d4
	context->frame->sample_rate = format->nSamplesPerSec;
Packit 1fb8d4
	context->frame->format = AV_SAMPLE_FMT_S16;
Packit 1fb8d4
Packit 1fb8d4
	if (context->encoder)
Packit 1fb8d4
	{
Packit 1fb8d4
		context->resampled->format = context->context->sample_fmt;
Packit 1fb8d4
		context->resampled->sample_rate = context->context->sample_rate;
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		context->resampled->format = AV_SAMPLE_FMT_S16;
Packit 1fb8d4
		context->resampled->sample_rate = format->nSamplesPerSec;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	context->resampled->channel_layout = layout;
Packit 1fb8d4
	context->resampled->channels = format->nChannels;
Packit 1fb8d4
Packit 1fb8d4
	if (context->context->frame_size > 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		context->buffered->channel_layout = context->resampled->channel_layout;
Packit 1fb8d4
		context->buffered->channels = context->resampled->channels;
Packit 1fb8d4
		context->buffered->format = context->resampled->format;
Packit 1fb8d4
		context->buffered->nb_samples = context->context->frame_size;
Packit 1fb8d4
Packit 1fb8d4
		if ((ret = av_frame_get_buffer(context->buffered, 1)) < 0)
Packit 1fb8d4
			goto fail;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	context->isOpen = TRUE;
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
fail:
Packit 1fb8d4
	ffmpeg_close_context(context);
Packit 1fb8d4
	return FALSE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
#if defined(SWRESAMPLE_FOUND)
Packit 1fb8d4
static BOOL ffmpeg_resample_frame(SwrContext* context,
Packit 1fb8d4
                                  AVFrame* in, AVFrame* out)
Packit 1fb8d4
{
Packit 1fb8d4
	int ret;
Packit 1fb8d4
Packit 1fb8d4
	if (!swr_is_initialized(context))
Packit 1fb8d4
	{
Packit 1fb8d4
		if ((ret = swr_config_frame(context, out, in)) < 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			const char* err = av_err2str(ret);
Packit 1fb8d4
			WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if ((ret = (swr_init(context))) < 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			const char* err = av_err2str(ret);
Packit 1fb8d4
			WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if ((ret = swr_convert_frame(context, out, in)) < 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		const char* err = av_err2str(ret);
Packit 1fb8d4
		WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
#else
Packit 1fb8d4
static BOOL ffmpeg_resample_frame(AVAudioResampleContext* context,
Packit 1fb8d4
                                  AVFrame* in, AVFrame* out)
Packit 1fb8d4
{
Packit 1fb8d4
	int ret;
Packit 1fb8d4
Packit 1fb8d4
	if (!avresample_is_open(context))
Packit 1fb8d4
	{
Packit 1fb8d4
		if ((ret = avresample_config(context, out, in)) < 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			const char* err = av_err2str(ret);
Packit 1fb8d4
			WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if ((ret = (avresample_open(context))) < 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			const char* err = av_err2str(ret);
Packit 1fb8d4
			WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if ((ret = avresample_convert_frame(context, out, in)) < 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		const char* err = av_err2str(ret);
Packit 1fb8d4
		WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
static BOOL ffmpeg_encode_frame(AVCodecContext* context, AVFrame* in,
Packit 1fb8d4
                                AVPacket* packet, wStream* out)
Packit 1fb8d4
{
Packit 1fb8d4
	int ret;
Packit 1fb8d4
	/* send the packet with the compressed data to the encoder */
Packit 1fb8d4
	ret = avcodec_send_frame(context, in);
Packit 1fb8d4
Packit 1fb8d4
	if (ret < 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		const char* err = av_err2str(ret);
Packit 1fb8d4
		WLog_ERR(TAG, "Error submitting the packet to the encoder %s [%d]",
Packit 1fb8d4
		         err, ret);
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	/* read all the output frames (in general there may be any number of them */
Packit 1fb8d4
	while (ret >= 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		ret = avcodec_receive_packet(context, packet);
Packit 1fb8d4
Packit 1fb8d4
		if ((ret == AVERROR(EAGAIN)) || (ret == AVERROR_EOF))
Packit 1fb8d4
			return TRUE;
Packit 1fb8d4
		else if (ret < 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			const char* err = av_err2str(ret);
Packit 1fb8d4
			WLog_ERR(TAG, "Error during encoding %s [%d]", err, ret);
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (!Stream_EnsureRemainingCapacity(out, packet->size))
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		Stream_Write(out, packet->data, packet->size);
Packit 1fb8d4
		av_packet_unref(packet);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL ffmpeg_fill_frame(AVFrame* frame, const AUDIO_FORMAT* inputFormat,
Packit 1fb8d4
                              const BYTE* data, size_t size)
Packit 1fb8d4
{
Packit 1fb8d4
	int ret, bpp;
Packit 1fb8d4
	frame->channels = inputFormat->nChannels;
Packit 1fb8d4
	frame->sample_rate = inputFormat->nSamplesPerSec;
Packit 1fb8d4
	frame->format = ffmpeg_sample_format(inputFormat);
Packit 1fb8d4
	frame->channel_layout = av_get_default_channel_layout(frame->channels);
Packit 1fb8d4
	bpp = av_get_bytes_per_sample(frame->format);
Packit 1fb8d4
	frame->nb_samples = size / inputFormat->nChannels / bpp;
Packit 1fb8d4
Packit 1fb8d4
	if ((ret = avcodec_fill_audio_frame(frame, frame->channels,
Packit 1fb8d4
	                                    frame->format,
Packit 1fb8d4
	                                    data, size, 1)) < 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		const char* err = av_err2str(ret);
Packit 1fb8d4
		WLog_ERR(TAG, "Error during audio frame fill %s [%d]", err, ret);
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
#if defined(SWRESAMPLE_FOUND)
Packit 1fb8d4
static BOOL ffmpeg_decode(AVCodecContext* dec_ctx, AVPacket* pkt,
Packit 1fb8d4
                          AVFrame* frame,
Packit 1fb8d4
                          SwrContext* resampleContext,
Packit 1fb8d4
                          AVFrame* resampled, wStream* out)
Packit 1fb8d4
#else
Packit 1fb8d4
static BOOL ffmpeg_decode(AVCodecContext* dec_ctx, AVPacket* pkt,
Packit 1fb8d4
                          AVFrame* frame,
Packit 1fb8d4
                          AVAudioResampleContext* resampleContext,
Packit 1fb8d4
                          AVFrame* resampled, wStream* out)
Packit 1fb8d4
#endif
Packit 1fb8d4
{
Packit 1fb8d4
	int ret;
Packit 1fb8d4
	/* send the packet with the compressed data to the decoder */
Packit 1fb8d4
	ret = avcodec_send_packet(dec_ctx, pkt);
Packit 1fb8d4
Packit 1fb8d4
	if (ret < 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		const char* err = av_err2str(ret);
Packit 1fb8d4
		WLog_ERR(TAG, "Error submitting the packet to the decoder %s [%d]",
Packit 1fb8d4
		         err, ret);
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	/* read all the output frames (in general there may be any number of them */
Packit 1fb8d4
	while (ret >= 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		ret = avcodec_receive_frame(dec_ctx, frame);
Packit 1fb8d4
Packit 1fb8d4
		if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF)
Packit 1fb8d4
			return TRUE;
Packit 1fb8d4
		else if (ret < 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			const char* err = av_err2str(ret);
Packit 1fb8d4
			WLog_ERR(TAG, "Error during decoding %s [%d]", err, ret);
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
#if defined(SWRESAMPLE_FOUND)
Packit 1fb8d4
		if (!swr_is_initialized(resampleContext))
Packit 1fb8d4
		{
Packit 1fb8d4
			if ((ret = swr_config_frame(resampleContext, resampled, frame)) < 0)
Packit 1fb8d4
			{
Packit 1fb8d4
#else
Packit 1fb8d4
		if (!avresample_is_open(resampleContext))
Packit 1fb8d4
		{
Packit 1fb8d4
			if ((ret = avresample_config(resampleContext, resampled, frame)) < 0)
Packit 1fb8d4
			{
Packit 1fb8d4
#endif
Packit 1fb8d4
				const char* err = av_err2str(ret);
Packit 1fb8d4
				WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
Packit 1fb8d4
				return FALSE;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
#if defined(SWRESAMPLE_FOUND)
Packit 1fb8d4
			if ((ret = (swr_init(resampleContext))) < 0)
Packit 1fb8d4
#else
Packit 1fb8d4
			if ((ret = (avresample_open(resampleContext))) < 0)
Packit 1fb8d4
#endif
Packit 1fb8d4
			{
Packit 1fb8d4
				const char* err = av_err2str(ret);
Packit 1fb8d4
				WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
Packit 1fb8d4
				return FALSE;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
#if defined(SWRESAMPLE_FOUND)
Packit 1fb8d4
		if ((ret = swr_convert_frame(resampleContext, resampled, frame)) < 0)
Packit 1fb8d4
#else
Packit 1fb8d4
		if ((ret = avresample_convert_frame(resampleContext, resampled, frame)) < 0)
Packit 1fb8d4
#endif
Packit 1fb8d4
		{
Packit 1fb8d4
			const char* err = av_err2str(ret);
Packit 1fb8d4
			WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		{
Packit 1fb8d4
			const size_t data_size = resampled->channels * resampled->nb_samples * 2;
Packit 1fb8d4
			Stream_EnsureRemainingCapacity(out, data_size);
Packit 1fb8d4
			Stream_Write(out, resampled->data[0], data_size);
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL freerdp_dsp_ffmpeg_supports_format(const AUDIO_FORMAT* format, BOOL encode)
Packit 1fb8d4
{
Packit 1fb8d4
	enum AVCodecID id = ffmpeg_get_avcodec(format);
Packit 1fb8d4
Packit 1fb8d4
	if (ffmpeg_codec_is_filtered(id, encode))
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (encode)
Packit 1fb8d4
		return avcodec_find_encoder(id) != NULL;
Packit 1fb8d4
	else
Packit 1fb8d4
		return avcodec_find_decoder(id) != NULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
FREERDP_DSP_CONTEXT* freerdp_dsp_ffmpeg_context_new(BOOL encode)
Packit 1fb8d4
{
Packit 1fb8d4
	FREERDP_DSP_CONTEXT* context;
Packit 1fb8d4
	avcodec_register_all();
Packit 1fb8d4
	context = calloc(1, sizeof(FREERDP_DSP_CONTEXT));
Packit 1fb8d4
Packit 1fb8d4
	if (!context)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	context->encoder = encode;
Packit 1fb8d4
	return context;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void freerdp_dsp_ffmpeg_context_free(FREERDP_DSP_CONTEXT* context)
Packit 1fb8d4
{
Packit 1fb8d4
	if (context)
Packit 1fb8d4
	{
Packit 1fb8d4
		ffmpeg_close_context(context);
Packit 1fb8d4
		free(context);
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL freerdp_dsp_ffmpeg_context_reset(FREERDP_DSP_CONTEXT* context,
Packit 1fb8d4
                                      const AUDIO_FORMAT* targetFormat)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!context || !targetFormat)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	ffmpeg_close_context(context);
Packit 1fb8d4
	context->format = *targetFormat;
Packit 1fb8d4
	return ffmpeg_open_context(context);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL freerdp_dsp_ffmpeg_encode(FREERDP_DSP_CONTEXT* context, const AUDIO_FORMAT* format,
Packit 1fb8d4
                               const BYTE* data, size_t length, wStream* out)
Packit 1fb8d4
{
Packit 1fb8d4
	int rc;
Packit 1fb8d4
Packit 1fb8d4
	if (!context || !format || !data || !out || !context->encoder)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (!context || !data || !out)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	/* Create input frame */
Packit 1fb8d4
	if (!ffmpeg_fill_frame(context->frame, format, data, length))
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	/* Resample to desired format. */
Packit 1fb8d4
	if (!ffmpeg_resample_frame(context->rcontext,
Packit 1fb8d4
	                           context->frame,
Packit 1fb8d4
	                           context->resampled))
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (context->context->frame_size <= 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		return ffmpeg_encode_frame(context->context, context->resampled,
Packit 1fb8d4
		                           context->packet, out);
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		int copied = 0;
Packit 1fb8d4
		int rest = context->resampled->nb_samples;
Packit 1fb8d4
Packit 1fb8d4
		do
Packit 1fb8d4
		{
Packit 1fb8d4
			int inSamples = rest;
Packit 1fb8d4
Packit 1fb8d4
			if (inSamples + context->bufferedSamples > context->context->frame_size)
Packit 1fb8d4
				inSamples = context->context->frame_size - context->bufferedSamples;
Packit 1fb8d4
Packit 1fb8d4
			rc = av_samples_copy(context->buffered->extended_data, context->resampled->extended_data,
Packit 1fb8d4
			                     context->bufferedSamples, copied, inSamples,
Packit 1fb8d4
			                     context->context->channels, context->context->sample_fmt);
Packit 1fb8d4
			rest -= inSamples;
Packit 1fb8d4
			copied += inSamples;
Packit 1fb8d4
			context->bufferedSamples += inSamples;
Packit 1fb8d4
Packit 1fb8d4
			if (context->context->frame_size <= context->bufferedSamples)
Packit 1fb8d4
			{
Packit 1fb8d4
				/* Encode in desired format. */
Packit 1fb8d4
				if (!ffmpeg_encode_frame(context->context, context->buffered,
Packit 1fb8d4
				                         context->packet, out))
Packit 1fb8d4
					return FALSE;
Packit 1fb8d4
Packit 1fb8d4
				context->bufferedSamples = 0;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
		while (rest > 0);
Packit 1fb8d4
Packit 1fb8d4
		return TRUE;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL freerdp_dsp_ffmpeg_decode(FREERDP_DSP_CONTEXT* context, const AUDIO_FORMAT* srcFormat,
Packit 1fb8d4
                               const BYTE* data, size_t length, wStream* out)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!context || !srcFormat || !data || !out || context->encoder)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	av_init_packet(context->packet);
Packit 1fb8d4
	context->packet->data = (uint8_t*)data;
Packit 1fb8d4
	context->packet->size = length;
Packit 1fb8d4
	return ffmpeg_decode(context->context, context->packet, context->frame,
Packit 1fb8d4
	                     context->rcontext, context->resampled, out);
Packit 1fb8d4
}