RetroArch/record/ffemu.c

193 lines
4.3 KiB
C
Raw Normal View History

2011-01-03 17:51:17 +01:00
#include <libavcodec/avcodec.h>
#include <libavutil/mathematics.h>
#include <libavformat/avformat.h>
#include <stdint.h>
#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
#include <stdbool.h>
#include "ffemu.h"
struct video_info
{
bool enabled;
AVCodecContext *codec;
AVFrame *frame;
FILE *file;
char *file_name;
uint8_t *outbuf;
size_t outbuf_size;
} video;
struct audio_info
{
bool enabled;
AVCodecContext *codec;
int16_t *buffer;
size_t frames_in_buffer;
void *outbuf;
size_t outbuf_size;
FILE *file;
char *file_name;
} audio;
struct ffemu
{
struct video_info video;
struct audio_info audio;
struct ffemu_params params;
};
static int init_video(struct video_info *video, struct ffemu_params *param)
{
(void)video;
(void)param;
return -1;
}
static int map_audio_codec(ffemu_audio_codec codec)
{
(void)codec;
return CODEC_ID_AAC;
}
static int init_audio(struct audio_info *audio, struct ffemu_params *param)
{
AVCodec *codec = avcodec_find_encoder(map_audio_codec(param->acodec));
if (!codec)
return -1;
audio->codec = avcodec_alloc_context();
audio->codec->bit_rate = 128000;
audio->codec->sample_rate = param->samplerate;
audio->codec->channels = param->channels;
if (avcodec_open(audio->codec, codec) != 0)
return -1;
audio->buffer = av_malloc(audio->codec->frame_size * param->channels * sizeof(int16_t));
if (!audio->buffer)
return -1;
audio->file = fopen("/tmp/audio.aac", "wb");
if (!audio->file)
return -1;
audio->outbuf_size = 50000;
audio->outbuf = av_malloc(audio->outbuf_size);
if (!audio->outbuf)
return -1;
return 0;
}
ffemu_t *ffemu_new(const struct ffemu_params *params)
{
avcodec_init();
av_register_all();
ffemu_t *handle = calloc(1, sizeof(*handle));
if (!handle)
goto error;
handle->params = *params;
if (handle->params.vcodec != FFEMU_VIDEO_NONE)
handle->video.enabled = true;
if (handle->params.acodec != FFEMU_AUDIO_NONE)
handle->audio.enabled = true;
if (handle->video.enabled)
if (init_video(&handle->video, &handle->params) < 0)
goto error;
if (handle->audio.enabled)
if (init_audio(&handle->audio, &handle->params) < 0)
goto error;
return handle;
error:
ffemu_free(handle);
return NULL;
}
void ffemu_free(ffemu_t *handle)
{
if (handle)
{
if (handle->audio.codec)
{
avcodec_close(handle->audio.codec);
av_free(handle->audio.codec);
}
if (handle->audio.buffer)
av_free(handle->audio.buffer);
if (handle->video.codec)
{
avcodec_close(handle->video.codec);
av_free(handle->video.codec);
}
if (handle->video.frame)
av_free(handle->video.frame);
if (handle->video.file)
fclose(handle->video.file);
if (handle->audio.file)
fclose(handle->audio.file);
free(handle);
}
}
int ffemu_push_video(ffemu_t *handle, const struct ffemu_video_data *data)
{
(void)handle;
(void)data;
if (!handle->video.enabled)
return -1;
return 0;
}
int ffemu_push_audio(ffemu_t *handle, const struct ffemu_audio_data *data)
{
size_t written_frames = 0;
while (written_frames < data->frames)
{
size_t can_write = handle->audio.codec->frame_size - handle->audio.frames_in_buffer;
size_t write_frames = data->frames > can_write ? can_write : data->frames;
memcpy(handle->audio.buffer + handle->audio.frames_in_buffer * handle->params.channels,
data->data + written_frames * handle->params.channels,
write_frames * handle->params.channels * sizeof(int16_t));
written_frames += write_frames;
handle->audio.frames_in_buffer += write_frames;
if (handle->audio.frames_in_buffer == (size_t)handle->audio.codec->frame_size)
{
size_t out_size = avcodec_encode_audio(handle->audio.codec, handle->audio.outbuf, handle->audio.outbuf_size, handle->audio.buffer);
fwrite(handle->audio.outbuf, 1, out_size, handle->audio.file);
handle->audio.frames_in_buffer = 0;
}
}
return 0;
}
int ffemu_mux(ffemu_t *handle, const char *path, const struct ffemu_muxer *muxer)
{
(void)handle;
(void)path;
(void)muxer;
return -1;
}