⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ffmpeg_raw.cpp

📁 彩信浏览器
💻 CPP
字号:
// This file is part of Ambulant Player, www.ambulantplayer.org.//// Copyright (C) 2003-2007 Stichting CWI, // Kruislaan 413, 1098 SJ Amsterdam, The Netherlands.//// Ambulant Player 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.//// Ambulant Player 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 Ambulant Player; if not, write to the Free Software// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA#include "ambulant/net/ffmpeg_common.h"#include "ambulant/net/ffmpeg_raw.h" #include "ambulant/net/ffmpeg_factory.h" #include "ambulant/lib/logger.h"#include "ambulant/net/url.h"// WARNING: turning on AM_DBG globally for the ffmpeg code seems to trigger// a condition that makes the whole player hang or collapse. So you probably// shouldn't do it:-)//#define AM_DBG#ifndef AM_DBG#define AM_DBG if(0)#endif using namespace ambulant;using namespace net;raw_datasource_factory *ambulant::net::get_ffmpeg_raw_datasource_factory(){#if 0	static raw_datasource_factory *s_factory;		if (!s_factory) s_factory = new ffmpeg_raw_datasource_factory();	return s_factory;#else	return new ffmpeg_raw_datasource_factory();#endif}#define DEFAULT_BUFFER_SIZE 128*1024datasource* ffmpeg_raw_datasource_factory::new_raw_datasource(const net::url& url){		AM_DBG lib::logger::get_logger()->debug("ffmpeg_raw_datasource_factory::new_raw_datasource(%s)", repr(url).c_str());	URLContext *context = detail::ffmpeg_rawreader::supported(url);	if (!context) {		AM_DBG lib::logger::get_logger()->debug("ffmpeg_raw_datasource_factory::new_raw_datasource: no support for %s", repr(url).c_str());		return NULL;	}	detail::ffmpeg_rawreader *thread = new detail::ffmpeg_rawreader(context);	datasource *ds = new ffmpeg_raw_datasource(url, context, thread);	if (ds == NULL) {		AM_DBG lib::logger::get_logger()->debug("ffmpeg_raw_datasource_factory::new_raw_datasource: could not allocate ffmpeg_video_datasource");		thread->cancel();		return NULL;	}	thread->start();	return ds;}// **************************** ffmpeg_rawreader *****************************detail::ffmpeg_rawreader::ffmpeg_rawreader(URLContext *con):   m_con(con),	m_sink(NULL){	AM_DBG lib::logger::get_logger()->debug("ffmpeg_rawreader::ffmpeg_rawreader() m_con=0x%x", con);}detail::ffmpeg_rawreader::~ffmpeg_rawreader(){	m_lock.enter();	AM_DBG lib::logger::get_logger()->debug("ffmpeg_rawreader::~ffmpeg_rawreader() m_con=0x%x", m_con);	if (m_con) url_close(m_con);	m_con = NULL;	m_lock.leave();}URLContext *detail::ffmpeg_rawreader::supported(const net::url& url){	ffmpeg_init();	// Setup struct to allow ffmpeg to determine whether it supports this	URLContext *ic = NULL;	int err = url_open(&ic, url.get_url().c_str(), URL_RDONLY);	if (err) {		AM_DBG lib::logger::get_logger()->debug("ffmpeg_rawreader::supported(%s): url_open returned error %d, ic=0x%x", repr(url).c_str(), err, (void*)ic);		if (ic) {			url_close(ic);			ic = NULL;		}	}	return ic;}voiddetail::ffmpeg_rawreader::cancel(){	if (is_running())		stop();	release();}void detail::ffmpeg_rawreader::set_datasink(detail::rawdatasink *parent){	m_lock.enter();	assert(m_sink == 0);	m_sink = parent;	m_lock.leave();}unsigned longdetail::ffmpeg_rawreader::run(){	m_lock.enter();	AM_DBG lib::logger::get_logger()->debug("ffmpeg_rawreader::run: started");	while (!exit_requested()) {		uint8_t *sinkbuffer;		int sinkbuffersize;				sinkbuffersize = m_sink->get_sinkbuffer(&sinkbuffer);		if (sinkbuffersize == 0) {			m_lock.leave();			sleep(1);			m_lock.enter();		} else {			int bytecount;			AM_DBG lib::logger::get_logger("ffmpeg_rawreader::run: calling url_read(size=%d)", sinkbuffersize);			bytecount = url_read(m_con, sinkbuffer, sinkbuffersize);			AM_DBG lib::logger::get_logger("ffmpeg_rawreader::run:url_read() returned %d", sinkbuffersize);			if (m_sink && bytecount >= 0)				m_sink->pushdata(bytecount);			if (bytecount <= 0 || !m_sink)				break;		}	}	AM_DBG lib::logger::get_logger()->debug("ffmpeg_rawreader::run: final sinkdata(0)");	if (m_sink) m_sink->pushdata(-1);	m_lock.leave();	return 0;}		// **************************** ffmpeg_raw_datasource *****************************ffmpeg_raw_datasource::ffmpeg_raw_datasource(const net::url& url, URLContext *context,	detail::ffmpeg_rawreader *thread):	m_url(url),	m_con(context),	m_src_end_of_file(false),	m_event_processor(NULL),	m_thread(thread),	m_client_callback(NULL){	AM_DBG lib::logger::get_logger()->debug("ffmpeg_raw_datasource::ffmpeg_raw_datasource() -> 0x%x", (void*)this);	if (!m_thread) {		lib::logger::get_logger()->error(gettext("%s: Cannot start reader thread"), url.get_url().c_str());		m_src_end_of_file = true;		return;	}	m_thread->set_datasink(this);}ffmpeg_raw_datasource::~ffmpeg_raw_datasource(){	AM_DBG lib::logger::get_logger()->debug("ffmpeg_raw_datasource::~ffmpeg_raw_datasource(0x%x)", (void*)this);	stop();}voidffmpeg_raw_datasource::stop(){	m_lock.enter();	AM_DBG lib::logger::get_logger()->debug("ffmpeg_raw_datasource::stop(0x%x)", (void*)this);	if (m_thread) {		detail::ffmpeg_rawreader *tmpthread = m_thread;		m_thread = NULL;		m_lock.leave();		tmpthread->cancel();		m_lock.enter();	}	m_thread = NULL;	AM_DBG lib::logger::get_logger()->debug("ffmpeg_raw_datasource::stop: thread stopped");	m_con = NULL; // owned by the thread	if (m_client_callback) delete m_client_callback;	m_client_callback = NULL;	m_lock.leave();}	void ffmpeg_raw_datasource::start(ambulant::lib::event_processor *evp, ambulant::lib::event *callbackk){	m_lock.enter();		if (m_client_callback != NULL) {		AM_DBG lib::logger::get_logger()->debug("ffmpeg_raw_datasource::start(): m_client_callback already set!");		delete m_client_callback;		m_client_callback = NULL;	}	if (m_buffer.buffer_not_empty() || _end_of_file() ) {		// We have data (or EOF) available. Don't bother starting up our source again, in stead		// immedeately signal our client again		if (evp && callbackk) {			AM_DBG lib::logger::get_logger()->debug("ffmpeg_raw_datasource::start: trigger client callback");			evp->add_event(callbackk, 0, ambulant::lib::ep_med);		} else {			lib::logger::get_logger()->debug("Internal error: ffmpeg_raw_datasource::start(): no client callback!");			lib::logger::get_logger()->warn(gettext("Programmer error encountered"));		}	} else {		// We have no data available. Start our source, and in our data available callback we		// will signal the client.		m_client_callback = callbackk;		m_event_processor = evp;	}	m_lock.leave();} void ffmpeg_raw_datasource::readdone(int len){	m_lock.enter();	m_buffer.readdone(len);	AM_DBG lib::logger::get_logger()->debug("ffmpeg_raw_datasource.readdone : done with %d bytes", len);//	restart_input();	m_lock.leave();}int ffmpeg_raw_datasource::get_sinkbuffer(uint8_t **datap){	m_lock.enter();	if (m_buffer.buffer_full()) {		*datap = NULL;		m_lock.leave();		return 0;	}	// Otherwise we just take a guess. 	*datap = (uint8_t *)m_buffer.get_write_ptr(DEFAULT_BUFFER_SIZE);	m_lock.leave();	if (*datap)		return DEFAULT_BUFFER_SIZE;	else		return 0;}voidffmpeg_raw_datasource::pushdata(int sz){	m_lock.enter();	if (sz >= 0) m_buffer.pushdata(sz);	if (sz <= 0)		m_src_end_of_file = true;	if ( m_client_callback && (m_buffer.buffer_not_empty() || m_src_end_of_file ) ) {		AM_DBG lib::logger::get_logger()->debug("ffmpeg_raw_datasource::pushdata(): calling client callback (%d, %d)", m_buffer.size(), m_src_end_of_file);		m_event_processor->add_event(m_client_callback, 0, ambulant::lib::ep_med);		m_client_callback = NULL;		//m_event_processor = NULL;	} else {		AM_DBG lib::logger::get_logger()->debug("ffmpeg_raw_datasource::pushdata(): No client callback!");	}	m_lock.leave();}bool ffmpeg_raw_datasource::end_of_file(){	m_lock.enter();	bool rv = _end_of_file();	m_lock.leave();	return rv;}bool ffmpeg_raw_datasource::_end_of_file(){	// private method - no need to lock	if (m_buffer.buffer_not_empty()) return false;	return m_src_end_of_file;}char* ffmpeg_raw_datasource::get_read_ptr(){	m_lock.enter();	char *rv = m_buffer.get_read_ptr();	m_lock.leave();	return rv;}int ffmpeg_raw_datasource::size() const{		return m_buffer.size();}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -