- merge v0.01 --> Añadido fileselector - Añadidas fuentes de Gem y Pure Data - pix2jpg incluído en Gem. Archivos de construcción de Gem modificados. - Añadido fichero ompiling.txt con instrucciones de compilación
436 lines
12 KiB
C++
436 lines
12 KiB
C++
////////////////////////////////////////////////////////
|
|
//
|
|
// GEM - Graphics Environment for Multimedia
|
|
//
|
|
// zmoelnig@iem.kug.ac.at
|
|
//
|
|
// Implementation file
|
|
//
|
|
// Copyright (c) 1997-1999 Mark Danks.
|
|
// Copyright (c) Günther Geiger.
|
|
// Copyright (c) 2001-2011 IOhannes m zmölnig. forum::für::umläute. IEM. zmoelnig@iem.at
|
|
// For information on usage and redistribution, and for a DISCLAIMER OF ALL
|
|
// WARRANTIES, see the file, "GEM.LICENSE.TERMS" in this distribution.
|
|
//
|
|
/////////////////////////////////////////////////////////
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#ifdef HAVE_LIBGMERLIN_AVDEC
|
|
# define HAVE_GMERLIN
|
|
#endif
|
|
|
|
|
|
#ifdef HAVE_GMERLIN
|
|
#include "filmGMERLIN.h"
|
|
#include "plugins/PluginFactory.h"
|
|
#include "Gem/RTE.h"
|
|
#include "Gem/Properties.h"
|
|
|
|
|
|
//#define GEM_FILMGMERLIN_TRACKSWITCH 1
|
|
|
|
using namespace gem::plugins;
|
|
|
|
REGISTER_FILMFACTORY("gmerlin", filmGMERLIN);
|
|
|
|
/////////////////////////////////////////////////////////
|
|
//
|
|
// filmGMERLIN
|
|
//
|
|
/////////////////////////////////////////////////////////
|
|
// Constructor
|
|
//
|
|
/////////////////////////////////////////////////////////
|
|
|
|
filmGMERLIN :: filmGMERLIN(void) :
|
|
m_numFrames(-1), m_numTracks(-1),
|
|
m_file(NULL),
|
|
m_opt(NULL),
|
|
m_track(0), m_stream(0),
|
|
m_gframe(NULL),
|
|
m_finalframe(NULL),
|
|
m_gconverter(NULL),
|
|
m_fps(0.), m_fps_num(1), m_fps_denum(1),
|
|
m_next_timestamp(0),
|
|
#ifdef USE_FRAMETABLE
|
|
m_frametable(NULL),
|
|
#endif
|
|
m_doConvert(false)
|
|
{
|
|
m_gconverter=gavl_video_converter_create ();
|
|
}
|
|
|
|
////////////////////////////////////////////////////////
|
|
// Destructor
|
|
//
|
|
/////////////////////////////////////////////////////////
|
|
filmGMERLIN :: ~filmGMERLIN(void)
|
|
{
|
|
close();
|
|
if(m_gconverter)gavl_video_converter_destroy(m_gconverter);m_gconverter=NULL;
|
|
}
|
|
|
|
void filmGMERLIN :: close(void)
|
|
{
|
|
if(m_file)bgav_close(m_file);m_file=NULL;
|
|
#ifdef USE_FRAMETABLE
|
|
if(m_frametable)gavl_frame_table_destroy(m_frametable); m_frametable=NULL;
|
|
#endif
|
|
/* LATER: free frame buffers */
|
|
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////
|
|
// logging
|
|
//
|
|
/////////////////////////////////////////////////////////
|
|
void filmGMERLIN::log(bgav_log_level_t level, const char *log_domain, const char *message)
|
|
{
|
|
switch(level) {
|
|
case BGAV_LOG_DEBUG:
|
|
verbose(2, "[pix_film:%s] %s", log_domain, message);
|
|
break;
|
|
case BGAV_LOG_INFO:
|
|
verbose(1, "[pix_film:%s] %s", log_domain, message);
|
|
break;
|
|
case BGAV_LOG_WARNING:
|
|
post("[pix_film:%s] %s", log_domain, message);
|
|
break;
|
|
case BGAV_LOG_ERROR:
|
|
error("[pix_film:%s!] %s", log_domain, message);
|
|
break;
|
|
default:break;
|
|
}
|
|
}
|
|
void filmGMERLIN::log_callback (void *data, bgav_log_level_t level, const char *log_domain, const char *message)
|
|
{
|
|
((filmGMERLIN*)(data))->log(level, log_domain, message);
|
|
}
|
|
|
|
bool filmGMERLIN :: isThreadable(void) {
|
|
if(m_numFrames<0)
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////
|
|
// really open the file !
|
|
//
|
|
/////////////////////////////////////////////////////////
|
|
bool filmGMERLIN :: open(const std::string sfilename, const gem::Properties&wantProps)
|
|
{
|
|
close();
|
|
|
|
m_track=0;
|
|
|
|
m_file = bgav_create();
|
|
if(!m_file) return false;
|
|
m_opt = bgav_get_options(m_file);
|
|
if(!m_opt) return false;
|
|
|
|
/*
|
|
bgav_options_set_connect_timeout(m_opt, connect_timeout);
|
|
bgav_options_set_read_timeout(m_opt, read_timeout);
|
|
bgav_options_set_network_bandwidth(m_opt, network_bandwidth);
|
|
*/
|
|
bgav_options_set_seek_subtitles(m_opt, 0);
|
|
bgav_options_set_sample_accurate(m_opt, 1);
|
|
|
|
bgav_options_set_log_callback(m_opt,
|
|
log_callback,
|
|
this);
|
|
|
|
const char*filename=sfilename.c_str();
|
|
|
|
if(!strncmp(filename, "vcd://", 6))
|
|
{
|
|
if(!bgav_open_vcd(m_file, filename + 5))
|
|
{
|
|
//error("Could not open VCD Device %s", filename + 5);
|
|
return false;
|
|
}
|
|
}
|
|
else if(!strncmp(filename, "dvd://", 6))
|
|
{
|
|
if(!bgav_open_dvd(m_file, filename + 5))
|
|
{
|
|
//error("Could not open DVD Device %s", filename + 5);
|
|
return false;
|
|
}
|
|
}
|
|
else if(!strncmp(filename, "dvb://", 6))
|
|
{
|
|
if(!bgav_open_dvb(m_file, filename + 6))
|
|
{
|
|
//error("Could not open DVB Device %s", filename + 6);
|
|
return false;
|
|
}
|
|
}
|
|
else {
|
|
if(!bgav_open(m_file, filename)) {
|
|
//error("Could not open file %s", filename);
|
|
close();
|
|
|
|
return false;
|
|
}
|
|
}
|
|
if(bgav_is_redirector(m_file))
|
|
{
|
|
int i=0;
|
|
int num_urls=bgav_redirector_get_num_urls(m_file);
|
|
post("Found redirector:");
|
|
for(i = 0; i < num_urls; i++)
|
|
{
|
|
post("#%d: '%s' -> %s", i, bgav_redirector_get_name(m_file, i), bgav_redirector_get_url(m_file, i));
|
|
}
|
|
for(i = 0; i < num_urls; i++) {
|
|
filename=(char*)bgav_redirector_get_url(m_file, i);
|
|
close();
|
|
if (open(filename, wantProps)) {
|
|
return true;
|
|
}
|
|
}
|
|
/* couldn't open a redirected stream */
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
* ok, we have been able to open the "file"
|
|
* now get some information from it...
|
|
*/
|
|
m_numTracks = bgav_num_tracks(m_file);
|
|
// LATER: check whether this track has a video-stream...
|
|
int numvstreams=bgav_num_video_streams (m_file, m_track);
|
|
if(numvstreams) {
|
|
bgav_select_track(m_file, m_track);
|
|
} else {
|
|
post("track %d does not contain a video-stream: skipping");
|
|
}
|
|
|
|
bgav_set_video_stream(m_file, m_stream, BGAV_STREAM_DECODE);
|
|
if(!bgav_start(m_file)) {
|
|
close();
|
|
return false;
|
|
}
|
|
m_next_timestamp=bgav_video_start_time(m_file, m_track);
|
|
|
|
gavl_video_format_t*gformat = (gavl_video_format_t*)bgav_get_video_format (m_file, m_stream);
|
|
m_gframe = gavl_video_frame_create_nopad(gformat);
|
|
|
|
|
|
gavl_video_format_t finalformat[1];
|
|
finalformat->frame_width = gformat->frame_width;
|
|
finalformat->frame_height = gformat->frame_height;
|
|
finalformat->image_width = gformat->image_width;
|
|
finalformat->image_height = gformat->image_height;
|
|
finalformat->pixel_width = gformat->pixel_width;
|
|
finalformat->pixel_height = gformat->pixel_height;
|
|
finalformat->frame_duration = gformat->frame_duration;
|
|
finalformat->timescale = gformat->timescale;
|
|
|
|
#ifdef __APPLE__
|
|
/* GAVL_UYVY definitely works on PowerPC
|
|
* m.e.grimm also reported that he had to use UYVY,
|
|
* i assume he is on x86_64;
|
|
* so make it the default for now
|
|
*/
|
|
finalformat->pixelformat=GAVL_UYVY;
|
|
/* don't know what this is exactly: */
|
|
//finalformat->pixelformat=GAVL_YUY2;
|
|
#else
|
|
finalformat->pixelformat=GAVL_RGBA_32;
|
|
#endif
|
|
|
|
m_finalframe = gavl_video_frame_create_nopad(finalformat);
|
|
m_doConvert= (gavl_video_converter_init (m_gconverter, gformat, finalformat)>0);
|
|
m_image.image.xsize=gformat->frame_width;
|
|
m_image.image.ysize=gformat->frame_height;
|
|
#ifdef __APPLE__
|
|
m_image.image.setCsizeByFormat(GL_YUV422_GEM);
|
|
#else
|
|
m_image.image.setCsizeByFormat(GL_RGBA);
|
|
#endif
|
|
m_image.image.notowned=true;
|
|
m_image.image.upsidedown=true;
|
|
m_image.newfilm=true;
|
|
|
|
if(gformat->frame_duration) {
|
|
m_fps = gformat->timescale / gformat->frame_duration;
|
|
} else {
|
|
m_fps = gformat->timescale;
|
|
}
|
|
|
|
m_fps_num=gformat->timescale;
|
|
m_fps_denum=gformat->frame_duration;
|
|
|
|
m_numFrames=-1;
|
|
#ifdef USE_FRAMETABLE
|
|
m_frametable=bgav_get_frame_table(m_file, m_track);
|
|
if(m_frametable)
|
|
m_numFrames=gavl_frame_table_num_frames (m_frametable);
|
|
#endif
|
|
|
|
gavl_time_t dur=bgav_get_duration (m_file, m_track);
|
|
if(m_numFrames<0)
|
|
if(dur!=GAVL_TIME_UNDEFINED)
|
|
m_numFrames = gavl_time_to_frames(m_fps_num,
|
|
m_fps_denum,
|
|
dur);
|
|
|
|
return true;
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////
|
|
// render
|
|
//
|
|
/////////////////////////////////////////////////////////
|
|
pixBlock* filmGMERLIN :: getFrame(void){
|
|
if(!m_file)return NULL;
|
|
bgav_read_video(m_file, m_gframe, m_stream);
|
|
if(m_doConvert) {
|
|
gavl_video_convert (m_gconverter, m_gframe, m_finalframe);
|
|
m_image.image.data=m_finalframe->planes[0];
|
|
} else {
|
|
m_image.image.data=m_gframe->planes[0];
|
|
}
|
|
m_image.newimage=true;
|
|
m_image.image.upsidedown=true;
|
|
|
|
m_next_timestamp=m_gframe->timestamp+m_gframe->duration;
|
|
return &m_image;
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////
|
|
// changeFrame
|
|
//
|
|
/////////////////////////////////////////////////////////
|
|
film::errCode filmGMERLIN :: changeImage(int imgNum, int trackNum){
|
|
if(trackNum<0) {
|
|
/* just automatically proceed to the next frame: this might speed up things for linear decoding */
|
|
return film::SUCCESS;
|
|
}
|
|
|
|
if(!m_file)return film::FAILURE;
|
|
|
|
#if GEM_FILMGMERLIN_TRACKSWITCH
|
|
// LATER implement track-switching
|
|
// this really shares a lot of code with open() so it should go into a separate function
|
|
if(trackNum) {
|
|
if(m_numTracks>trackNum || trackNum<0) {
|
|
post("selected invalid track %d of %d", trackNum, m_numTracks);
|
|
} else {
|
|
int numvstreams=bgav_num_video_streams (m_file, m_track);
|
|
post("track %d contains %d video streams", m_track, numvstreams);
|
|
if(numvstreams) {
|
|
bgav_select_track(m_file, m_track);
|
|
#ifdef USE_FRAMETABLE
|
|
if(m_frametable) {
|
|
gavl_frame_table_destroy(m_frametable);
|
|
m_frametable=bgav_get_frame_table(m_file, m_track);
|
|
}
|
|
#endif
|
|
} else {
|
|
post("track %d does not contain a video-stream: skipping");
|
|
}
|
|
}
|
|
}
|
|
#endif /* GEM_FILMGMERLIN_TRACKSWITCH */
|
|
|
|
if(imgNum>=m_numFrames || imgNum<0)return film::FAILURE;
|
|
|
|
if(bgav_can_seek(m_file)) {
|
|
if(0) {
|
|
#ifdef USE_FRAMETABLE
|
|
} else if(m_frametable) {
|
|
// no assumptions about fixed framerate
|
|
int64_t seekpos = gavl_frame_table_frame_to_time(m_frametable, imgNum, NULL);
|
|
bgav_seek_video(m_file, m_track, seekpos);
|
|
return film::SUCCESS;
|
|
#endif
|
|
} else {
|
|
// assuming fixed framerate
|
|
/*
|
|
Plaum: "Relying on a constant framerate is not good."
|
|
m_fps_denum and m_fps_num are set only once!
|
|
*/
|
|
int64_t seekposOrg = imgNum*m_fps_denum;
|
|
int64_t seekpos = seekposOrg;
|
|
|
|
#if 0
|
|
// LATER: set a minimum frame offset
|
|
// keep in mind that m_fps_denum could be 1!
|
|
// so it's better to calculate the difference in milliseconds and compare
|
|
int64_t diff=m_next_timestamp-seekpos;
|
|
# define TIMESTAMP_OFFSET_MAX 5
|
|
if(diff<TIMESTAMP_OFFSET_MAX && diff>(TIMESTAMP_OFFSET_MAX * -1)) {
|
|
// hey we are already there...
|
|
return film::SUCCESS;
|
|
}
|
|
#endif
|
|
|
|
bgav_seek_scaled(m_file, &seekpos, m_fps_num);
|
|
if(seekposOrg == seekpos)
|
|
return film::SUCCESS;
|
|
/* never mind: always return success... */
|
|
return film::SUCCESS;
|
|
}
|
|
}
|
|
|
|
return film::FAILURE;
|
|
}
|
|
///////////////////////////////
|
|
// Properties
|
|
bool filmGMERLIN::enumProperties(gem::Properties&readable,
|
|
gem::Properties&writeable) {
|
|
readable.clear();
|
|
writeable.clear();
|
|
|
|
gem::any value;
|
|
value=0.;
|
|
readable.set("fps", value);
|
|
readable.set("frames", value);
|
|
readable.set("width", value);
|
|
readable.set("height", value);
|
|
|
|
return false;
|
|
}
|
|
|
|
void filmGMERLIN::setProperties(gem::Properties&props) {
|
|
}
|
|
|
|
void filmGMERLIN::getProperties(gem::Properties&props) {
|
|
std::vector<std::string> keys=props.keys();
|
|
gem::any value;
|
|
double d;
|
|
unsigned int i=0;
|
|
for(i=0; i<keys.size(); i++) {
|
|
std::string key=keys[i];
|
|
props.erase(key);
|
|
if("fps"==key) {
|
|
d=m_fps;
|
|
value=d; props.set(key, value);
|
|
}
|
|
if("frames"==key && m_numFrames>=0) {
|
|
d=m_numFrames;
|
|
value=d; props.set(key, value);
|
|
}
|
|
if("tracks"==key && m_numTracks>=0) {
|
|
d=m_numTracks;
|
|
value=d; props.set(key, value);
|
|
}
|
|
if("width"==key) {
|
|
d=m_image.image.xsize;
|
|
value=d; props.set(key, value);
|
|
}
|
|
if("height"==key) {
|
|
d=m_image.image.ysize;
|
|
value=d; props.set(key, value);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
#endif // GMERLIN
|