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

src.com.ibm.as400.access.ClientAccessDataStream Maven / Gradle / Ivy

There is a newer version: 20.0.8
Show newest version
///////////////////////////////////////////////////////////////////////////////
//                                                                             
// 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