All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.antstreaming.rtsp.PacketReceiverRunnable Maven / Gradle / Ivy

package com.antstreaming.rtsp;

import static org.bytedeco.javacpp.avcodec.AV_CODEC_FLAG_GLOBAL_HEADER;
import static org.bytedeco.javacpp.avcodec.av_packet_unref;
import static org.bytedeco.javacpp.avcodec.avcodec_parameters_copy;
import static org.bytedeco.javacpp.avformat.AVFMT_GLOBALHEADER;
import static org.bytedeco.javacpp.avformat.AVFMT_NOFILE;
import static org.bytedeco.javacpp.avformat.AVIO_FLAG_WRITE;
import static org.bytedeco.javacpp.avformat.av_find_input_format;
import static org.bytedeco.javacpp.avformat.av_interleaved_write_frame;
import static org.bytedeco.javacpp.avformat.av_read_frame;
import static org.bytedeco.javacpp.avformat.av_write_trailer;
import static org.bytedeco.javacpp.avformat.avformat_alloc_output_context2;
import static org.bytedeco.javacpp.avformat.avformat_close_input;
import static org.bytedeco.javacpp.avformat.avformat_find_stream_info;
import static org.bytedeco.javacpp.avformat.avformat_free_context;
import static org.bytedeco.javacpp.avformat.avformat_new_stream;
import static org.bytedeco.javacpp.avformat.avformat_open_input;
import static org.bytedeco.javacpp.avformat.avformat_write_header;
import static org.bytedeco.javacpp.avformat.avio_closep;
import static org.bytedeco.javacpp.avutil.AV_ROUND_NEAR_INF;
import static org.bytedeco.javacpp.avutil.AV_ROUND_PASS_MINMAX;
import static org.bytedeco.javacpp.avutil.av_dict_set;
import static org.bytedeco.javacpp.avutil.av_rescale_q;
import static org.bytedeco.javacpp.avutil.av_rescale_q_rnd;

import java.io.File;
import java.io.FileOutputStream;
import java.io.OutputStream;
import java.net.URI;
import java.util.Date;

import org.apache.mina.core.session.IoSession;
import org.bytedeco.javacpp.PointerPointer;
import org.bytedeco.javacpp.avcodec.AVPacket;
import org.bytedeco.javacpp.avformat;
import org.bytedeco.javacpp.avformat.AVFormatContext;
import org.bytedeco.javacpp.avformat.AVIOContext;
import org.bytedeco.javacpp.avformat.AVInputFormat;
import org.bytedeco.javacpp.avformat.AVStream;
import org.bytedeco.javacpp.avutil.AVDictionary;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;

import com.antstreaming.rtsp.protocol.RtspCode;
import com.antstreaming.rtsp.protocol.RtspHeaderCode;
import com.antstreaming.rtsp.protocol.RtspResponse;
import com.antstreaming.rtsp.session.DateUtil;

public class PacketReceiverRunnable implements Runnable {

	private AVFormatContext inputFormatCtx;
	private long[] lastDTS;
	private AVFormatContext outputRTMPFormatContext;
	private boolean closeRequest = false;
	private File sdpFile;

	private Logger logger = LoggerFactory.getLogger(PacketReceiverRunnable.class);
	private String url;
	private StringBuilder liveStreamSdpDef;
	private String announcedStreamName;
	private IoSession session;
	private String sessionKey;
	private String cseq;
	private ThreadPoolTaskScheduler mTaskScheduler;


	public PacketReceiverRunnable(ThreadPoolTaskScheduler mTaskScheduler, String cseq, String sessionKey,
			IoSession session, String announcedStreamName, StringBuilder liveStreamSdpDef, String url) {
		this.mTaskScheduler = mTaskScheduler;
		this.cseq = cseq;
		this.sessionKey = sessionKey;
		this.session = session;
		this.announcedStreamName = announcedStreamName;
		this.liveStreamSdpDef = liveStreamSdpDef;
		this.url = url;
	}

	public boolean prepare_input_context(ThreadPoolTaskScheduler mTaskScheduler, final String cseq, final String sessionKey, final IoSession session, String announcedStreamName, StringBuilder liveStreamSdpDef, String streamUrl) {
		
		try {
			sdpFile = new File(announcedStreamName + ".sdp");
			
			try (OutputStream fos = new FileOutputStream(sdpFile)) {
				fos.write(liveStreamSdpDef.toString().getBytes());
			}

			inputFormatCtx = new AVFormatContext(null);
			

			AVDictionary options = new AVDictionary();
			int ret = av_dict_set(options, "protocol_whitelist", "file,crypto,udp,rtp", 0);
			if (ret < 0) {
				logger.debug("cannot set protocol_whitelist");
				return false;
			}
			
			AVInputFormat sdpFormat = av_find_input_format("sdp");


			if (avformat_open_input(inputFormatCtx, sdpFile.getAbsolutePath(), sdpFormat, options) != 0) {
				logger.debug("Could not open rtp for demuxing");
				return false;
			}
			
			mTaskScheduler.schedule(new Runnable() {
				@Override
				public void run() {
					RtspResponse response = new RtspResponse();
					response.setCode(RtspCode.OK);
					response.setHeader(RtspHeaderCode.CSEQ, cseq);
					response.setHeader(RtspHeaderCode.DATE, DateUtil.getGmtDate());
					response.setHeader(RtspHeaderCode.SESSION, sessionKey);
					session.write(response);
				}
			}, new Date());

			if (avformat_find_stream_info(inputFormatCtx, (PointerPointer)null) < 0) {
				logger.debug("Could not get stream info");
				return false;
			}


			outputRTMPFormatContext = new AVFormatContext(null);

			ret = avformat_alloc_output_context2(outputRTMPFormatContext, null, "flv", null);

			lastDTS = new long[this.inputFormatCtx.nb_streams()];
			for (int i=0; i < inputFormatCtx.nb_streams(); i++) {
				AVStream inStream = inputFormatCtx.streams(i);
				AVStream outStream = avformat_new_stream(outputRTMPFormatContext, inStream.codec().codec());

				
				ret = avcodec_parameters_copy(outStream.codecpar(), inStream.codecpar());
				
				if (ret < 0) {
					logger.debug("Cannot get codec parameters\n");
					return false;
				}

				outStream.codec().codec_tag(0);
				
				if ((outputRTMPFormatContext.oformat().flags() & AVFMT_GLOBALHEADER) != 0) {
					outStream.codec().flags(outStream.codec().flags() | AV_CODEC_FLAG_GLOBAL_HEADER);
				}
				
				//initialize last decoding time stamp reference value
				lastDTS[i] = -1;
			}


			if ((outputRTMPFormatContext.flags() & AVFMT_NOFILE) == 0) 
			{
				AVIOContext pb = new AVIOContext(null);

				URI url = new URI(streamUrl);
				String urlStr = "rtmp://" + url.getHost() + "/" + url.getPath();
				//				logger.debug("rtmp url: " + urlStr);
				
				ret = avformat.avio_open(pb,  urlStr, AVIO_FLAG_WRITE);
				outputRTMPFormatContext.pb(pb);

				ret = avformat_write_header(outputRTMPFormatContext, (AVDictionary)null);
				if (ret < 0) {
					logger.debug("Cannot write header to rtmp\n");
					return false;
				}
			}
		}
		catch (Exception e) {
			e.printStackTrace();
		}
		

		return true;
	}
	
	//TODO: check that if process interrupts soomehow, there is no memory leakage

	@Override
	public void run() {


		if (!prepare_input_context(mTaskScheduler, cseq, sessionKey, session, announcedStreamName, liveStreamSdpDef, url)) {
			closeInternal();
			return;
		}
		
		while(true) {
			//TODO check garbage collection and memory allocations on jvisualvm
			AVPacket pkt = new AVPacket();

			int ret = av_read_frame(inputFormatCtx, pkt);
			if (ret<0) {
				closeInternal();
				break;
			}
			int packetIndex = pkt.stream_index();

			AVStream in_stream = inputFormatCtx.streams(packetIndex);
			AVStream out_stream2 = outputRTMPFormatContext.streams(packetIndex);
			
		//	if (pkt.dts() != avutil.AV_NOPTS_VALUE && lastDTS[packetIndex] == -1) {
		//		lastDTS[packetIndex] = pkt.dts();
		//	}
			if (pkt.dts() < 0) {
				continue;
			}
			
			if (lastDTS[packetIndex] >= pkt.dts()) {
				logger.warn("dts timestamps are not in correct order last dts:"  + lastDTS[packetIndex] 
						+ " current dts:" + pkt.dts() + " fixing problem by adding offset");

				pkt.dts(lastDTS[packetIndex] + 1);
			}

			lastDTS[packetIndex] = pkt.dts();
			if (pkt.dts() > pkt.pts()) {
				pkt.pts(pkt.dts());
			}

			//sending rtmp
			pkt.pts(av_rescale_q_rnd(pkt.pts(), in_stream.time_base(), out_stream2.time_base(), AV_ROUND_NEAR_INF|AV_ROUND_PASS_MINMAX));
			pkt.dts(av_rescale_q_rnd(pkt.dts(), in_stream.time_base(), out_stream2.time_base(), AV_ROUND_NEAR_INF|AV_ROUND_PASS_MINMAX));
			pkt.duration(av_rescale_q(pkt.duration(), in_stream.time_base(), out_stream2.time_base()));
			pkt.pos(-1);


			ret = av_interleaved_write_frame(outputRTMPFormatContext, pkt);
			if (ret < 0) {
				logger.debug("Error muxing rtmp packet\n");
			}

			av_packet_unref(pkt);

			if (closeRequest) {
				closeInternal();
				break;
			}
		}
	}

	public void closeInternal() {
		logger.warn("closing rtmp format context");
		av_write_trailer(outputRTMPFormatContext);


		logger.warn("closing rtsp input format context");
		// Close the video file
		if (inputFormatCtx != null) {
			avformat_close_input(inputFormatCtx);
		}


		// close output 
		if (outputRTMPFormatContext != null && ((outputRTMPFormatContext.oformat().flags() & AVFMT_NOFILE) == 0)) {
			logger.warn("closing rtmp format context pb");
			avio_closep(outputRTMPFormatContext.pb());
		}
		avformat_free_context(outputRTMPFormatContext);


		logger.debug("deleting sdpfile");
		if (sdpFile != null && sdpFile.exists()) {
			sdpFile.delete();
			sdpFile = null;
		}
	}

	public synchronized void closeMuxer() {
		closeRequest = true;

	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy