Blame D4StreamUnMarshaller.h

Packit a4aae4
// D4StreamUnMarshaller.h
Packit a4aae4
Packit a4aae4
// -*- mode: c++; c-basic-offset:4 -*-
Packit a4aae4
Packit a4aae4
// This file is part of libdap, A C++ implementation of the OPeNDAP Data
Packit a4aae4
// Access Protocol.
Packit a4aae4
Packit a4aae4
// Copyright (c) 2012 OPeNDAP, Inc.
Packit a4aae4
// Author: James Gallagher <jgallagher@opendap.org>
Packit a4aae4
//
Packit a4aae4
// This library is free software; you can redistribute it and/or
Packit a4aae4
// modify it under the terms of the GNU Lesser General Public
Packit a4aae4
// License as published by the Free Software Foundation; either
Packit a4aae4
// version 2.1 of the License, or (at your option) any later version.
Packit a4aae4
//
Packit a4aae4
// This library is distributed in the hope that it will be useful,
Packit a4aae4
// but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit a4aae4
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit a4aae4
// Lesser General Public License for more details.
Packit a4aae4
//
Packit a4aae4
// You should have received a copy of the GNU Lesser General Public
Packit a4aae4
// License along with this library; if not, write to the Free Software
Packit a4aae4
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
Packit a4aae4
//
Packit a4aae4
// You can contact OPeNDAP, Inc. at PO Box 112, Saunderstown, RI. 02874-0112.
Packit a4aae4
Packit a4aae4
#ifndef I_D4StreamUnMarshaller_h
Packit a4aae4
#define I_D4StreamUnMarshaller_h 1
Packit a4aae4
Packit a4aae4
#include <iostream>
Packit a4aae4
Packit a4aae4
// See comment in D4StreamMarshaller
Packit a4aae4
#define USE_XDR_FOR_IEEE754_ENCODING 0
Packit a4aae4
Packit a4aae4
#if USE_XDR_FOR_IEEE754_ENCODING
Packit a4aae4
#ifdef WIN32
Packit a4aae4
#include <rpc.h>
Packit a4aae4
#include <winsock2.h>
Packit a4aae4
#include <xdr.h>
Packit a4aae4
#else
Packit a4aae4
#include <rpc/types.h>
Packit a4aae4
#include <netinet/in.h>
Packit a4aae4
#include <rpc/xdr.h>
Packit a4aae4
#endif
Packit a4aae4
#endif
Packit a4aae4
Packit a4aae4
#include <crc.h>
Packit a4aae4
Packit a4aae4
// #include "Type.h"
Packit a4aae4
#include "dods-datatypes.h"
Packit a4aae4
#include "UnMarshaller.h"
Packit a4aae4
#include "InternalErr.h"
Packit a4aae4
Packit a4aae4
#include "util.h"
Packit a4aae4
#include "debug.h"
Packit a4aae4
Packit a4aae4
using std::istream;
Packit a4aae4
Packit a4aae4
namespace libdap {
Packit a4aae4
Packit a4aae4
class Vector;
Packit a4aae4
Packit a4aae4
/** @brief Read data from the stream made by D4StreamMarshaller.
Packit a4aae4
 */
Packit a4aae4
class D4StreamUnMarshaller: public UnMarshaller {
Packit a4aae4
public:
Packit a4aae4
    const static unsigned int c_checksum_length = 4;
Packit a4aae4
Packit a4aae4
private:
Packit a4aae4
    istream &d_in;
Packit a4aae4
    bool d_twiddle_bytes;
Packit a4aae4
Packit a4aae4
#if USE_XDR_FOR_IEEE754_ENCODING
Packit a4aae4
    // These are used for reals that need to be converted from IEEE 754
Packit a4aae4
    XDR d_source;
Packit a4aae4
    char d_buf[sizeof(dods_float64)];
Packit a4aae4
#endif
Packit a4aae4
Packit a4aae4
    D4StreamUnMarshaller();
Packit a4aae4
    D4StreamUnMarshaller(const D4StreamUnMarshaller &);
Packit a4aae4
    D4StreamUnMarshaller & operator=(const D4StreamUnMarshaller &);
Packit a4aae4
#if USE_XDR_FOR_IEEE754_ENCODING
Packit a4aae4
    void m_deserialize_reals(char *val, int64_t num, int width, Type type);
Packit a4aae4
#endif
Packit a4aae4
    void m_twidle_vector_elements(char *vals, int64_t num, int width);
Packit a4aae4
Packit a4aae4
public:
Packit a4aae4
    D4StreamUnMarshaller(istream &in, bool twiddle_bytes);
Packit a4aae4
    D4StreamUnMarshaller(istream &in);
Packit a4aae4
    virtual ~D4StreamUnMarshaller();
Packit a4aae4
Packit a4aae4
    void set_twiddle_bytes(bool twiddle) { d_twiddle_bytes = twiddle; }
Packit a4aae4
Packit a4aae4
    /**
Packit a4aae4
     * @brief Is the data source we are reading from a big-endian machine?
Packit a4aae4
     * We need this because the value of the CRC32 checksum is dependent on
Packit a4aae4
     * byte order.
Packit a4aae4
     *
Packit a4aae4
     * @note This is somewhat tortured logic, but if this host is big-endian and
Packit a4aae4
     * twiddle_bytes is not true, then the remote host must be big-endian. Similarly,
Packit a4aae4
     * if this host is not big-endian and twiddle_bytes is true, then the remote
Packit a4aae4
     * host must be big-endian
Packit a4aae4
     */
Packit a4aae4
    bool is_source_big_endian() const { return (is_host_big_endian() && !d_twiddle_bytes)
Packit a4aae4
                                               || (!is_host_big_endian() && d_twiddle_bytes); }
Packit a4aae4
Packit a4aae4
    Crc32::checksum get_checksum();
Packit a4aae4
    string get_checksum_str();
Packit a4aae4
    int64_t get_count();
Packit a4aae4
Packit a4aae4
    virtual void get_byte(dods_byte &val;;
Packit a4aae4
    virtual void get_int8(dods_int8 &val;;
Packit a4aae4
Packit a4aae4
    virtual void get_int16(dods_int16 &val;;
Packit a4aae4
    virtual void get_int32(dods_int32 &val;;
Packit a4aae4
Packit a4aae4
    virtual void get_int64(dods_int64 &val;;
Packit a4aae4
Packit a4aae4
    virtual void get_float32(dods_float32 &val;;
Packit a4aae4
    virtual void get_float64(dods_float64 &val;;
Packit a4aae4
Packit a4aae4
    virtual void get_uint16(dods_uint16 &val;;
Packit a4aae4
    virtual void get_uint32(dods_uint32 &val;;
Packit a4aae4
Packit a4aae4
    virtual void get_uint64(dods_uint64 &val;;
Packit a4aae4
Packit a4aae4
    virtual void get_str(string &val;;
Packit a4aae4
    virtual void get_url(string &val;;
Packit a4aae4
Packit a4aae4
    virtual void get_opaque(char *, unsigned int) {
Packit a4aae4
    	throw InternalErr(__FILE__, __LINE__, "Not implemented for DAP4, use get_opaque_dap4() instead.");
Packit a4aae4
    }
Packit a4aae4
Packit a4aae4
    virtual void get_opaque_dap4(char **val, int64_t &len;;
Packit a4aae4
    virtual void get_opaque_dap4( vector<uint8_t> &val );
Packit a4aae4
Packit a4aae4
    virtual void get_int(int &) {
Packit a4aae4
        throw InternalErr(__FILE__, __LINE__, "Not implemented for DAP4");
Packit a4aae4
    }
Packit a4aae4
Packit a4aae4
    // Note that DAP4 assumes clients know the size of arrays when they
Packit a4aae4
    // read the data; it's the 'varying' get methods that read & return the
Packit a4aae4
    // number of elements. These methods are here to appease the UnMarshaller
Packit a4aae4
    // 'interface' code
Packit a4aae4
    virtual void get_vector(char **, unsigned int &, Vector &) {
Packit a4aae4
        throw InternalErr(__FILE__, __LINE__, "Not implemented for DAP4");
Packit a4aae4
    }
Packit a4aae4
Packit a4aae4
    virtual void get_vector(char **, unsigned int &, int, Vector & ) {
Packit a4aae4
        throw InternalErr(__FILE__, __LINE__, "Not implemented for DAP4");
Packit a4aae4
    }
Packit a4aae4
Packit a4aae4
    virtual void get_vector(char *val, int64_t num_bytes);
Packit a4aae4
    virtual void get_vector(char *val, int64_t num_elem, int elem_size);
Packit a4aae4
    virtual void get_vector_float32(char *val, int64_t num_elem);
Packit a4aae4
    virtual void get_vector_float64(char *val, int64_t num_elem);
Packit a4aae4
Packit a4aae4
    virtual void dump(ostream &strm) const;
Packit a4aae4
};
Packit a4aae4
Packit a4aae4
} // namespace libdap
Packit a4aae4
Packit a4aae4
#endif // I_D4StreamUnMarshaller_h
Packit a4aae4