Skip to content
This repository has been archived by the owner on Apr 19, 2024. It is now read-only.

Commit

Permalink
Merge pull request #132 from Sensirion/update-fix-point-aritmetic
Browse files Browse the repository at this point in the history
Fix fix16_mul() to work properly on 8-bit MCUs
  • Loading branch information
MBjoern authored May 7, 2021
2 parents d4fe77a + 1f80d8e commit 240e09f
Show file tree
Hide file tree
Showing 3 changed files with 49 additions and 40 deletions.
3 changes: 3 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0

## [Unreleased]

* [`fixed`] Fix fix16_mul() to work properly with 8-bit PIC compilers (and
possibly others)

## [7.1.1] - 2020-12-14

* [`changed`] Update embedded-common to 0.1.0 to improve compatibility when
Expand Down
79 changes: 40 additions & 39 deletions sgp40_voc_index/sensirion_voc_algorithm.c
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright (c) 2020, Sensirion AG
* Copyright (c) 2021, Sensirion AG
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
Expand Down Expand Up @@ -45,12 +45,12 @@
/*!< fix16_t value of 1 */
#define FIX16_ONE 0x00010000

inline fix16_t fix16_from_int(int32_t a) {
static inline fix16_t fix16_from_int(int32_t a) {
return a * FIX16_ONE;
}

inline int32_t fix16_cast_to_int(fix16_t a) {
return (a >> 16);
static inline int32_t fix16_cast_to_int(fix16_t a) {
return (a >= 0) ? (a >> 16) : -((-a) >> 16);
}

/*! Multiplies the two given fix16_t's and returns the result. */
Expand All @@ -75,14 +75,16 @@ static fix16_t fix16_mul(fix16_t inArg0, fix16_t inArg1) {
// AD
// AC
// |----| 64 bit product
int32_t A = (inArg0 >> 16), C = (inArg1 >> 16);
uint32_t B = (inArg0 & 0xFFFF), D = (inArg1 & 0xFFFF);
uint32_t absArg0 = (uint32_t)((inArg0 >= 0) ? inArg0 : (-inArg0));
uint32_t absArg1 = (uint32_t)((inArg1 >= 0) ? inArg1 : (-inArg1));
uint32_t A = (absArg0 >> 16), C = (absArg1 >> 16);
uint32_t B = (absArg0 & 0xFFFF), D = (absArg1 & 0xFFFF);

int32_t AC = A * C;
int32_t AD_CB = A * D + C * B;
uint32_t AC = A * C;
uint32_t AD_CB = A * D + C * B;
uint32_t BD = B * D;

int32_t product_hi = AC + (AD_CB >> 16);
uint32_t product_hi = AC + (AD_CB >> 16);

// Handle carry from lower 32 bits to upper part of result.
uint32_t ad_cb_temp = AD_CB << 16;
Expand All @@ -91,30 +93,29 @@ static fix16_t fix16_mul(fix16_t inArg0, fix16_t inArg1) {
product_hi++;

#ifndef FIXMATH_NO_OVERFLOW
// The upper 17 bits should all be the same (the sign).
if (product_hi >> 31 != product_hi >> 15)
return FIX16_OVERFLOW;
// The upper 17 bits should all be zero.
if (product_hi >> 15)
return (fix16_t)FIX16_OVERFLOW;
#endif

#ifdef FIXMATH_NO_ROUNDING
return (product_hi << 16) | (product_lo >> 16);
fix16_t result = (fix16_t)((product_hi << 16) | (product_lo >> 16));
if ((inArg0 < 0) != (inArg1 < 0))
result = -result;
return result;
#else
// Subtracting 0x8000 (= 0.5) and then using signed right shift
// achieves proper rounding to result-1, except in the corner
// case of negative numbers and lowest word = 0x8000.
// To handle that, we also have to subtract 1 for negative numbers.
// Adding 0x8000 (= 0.5) and then using right shift
// achieves proper rounding to result.
// Handle carry from lower to upper part.
uint32_t product_lo_tmp = product_lo;
product_lo -= 0x8000;
product_lo -= (uint32_t)product_hi >> 31;
if (product_lo > product_lo_tmp)
product_hi--;

// Discard the lowest 16 bits. Note that this is not exactly the same
// as dividing by 0x10000. For example if product = -1, result will
// also be -1 and not 0. This is compensated by adding +1 to the result
// and compensating this in turn in the rounding above.
fix16_t result = (product_hi << 16) | (product_lo >> 16);
result += 1;
product_lo += 0x8000;
if (product_lo < product_lo_tmp)
product_hi++;

// Discard the lowest 16 bits and convert back to signed result.
fix16_t result = (fix16_t)((product_hi << 16) | (product_lo >> 16));
if ((inArg0 < 0) != (inArg1 < 0))
result = -result;
return result;
#endif
}
Expand All @@ -126,10 +127,10 @@ static fix16_t fix16_div(fix16_t a, fix16_t b) {
// platforms without hardware divide.

if (b == 0)
return FIX16_MINIMUM;
return (fix16_t)FIX16_MINIMUM;

uint32_t remainder = (a >= 0) ? a : (-a);
uint32_t divider = (b >= 0) ? b : (-b);
uint32_t remainder = (uint32_t)((a >= 0) ? a : (-a));
uint32_t divider = (uint32_t)((b >= 0) ? b : (-b));

uint32_t quotient = 0;
uint32_t bit = 0x10000;
Expand All @@ -142,7 +143,7 @@ static fix16_t fix16_div(fix16_t a, fix16_t b) {

#ifndef FIXMATH_NO_OVERFLOW
if (!bit)
return FIX16_OVERFLOW;
return (fix16_t)FIX16_OVERFLOW;
#endif

if (divider & 0x80000000) {
Expand Down Expand Up @@ -173,13 +174,13 @@ static fix16_t fix16_div(fix16_t a, fix16_t b) {
}
#endif

fix16_t result = quotient;
fix16_t result = (fix16_t)quotient;

/* Figure out the sign of result */
if ((a ^ b) & 0x80000000) {
if ((a < 0) != (b < 0)) {
#ifndef FIXMATH_NO_OVERFLOW
if (result == FIX16_MINIMUM)
return FIX16_OVERFLOW;
return (fix16_t)FIX16_OVERFLOW;
#endif

result = -result;
Expand All @@ -191,7 +192,7 @@ static fix16_t fix16_div(fix16_t a, fix16_t b) {
static fix16_t fix16_sqrt(fix16_t x) {
// It is assumed that x is not negative

uint32_t num = x;
uint32_t num = (uint32_t)x;
uint32_t result = 0;
uint32_t bit;
uint8_t n;
Expand Down Expand Up @@ -246,9 +247,9 @@ static fix16_t fix16_sqrt(fix16_t x) {
}

static fix16_t fix16_exp(fix16_t x) {
// Function to approximate exp(); optimized more for code size than speed
// Function to approximate exp(); optimized more for code size than speed

// exp(x) for x = +/- {1, 1/8, 1/64, 1/512}
// exp(x) for x = +/- {1, 1/8, 1/64, 1/512}
#define NUM_EXP_VALUES 4
static const fix16_t exp_pos_values[NUM_EXP_VALUES] = {
F16(2.7182818), F16(1.1331485), F16(1.0157477), F16(1.0019550)};
Expand Down Expand Up @@ -801,4 +802,4 @@ VocAlgorithm__adaptive_lowpass__process(VocAlgorithmParams* params,
((fix16_mul((F16(1.) - a3), params->m_Adaptive_Lowpass___X3)) +
(fix16_mul(a3, sample)));
return params->m_Adaptive_Lowpass___X3;
}
}
7 changes: 6 additions & 1 deletion sgp40_voc_index/sensirion_voc_algorithm.h
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright (c) 2020, Sensirion AG
* Copyright (c) 2021, Sensirion AG
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
Expand Down Expand Up @@ -43,6 +43,11 @@ typedef int32_t fix16_t;
#define F16(x) \
((fix16_t)(((x) >= 0) ? ((x)*65536.0 + 0.5) : ((x)*65536.0 - 0.5)))

// Should be set by the building toolchain
#ifndef LIBRARY_VERSION_NAME
#define LIBRARY_VERSION_NAME "custom build"
#endif

#define VocAlgorithm_SAMPLING_INTERVAL (1.)
#define VocAlgorithm_INITIAL_BLACKOUT (45.)
#define VocAlgorithm_VOC_INDEX_GAIN (230.)
Expand Down

0 comments on commit 240e09f

Please sign in to comment.