qpc/source/qs_fp.c
Quantum Leaps 28ef7ac6ce 5.9.0
2017-05-17 13:16:32 -04:00

130 lines
4.3 KiB
C

/**
* @file
* @brief QS floating point output implementation
* @ingroup qs
* @cond
******************************************************************************
* Last updated for version 5.7.0
* Last updated on 2016-09-08
*
* Q u a n t u m L e a P s
* ---------------------------
* innovating embedded systems
*
* Copyright (C) Quantum Leaps, LLC. All rights reserved.
*
* 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
* by the Free Software Foundation, either version 3 of the License, or
* (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:
* https://state-machine.com
* mailto:info@state-machine.com
******************************************************************************
* @endcond
*/
#include "qs_port.h" /* QS port */
#include "qs_pkg.h"
/****************************************************************************/
/**
* @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
* This function is only to be used through macros, never in the
* client code directly.
*/
void QS_f64(uint8_t format, float64_t d) {
union F64Rep {
float64_t d;
struct UInt2 {
uint32_t u1;
uint32_t u2;
} i;
} fu64; /* the internal binary representation */
uint8_t chksum = QS_priv_.chksum;
uint8_t *buf = QS_priv_.buf;
QSCtr head = QS_priv_.head;
QSCtr end = QS_priv_.end;
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 };
fu64.d = d; /* assign the binary representation */
QS_priv_.used += (QSCtr)9; /* 9 bytes about to be added */
QS_INSERT_ESC_BYTE(format) /* insert the format byte */
/* 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;
}
/* output 4 bytes from fu64.i.u1 ... */
for (i = (uint32_t)4; i != (uint32_t)0; --i) {
format = (uint8_t)fu64.i.u1;
QS_INSERT_ESC_BYTE(format)
fu64.i.u1 >>= 8;
}
/* output 4 bytes from fu64.i.u2 ... */
for (i = (uint32_t)4; i != (uint32_t)0; --i) {
format = (uint8_t)fu64.i.u2;
QS_INSERT_ESC_BYTE(format)
fu64.i.u2 >>= 8;
}
QS_priv_.head = head; /* save the head */
QS_priv_.chksum = chksum; /* save the checksum */
}