2014-04-06 11:43:13 -04:00
|
|
|
/**
|
2015-04-28 13:45:35 -04:00
|
|
|
* @file
|
|
|
|
* @brief QS floating point output implementation
|
|
|
|
* @ingroup qs
|
|
|
|
* @cond
|
2014-04-06 11:43:13 -04:00
|
|
|
******************************************************************************
|
2016-09-09 13:14:46 -04:00
|
|
|
* Last updated for version 5.7.0
|
|
|
|
* Last updated on 2016-09-08
|
2012-08-14 18:07:04 -04:00
|
|
|
*
|
|
|
|
* Q u a n t u m L e a P s
|
|
|
|
* ---------------------------
|
|
|
|
* innovating embedded systems
|
|
|
|
*
|
2017-05-17 13:16:32 -04:00
|
|
|
* Copyright (C) Quantum Leaps, LLC. All rights reserved.
|
2012-08-14 18:07:04 -04:00
|
|
|
*
|
|
|
|
* This program is open source software: you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as published
|
2013-10-16 16:44:03 -04:00
|
|
|
* by the Free Software Foundation, either version 3 of the License, or
|
2012-08-14 18:07:04 -04:00
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* Alternatively, this program may be distributed and modified under the
|
|
|
|
* terms of Quantum Leaps commercial licenses, which expressly supersede
|
|
|
|
* the GNU General Public License and are specifically designed for
|
|
|
|
* licensees interested in retaining the proprietary status of their code.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* Contact information:
|
2017-05-17 13:16:32 -04:00
|
|
|
* https://state-machine.com
|
|
|
|
* mailto:info@state-machine.com
|
2014-04-06 11:43:13 -04:00
|
|
|
******************************************************************************
|
2015-04-28 13:45:35 -04:00
|
|
|
* @endcond
|
2014-04-06 11:43:13 -04:00
|
|
|
*/
|
|
|
|
#include "qs_port.h" /* QS port */
|
2012-08-14 18:07:04 -04:00
|
|
|
#include "qs_pkg.h"
|
|
|
|
|
2014-04-06 11:43:13 -04:00
|
|
|
/****************************************************************************/
|
2012-08-14 18:07:04 -04:00
|
|
|
/**
|
2015-04-28 13:45:35 -04:00
|
|
|
* @note This function is only to be used through macros, never in the
|
|
|
|
* client code directly.
|
|
|
|
*/
|
|
|
|
void QS_f32(uint8_t format, float32_t f) {
|
|
|
|
union F32Rep {
|
|
|
|
float32_t f;
|
|
|
|
uint32_t u;
|
|
|
|
} fu32; /* the internal binary representation */
|
|
|
|
uint8_t chksum = QS_priv_.chksum; /* put in a temporary (register) */
|
|
|
|
uint8_t *buf = QS_priv_.buf; /* put in a temporary (register) */
|
|
|
|
QSCtr head = QS_priv_.head; /* put in a temporary (register) */
|
|
|
|
QSCtr end = QS_priv_.end; /* put in a temporary (register) */
|
|
|
|
int_fast8_t i;
|
|
|
|
|
|
|
|
fu32.f = f; /* assign the binary representation */
|
|
|
|
|
|
|
|
QS_priv_.used += (QSCtr)5; /* 5 bytes about to be added */
|
|
|
|
QS_INSERT_ESC_BYTE(format) /* insert the format byte */
|
|
|
|
|
|
|
|
/* insert 4 bytes... */
|
|
|
|
for (i = (int_fast8_t)4; i != (int_fast8_t)0; --i) {
|
|
|
|
format = (uint8_t)fu32.u;
|
|
|
|
QS_INSERT_ESC_BYTE(format)
|
|
|
|
fu32.u >>= 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
QS_priv_.head = head; /* save the head */
|
|
|
|
QS_priv_.chksum = chksum; /* save the checksum */
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************/
|
|
|
|
/**
|
|
|
|
* @description
|
2014-04-06 11:43:13 -04:00
|
|
|
* This function is only to be used through macros, never in the
|
|
|
|
* client code directly.
|
2012-08-14 18:07:04 -04:00
|
|
|
*/
|
|
|
|
void QS_f64(uint8_t format, float64_t d) {
|
|
|
|
union F64Rep {
|
|
|
|
float64_t d;
|
|
|
|
struct UInt2 {
|
2013-09-23 14:34:35 -04:00
|
|
|
uint32_t u1;
|
|
|
|
uint32_t u2;
|
2012-08-14 18:07:04 -04:00
|
|
|
} i;
|
2014-04-06 11:43:13 -04:00
|
|
|
} fu64; /* the internal binary representation */
|
2013-09-23 14:34:35 -04:00
|
|
|
uint8_t chksum = QS_priv_.chksum;
|
|
|
|
uint8_t *buf = QS_priv_.buf;
|
|
|
|
QSCtr head = QS_priv_.head;
|
|
|
|
QSCtr end = QS_priv_.end;
|
2016-09-09 13:14:46 -04:00
|
|
|
uint32_t i;
|
|
|
|
/* static constant untion to detect endianness of the machine */
|
|
|
|
static union U32Rep {
|
|
|
|
uint32_t u32;
|
|
|
|
uint8_t u8;
|
|
|
|
} const endian = { (uint32_t)1 };
|
2012-08-14 18:07:04 -04:00
|
|
|
|
2014-04-06 11:43:13 -04:00
|
|
|
fu64.d = d; /* assign the binary representation */
|
2012-08-14 18:07:04 -04:00
|
|
|
|
2014-04-06 11:43:13 -04:00
|
|
|
QS_priv_.used += (QSCtr)9; /* 9 bytes about to be added */
|
|
|
|
QS_INSERT_ESC_BYTE(format) /* insert the format byte */
|
2012-08-14 18:07:04 -04:00
|
|
|
|
2016-09-09 13:14:46 -04:00
|
|
|
/* is this a big-endian machine? */
|
|
|
|
if (endian.u8 == (uint8_t)0) {
|
|
|
|
/* swap fu64.i.u1 <-> fu64.i.u2... */
|
|
|
|
i = fu64.i.u1;
|
|
|
|
fu64.i.u1 = fu64.i.u2;
|
|
|
|
fu64.i.u2 = i;
|
|
|
|
}
|
|
|
|
|
2014-04-06 11:43:13 -04:00
|
|
|
/* output 4 bytes from fu64.i.u1 ... */
|
2016-09-09 13:14:46 -04:00
|
|
|
for (i = (uint32_t)4; i != (uint32_t)0; --i) {
|
2013-09-23 14:34:35 -04:00
|
|
|
format = (uint8_t)fu64.i.u1;
|
|
|
|
QS_INSERT_ESC_BYTE(format)
|
|
|
|
fu64.i.u1 >>= 8;
|
|
|
|
}
|
2014-04-06 11:43:13 -04:00
|
|
|
|
|
|
|
/* output 4 bytes from fu64.i.u2 ... */
|
2016-09-09 13:14:46 -04:00
|
|
|
for (i = (uint32_t)4; i != (uint32_t)0; --i) {
|
2013-09-23 14:34:35 -04:00
|
|
|
format = (uint8_t)fu64.i.u2;
|
|
|
|
QS_INSERT_ESC_BYTE(format)
|
|
|
|
fu64.i.u2 >>= 8;
|
|
|
|
}
|
|
|
|
|
2014-04-06 11:43:13 -04:00
|
|
|
QS_priv_.head = head; /* save the head */
|
|
|
|
QS_priv_.chksum = chksum; /* save the checksum */
|
2012-08-14 18:07:04 -04:00
|
|
|
}
|