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

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

There is a newer version: 20.0.8
Show newest version
///////////////////////////////////////////////////////////////////////////////
//                                                                             
// JTOpen (IBM Toolbox for Java - OSS version)                              
//                                                                             
// Filename: DDMObjectDataStream.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-2002 International Business Machines Corporation and     
// others. All rights reserved.                                                
//                                                                             
///////////////////////////////////////////////////////////////////////////////
//
// @A1 - 09/19/2007 - The NULL Field Byte Map array must be set based on the 
//       maximum number of fields for any format in a given file.  See further 
//       detailed explaination below for the getObjectS38BUF() method.
//                                                                             
///////////////////////////////////////////////////////////////////////////////

package com.ibm.as400.access;

import java.io.IOException;
import java.io.CharConversionException;
import java.io.UnsupportedEncodingException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.Hashtable;

/**
 *DDM object data stream.
**/
class DDMObjectDataStream extends DDMDataStream
{
  private static final String copyright = "Copyright (C) 1997-2002 International Business Machines Corporation and others.";

  /**
   *Constructs a DDMObjectDataStream object.
  **/
  DDMObjectDataStream()
  {
    super();
  }

  /**
   *Constructs a DDMObjectDataStream object.
   *@param data the data with which to populate the object.
  **/
  DDMObjectDataStream(byte[] data)
  {
    super(data);
  }

  /**
   *Constructs a DDMObjectDataStream object.
   *@param length the length of the data stream.
  **/
  DDMObjectDataStream(int length)
  {
    super(length);
    // Initialize the header:
    //  Don't continue on error, not chained, GDS id = D0, type = RQSDSS,
    //  no same request correlation.
    setContinueOnError(false);
    setIsChained(false);
    setGDSId((byte)0xD0);
    setHasSameRequestCorrelation(false);
    setType(3);
  }

  /**
   *Returns a new instance of a DDMObjectDataStream.
   *used by the DDMDataStream.construct() method.
   *@return a new instance of a DDMObjectDataStream.
  **/
  public Object getNewDataStream()
  {
    return new DDMObjectDataStream();
  }

  /**
   *Returns a unique identifier for this type of object.
   *@return a unique identifier for this type of object.
  **/
  public int hashCode()
  {
    return 3;  // Object
  }

  /**
   *Returns the S38BUF object data stream.
   *@param records the data in the buffer.
   *@param recordIncrement the bytes between the start of each record.
   *@param maxNumberOfFieldsPerFormatInFile indicates the largest of the  @A1A
   * record.getNumberOfFields() values for this file.  A physical file 
   * has only one record format so maxNumberOfFieldsPerFormatInFile would
   * be the same as getNumberOfFields().  However, a multi-format logical
   * file may have multiple record formats.  maxNumberOfFieldsPerFormatInFile
   * should be set to the largest getNumberOfFields() value for the various
   * record formats in a multi-format logical file.  In some cases, 
   * getObjectS38BUF() may be called before the recordFormat's are known. In
   * that case, maxNumberOfFieldsPerFormatInFile may be set to -1 to obtain
   * equivalent behavior prior to the addition of this new parameter.
   *@return S38BUF data stream
   * Term = S38BUF
   * Size = 6 --> Header (0-5)
   *        2 --> LL S38BUF term and parm (6,7)
   *        2 --> CP S38BUF code point (8,9)
   *@exception CharConversionException If an error occurs during conversion.
   *@exception UnsupportedEncodingException If an error occurs during conversion.
   *
   * #SSPDDM1 - Changed method to accept isSSPFile parameter and skip null field
   *            map if it is set to true
  **/
  static DDMObjectDataStream[] getObjectS38BUF(Record[] records,
                                               DDMS38OpenFeedback openFeedback,
                                               boolean isSSPFile,         // #SSPDDM1
                                               int maxNumberOfFieldsPerFormatInFile) //@A1C
  throws CharConversionException,
  UnsupportedEncodingException
  {
    // Get the record format, the record increment, and the blocking
    // factor.
    RecordFormat format = records[0].getRecordFormat();
    int recordIncrement = openFeedback.getRecordIncrement();
    int blockingFactor = openFeedback.getMaxNumberOfRecordsTransferred();

    // Instantiate an array of data streams.
    DDMObjectDataStream[] dataStreams =
    new DDMObjectDataStream[records.length / blockingFactor +
                            (records.length % blockingFactor == 0 ? 0 : 1)];

    // Create a data stream every 'blockingFactor' records.
    for (int dataStreamIndex = 0, recordIndex = 0;
        dataStreamIndex < dataStreams.length; dataStreamIndex++)
    {
      // We can only copy 'blockingFactor' records per data stream.
      // Calculate the end index of the last record to be copied to the
      // current data stream.
      int endIndex = recordIndex + blockingFactor;
      if (endIndex > records.length)
      {
        endIndex = records.length;
      }

      // Compute the total data stream length, the record data offset, and
      // the S38BUF LL.  If the record data length is greater than the
      // maximum data stream length, we must account for extra bookkeeping
      // data.
      int fourByteLength = 0;
      int recDataLength =  recordIncrement * (endIndex - recordIndex);
      int recordOffset = 10;
      int dsLength = recDataLength + 10;
      int s38BUFLL = recDataLength + 4;
      if (recDataLength > MAX_DATA_STREAM_LEN - 10)
      {
        if (recDataLength > MAX_DATA_STREAM_LEN - 4)
        {
          // We need the four byte length indicator.
          fourByteLength = recDataLength;
          recDataLength -= 4;
          dsLength += 4; // 4 byte length indicator
          s38BUFLL = 0x8008;
          recordOffset = 14;
        }

        // Subtract from record data length the amount before the packets.
        recDataLength -= (MAX_DATA_STREAM_LEN - 10);
      }

      // Instantiate the data stream.
      dataStreams[dataStreamIndex] = new DDMObjectDataStream(dsLength);

      // Set the S38BUF LL, and CP.
      dataStreams[dataStreamIndex].set16bit(s38BUFLL, 6);
      dataStreams[dataStreamIndex].set16bit(DDMTerm.S38BUF, 8);

      // Set the four byte length indicator if needed.
      if (fourByteLength != 0)
      {
        dataStreams[dataStreamIndex].set32bit(fourByteLength, 10);
      }

      // For each record, write the record data and the null field byte
      // map after the record data.
      for (; recordIndex < endIndex; recordIndex++,
          recordOffset += recordIncrement)
      {
        // Copy the record data to the data stream.
        byte[] recordData = records[recordIndex].getContents();
        System.arraycopy(recordData, 0, dataStreams[dataStreamIndex].data_,
                         recordOffset, recordData.length);

        // Write the null field byte map array after the record data.  It
        // immediately preceeds the next record. 0xf1 = null, 0xf0 != null
        // There may be a gap between the end of the record data and the
        // start of the null field byte map.
        int numFields = records[recordIndex].getNumberOfFields();
        // Skip writing the null field map for SSP files					    	// #SSPDDM1
        if (!isSSPFile)                                 // #SSPDDM1
        {
          // The NULL byte field map is left justified and has as many bytes as there @A1A 
          // are fields in the file.  In the case of a multi-format logical file, some
          // formats may have more/less getNumberOfFields() than others.  In that case,
          // the NULL byte field map has entries for the format with the largest 
          // getNumberOfFields().
          // The parameter maxNumberOfFieldsPerFormatInFile is the largest of the 
          // record format's getNumberOfFields() values.
          // Therefore, if we are writing a    
          // record to a format that has 2 fields we need to set two entries into the NULL
          // byte field map.  However, since the file could be a multi-format logical file
          // there may be other recordFormats in the file that have more fields than the one
          // currently being written to.  So, for example if maxNumberOfFieldsPerFormatInFile
          // is 3 then we need to start writing the 0xf1 or 0xf0 values at an offset as if 
          // there were 3 fields, but only need to set 2 values (if there are 2 fields in 
          // this particular record's recordFormat)
          // Previously, the code was setting the 0xf0 and 0xf1 values in the NULL byte 
          // field map right-justified because we did not know/care about  
          // maxNumberOfFieldsPerFormatInFile. This would work for physical files, single 
          // format logical files, and multi-format logical files only if each of the formats 
          // in the multi-format logical file had the exact same number of fields.
          // #SSPDDM1
          // If maxNumberOfFieldsPerFormatInFile
          if (maxNumberOfFieldsPerFormatInFile == -1)                                      //@A1A
          {
            maxNumberOfFieldsPerFormatInFile = numFields;                                  //@A1A
          }
          for (int f = 0, fieldOffset = recordOffset +                                     //@A1C
               (recordIncrement - maxNumberOfFieldsPerFormatInFile); f < numFields; fieldOffset++, f++)
          {
            dataStreams[dataStreamIndex].data_[fieldOffset] =
            (records[recordIndex].isNullField(f) ? (byte) 0xf1 : (byte) 0xf0);
          }
        }                                        // #SSPDDM1
      }
    }

    return dataStreams;
  }

/* COMMENT OUT UNUSED METHOD                                                                @A1D
  // #SSPDDM1 - method changed to all overloaded method with isSSP parameter
  static DDMObjectDataStream[] getObjectS38BUF(Record[] records,
                                               DDMS38OpenFeedback openFeedback)
  throws CharConversionException,
  UnsupportedEncodingException
  {
    return getObjectS38BUF(records, openFeedback, false);  // #SSPDDM1 - Call with isSSP default to false
  }    
*/
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy