src.com.ibm.as400.access.DDMObjectDataStream Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jt400 Show documentation
Show all versions of jt400 Show documentation
The Open Source version of the IBM Toolbox for Java
///////////////////////////////////////////////////////////////////////////////
//
// 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.
**/
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