/* The receiver for the Giotto domain.
Copyright (c) 1998-2005 The Regents of the University of California.
All rights reserved.
Permission is hereby granted, without written agreement and without
license or royalty fees, to use, copy, modify, and distribute this
software and its documentation for any purpose, provided that the above
copyright notice and the following two paragraphs appear in all copies
of this software.
IN NO EVENT SHALL THE UNIVERSITY OF CALIFORNIA BE LIABLE TO ANY PARTY
FOR DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN IF
THE UNIVERSITY OF CALIFORNIA HAS BEEN ADVISED OF THE POSSIBILITY OF
SUCH DAMAGE.
THE UNIVERSITY OF CALIFORNIA SPECIFICALLY DISCLAIMS ANY WARRANTIES,
INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE
PROVIDED HEREUNDER IS ON AN "AS IS" BASIS, AND THE UNIVERSITY OF
CALIFORNIA HAS NO OBLIGATION TO PROVIDE MAINTENANCE, SUPPORT, UPDATES,
ENHANCEMENTS, OR MODIFICATIONS.
PT_COPYRIGHT_VERSION_2
COPYRIGHTENDKEY
*/
package ptolemy.domains.giotto.kernel;
import ptolemy.actor.AbstractReceiver;
import ptolemy.actor.IOPort;
import ptolemy.actor.NoRoomException;
import ptolemy.actor.NoTokenException;
import ptolemy.actor.StateReceiver;
import ptolemy.data.Token;
import ptolemy.kernel.util.IllegalActionException;
//////////////////////////////////////////////////////////////////////////
//// GiottoReceiver
/**
A receiver for the Giotto domain. It uses double-buffering. An actor
may write to a receiver using put at any time. However, a new token
will only be available to get if update has been called before. The
update method makes the token from the last put invocation available
to get. The GiottoDirector delays calls to update according to the
Giotto semantics.
@author Christoph Meyer Kirsch and Edward A. Lee
@version $Id: GiottoReceiver.java,v 1.37 2005/07/08 19:58:25 cxh Exp $
@since Ptolemy II 1.0
@Pt.ProposedRating Yellow (cm)
@Pt.AcceptedRating Yellow (liuj)
@see GiottoDirector
*/
public class GiottoReceiver extends AbstractReceiver implements StateReceiver {
/** Construct an empty GiottoReceiver with no container.
*/
public GiottoReceiver() {
super();
}
/** Construct an empty GiottoReceiver with the specified container.
* @param container The container.
* @exception IllegalActionException If the container does
* not accept this receiver.
*/
public GiottoReceiver(IOPort container) throws IllegalActionException {
super(container);
}
///////////////////////////////////////////////////////////////////
//// public methods ////
/** Clear this receiver of any contained tokens.
* FIXME.
* Should rename and replace all the reset() with clear().
*/
public void clear() {
reset();
}
/** Get the contained and available token, i.e., get the last
* token that has been put into the receiver before the last
* update.
* @return A token.
* @exception NoTokenException If no token is available.
*/
public Token get() throws NoTokenException {
if (_token == null) {
throw new NoTokenException(getContainer(),
"Attempt to get data from an empty receiver.");
}
return _token;
}
/** Return true, since writing to this receiver is always allowed.
* @return True.
*/
public boolean hasRoom() {
return true;
}
/** Return true, since writing to this receiver is always allowed.
* @param numberOfTokens The size of tokens to be written to the receiver.
* @return True.
*/
public final boolean hasRoom(int numberOfTokens) {
return true;
}
/** Return true if there is a token available. A token is
* available whenever put() has been called at least once followed
* by a call to the update() method.
* @return True if there is a token available.
*/
public boolean hasToken() {
return (_token != null);
}
/** Return true if the receiver has at least one token available.
* Any number of calls to get() is allowed and will return the same
* available token.
* @param numberOfTokens The number of tokens available in this receiver.
* @return True if there is a token available.
*/
public final boolean hasToken(int numberOfTokens) {
return (_token != null);
}
/** Put a token into this receiver. Any token which has been put
* into the receiver before without calling update will be lost.
* The token becomes available to the get() method only after
* update() is called.
* <p>
* Note that putting a null into this receiver will leave the
* receiver empty after update. The receiver does not check
* against this but expects that IOPort will always put
* non-null tokens into receivers.
* @param token The token to be put into this receiver.
* @exception NoRoomException Not thrown in this base class.
*/
public void put(Token token) throws NoRoomException {
_nextToken = token;
}
/** Get the contained and available token, i.e., get the last
* token that has been put into the receiver before the last
* update and reset the _token only.
* @return A token.
* @exception NoTokenException If no token is available.
*/
public Token remove() throws NoTokenException {
if (_token == null) {
throw new NoTokenException(getContainer(),
"Attempt to get data from an empty receiver.");
}
Token buffer = _token;
_token = null;
return buffer;
}
/** Reset the receiver by removing all tokens from the receiver.
*/
public void reset() {
_token = null;
_nextToken = null;
}
/** Update the receiver by making the last token that has been
* passed to put() available to get().
*/
public void update() {
_token = _nextToken;
}
///////////////////////////////////////////////////////////////////
//// private variables ////
// The next token.
private Token _nextToken = null;
// The token available for reading.
private Token _token = null;
}