You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

422 lines
12 KiB

/*****************************************************************************
* transcode.c: test for transcoding pipeline
*****************************************************************************
* Copyright (C) 2021 VideoLabs
*
* Author: Alexandre Janniaux <ajanni@videolabs.io>
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as published
* by the Free Software Foundation; either version 2.1 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
*****************************************************************************/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
/* Define a builtin module for mocked parts */
#define MODULE_NAME test_transcode_mock
#undef VLC_DYNAMIC_PLUGIN
#include "../../libvlc/test.h"
#include <vlc_common.h>
#include <vlc_plugin.h>
#include <vlc_access.h>
#include <vlc_demux.h>
#include <vlc_codec.h>
#include <vlc_window.h>
#include <vlc_interface.h>
#include <vlc_player.h>
#include <vlc_filter.h>
#include <vlc_threads.h>
#include <vlc_sout.h>
#include <vlc_frame.h>
#include <limits.h>
#include "transcode.h"
#include "../lib/libvlc_internal.h"
static const char dec_dev_arg[] = "--dec-dev=" MODULE_STRING;
const char vlc_module_name[] = MODULE_STRING;
static size_t current_scenario = 0;
static vlc_cond_t player_cond = VLC_STATIC_COND;
static void DecoderDeviceClose(struct vlc_decoder_device *device)
{ VLC_UNUSED(device); }
static const struct vlc_decoder_device_operations decoder_device_ops =
{
.close = DecoderDeviceClose,
};
static int OpenDecoderDevice(
struct vlc_decoder_device *device,
vlc_window_t *window
) {
VLC_UNUSED(window);
device->ops = &decoder_device_ops;
/* Pick any valid one, we'll not use the module which can make use of
* the private parts. */
device->type = VLC_DECODER_DEVICE_VAAPI;
return VLC_SUCCESS;
}
static int DecoderDecode(decoder_t *dec, vlc_frame_t *frame)
{
if (frame == NULL)
return VLC_SUCCESS;
const picture_resource_t resource = {
.p_sys = NULL,
};
picture_t *pic = picture_NewFromResource(&dec->fmt_out.video, &resource);
assert(pic);
pic->date = frame->i_pts;
vlc_frame_Release(frame);
struct transcode_scenario *scenario = &transcode_scenarios[current_scenario];
assert(scenario->decoder_decode != NULL);
return scenario->decoder_decode(dec, pic);
}
static void CloseDecoder(vlc_object_t *obj)
{
decoder_t *dec = (decoder_t*)obj;
struct vlc_video_context *vctx = dec->p_sys;
if (vctx)
vlc_video_context_Release(vctx);
}
static int OpenDecoder(vlc_object_t *obj)
{
decoder_t *dec = (decoder_t*)obj;
struct vlc_decoder_device *device = decoder_GetDecoderDevice(dec);
assert(device);
vlc_decoder_device_Release(device);
dec->pf_decode = DecoderDecode;
// Necessary ?
es_format_Clean(&dec->fmt_out);
es_format_Copy(&dec->fmt_out, dec->fmt_in);
struct transcode_scenario *scenario = &transcode_scenarios[current_scenario];
assert(scenario->decoder_setup != NULL);
scenario->decoder_setup(dec);
msg_Dbg(obj, "Decoder chroma %4.4s -> %4.4s size %ux%u",
(const char *)&dec->fmt_in->i_codec,
(const char *)&dec->fmt_out.i_codec,
dec->fmt_out.video.i_width, dec->fmt_out.video.i_height);
return VLC_SUCCESS;
}
static int OpenFilter(filter_t *filter)
{
static const struct vlc_filter_operations ops = {
.filter_video = NULL,
.close = NULL,
};
filter->ops = &ops;
return VLC_SUCCESS;
}
static picture_t *ConverterFilter(filter_t *filter, picture_t *input)
{
video_format_Clean(&input->format);
video_format_Copy(&input->format, &filter->fmt_out.video);
return input;
}
static int OpenConverter(filter_t *filter)
{
msg_Dbg(filter, "converter chroma %4.4s -> %4.4s size %ux%u -> %ux%u",
(const char *)&filter->fmt_in.i_codec,
(const char *)&filter->fmt_out.i_codec,
filter->fmt_in.video.i_width, filter->fmt_in.video.i_height,
filter->fmt_out.video.i_width, filter->fmt_out.video.i_height);
struct transcode_scenario *scenario = &transcode_scenarios[current_scenario];
assert(scenario->converter_setup != NULL);
scenario->converter_setup(filter);
static const struct vlc_filter_operations ops = {
.filter_video = ConverterFilter,
.close = NULL,
};
filter->ops = &ops;
return VLC_SUCCESS;
}
static vlc_frame_t *EncodeVideo(encoder_t *enc, picture_t *pic)
{
if (pic == NULL)
return NULL;
assert(pic->format.i_chroma == enc->fmt_in.video.i_chroma);
vlc_frame_t *frame = vlc_frame_Alloc(4);
struct transcode_scenario *scenario = &transcode_scenarios[current_scenario];
if (scenario->encoder_encode != NULL)
scenario->encoder_encode(enc, pic);
return frame;
}
static void CloseEncoder(encoder_t *enc)
{
struct transcode_scenario *scenario = &transcode_scenarios[current_scenario];
if (scenario->encoder_close != NULL)
scenario->encoder_close(enc);
}
static int OpenEncoder(vlc_object_t *obj)
{
encoder_t *enc = (encoder_t *)obj;
enc->p_sys = NULL;
struct transcode_scenario *scenario = &transcode_scenarios[current_scenario];
assert(scenario->encoder_setup != NULL);
scenario->encoder_setup(enc);
msg_Dbg(obj, "Encoder chroma %4.4s -> %4.4s size %ux%u -> %ux%u",
(const char *)&enc->fmt_in.i_codec,
(const char *)&enc->fmt_out.i_codec,
enc->fmt_in.video.i_width, enc->fmt_in.video.i_height,
enc->fmt_out.video.i_width, enc->fmt_out.video.i_height);
static const struct vlc_encoder_operations ops =
{
.encode_video = EncodeVideo,
.close = CloseEncoder,
};
enc->ops = &ops;
return VLC_SUCCESS;
}
static int ErrorCheckerSend(sout_stream_t *stream, void *id, vlc_frame_t *f)
{
struct transcode_scenario *scenario = &transcode_scenarios[current_scenario];
int ret = sout_StreamIdSend(stream->p_next, id, f);
if (ret != VLC_SUCCESS)
scenario->report_error(stream);
return VLC_SUCCESS;
}
static void *ErrorCheckerAdd(sout_stream_t *stream,
const es_format_t *fmt,
const char *es_id)
{ return sout_StreamIdAdd(stream->p_next, fmt, es_id); }
static void ErrorCheckerDel(sout_stream_t *stream, void *id)
{ sout_StreamIdDel(stream->p_next, id); };
static int OpenErrorChecker(vlc_object_t *obj)
{
sout_stream_t *stream = (sout_stream_t *)obj;
static const struct sout_stream_operations ops = {
.add = ErrorCheckerAdd,
.del = ErrorCheckerDel,
.send = ErrorCheckerSend,
};
stream->ops = &ops;
return VLC_SUCCESS;
}
static int OutputCheckerSend(sout_stream_t *stream, void *id, vlc_frame_t *f)
{
(void)stream; (void)id;
struct transcode_scenario *scenario = &transcode_scenarios[current_scenario];
assert(scenario->report_output != NULL);
scenario->report_output(f);
vlc_frame_ChainRelease(f);
return VLC_SUCCESS;
}
static void *OutputCheckerAdd(sout_stream_t *stream, const es_format_t *fmt,
const char *es_id)
{
(void)stream; (void)fmt; (void)es_id;
return (void*)0x42;
}
static void OutputCheckerDel(sout_stream_t *stream, void *id)
{ (void)stream; (void)id; }
static int OpenOutputChecker(vlc_object_t *obj)
{
sout_stream_t *stream = (sout_stream_t *)obj;
static const struct sout_stream_operations ops = {
.add = OutputCheckerAdd,
.del = OutputCheckerDel,
.send = OutputCheckerSend,
};
stream->ops = &ops;
return VLC_SUCCESS;
}
static void on_state_changed(vlc_player_t *player, enum vlc_player_state state, void *opaque)
{
(void)player; (void)state; (void) opaque;
vlc_cond_signal(&player_cond);
}
static void play_scenario(intf_thread_t *intf, struct transcode_scenario *scenario)
{
transcode_scenario_init();
input_item_t *media = input_item_New(scenario->source, "dummy");
assert(media);
/* TODO: Codec doesn't seem to have effect in transcode:
* - add a test that --codec works?
* - do not use --codec at all here? */
input_item_AddOption(media, scenario->sout, VLC_INPUT_OPTION_TRUSTED);
var_Create(intf, "codec", VLC_VAR_STRING);
var_SetString(intf, "codec", MODULE_STRING);
var_Create(intf, "sout-transcode-venc", VLC_VAR_STRING);
var_SetString(intf, "sout-transcode-venc", MODULE_STRING);
var_Create(intf, "sout-transcode-vcodec", VLC_VAR_STRING);
var_SetString(intf, "sout-transcode-vcodec", "test");
vlc_player_t *player = vlc_player_New(&intf->obj,
VLC_PLAYER_LOCK_NORMAL);
assert(player);
static const struct vlc_player_cbs player_cbs = {
.on_state_changed = on_state_changed,
};
vlc_player_Lock(player);
vlc_player_listener_id *listener =
vlc_player_AddListener(player, &player_cbs, NULL);
vlc_player_SetCurrentMedia(player, media);
vlc_player_Start(player);
vlc_player_Unlock(player);
transcode_scenario_wait(scenario);
vlc_player_Lock(player);
vlc_player_Stop(player);
while (vlc_player_GetState(player) != VLC_PLAYER_STATE_STOPPED)
vlc_player_CondWait(player, &player_cond);
vlc_player_RemoveListener(player, listener);
vlc_player_Unlock(player);
transcode_scenario_check(scenario);
vlc_player_Delete(player);
input_item_Release(media);
var_Destroy(intf, "sout-transcode-vcodec");
var_Destroy(intf, "sout-transcode-venc");
}
static int OpenIntf(vlc_object_t *obj)
{
intf_thread_t *intf = (intf_thread_t*)obj;
while (current_scenario < transcode_scenarios_count)
{
msg_Info(intf, " - Running transcode scenario %zu", current_scenario);
play_scenario(intf, &transcode_scenarios[current_scenario]);
current_scenario++;
}
return VLC_SUCCESS;
}
/**
* Inject the mocked modules as a static plugin:
* - access for triggering the correct decoder
* - decoder for generating video format and context
* - filter for generating video format and context
* - encoder to check the previous video format and context
**/
vlc_module_begin()
set_callbacks(OpenDecoder, CloseDecoder)
set_capability("video decoder", INT_MAX)
add_submodule()
set_callback(OpenErrorChecker)
set_capability("sout filter", 0)
add_shortcut("error_checker")
add_submodule()
set_callback(OpenOutputChecker)
set_capability("sout output", 0)
add_shortcut("output_checker")
add_submodule()
set_callback_dec_device(OpenDecoderDevice, 0)
add_submodule()
set_callback_video_filter(OpenFilter)
add_submodule()
set_callback_video_converter(OpenConverter, INT_MAX)
add_submodule()
set_callback(OpenEncoder)
set_capability("video encoder", 0)
add_submodule()
set_callback(OpenIntf)
set_capability("interface", 0)
vlc_module_end()
VLC_EXPORT const vlc_plugin_cb vlc_static_modules[] = {
VLC_SYMBOL(vlc_entry),
NULL
};
int main( int argc, char **argv )
{
(void)argc; (void)argv;
#ifndef ENABLE_SOUT
(void) libvlc_InternalPlay;
return 77;
#endif
test_init();
const char * const args[] = {
"-vvv", "--vout=dummy", "--aout=dummy", "--text-renderer=dummy",
"--no-auto-preparse", dec_dev_arg
};
libvlc_instance_t *vlc = libvlc_new(ARRAY_SIZE(args), args);
libvlc_InternalAddIntf(vlc->p_libvlc_int, MODULE_STRING);
libvlc_InternalPlay(vlc->p_libvlc_int);
libvlc_release(vlc);
assert(transcode_scenarios_count == current_scenario);
return 0;
}