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

uk.ac.starlink.fits.ColumnReader Maven / Gradle / Ivy

There is a newer version: 4.3
Show newest version
package uk.ac.starlink.fits;

import java.io.IOException;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.nio.charset.Charset;
import java.util.logging.Logger;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import uk.ac.starlink.table.TableFormatException;
import uk.ac.starlink.table.Tables;

/**
 * Abstract class defining what needs to be done to read from a
 * stream and return an object representing a value in a given
 * table column for a BINTABLE FITS table.
 *
 * @author  Mark Taylor
 * @since   8 Jul 2008
 */
abstract class ColumnReader {

    private final Class clazz_;
    private final int[] shape_;
    private final int length_;
    private final ColFlags flags_;
    private static final Logger logger_ =
        Logger.getLogger( "uk.ac.starlink.fits" );
    private static final Charset US_ASCII = Charset.forName( "US-ASCII" );

    /** Policy for dealing with offset long values. */
    static final boolean OFFSET_LONG_TO_STRING = true;

    /**
     * Constructs a new reader with a given content class, shape and length.
     *
     * @param clazz  the class which readValue will return
     * @param shape  the shape to be imposed on the array returned by
     *               readValue, or null if that
     *               returns a scalar
     * @param length  the number of bytes readValue reads from
     *                the stream
     * @param flags  additional information
     */
    ColumnReader( Class clazz, int[] shape, int length, ColFlags flags ) {
        clazz_ = clazz;
        shape_ = shape;
        length_ = length;
        flags_ = flags;
    }

    /**
     * Constructs a scalar reader with a given content class and length.
     *
     * @param clazz  the class which readValue will return
     *               (shouldn't be an array)
     * @param length  the number of bytes readValue reads from
     *                the stream
     * @param flags  additional information
     */
    ColumnReader( Class clazz, int length, ColFlags flags ) {
        this( clazz, null, length, flags );
    }

    /**
     * Reads bytes from a stream to return an object.
     *
     * @param  stream containing bytes to turn into an object
     * @return  an object read from the stream of type
     *          getContentClass (or null)
     */
    abstract Object readValue( BasicInput stream ) throws IOException;

    /**
     * Returns the class which objects returned by readValue
     * will belong to.
     *
     * @return  value class
     */
    Class getContentClass() {
        return clazz_;
    }

    /**
     * Returns the shape imposed on array elements in the sense of
     * ValueInfo.getShape();
     *
     * @param  shape, or null for scalars
     */
    int[] getShape() {
        return shape_;
    }

    /**
     * Returns string size in the sense of ValueInfo.getElementSize().
     *
     * @return element size, or -1 if not applicable
     */
    int getElementSize() {
        return -1;
    }

    /**
     * Returns the number of bytes each call to readValue reads
     * from the stream.
     *
     * @return  byte count
     */
    int getLength() {
        return length_;
    }

    /**
     * Indicates whether this reader reads (scalar or array) values that
     * are in the unsigned byte range, 0..255.  This usually returns false,
     * but may be true in some cases for short integer values.
     *
     * @return   true iff short int values have been deserialized from
     *           unsigned byte values
     */
    boolean isUnsignedByte() {
        return flags_.isUnsignedByte_;
    }

    /**
     * If this reader represents a long value (or array of long values)
     * with a non-zero integer offset (TZEROn integer and non-zero,
     * TSCALn zero) that is represented by a String, this gives the
     * integer offset which has been applied.
     * If that's not the case, the return value is null.
     *
     * @return   offset for stringified long integers, or null
     */
    BigInteger getLongOffset() {
        return flags_.longOffset_;
    }

    /**
     * Constructs a ColumnReader object suitable for reading a given column
     * of a table.
     *
     * @param   tform  TFORM string from FITS header for column
     * @param   scale  factor to scale numerical values by
     * @param   zeroNum   offset to add to numerical values
     * @param   hasBlank  true if a magic value is regarded as blank
     * @param   blank   value represnting magic null value
     *                  (only used if hasBlank is true)
     * @param   tdims  dimensions specified by TDIMS card, or null if none
     *                 given
     * @param   ttype  column name
     * @param   heapStart   offset of heap into HDU data part, or -1 if no
     *                      heap or no random access is available
     * @return  a reader suitable for reading this type of column
     */
    public static ColumnReader createColumnReader( String tform, double scale,
                                                   Number zeroNum,
                                                   boolean hasBlank,
                                                   long blank, int[] tdims,
                                                   String ttype,
                                                   final long heapStart )
            throws TableFormatException {

        /* Parse TFORM to find repeat count and data type. */
        Matcher fmatch = Pattern.compile( "([0-9]*)([LXBIJKAEDCMPQ])(.*)" )
                                .matcher( tform );
        if ( ! fmatch.lookingAt() ) {
            throw new TableFormatException( "Error parsing TFORM value "
                                          + tform );
        }
        String scount = fmatch.group( 1 );
        final int count = scount.length() == 0
                  ? 1
                  : Integer.parseInt( scount );
        char type = fmatch.group( 2 ).charAt( 0 );
        String matchA = fmatch.group( 3 ).trim();

        /* Work out a sensible dims array which may or may not be the same
         * as the TDIMS value. */
        int[] dims;
        if ( type == 'P' || type == 'Q' ) {

            /* In most cases, an array of unknown size and shape 
             * is appropriate. */
            if ( tdims == null ) {
                dims = new int[] { -1 };
            }

            /* If a TDIMn shape has been supplied, just pass that on.
             * I don't really see how it's going to make sense, but software
             * downstream might be able to do something with it (and hopefully
             * will not get too confused by the fact that the size of elements
             * does not match their declared size). */
            else {
                dims = tdims;
            }
        }
        else {
            if ( count == 1 ) {
                dims = null;
            }
            else if ( tdims == null ) {
                dims = new int[] { count };
            }
            else {
                int nel = 1;
                for ( int i = 0; i < tdims.length; i++ ) {
                    nel *= tdims[ i ];
                }
                dims = nel == count ? tdims : new int[] { count };
            }
        }

        /* Variable sized array case ('P' or 'Q' descriptors). */
        if ( type == 'P' ) {

            /* If will be doing random access and know the start of the heap
             * we can cope with this. */
            if ( heapStart >= 0 ) {
                char vtype = matchA.charAt( 0 );
                final ArrayReader aReader =
                    createArrayReader( vtype, scale, zeroNum,
                                       hasBlank, blank, dims );
                return new ColumnReader( aReader.getContentClass(),
                                         aReader.getShape(), 8,
                                         aReader.flags_ ) {
                    Object readValue( BasicInput stream ) throws IOException {
                        int nel = stream.readInt();
                        int heapOffset = stream.readInt();
                        if ( nel > 0 ) {
                            long point = stream.getOffset();
                            stream.seek( heapStart + heapOffset );
                            Object array = aReader.readArray( stream, nel );
                            stream.seek( point );
                            return array;
                        }
                        else {
                            return aReader.readArray( stream, 0 );
                        }
                    }
                    int getElementSize() {
                        return aReader.getElementSize();
                    }
                };
            }

            /* Otherwise we can't do variable sized arrays. */
            else {
                logger_.warning( "Column " + ttype + "(TFORM=" + tform + ") - "
                               + "variable length arrays not supported "
                               + "in sequential mode" );
                final String value = "?";
                return new ColumnReader( String.class, 8, ColFlags.NONE ) {
                    Object readValue( BasicInput stream ) throws IOException {
                        int nel = stream.readInt();
                        int offset = stream.readInt();
                        return nel > 0 ? value : "";
                    }
                    int getElementSize() {
                        return value.length();
                    }
                };
            }
        }

        else if ( type == 'Q' ) {
            if ( heapStart >= 0 ) {
                char vtype = matchA.charAt( 0 );
                final ArrayReader aReader =
                    createArrayReader( vtype, scale, zeroNum,
                                       hasBlank, blank, dims );
                return new ColumnReader( aReader.getContentClass(),
                                         aReader.getShape(), 16,
                                         aReader.flags_ ) {
                    Object readValue( BasicInput stream ) throws IOException {
                        long lnel = stream.readLong();
                        long heapOffset = stream.readLong();
                        int nel = Tables.checkedLongToInt( lnel );
                        if ( nel > 0 ) {
                            long point = stream.getOffset();
                            stream.seek( heapStart + heapOffset );
                            Object array = aReader.readArray( stream, nel );
                            stream.seek( point );
                            return array;
                        }
                        else {
                            return aReader.readArray( stream, 0 );
                        }
                    }
                    int getElementSize() {
                        return aReader.getElementSize();
                    }
                };
            }
            else {
                logger_.warning( "Column " + ttype + "(TFORM=" + tform + ") - "
                               + "variable length arrays not supported "
                               + "in sequential mode" );
                final String value = "?";
                return new ColumnReader( String.class, 16, ColFlags.NONE ) {
                    Object readValue( BasicInput stream ) throws IOException {
                        long nel = stream.readLong();
                        long offset = stream.readLong();
                        return nel > 0 ? value : "";
                    }
                    int getElementSize() {
                        return value.length();
                    }
                };
            }
        }

        /* Scalar value case. */
        else if ( count == 1 ) {
            return createScalarColumnReader( type, scale, zeroNum,
                                             hasBlank, blank );
        }

        /* Fixed size array case. */
        else {

            /* Special case: fixed length Substring Array Convention
             * (rAw means array of w-character strings).  This seems to be
             * an alternative to doing the same thing using TDIMnn. */
            if ( type == 'A' && matchA.matches( "[0-9]+" ) &
                 dims.length == 1 ) {
                int sleng = Integer.parseInt( matchA );
                if ( dims[ 0 ] % sleng == 0 ) {
                    dims = new int[] { sleng, dims[ 0 ] / sleng };
                }
            }

            /* Construct and return the array column reader. */
            final ArrayReader aReader =
                createArrayReader( type, scale, zeroNum,
                                   hasBlank, blank, dims );
            boolean isComplex = type == 'C' || type == 'M';
            final int primCount = ( isComplex ? 2 : 1 ) * count;
            return new ColumnReader( aReader.getContentClass(),
                                     aReader.getShape(),
                                     aReader.getByteCount( primCount ),
                                     aReader.flags_ ) {
                Object readValue( BasicInput stream ) throws IOException {
                    return aReader.readArray( stream, primCount );
                }
                int getElementSize() {
                    return aReader.getElementSize();
                }
            };
        }
    }

    /** 
     * Returns a new column reader for a scalar column.
     *
     * @param   type  TFORM data type character
     * @param   scale  factor to scale numerical values by
     * @param   zeroNum   offset to add to numerical values
     * @param   hasBlank  true if a magic value is regarded as blank
     * @param   blank   value represnting magic null value
     *                  (only used if hasBlank is true)
     * @return  new column reader
     */
    @SuppressWarnings("cast")
    private static ColumnReader createScalarColumnReader(
            char type, final double scale, final Number zeroNum,
            final boolean hasBlank, final long blank ) {
        final long lZero = zeroNum.longValue();
        final double dZero = zeroNum.doubleValue();
        final boolean isScaled = ( scale != 1.0 || dZero != 0.0 );
        final boolean isOffset = ( scale == 1.0 && dZero != 0.0 );
        final boolean intOffset = isOffset && isInteger( zeroNum );
        final ColumnReader reader;
        switch ( type ) {

            /* Logical. */
            case 'L':
                reader = new ColumnReader( Boolean.class, 1, ColFlags.NONE ) {
                    Object readValue( BasicInput stream )
                            throws IOException {    
                        switch ( stream.readByte() ) {
                            case (byte) 'T':        
                                return Boolean.TRUE; 
                            case (byte) 'F':
                                return Boolean.FALSE;
                            default:
                                return null;
                        }
                    } 
                };
                return reader;

            /* Bit vector (1-element). */
            case 'X':
                reader = new ColumnReader( Boolean.class, 1, ColFlags.NONE ) {
                    Object readValue( BasicInput stream )
                            throws IOException {
                        int b = stream.readByte();
                        return Boolean.valueOf( ( b & 0x80 ) != 0 );
                    }
                };
                return reader;

            /* Unsigned byte - this is a bit fiddly, since a java byte is
             * signed and a FITS byte is unsigned.  We cope with this in
             * general by reading the byte as a short.  However, in the
             * special case that the scaling is exactly right to store
             * a signed byte as an unsigned one (scale=1, zero=-128) we
             * can transform a FITS byte directly into a java one. */
            case 'B':
                final short mask = (short) 0x00ff;
                boolean shortable = intOffset && dZero >= Short.MIN_VALUE
                                              && dZero < Short.MAX_VALUE - 256;
                if ( dZero == -128.0 && scale == 1.0 ) {
                    reader = new ColumnReader( Byte.class, 1, ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            byte val = stream.readByte();
                            return ( hasBlank && val == (byte) blank )
                                 ? null
                                 : Byte.valueOf( (byte) ( val ^ (byte) 0x80 ) );
                        }
                    };
                }
                else if ( shortable ) {
                    final short sZero = (short) lZero;
                    reader = new ColumnReader( Short.class, 1, ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            byte val = stream.readByte();
                            return ( hasBlank && val == (byte) blank )
                                 ? null
                                 : Short.valueOf( (short)
                                                  ( ( val & mask ) + sZero ) );
                        }
                    };
                }
                else if ( isScaled ) {
                    reader = new ColumnReader( Float.class, 1, ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            byte val = stream.readByte();
                            return ( hasBlank && val == (byte) blank )
                                        ? null
                                        : new Float( ( val & mask )
                                                     * scale + dZero );
                        }
                    };
                }
                else {
                    reader = new ColumnReader( Short.class, 1,
                                               ColFlags.UNSIGNED_BYTE ) {
                        Object readValue( BasicInput stream )  
                                throws IOException {
                            byte val = stream.readByte();
                            return ( hasBlank && val == (byte) blank )
                                   ? null
                                   : Short.valueOf( (short) ( val & mask ) );
                        }
                    };
                }
                return reader;

            /* Short. */
            case 'I':
                boolean intable = intOffset
                               && dZero > Integer.MIN_VALUE - Short.MIN_VALUE
                               && dZero < Integer.MAX_VALUE - Short.MAX_VALUE;
                if ( intable ) {
                    final int iZero = (int) lZero;
                    reader = new ColumnReader( Integer.class, 2,
                                               ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            short val = stream.readShort();
                            return ( hasBlank && val == (short) blank )
                                 ? null
                                 : Integer.valueOf( (int) ( val + iZero ) );
                        }
                    };
                }
                else if ( isScaled ) {
                    reader = new ColumnReader( Float.class, 2, ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            short val = stream.readShort();
                            return ( hasBlank && val == (short) blank )
                                        ? null
                                        : new Float( (float)
                                                   ( val * scale + dZero ) );
                        }
                    };
                }
                else {
                    reader = new ColumnReader( Short.class, 2, ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            short val = stream.readShort();
                            return ( hasBlank && val == (short) blank )
                                        ? null
                                        : Short.valueOf( val );
                        }
                    };
                }
                return reader;

            /* Integer. */
            case 'J':
                boolean longable = intOffset
                                && dZero > Long.MIN_VALUE - Integer.MIN_VALUE
                                && dZero < Long.MAX_VALUE - Integer.MAX_VALUE;
                if ( longable ) {
                    reader = new ColumnReader( Long.class, 4, ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            long val = stream.readInt();
                            return ( hasBlank && val == (int) blank )
                                 ? null
                                 : Long.valueOf( (long) ( val + lZero ) );
                        }
                    };
                }
                else if ( isScaled ) {
                    reader = new ColumnReader( Double.class, 4,
                                               ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            int val = stream.readInt();
                            return ( hasBlank && val == (int) blank )
                                        ? null
                                        : new Double( val * scale + dZero );
                        }
                    };
                }
                else {
                    reader = new ColumnReader( Integer.class, 4,
                                               ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            int val = stream.readInt();
                            return ( hasBlank && val == (int) blank )
                                        ? null
                                        : Integer.valueOf( val );
                        }
                    };
                }
                return reader;

            /* Long. */
            case 'K':
                boolean isUnsignedLong =
                    zeroNum.equals( BintableStarTable.TWO63 ) && scale == 1.0;
                if ( intOffset || isUnsignedLong ) {
                    if ( OFFSET_LONG_TO_STRING ) {
                        final BigInteger bigZero =
                              zeroNum instanceof BigInteger
                            ? (BigInteger) zeroNum
                            : BigInteger.valueOf( zeroNum.longValue() );
                        ColFlags flags = ColFlags.createLongOffset( bigZero );
                        reader = new ColumnReader( String.class, 8, flags ) {
                            Object readValue( BasicInput stream )
                                    throws IOException {
                                long val = stream.readLong();
                                if ( hasBlank && val == (long) blank ) {
                                    return null;
                                }
                                else {
                                    return BigInteger.valueOf( val )
                                                     .add( bigZero )
                                                     .toString();
                                }
                            }
                        };
                    }
                    else if ( isUnsignedLong ) {
                        long lMin = Long.MIN_VALUE;
                        long lMax = -1L;
                        final LongRanger ranger =
                            new LongRanger( lMin, lMax, zeroNum, "null" );
                        reader = new ColumnReader( Long.class, 8,
                                                   ColFlags.NONE ) {
                            Object readValue( BasicInput stream )
                                    throws IOException {
                                long val = stream.readLong();
                                if ( hasBlank && val == (long) blank ) {
                                    return null;
                                }
                                else {
                                    return ranger.inRange( val )
                                         ? new Long( val + Long.MAX_VALUE + 1L )
                                         : null;
                                }
                            }
                        };
                    }
                    else {
                        long lMin = lZero < 0 ? Long.MIN_VALUE - lZero
                                              : Long.MIN_VALUE;
                        long lMax = lZero > 0 ? Long.MAX_VALUE - lZero
                                              : Long.MAX_VALUE;
                        final LongRanger ranger =
                           new LongRanger( lMin, lMax, zeroNum, "null" );
                        reader = new ColumnReader( Long.class, 8,
                                                   ColFlags.NONE ) {
                            Object readValue( BasicInput stream )
                                    throws IOException {
                                long val = stream.readLong();
                                if ( hasBlank && val == (long) blank ) {
                                    return null;
                                }
                                else {
                                    return ranger.inRange( val )
                                         ? new Long( val + lZero )
                                         : null;
                                }
                            }
                        };
                    }
                }
                else if ( isScaled ) {
                    reader = new ColumnReader( Double.class, 8,
                                               ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            long val = stream.readLong();
                            return ( hasBlank && val == (long) blank )
                                        ? null
                                        : new Double( val * scale + dZero );
                        }
                    };
                }
                else {
                    reader = new ColumnReader( Long.class, 8, ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            long val = stream.readLong();
                            return ( hasBlank && val == (long) blank )
                                        ? null
                                        : Long.valueOf( val );
                        }
                    };
                }
                return reader;

            /* Character. */
            case 'A':
                reader = new ColumnReader( Character.class, 1, ColFlags.NONE ) {
                    Object readValue( BasicInput stream )
                            throws IOException {
                        char c = (char) ( stream.readByte() & 0xff );
                        return c == '\0'
                             ? null
                             : Character.valueOf( c );
                    }
                };
                return reader;

            /* Floating point. */
            case 'E':
                if ( isScaled ) {
                    reader = new ColumnReader( Float.class, 4, ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            float val = stream.readFloat();
                            return new Float( val * scale + dZero );
                        }
                    };
                }
                else {
                    reader = new ColumnReader( Float.class, 4, ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            float val = stream.readFloat();
                            return new Float( val );
                        }
                    };
                }
                return reader;    

            /* Double precision. */
            case 'D':
                if ( isScaled ) {
                    reader = new ColumnReader( Double.class, 8,
                                               ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            double val = stream.readDouble();
                            return new Double( val * scale + dZero );
                        }
                    };
                }
                else {
                    reader = new ColumnReader( Double.class, 8,
                                               ColFlags.NONE ) {
                        Object readValue( BasicInput stream )
                                throws IOException {
                            double val = stream.readDouble();
                            return new Double( val );
                        }
                    };
                }
                return reader;

            /* Complex. */
            case 'C':
            case 'M':
                final int[] complexDims = new int[] { 2 };
                final ArrayReader complexReader =
                    type == 'C'
                        ? createFloatsArrayReader( complexDims, scale, dZero )
                        : createDoublesArrayReader( complexDims, scale, dZero );
                return new ColumnReader( complexReader.getContentClass(),
                                         complexDims,
                                         complexReader.getByteCount( 2 ),
                                         ColFlags.NONE ) {
                    Object readValue( BasicInput stream ) throws IOException {
                        return complexReader.readArray( stream, 2 );
                    }
                    int getElementSize() {
                        return complexReader.getElementSize();
                    }
                };

            default:
                throw new AssertionError( "Unknown TFORM type " + type );
        }
    }

    /**
     * Returns a new ArrayReader object for a given data type.
     *
     * @param   type  TFORM data type character
     * @param   scale  factor to scale numerical values by
     * @param   zeroNum   offset to add to numerical values
     * @param   hasBlank  true if a magic value is regarded as blank
     * @param   blank   value represnting magic null value
     *                  (only used if hasBlank is true)
     * @param   dims    apparent dimensions of FITS array 
     *                  (not necessarily same as dimensions of read data)
     * @return  new array reader
     */
    @SuppressWarnings("cast")
    private static ArrayReader createArrayReader( char type, final double scale,
                                                  final Number zeroNum,
                                                  final boolean hasBlank,
                                                  final long blank,
                                                  final int[] dims ) {
        final long lZero = zeroNum.longValue();
        final double dZero = zeroNum.doubleValue();
        final boolean isScaled = ( scale != 1.0 || dZero != 0.0 );
        final boolean isOffset = ( scale == 1.0 && dZero != 0.0 );
        final boolean intOffset = isOffset && isInteger( zeroNum );
        final ArrayReader reader;
        switch ( type ) {

            /* Logical. */
            case 'L':
                reader = new ArrayReader( boolean[].class, dims, 1,
                                          ColFlags.NONE ) {
                    Object readArray( BasicInput stream, int count )
                            throws IOException {
                        boolean[] value = new boolean[ count ];
                        for ( int i = 0; i < count; i++ ) {
                            value[ i ] = stream.readByte() == (byte) 'T';
                        }
                        return value;
                    }
                };
                return reader;

            /* Bits. */
            case 'X':
                reader = new ArrayReader( boolean[].class, dims, -1,
                                          ColFlags.NONE ) {
                    Object readArray( BasicInput stream, int count )
                            throws IOException {
                        boolean[] value = new boolean[ count ];
                        int ibit = 0;
                        int b = 0;
                        for ( int i = 0; i < count; i++ ) {
                            if ( ibit == 0 ) {
                                ibit = 8;
                                b = stream.readByte();
                            }
                            value[ i ] = ( b & 0x80 ) != 0;
                            b = b << 1;
                            ibit--;
                        }
                        return value;
                    }
                    int getByteCount( int nel ) {
                        return ( nel + 7 ) / 8;
                    }
                };
                return reader;

            /* Unsigned byte - this is a bit fiddly, since a java byte is
             * signed and a FITS byte is unsigned.  We cope with this in
             * general by reading the byte as a short.  However, in the
             * special case that the scaling is exactly right to store
             * a signed byte as an unsigned one (scale=1, zero=-128) we
             * can transform a FITS byte directly into a java one. */
            case 'B':
                final short mask = (short) 0x00ff;
                boolean shortable = intOffset && dZero >= Short.MIN_VALUE
                                              && dZero < Short.MAX_VALUE - 256;
                if ( dZero == -128.0 && scale == 1.0 ) {
                    reader = new ArrayReader( byte[].class, dims, 1,
                                              ColFlags.NONE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            byte[] value = new byte[ count ];
                            for ( int i = 0; i < count; i++ ) {
                                byte val = stream.readByte();
                                value[ i ] = (byte) ( val ^ (byte) 0x80 );
                            }
                            return value;
                        }
                    };
                }
                else if ( shortable ) {
                    final short sZero = (short) lZero;
                    reader = new ArrayReader( short[].class, dims, 1,
                                              ColFlags.NONE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            short[] value = new short[ count ];
                            for ( int i = 0; i < count; i++ ) {
                                byte val = stream.readByte();
                                value[ i ] = (short) ( ( val & mask ) + sZero );
                            }
                            return value;
                        }
                    };
                }
                else if ( isScaled ) {
                    reader = new ArrayReader( float[].class, dims, 1,
                                              ColFlags.NONE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            float[] value = new float[ count ];
                            for ( int i = 0; i < count; i++ ) {
                                byte val = stream.readByte();
                                value[ i ] =
                                    ( hasBlank && val == (byte) blank )
                                         ? Float.NaN
                                         : (float) ( ( val & mask )
                                                     * scale + dZero );
                            }
                            return value;
                        }
                    };
                }
                else {
                    reader = new ArrayReader( short[].class, dims, 1,
                                              ColFlags.UNSIGNED_BYTE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            short[] value = new short[ count ];
                            for ( int i = 0; i < count; i++ ) {
                                byte val = stream.readByte();
                                value[ i ] = (short) ( val & mask );
                            }
                            return value;
                        }
                    };
                }
                return reader;

            /* Short. */
            case 'I':
                boolean intable = intOffset
                               && dZero > Integer.MIN_VALUE - Short.MIN_VALUE
                               && dZero < Integer.MAX_VALUE - Short.MAX_VALUE;
                if ( intable ) {
                    final int iZero = (int) lZero;
                    reader = new ArrayReader( int[].class, dims, 2,
                                              ColFlags.NONE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            int[] value = new int[ count ];
                            for ( int i = 0; i < count; i++ ) {
                                short val = stream.readShort();
                                value[ i ] = (int) ( val + iZero );
                            }
                            return value;
                        }
                    };
                }
                else if ( isScaled ) {
                    reader = new ArrayReader( float[].class, dims, 2,
                                              ColFlags.NONE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            float[] value = new float[ count ];
                            for ( int i = 0; i < count; i++ ) {
                                short val = stream.readShort();
                                value[ i ] =
                                    ( hasBlank && val == (short) blank )
                                         ? Float.NaN
                                         : (float) ( val * scale + dZero );
                            }
                            return value;
                        }
                    };
                }
                else {
                    reader = new ArrayReader( short[].class, dims, 2,
                                              ColFlags.NONE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            short[] value = new short[ count ];
                            for ( int i = 0; i < count; i++ ) {
                                short val = stream.readShort();
                                value[ i ] = val;
                            }
                            return value;
                        }
                    };
                }
                return reader;

            /* Integer. */
            case 'J':
                boolean longable = intOffset
                                && dZero > Long.MIN_VALUE - Integer.MIN_VALUE
                                && dZero < Long.MAX_VALUE - Integer.MAX_VALUE;
                if ( longable ) {
                    reader = new ArrayReader( long[].class, dims, 4,
                                              ColFlags.NONE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            long[] value = new long[ count ];
                            for ( int i = 0; i < count; i++ ) {
                                int val = stream.readInt();
                                value[ i ] = (long) ( val + lZero );
                            }
                            return value;
                        }
                    };
                }
                else if ( isScaled ) {
                    reader = new ArrayReader( double[].class, dims, 4,
                                              ColFlags.NONE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            double[] value = new double[ count ];
                            for ( int i = 0; i < count; i++ ) {
                                int val = stream.readInt();
                                value[ i ] =
                                    ( hasBlank && val == (int) blank )
                                         ? Double.NaN
                                         : val * scale + dZero;
                            }
                            return value;
                        }
                    };
                }
                else {
                    reader = new ArrayReader( int[].class, dims, 4,
                                              ColFlags.NONE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            int[] value = new int[ count ];
                            for ( int i = 0; i < count; i++ ) {
                                int val = stream.readInt();
                                // can't do anything with a blank
                                value[ i ] = val;
                            }
                            return value;
                        }
                    };
                }
                return reader;

            /* Long. */
            case 'K':
                boolean isUnsignedLong =
                    zeroNum.equals( BintableStarTable.TWO63 ) && scale == 1.0; 
                if ( intOffset || isUnsignedLong ) {
                    if ( OFFSET_LONG_TO_STRING ) {
                        final BigInteger bigZero =
                              zeroNum instanceof BigInteger
                            ? (BigInteger) zeroNum
                            : BigInteger.valueOf( zeroNum.longValue() );
                        ColFlags flags = ColFlags.createLongOffset( bigZero );
                        reader = new ArrayReader( String[].class, dims,
                                                  8, flags ) {
                            Object readArray( BasicInput stream, int count )
                                    throws IOException {
                                String[] value = new String[ count ];
                                for ( int i = 0; i < count; i++ ) {
                                    long val = stream.readLong();
                                    if ( ! hasBlank || val != (long) blank ) {
                                        value[ i ] = BigInteger.valueOf( val )
                                                               .add( bigZero )
                                                               .toString();
                                    }
                                }
                                return value;
                            }
                        };
                    }
                    else if ( isUnsignedLong ) {
                        long lMin = Long.MIN_VALUE;
                        long lMax = -1L;
                        final LongRanger ranger =
                            new LongRanger( lMin, lMax, zeroNum,
                                            Long.toString( Long.MIN_VALUE ) );
                        reader = new ArrayReader( long[].class, dims,
                                                  8, ColFlags.NONE ) {
                            Object readArray( BasicInput stream, int count )
                                    throws IOException {
                                long[] value = new long[ count ];
                                for ( int i = 0; i < count; i++ ) {
                                    long val = stream.readLong();
                                    value[ i ] = ranger.inRange( val )
                                               ? val + Long.MAX_VALUE + 1L
                                               : Long.MIN_VALUE;
                                }
                                return value;
                            }
                        };
                    }
                    else {
                        long lMax = lZero > 0 ? Long.MAX_VALUE - lZero
                                              : Long.MAX_VALUE;
                        long lMin = lZero < 0 ? Long.MIN_VALUE - lZero
                                              : Long.MIN_VALUE;
                        final LongRanger ranger =
                            new LongRanger( lMin, lMax, zeroNum,
                                            Long.toString( Long.MIN_VALUE ) );
                        reader = new ArrayReader( long[].class, dims,
                                                  8, ColFlags.NONE ) {
                            Object readArray( BasicInput stream, int count )
                                    throws IOException {
                                long[] value = new long[ count ];
                                for ( int i = 0; i < count; i++ ) {
                                    long val = stream.readLong();
                                    value[ i ] = ranger.inRange( val )
                                               ? val + lZero
                                               : Long.MIN_VALUE;
                                }
                                return value;
                            }
                        };
                    }
                }
                else if ( isScaled ) {
                    reader = new ArrayReader( double[].class, dims, 8,
                                              ColFlags.NONE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            double[] value = new double[ count ];
                            for ( int i = 0; i < count; i++ ) {
                                long val = stream.readLong();
                                value[ i ] =
                                    ( hasBlank && val == (long) blank )
                                         ? Double.NaN
                                         : val * scale + dZero;
                            }
                            return value;
                        }
                    };
                }
                else {
                    reader = new ArrayReader( long[].class, dims, 8,
                                              ColFlags.NONE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            long[] value = new long[ count ];
                            for ( int i = 0; i < count; i++ ) {
                                long val = stream.readLong();
                                // can't do anything with a blank
                                value[ i ] = val;
                            }
                            return value;
                        }
                    };
                }
                return reader;

            /* Characters. */
            case 'A':
                if ( dims.length == 1 ) {
                    reader = new ArrayReader( String.class, null, 1,
                                              ColFlags.NONE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            return readString( stream, count );
                        }
                        int getElementSize() {
                            return dims[ 0 ];
                        }
                    };
                }
                else {
                    int nel = 1;
                    for ( int i = 1; i < dims.length; i++ ) {
                        nel *= dims[ i ];
                    }
                    final int stringLength = dims[ 0 ];
                    final int nString = nel;
                    int[] shape = new int[ dims.length - 1 ];
                    System.arraycopy( dims, 1, shape, 0, dims.length - 1 );
                    reader = new ArrayReader( String[].class, shape, 1,
                                              ColFlags.NONE ) {
                        Object readArray( BasicInput stream, int count )
                                throws IOException {
                            int nString =
                                ( count + stringLength - 1 ) / stringLength;
                            String[] value = new String[ nString ];
                            for ( int i = 0; i < nString; i++ ) {
                                int nchar = Math.min( count, stringLength );
                                value[ i ] = readString( stream, nchar );
                                count -= nchar;
                            }
                            assert count == 0;
                            return value;
                        }
                        int getElementSize() {
                            return stringLength;
                        }
                    };
                }
                return reader;

            /* Floating point. */
            case 'E':
                return createFloatsArrayReader( dims, scale, dZero );

            /* Double precision. */
            case 'D':
                return createDoublesArrayReader( dims, scale, dZero );

            /* Single precision complex. */
            case 'C':
                return createFloatsArrayReader( complexShape( dims ),
                                                scale, dZero );

            /* Double precision complex. */
            case 'M':
                return createDoublesArrayReader( complexShape( dims ),
                                                 scale, dZero );

            /* No known TFORM type. */
            default:
                throw new AssertionError( "Unknown TFORM type " + type );
        }
    }

    /**
     * Returns a new array reader for reading single precision floating point
     * values.
     *
     * @param  shape  shape of the array to read
     * @param   scale  factor to scale numerical values by
     * @param   zero   offset to add to numerical values
     * @return  new array reader
     */
    private static ArrayReader createFloatsArrayReader( int[] shape,
                                                        final double scale,
                                                        final double zero ) {
        final boolean isScaled = scale != 1.0 || zero != 0.0;
        if ( isScaled ) {
            return new ArrayReader( float[].class, shape, 4, ColFlags.NONE ) {
                Object readArray( BasicInput stream, int count ) 
                        throws IOException {
                    float[] value = new float[ count ];
                    for ( int i = 0; i < count; i++ ) {
                        float val = stream.readFloat();
                        value[ i ] = (float) ( val * scale + zero );
                    }
                    return value;
                }
            };
        }
        else {
            return new ArrayReader( float[].class, shape, 4, ColFlags.NONE ) {
                Object readArray( BasicInput stream, int count )
                        throws IOException {
                    float[] value = new float[ count ];
                    for ( int i = 0; i < count; i++ ) {
                        value[ i ] = stream.readFloat();
                    }
                    return value;
                }
            };
        }
    }

    /**
     * Returns a new array reader for reading double precision floating point
     * values.
     *
     * @param  shape  shape of the array to read
     * @param   scale  factor to scale numerical values by
     * @param   zero   offset to add to numerical values
     * @return  new array reader
     */
    private static ArrayReader createDoublesArrayReader( int[] shape,
                                                         final double scale,
                                                         final double zero ) {
        final boolean isScaled = scale != 1.0 || zero != 0.0;
        if ( isScaled ) {
            return new ArrayReader( double[].class, shape, 8, ColFlags.NONE ) {
                Object readArray( BasicInput stream, int count )
                        throws IOException {
                    double[] value = new double[ count ];
                    for ( int i = 0; i < count; i++ ) {
                        double val = stream.readDouble();
                        value[ i ] = val * scale + zero;
                    }
                    return value;
                }
            };
        }
        else {
            return new ArrayReader( double[].class, shape, 8, ColFlags.NONE ) {
                Object readArray( BasicInput stream, int count )
                        throws IOException {
                    double[] value = new double[ count ];
                    for ( int i = 0; i < count; i++ ) {
                        value[ i ] = stream.readDouble();
                    }
                    return value;
                }
            };
        }
    }

    /**
     * Reads a string from a data stream.
     * A fixed number of bytes are read from the stream, but the returned
     * object is a variable-length string with trailing spaces omitted.
     * If it's all spaces, null is returned.
     *
     * @param  stream  the stream to read from
     * @param  count  number of bytes to read from the stream
     * @return  string read
     */
    private static String readString( BasicInput stream, int count )
            throws IOException {
        byte[] bbuf = new byte[ count ];
        stream.readBytes( bbuf );
        int last = -1;
        boolean end = false;
        for ( int i = 0; i < count && !end; i++ ) {
            switch ( bbuf[ i ] ) {
                case 0:
                    end = true;
                    break;
                case (byte) ' ':
                    break;
                default:
                    last = i;
            }
        }
        int leng = last + 1;
        return leng == 0 ? null
                         : new String( bbuf, 0, leng, US_ASCII );
    }

    /**
     * Returns a dimensions array based on a given one, but with an extra
     * dimension of extent 2 prepended to the list.
     *
     * @param  dims  intial dimensions array (null is interpreted
     *               as a zero-dimensional array
     * @return  like dims but with a 2 at the start
     */
    private static int[] complexShape( int[] dims ) {
        if ( dims == null ) {
            return new int[] { 2 };
        }
        else {
            int[] shape = new int[ dims.length + 1 ];
            shape[ 0 ] = 2;
            System.arraycopy( dims, 0, shape, 1, dims.length );
            return shape;
        }
    }

    /**
     * Indicates whether a Number value is an integer type or not.
     * Makes some assumptions about the types it may have been passed.
     *
     * @param  num   value
     * @return   true  iff num is an integer type
     */
    private static boolean isInteger( Number num ) {
        if ( num instanceof Byte ||
             num instanceof Short ||
             num instanceof Integer ||
             num instanceof Long ) {
            return true;
        }
        else if ( num instanceof BigInteger ) {
            BigInteger bnum = (BigInteger) num;
            return bnum.compareTo( BigInteger.valueOf( Long.MIN_VALUE ) ) >= 0
                && bnum.compareTo( BigInteger.valueOf( Long.MAX_VALUE ) ) <= 0;
        }
        else {
            assert num instanceof Float ||
                   num instanceof Double ||
                   num instanceof BigDecimal;
            return false;
        }
    }

    /**
     * Assesses whether long values are within a given range.
     * The first time that a value out of range is encountered, 
     * a warning message is written through the logging system.
     */
    private static class LongRanger {
        private final long lMin_;
        private final long lMax_;
        private final Number zeroNum_;
        private final String failReturn_;
        private boolean hasWarned_;

        /**
         * Constructor.
         *
         * @param   lMin  minimum permitted value
         * @param   lMax  maximum permitted value
         * @param   zeroNum   offset value (used in log messages)
         * @param   failReturn  substitute value in case of range miss
         *                      (used in log messages)
         */
        LongRanger( long lMin, long lMax, Number zeroNum, String failReturn  ) {
            lMin_ = lMin;
            lMax_ = lMax;
            zeroNum_ = zeroNum;
            failReturn_ = failReturn;
        }

        /**
         * Determines whether a given value is in range.
         * Warns through logging system the first time false is returned.
         *
         * @param  lval   value to assess
         * @return  true iff lMin <= lval <= lMax
         */
        boolean inRange( long lval ) {
            if ( lval >= lMin_ && lval <= lMax_ ) {
                return true;
            }
            else {
                if ( ! hasWarned_ ) {
                    String msg = "Cannot represent large offset long values"
                               + " - will return " + failReturn_
                               + " (offset=" + zeroNum_ + ";"
                               + " first value=" + lval + ")";
                    logger_.warning( msg );
                    hasWarned_ = true;
                }
                return false;
            }
        }
    }

    /**
     * Abstract class defining an object which can read an array of similarly
     * typed values from a stream.
     */
    private static abstract class ArrayReader {

        private final Class clazz_;
        private final int[] shape_;
        private final int elBytes_;
        private final ColFlags flags_;

        /**
         * Constructor.
         *
         * @param   clazz  class of values read by this reader
         * @param   shape  shape of values read by this reader
         * @param   elBytes  number of bytes read from stream for each element
         * @param   flags  additional information
         */
        ArrayReader( Class clazz, int[] shape, int elBytes,
                     ColFlags flags ) {
            clazz_ = clazz;
            shape_ = shape;
            elBytes_ = elBytes;
            flags_ = flags;
        }

        /**
         * Reads an array from the current position in a stream.
         *
         * @param  stream   stream to read from
         * @param  count   number of items to read
         */
        abstract Object readArray( BasicInput stream, int count )
                throws IOException;

        /** 
         * Returns the class of objects returned from readArray.
         *
         * @return  content class
         */
        Class getContentClass() {
            return clazz_;
        }

        /**
         * Returns the shape of returned values in the sense of 
         * ValueInfo.getShape();
         *
         * @param  shape, or null for scalars
         */
        int[] getShape() {
            return shape_;
        }

        /**
         * Returns string size in the sense of ValueInfo.getElementSize().
         *
         * @return element size, or -1 if not applicable
         */
        int getElementSize() {
            return -1;
        }

        /**
         * Returns the number of bytes read from the input stream 
         * when count elements are read.
         *
         * @param  count  number of elements read from array
         * @return  number of bytes read
         */
        int getByteCount( int count ) {
            return elBytes_ * count;
        }
    }

    /**
     * Represents miscellaneous information about a column reader.
     */
    private static class ColFlags {
        final boolean isUnsignedByte_;
        final BigInteger longOffset_;

        /** Instance indicating no flags set. */
        final static ColFlags NONE = new ColFlags( false, null );

        /** Instance indicating unsigned byte. */
        final static ColFlags UNSIGNED_BYTE = new ColFlags( true, null );

        /**
         * Constructor.
         *
         * @param  isUnsignedByte  true to indicate reads that are in the
         *                         unsigned byte range, 0..255
         * @param  longOffset   offset value for offset stringified long values
         */
        ColFlags( boolean isUnsignedByte, BigInteger longOffset ) {
            isUnsignedByte_ = isUnsignedByte;
            longOffset_ = longOffset;
        }

        /**
         * Creates an instance indicating stringified offset long values.
         *
         * @param  off  offset value
         */
        static ColFlags createLongOffset( BigInteger off ) {
            return new ColFlags( false, off );
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy