src.com.ibm.as400.access.ClientAccessDataStream Maven / Gradle / Ivy
///////////////////////////////////////////////////////////////////////////////
//
// JTOpen (IBM Toolbox for Java - OSS version)
//
// Filename: ClientAccessDataStream.java
//
// The source code contained herein is licensed under the IBM Public License
// Version 1.0, which has been approved by the Open Source Initiative.
// Copyright (C) 1997-2001 International Business Machines Corporation and
// others. All rights reserved.
//
///////////////////////////////////////////////////////////////////////////////
package com.ibm.as400.access;
import java.io.IOException;
import java.io.InputStream;
import java.util.Hashtable;
/** Base class for client access server data streams.
* Provides methods to access common client access data stream header.
*
*/
public class ClientAccessDataStream extends DataStream
{
static final String copyright = "Copyright (C) 1997-2001 International Business Machines Corporation and others.";
protected static final int HEADER_LENGTH = 20;
private static final CADSPool basePool_ = new CADSPool(); //@P0A - base datastream pool
private class InUseLock extends Object {} //@C7A
private InUseLock inUseLock_ = new InUseLock();
// Note: All references to inUse_ in subclass should be followed by
// a call to canUse, which which acquire the lock to validate that
// the DataStream can actually be used.
boolean inUse_; //@P0A
private CADSPool fromPool_ = null;
private int fromPoolIndex_ = 0;
// Note: The following method is called by AS400ThreadedServer and AS400NoThreadServer.
// Construct an appropriate client access data stream object. Read from the InputStream to obtain the data stream data for the object.
// @param is InputStream from which to read to obtain the data stream contents.
// @param dataStreams Hashtable containing instances of data stream objects to receive into. This table is searched first when a reply comes in. If found the datastream will be removed from here as it is received.
// @param dataStreams Prototypes Hashtable containing data stream objects from which to obtain a model for this object.
// @exception IOException Data read from the input stream is less than 20 bytes or we are unable to read from the input stream for some other reason.
// @return ClientAccessDataStream object.
static final ClientAccessDataStream construct(InputStream is, Hashtable dataStreams, Hashtable dataStreamPrototypes, AS400ImplRemote system, int connectionID) throws IOException
{
// Construct a client access data stream to receive the data stream header. By using the default constructor for ClientAccessDataStream, we get a data stream of size HEADER_LENGTH.
//@P0D ClientAccessDataStream baseDataStream = new ClientAccessDataStream();
ClientAccessDataStream baseDataStream = basePool_.getUnusedStream(); //@P0A
try
{
// Receive the header.
byte[] data = baseDataStream.data_;
if (readFromStream(is, data, 0, HEADER_LENGTH, connectionID) < HEADER_LENGTH)
{
if (Trace.traceOn_) Trace.log(Trace.ERROR, "Failed to read all of the data stream header."); //@P0C
baseDataStream.returnToPool(); baseDataStream=null;
throw new ConnectionDroppedException(ConnectionDroppedException.CONNECTION_DROPPED);
}
// int length2 = ((data[0] & 0xFF) << 24) + ((data[1] & 0xFF) << 16) + ((data[2] & 0xFF) << 8) + (data[3] & 0xFF);
int length = baseDataStream.getLength();
if (baseDataStream.data_[6] != (byte)0xE0)
{
// boolean traceTurnedOn = false;
// if (!Trace.traceOn_) {
// traceTurnedOn = true;
// }
// Debugging code.
// if (! DataStream.traceOpened) {
// SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMdd_HHmm");
// Trace.setFileName("/tmp/toolboxTrace."+sdf.format(new Date())+".txt");
// DataStream.traceOpened=true;
// }
// if (!Trace.traceOn_) {
// traceTurnedOn = true;
// Trace.setTraceAllOn(true);
// Trace.setTraceOn(true);
// }
// Trace.log(Trace.ERROR, "Debug0601: Incorrect data stream header detected. baseDataStream.data_("+baseDataStream.data_.toString()+") length=("+length+")",
// baseDataStream.data_, 0, HEADER_LENGTH);
// Trace.log(Trace.ERROR, "Debug0601: Incorrect data stream header detected. data_("+data.toString()+") length=("+length2+")",
// data, 0, HEADER_LENGTH);
if (Trace.traceOn_) {
Trace.log(Trace.ERROR, "Incorrect data stream header detected.",
baseDataStream.data_, 0, HEADER_LENGTH);
}
// Debugging code
//if (traceTurnedOn) {
// Trace.setTraceAllOn(false);
// Trace.setTraceOn(false);
//}
baseDataStream.returnToPool(); baseDataStream=null;
is.skip(is.available()); // disregard the rest of this data stream
// Debug...
// Just hang the thread
// while (true) {
// try {
// Thread.sleep(100000);
// } catch (Exception e) {
// break;
// }
//}
throw new InternalErrorException(InternalErrorException.DATA_STREAM_UNKNOWN);
}
// First look for an instance data stream.
// If we found it remove it since instance datastreams are only used once.
// Print is the only thing that uses this.
ClientAccessDataStream newDataStream = (ClientAccessDataStream)dataStreams.get(baseDataStream);
if(newDataStream != null && newDataStream instanceof NPDataStream)
{
NPDataStream npds = (NPDataStream)newDataStream;
if(baseDataStream.getCorrelation() == npds.getCorrelationID())
dataStreams.remove(baseDataStream);
}
if(newDataStream != null && !(newDataStream instanceof NPDataStream))
{
newDataStream = (ClientAccessDataStream)dataStreams.remove(baseDataStream);
}
if (newDataStream == null) //@P0C
{
// If we couldn't find an instance datastream to receive into, look for a prototype data stream to generate one with.
ClientAccessDataStream modelDataStream = (ClientAccessDataStream)dataStreamPrototypes.get(baseDataStream);
if (modelDataStream == null)
{
// No model was found in the hash table, so we will return a generic data stream.
newDataStream = new ClientAccessDataStream();
}
else
{
// Get a new instance of the data stream.
newDataStream = (ClientAccessDataStream)modelDataStream.getNewDataStream();
if (newDataStream == null)
{
newDataStream = new ClientAccessDataStream();
}
}
}
newDataStream.system_ = system;
if (Trace.traceOn_) newDataStream.setConnectionID(connectionID);
// Initialize the header section of the new data stream.
int nowLength = baseDataStream.getLength();
// Debugging code
// if ((nowLength != length) || (length != length2) || (data != baseDataStream.data_ )) {
// if (! DataStream.traceOpened) {
// SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMdd_HHmm");
//
// Trace.setFileName("/tmp/toolboxTrace."+sdf.format(new Date())+".txt");
// DataStream.traceOpened=true;
// }
//
// boolean traceTurnedOn = false;
// if (!Trace.traceOn_) {
// traceTurnedOn = true;
// Trace.setTraceAllOn(true);
// Trace.setTraceOn(true);
// }
// if (nowLength != length) {
// Trace.log(Trace.DATASTREAM, "Debug0601: Buffer corrupted.. Original length="+length+" nowLength="+nowLength);
// }
// if (length != length2) {
// Trace.log(Trace.DATASTREAM, "Debug0601: Buffer corrupted.. Original length="+length+" length2="+length2);
// }
// if (data != baseDataStream.data_) {
// Trace.log(Trace.DATASTREAM, "Debug0601: Buffer corrupted.. data="+data+" baseDataStream.data_="+baseDataStream.data_);
// }
// if (traceTurnedOn) {
// Trace.setTraceAllOn(false);
// Trace.setTraceOn(false);
// }
//}
//
// TODO: Restructure this code so that a new byte array is not always allocated.
//
newDataStream.data_ = new byte[nowLength];
System.arraycopy(baseDataStream.data_, 0, newDataStream.data_, 0, HEADER_LENGTH);
if (newDataStream.data_.length - HEADER_LENGTH > 0)
{
// Receive any remaining bytes.
// The number of bytes to read is calculated from newDataStream.data_.length - HEADER_LENGTH
newDataStream.readAfterHeader(is);
}
return newDataStream;
}
finally
{
if (baseDataStream != null) {
baseDataStream.returnToPool(); baseDataStream = null;
}
}
}
// Constructs an empty ClientAccessDataStream object.
protected ClientAccessDataStream()
{
super(HEADER_LENGTH);
}
// Constructs a ClientAccessDataStream object.
// @param data Byte array with which to initialize this data stream.
ClientAccessDataStream(byte[] ds)
{
super(HEADER_LENGTH, ds);
}
// Constructs an empty ClientAccessDataStream object that is pooled.
ClientAccessDataStream(CADSPool pool, int poolIndex) {
super(HEADER_LENGTH);
fromPool_ = pool;
fromPoolIndex_ = poolIndex;
}
// Retrieve the request correlation for this data stream. The return value may be invalid if it has not been set.
// @return The request correlation number.
int getCorrelation()
{
return get32bit(12);
}
// Retrieve the CS instance for the data stream. The return value may be invalid if it has not been set.
// @return The CS instance of the data stream.
int getCSInstance()
{
return get32bit(8);
}
// Retrieve the header ID for the data stream. The return value may be invalid if it has not been set.
// @return The ID of the data stream.
int getHeaderID()
{
return get16bit(4);
}
// Retrieve the total length of the data stream. The return value may be invalid if it has not been set.
// @return The total length of this data stream.
int getLength()
{
return get32bit(0);
}
// Retrieve the request/reply ID of the data stream. The return value may be invalid if it has not been set.
// @return The request/reply ID of this data stream.
int getReqRepID()
{
//@P0D return get16bit(18);
return((data_[18] & 0xFF) << 8) + (data_[19] & 0xFF); //@P0A
}
// Retrieve the server ID of the data stream. The return value may be invalid if it has not been set.
// @return The server ID of this data stream.
int getServerID()
{
return get16bit(6);
}
// Retrieve the template length of the data stream. The return value may be invalid if it has not been set.
// @return The template length of this data stream.
int getTemplateLen()
{
return get16bit(16);
}
// Retrieve the hash code of the data stream. The return value may be invalid if it has not been set.
// Note: Reply data stream sub-classes should override this method to return the request/reply id directly instead of calling getReqRepId().
// @return The hash code of this data stream.
public int hashCode()
{
//@P0D return getReqRepID();
return((data_[18] & 0xFF) << 8) + (data_[19] & 0xFF); //@P0A
}
// Set the request correlation for this data stream.
// @param id The request correlation number.
protected void setCorrelation(int id)
{
set32bit(id, 12);
}
// Set the CS instance for this data stream.
// @param id The CS instance.
protected void setCSInstance(int id)
{
set32bit(id, 8);
}
// Set the header ID for the data stream. It should be set to 0 for most of the Client Access servers.
// @param id The header ID to set.
protected void setHeaderID(int id)
{
set16bit(id, 4);
}
// Set the length of the data stream. This is the total length of the data stream.
// @param len The length of the data stream.
protected void setLength(int len)
{
//@P0D set32bit(len, 0);
data_[0] = (byte)(len >>> 24);
data_[1] = (byte)(len >>> 16);
data_[2] = (byte)(len >>> 8);
data_[3] = (byte) len;
}
// Set the request/reply ID for the data stream.
// @param id The request/reply ID to set.
protected void setReqRepID(int id)
{
set16bit(id, 18);
}
// Set the server ID for the data stream. This is the ID of the server to talk to.
// @param id The ID of the server.
protected void setServerID(int id)
{
set16bit(id, 6);
}
// Set the template length for the data stream.
// @param len The template length.
protected void setTemplateLen(int len)
{
set16bit(len, 16);
}
//
// Indicate that the buffer can be returned to the pool. In the past, the pooling implementation
// just set inUse_=false to return to the pool. This is provided so that the request buffer can be resized
// by inheriting classes
//
void returnToPool() throws InternalErrorException { // @A7C
synchronized(inUseLock_) {
if (inUse_) {
// Use this to find places where the object is used after it is returned to the pool
// Note: For DBBaseRequestDS objects, the data_ pointer has been set to the one for the
// DBStorage object.
// if (data_ != null) {
// Arrays.fill(data_, (byte) 0xeb);
// }
inUse_ = false;
// if (DBDSPool.monitor && this instanceof DBReplyRequestedDS) {
// System.out.println("Freeing "+((DBReplyRequestedDS) this).poolIndex);
// }
} else {
// This is an error case. You cannot double free a buffer
throw new InternalErrorException(InternalErrorException.UNKNOWN);
}
}
// Let the pool know that something was returned. This should speed up searches through the pool.
if (fromPool_ != null) {
fromPool_.returnToPool(this, fromPoolIndex_);
}
}
/**
* Can this be used. If not, false is returned.
* If it can be used, then inUse_ is set to return and true is returned
* @return true if this can be used
*/
public boolean canUse() {
synchronized (inUseLock_) {
if (inUse_) {
return false;
} else {
inUse_ = true;
return true;
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy