Skip to content

Commit

Permalink
Add support for float binary pattern matching
Browse files Browse the repository at this point in the history
Add support to bs_get_float2 in ordet to enable
<<F:32/float>> = Bin.

Signed-off-by: Davide Bettio <davide@uninstall.it>
  • Loading branch information
bettio committed Nov 9, 2023
1 parent 81c3b57 commit 9ce1a1d
Show file tree
Hide file tree
Showing 4 changed files with 195 additions and 0 deletions.
121 changes: 121 additions & 0 deletions src/libAtomVM/bitstring.c
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,8 @@

#include "bitstring.h"

#include <assert.h>

static inline uint64_t from_le64(uint64_t value)
{
return ((((value) &0xFF) << 56) | (((value) &0xFF00) << 40) | (((value) &0xFF0000) << 24) | (((value) &0xFF000000) << 8) | (((value) &0xFF00000000) >> 8) | (((value) &0xFF0000000000) >> 24) | (((value) &0xFF000000000000) >> 40) | (((value) &0xFF00000000000000) >> 56));
Expand Down Expand Up @@ -389,3 +391,122 @@ void bitstring_copy_bits_incomplete_bytes(uint8_t *dst, size_t bits_offset, cons
}
*dst = dest_byte;
}

bool bitstring_extract_f32(
term src_bin, size_t offset, avm_int_t n, enum BitstringFlags bs_flags, avm_float_t *dst)
{
unsigned long capacity = term_binary_size(src_bin);
if (8 * capacity - offset < (unsigned long) n) {
return false;
}

if ((offset & 0x7) == 0) {
int byte_offset = offset >> 3;
const uint8_t *src = (const uint8_t *) term_binary_data(src_bin) + byte_offset;

union
{
uint32_t bits;
float fvalue;
} f32;

if (bs_flags & LittleEndianIntegerMask) {
f32.bits = READ_32LE_UNALIGNED(src);
} else {
f32.bits = READ_32_UNALIGNED(src);
}
*dst = f32.fvalue;
return true;
} else {
return false;
}
}

static bool bin64_to_f32(uint64_t bin64, float *out)
{
int sign = (int) ((bin64 >> 63) & 1);
int exponent = (int) ((bin64 >> 52) & 0x7FF) - 1023;
uint64_t mantissa = (bin64 >> 28) & 0xFFFFFF;

if (exponent == 1024) {
// not a number
return false;
} else if (exponent > 127) {
// too big
return false;
} else if (exponent < -150) {
// safely convert to 0
exponent = -127;
mantissa = 0;
} else if (exponent < -126) {
// denormalized
mantissa |= 0x1000000;
mantissa >>= (-126 - exponent);
exponent = -127;
}

mantissa = (mantissa + 1) >> 1;
if (mantissa & 0x800000) {
exponent += 1;
mantissa &= 0x7FFFFF;
mantissa >>= 1;
}

union
{
uint32_t u32;
float f32;
} temp;

temp.u32 = mantissa;
temp.u32 |= (uint32_t) (exponent + 127) << 23;
temp.u32 |= (uint32_t) sign << 31;

*out = temp.f32;

return false;
}

bool bitstring_extract_f64(
term src_bin, size_t offset, avm_int_t n, enum BitstringFlags bs_flags, avm_float_t *dst)
{
unsigned long capacity = term_binary_size(src_bin);
if (8 * capacity - offset < (unsigned long) n) {
return false;
}

if ((offset & 0x7) == 0) {
int byte_offset = offset >> 3;
const uint8_t *src = (const uint8_t *) term_binary_data(src_bin) + byte_offset;

uint64_t bin64;
if (bs_flags & LittleEndianIntegerMask) {
bin64 = READ_64LE_UNALIGNED(src);
} else {
bin64 = READ_64_UNALIGNED(src);
}

_Static_assert(
(sizeof(avm_float_t) == 4) || (sizeof(avm_float_t) == 8), "Unsupported double size");

if (sizeof(avm_float_t) == 8) {
union
{
uint64_t bits;
double fvalue;
} f64;

f64.bits = bin64;
*dst = f64.fvalue;
} else if (sizeof(avm_float_t) == 4) {
float f32;
bool res = bin64_to_f32(bin64, &f32);
*dst = f32;
return res;
}
return true;

} else {
return false;
}
}
5 changes: 5 additions & 0 deletions src/libAtomVM/bitstring.h
Original file line number Diff line number Diff line change
Expand Up @@ -541,6 +541,11 @@ static inline void bitstring_copy_bits(uint8_t *dst, size_t bits_offset, const u
}
}

bool bitstring_extract_f32(
term src_bin, size_t offset, avm_int_t n, enum BitstringFlags bs_flags, avm_float_t *dst);
bool bitstring_extract_f64(
term src_bin, size_t offset, avm_int_t n, enum BitstringFlags bs_flags, avm_float_t *dst);

#ifdef __cplusplus
}
#endif
Expand Down
1 change: 1 addition & 0 deletions src/libAtomVM/opcodes.h
Original file line number Diff line number Diff line change
Expand Up @@ -110,6 +110,7 @@
#define OP_IS_FUNCTION2 115
#define OP_BS_START_MATCH2 116
#define OP_BS_GET_INTEGER2 117
#define OP_BS_GET_FLOAT2 118
#define OP_BS_GET_BINARY2 119
#define OP_BS_SKIP_BITS2 120
#define OP_BS_TEST_TAIL2 121
Expand Down
68 changes: 68 additions & 0 deletions src/libAtomVM/opcodesswitch.h
Original file line number Diff line number Diff line change
Expand Up @@ -4761,6 +4761,74 @@ HOT_FUNC int scheduler_entry_point(GlobalContext *glb)
break;
}

case OP_BS_GET_FLOAT2: {
uint32_t fail;
DECODE_LABEL(fail, pc)
term src;
DECODE_COMPACT_TERM(src, pc);
term arg2;
DECODE_COMPACT_TERM(arg2, pc);
term size;
DECODE_COMPACT_TERM(size, pc);
uint32_t unit;
DECODE_LITERAL(unit, pc);
uint32_t flags_value;
DECODE_LITERAL(flags_value, pc);

#ifdef IMPL_CODE_LOADER
TRACE("bs_get_float2/7\n");
#endif

#ifdef IMPL_EXECUTE_LOOP
VERIFY_IS_MATCH_STATE(src, "bs_get_float");
VERIFY_IS_INTEGER(size, "bs_get_float");

avm_int_t size_val = term_to_int(size);

TRACE("bs_get_float2/7, fail=%u src=%p size=%u unit=%u flags=%x\n", (unsigned) fail, (void *) src, (unsigned) size_val, (unsigned) unit, (int) flags_value);

avm_int_t increment = size_val * unit;
avm_float_t value;
term bs_bin = term_get_match_state_binary(src);
avm_int_t bs_offset = term_get_match_state_offset(src);
bool status;
switch (size_val) {
case 32:
status = bitstring_extract_f32(bs_bin, bs_offset, increment, flags_value, &value);
break;
case 64:
status = bitstring_extract_f64(bs_bin, bs_offset, increment, flags_value, &value);
break;
default:
TRACE("bs_get_float2: error extracting float.\n");
status = false;
JUMP_TO_ADDRESS(mod->labels[fail]);
break;
}
if (UNLIKELY(!status)) {
TRACE("bs_get_float2: error extracting float.\n");
JUMP_TO_ADDRESS(mod->labels[fail]);
} else {
term_set_match_state_offset(src, bs_offset + increment);

if (UNLIKELY(memory_ensure_free_opt(ctx, FLOAT_SIZE, MEMORY_NO_GC) != MEMORY_GC_OK)) {
ctx->x[0] = ERROR_ATOM;
ctx->x[1] = OUT_OF_MEMORY_ATOM;
return term_invalid_term();
}
term t = term_from_float(value, &ctx->heap);
#endif

dreg_t dreg;
DECODE_DEST_REGISTER(dreg, pc);

#ifdef IMPL_EXECUTE_LOOP
WRITE_REGISTER(dreg, t);
}
#endif
break;
}

case OP_BS_GET_BINARY2: {
uint32_t fail;
DECODE_LABEL(fail, pc)
Expand Down

0 comments on commit 9ce1a1d

Please sign in to comment.