src/jlibrtp/AppCallerThread.java
author nikita@nikita-rack
Thu, 09 Apr 2009 20:26:58 +0200
changeset 99 8de21ac527ce
parent 13 e684f11070d5
permissions -rw-r--r--
revert pour refaire un push propre

/**
 * Java RTP Library (jlibrtp)
 * Copyright (C) 2006 Arne Kepp
 * 
 * This library 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 library 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 library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 */
package jlibrtp;

import java.util.Enumeration;

/**
 * The purpose of this thread is to check whether there are packets ready from 
 * any participants.
 * 
 * It should sleep when not in use, and be woken up by a condition variable.
 * 
 * Optionally, if we do jitter-control, the condition variable should have a max waiting period 
 * equal to how often we need to push data.
 * 
 * @author Arne Kepp
 */
public class AppCallerThread extends Thread {
	/**  The parent RTP Session */
	RTPSession rtpSession;
	/**  The applications interface, where the callback methods are called */
	RTPAppIntf appl;
	
	/**
	 * Instatiates the AppCallerThread
	 * 
	 * @param session the RTPSession with participants etc
	 * @param rtpApp the interface to which data is given
	 */
	protected AppCallerThread(RTPSession session, RTPAppIntf rtpApp) {
		rtpSession = session;
		appl = rtpApp;
		if(RTPSession.rtpDebugLevel > 1) {
			System.out.println("<-> AppCallerThread created");
		}  
	}
	
	/**
	 * The AppCallerThread will run in this loop until the RTPSession
	 * is terminated.
	 * 
	 * Whenever an RTP packet is received it will loop over the
	 * participants to check for packet buffers that have available
	 * frame.
	 */
	public void run() {
		if(RTPSession.rtpDebugLevel > 3) {
			System.out.println("-> AppCallerThread.run()");
		}
		
		while(rtpSession.endSession == false) {
			
			rtpSession.pktBufLock.lock();
		    try {
				if(RTPSession.rtpDebugLevel > 4) {
					System.out.println("<-> AppCallerThread going to Sleep");
				}
				
				try { rtpSession.pktBufDataReady.await(); } 
					catch (Exception e) { System.out.println("AppCallerThread:" + e.getMessage());}
					
		    	// Next loop over all participants and check whether they have anything for us.
				Enumeration<Participant> enu = rtpSession.partDb.getParticipants();
				
				while(enu.hasMoreElements()) {
					Participant p = enu.nextElement(); 
					
					boolean done = false;
					//System.out.println(p.ssrc + " " + !done +" " + p.rtpAddress 
					//		+ " " + rtpSession.naiveReception + " " + p.pktBuffer);
					//System.out.println("done: " + done + "  p.unexpected: " + p.unexpected);
					while(!done && (!p.unexpected || rtpSession.naiveReception) 
							&& p.pktBuffer != null && p.pktBuffer.length > 0) {

						DataFrame aFrame = p.pktBuffer.popOldestFrame();
						if(aFrame == null) {
							done = true;
						} else {
							appl.receiveData(aFrame, p);
						}
					}
				}
		    
		     } finally {
		       rtpSession.pktBufLock.unlock();
		     }
			
		}
		if(RTPSession.rtpDebugLevel > 3) {
			System.out.println("<- AppCallerThread.run() terminating");
		}  
	}

}