277 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			277 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //
 | |
| // byte_order_stuff.cpp
 | |
| // libscio
 | |
| //
 | |
| // Created by Bob Polis at 2020-02-14
 | |
| // Copyright (c) 2020 SwiftCoder. All rights reserved.
 | |
| //
 | |
| 
 | |
| #include "libscio.hpp"
 | |
| #include <cmath>
 | |
| #include <cstring>
 | |
| #include <stdexcept>
 | |
| 
 | |
| using namespace sc::io;
 | |
| 
 | |
| // data_streamer ---------------------------------------------------------------
 | |
| 
 | |
| data_streamer::data_streamer(byte_order bo) : _target_byte_order(bo) {}
 | |
| 
 | |
| byte_order data_streamer::cur_byte_order() {
 | |
| 	uint8_t ch[2] = {0x12, 0x34};
 | |
| 	uint16_t* p = reinterpret_cast<uint16_t*>(ch);
 | |
| 	return *p == 0x1234 ? byte_order::big_endian : byte_order::little_endian;
 | |
| }
 | |
| 
 | |
| std::string data_streamer::get4chars(std::istream& file) const {
 | |
| 	char buf[4];
 | |
| 	file.read(buf, 4);
 | |
| 	return std::string(buf, 4);
 | |
| }
 | |
| 
 | |
| uint32_t data_streamer::getui24(std::istream& file) const {
 | |
| 	char buf[4] = {0};
 | |
| 	int i = cur_byte_order() == byte_order::little_endian ? 0 : 1;
 | |
| 	file.read(&buf[i], 3);
 | |
| 	if (cur_byte_order() != _target_byte_order) {
 | |
| 		char tmp = buf[i];
 | |
| 		buf[i] = buf[i + 2];
 | |
| 		buf[i + 2] = tmp;
 | |
| 	}
 | |
| 	return *reinterpret_cast<uint32_t*>(buf);
 | |
| }
 | |
| 
 | |
| int32_t data_streamer::getsi24(std::istream& file) const {
 | |
| 	return static_cast<int32_t>(getui24(file));
 | |
| }
 | |
| 
 | |
| double data_streamer::getf80(std::istream& file) const {
 | |
| 	float_80 val;
 | |
| 	file.read(reinterpret_cast<char*>(val._val), sizeof(val._val));
 | |
| 	if (_target_byte_order != byte_order::big_endian) {
 | |
| 		byte_swap(val._val, sizeof(val._val));
 | |
| 	}
 | |
| 	return val;
 | |
| }
 | |
| 
 | |
| void data_streamer::put4chars(const std::string& s, std::ostream& file) const {
 | |
| 	if (s.size() < 4) {
 | |
| 		throw std::runtime_error("data_streamer::put4chars: string too short");
 | |
| 	}
 | |
| 	file.write(s.c_str(), 4);
 | |
| }
 | |
| 
 | |
| void data_streamer::putui24(uint32_t val, std::ostream& file) const {
 | |
| 	char* buf = reinterpret_cast<char*>(&val);
 | |
| 	int i = cur_byte_order() == byte_order::little_endian ? 0 : 1;
 | |
| 	if (cur_byte_order() != _target_byte_order) {
 | |
| 		char tmp = buf[i];
 | |
| 		buf[i] = buf[i + 2];
 | |
| 		buf[i + 2] = tmp;
 | |
| 	}
 | |
| 	file.write(&buf[i], 3);
 | |
| }
 | |
| 
 | |
| void data_streamer::putsi24(int32_t val, std::ostream& file) const {
 | |
| 	putui24(static_cast<uint32_t>(val), file);
 | |
| }
 | |
| 
 | |
| void data_streamer::putf80(double val, std::ostream& file) const {
 | |
| 	float_80 f80 {val};
 | |
| 	if (_target_byte_order != byte_order::big_endian) {
 | |
| 		byte_swap(f80._val, sizeof(f80._val));
 | |
| 	}
 | |
| 	file.write(reinterpret_cast<char*>(f80._val), sizeof(f80._val));
 | |
| }
 | |
| 
 | |
| // byte_order_changer ----------------------------------------------------------
 | |
| 
 | |
| byte_order_changer::byte_order_changer(data_streamer* ds, byte_order order) : _data_streamer {ds} {
 | |
| 	if (ds == nullptr) {
 | |
| 		std::cerr << "byte_order_changer: no data streamer given\n";
 | |
| 		return;
 | |
| 	}
 | |
| 	_saved_byte_order = _data_streamer->target_byte_order();
 | |
| 	_data_streamer->target_byte_order(order);
 | |
| }
 | |
| 
 | |
| byte_order_changer::~byte_order_changer() {
 | |
| 	if (_data_streamer) {
 | |
| 		_data_streamer->target_byte_order(_saved_byte_order);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| byte_order_changer::byte_order_changer(byte_order_changer&& other) {
 | |
| 	_data_streamer = other._data_streamer;
 | |
| 	_saved_byte_order = other._saved_byte_order;
 | |
| 	other._data_streamer = nullptr;
 | |
| }
 | |
| 
 | |
| byte_order_changer& byte_order_changer::operator=(byte_order_changer&& other) {
 | |
| 	if (&other != this) {
 | |
| 		_data_streamer = other._data_streamer;
 | |
| 		_saved_byte_order = other._saved_byte_order;
 | |
| 		other._data_streamer = nullptr;
 | |
| 	}
 | |
| 	return *this;
 | |
| }
 | |
| 
 | |
| // float_80 --------------------------------------------------------------------
 | |
| 
 | |
| /*-
 | |
|  * Copyright (c) 2005, 2006 by Marco Trillo <marcotrillo@gmail.com>
 | |
|  *
 | |
|  * Permission is hereby granted, free of charge, to any
 | |
|  * person obtaining a copy of this software and associated
 | |
|  * documentation files (the "Software"), to deal in the
 | |
|  * Software without restriction, including without limitation
 | |
|  * the rights to use, copy, modify, merge, publish,
 | |
|  * distribute, sublicense, and/or sell copies of the
 | |
|  * Software, and to permit persons to whom the Software is
 | |
|  * furnished to do so, subject to the following conditions:
 | |
|  *
 | |
|  * The above copyright notice and this permission notice
 | |
|  * shall be included in all copies or substantial portions of
 | |
|  * the Software.
 | |
|  *
 | |
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
 | |
|  * KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
 | |
|  * WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
 | |
|  * PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS
 | |
|  * OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
 | |
|  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
 | |
|  * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
 | |
|  * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * Infinite & NAN values for non-IEEE systems
 | |
|  */
 | |
| #ifndef HUGE_VAL
 | |
| #ifdef HUGE
 | |
| #define INFINITE_VALUE	HUGE
 | |
| #define NAN_VALUE	HUGE
 | |
| #endif
 | |
| #else
 | |
| #define INFINITE_VALUE	HUGE_VAL
 | |
| #define NAN_VALUE	HUGE_VAL
 | |
| #endif
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * IEEE 754 Extended Precision
 | |
|  *
 | |
|  * Implementation here is the 80-bit extended precision
 | |
|  * format of Motorola 68881, Motorola 68882 and Motorola
 | |
|  * 68040 FPUs, as well as Intel 80x87 FPUs.
 | |
|  *
 | |
|  * See:
 | |
|  *    http://www.freescale.com/files/32bit/doc/fact_sheet/BR509.pdf
 | |
|  */
 | |
| /*
 | |
|  * Exponent range: [-16383,16383]
 | |
|  * Precision for mantissa: 64 bits with no hidden bit
 | |
|  * Bias: 16383
 | |
|  */
 | |
| 
 | |
| float_80::float_80() : float_80(0) {}
 | |
| 
 | |
| float_80::float_80(double in) {
 | |
| 	int sgn, exp, shift;
 | |
| 	double fraction, t;
 | |
| 	unsigned int lexp, hexp;
 | |
| 	unsigned long low, high;
 | |
| 
 | |
| 	if (in == 0.0) {
 | |
| 		memset(_val, 0, 10);
 | |
| 		return;
 | |
| 	}
 | |
| 	if (in < 0.0) {
 | |
| 		in = fabs(in);
 | |
| 		sgn = 1;
 | |
| 	} else
 | |
| 		sgn = 0;
 | |
| 
 | |
| 	fraction = frexp(in, &exp);
 | |
| 
 | |
| 	if (exp == 0 || exp > 16384) {
 | |
| 		if (exp > 16384) /* infinite value */
 | |
| 			low = high = 0;
 | |
| 		else {
 | |
| 			low = 0x80000000;
 | |
| 			high = 0;
 | |
| 		}
 | |
| 		exp = 32767;
 | |
| 		goto done;
 | |
| 	}
 | |
| 	fraction = ldexp(fraction, 32);
 | |
| 	t = floor(fraction);
 | |
| 	low = (unsigned long) t;
 | |
| 	fraction -= t;
 | |
| 	t = floor(ldexp(fraction, 32));
 | |
| 	high = (unsigned long) t;
 | |
| 
 | |
| 	/* Convert exponents < -16382 to -16382 (then they will be
 | |
| 	 * stored as -16383) */
 | |
| 	if (exp < -16382) {
 | |
| 		shift = 0 - exp - 16382;
 | |
| 		high >>= shift;
 | |
| 		high |= (low << (32 - shift));
 | |
| 		low >>= shift;
 | |
| 		exp = -16382;
 | |
| 	}
 | |
| 	exp += 16383 - 1;	/* bias */
 | |
| 
 | |
| done:
 | |
| 	lexp = ((unsigned int) exp) >> 8;
 | |
| 	hexp = ((unsigned int) exp) & 0xFF;
 | |
| 
 | |
| 	/* big endian */
 | |
| 	_val[0] = ((unsigned char) sgn) << 7;
 | |
| 	_val[0] |= (unsigned char) lexp;
 | |
| 	_val[1] = (unsigned char) hexp;
 | |
| 	_val[2] = (unsigned char) (low >> 24);
 | |
| 	_val[3] = (unsigned char) ((low >> 16) & 0xFF);
 | |
| 	_val[4] = (unsigned char) ((low >> 8) & 0xFF);
 | |
| 	_val[5] = (unsigned char) (low & 0xFF);
 | |
| 	_val[6] = (unsigned char) (high >> 24);
 | |
| 	_val[7] = (unsigned char) ((high >> 16) & 0xFF);
 | |
| 	_val[8] = (unsigned char) ((high >> 8) & 0xFF);
 | |
| 	_val[9] = (unsigned char) (high & 0xFF);
 | |
| }
 | |
| 
 | |
| float_80::operator double() const {
 | |
| 	int sgn, exp;
 | |
| 	unsigned long low, high;
 | |
| 	double out;
 | |
| 
 | |
| 	/* Extract the components from the big endian buffer */
 | |
| 	sgn = (int) (_val[0] >> 7);
 | |
| 	exp = ((int) (_val[0] & 0x7F) << 8) | ((int) _val[1]);
 | |
| 	low = (((unsigned long) _val[2]) << 24)
 | |
| 		| (((unsigned long) _val[3]) << 16)
 | |
| 		| (((unsigned long) _val[4]) << 8) | (unsigned long) _val[5];
 | |
| 	high = (((unsigned long) _val[6]) << 24)
 | |
| 		| (((unsigned long) _val[7]) << 16)
 | |
| 		| (((unsigned long) _val[8]) << 8) | (unsigned long) _val[9];
 | |
| 
 | |
| 	if (exp == 0 && low == 0 && high == 0)
 | |
| 		return (sgn ? -0.0 : 0.0);
 | |
| 
 | |
| 	switch (exp) {
 | |
| 	case 32767:
 | |
| 		if (low == 0 && high == 0)
 | |
| 			return (sgn ? -INFINITE_VALUE : INFINITE_VALUE);
 | |
| 		else
 | |
| 			return (sgn ? -NAN_VALUE : NAN_VALUE);
 | |
| 	default:
 | |
| 		exp -= 16383;	/* unbias exponent */
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	out = ldexp((double) low, -31 + exp);
 | |
| 	out += ldexp((double) high, -63 + exp);
 | |
| 
 | |
| 	return (sgn ? -out : out);
 | |
| }
 |