diff --git a/src/libtriton/CMakeLists.txt b/src/libtriton/CMakeLists.txt index f8b664618..dc82de0cc 100644 --- a/src/libtriton/CMakeLists.txt +++ b/src/libtriton/CMakeLists.txt @@ -69,6 +69,7 @@ set(LIBTRITON_SOURCE_FILES stubs/x8664-ms-libc.cpp stubs/x8664-systemv-libc.cpp utils/coreUtils.cpp + utils/softfloat.cpp ) # Define all header files @@ -106,6 +107,7 @@ set(LIBTRITON_HEADER_FILES includes/triton/comparableFunctor.hpp includes/triton/context.hpp includes/triton/coreUtils.hpp + includes/triton/softfloat.hpp includes/triton/cpuInterface.hpp includes/triton/cpuSize.hpp includes/triton/dllexport.hpp @@ -157,6 +159,10 @@ set(LIBTRITON_HEADER_FILES includes/triton/z3ToTriton.hpp ) +set_source_files_properties(utils/softfloat.cpp PROPERTIES COMPILE_DEFINITIONS + ${CMAKE_CXX_BYTE_ORDER} +) + # Define all resource files set(LIBTRITON_RESOURCE_FILES includes/triton/version.hpp.in diff --git a/src/libtriton/arch/arm/aarch64/aarch64Cpu.cpp b/src/libtriton/arch/arm/aarch64/aarch64Cpu.cpp index d35fb47b2..e42f2c610 100644 --- a/src/libtriton/arch/arm/aarch64/aarch64Cpu.cpp +++ b/src/libtriton/arch/arm/aarch64/aarch64Cpu.cpp @@ -536,56 +536,7 @@ namespace triton { throw triton::exceptions::Disassembly("Aarch64Cpu::disassembly(): Cannot correctly decode FP operand"); } - auto encode_fp_imm = [size](double fp_value) { - static_assert(sizeof(float) == sizeof(triton::uint32), "Unexpected float type size"); - static_assert(sizeof(double) == sizeof(triton::uint64), "Unexpected double type size"); - - auto IEEE754_f32_to_f16 = [](float value) -> uint16_t { - uint32_t f; - std::memcpy(&f, &value, sizeof(uint32_t)); - uint16_t sign = (f >> 16) & 0x8000; - int16_t exponent = ((f >> 23) & 0xff) - 127 + 15; - uint16_t mantissa = (f >> 13) & 0x3ff; - - if (exponent <= 0) { - if (exponent < -10) { - return sign; - } - mantissa = (mantissa | 0x400) >> (1 - exponent); - return sign | mantissa; - } else if (exponent == 0xff - (127 - 15)) { - if (mantissa) { - return sign | 0x7fff; - } else { - return sign | 0x7c00; - } - } else if (exponent > 30) { - return sign | 0x7c00; - } - return sign | (exponent << 10) | mantissa; - }; - - if (size == sizeof(double)) { - triton::uint64 result; - std::memcpy(&result, &fp_value, sizeof(double)); - return result; - } - else if (size == sizeof(float)) { - float converted = static_cast(fp_value); - triton::uint32 conv_repr; - std::memcpy(&conv_repr, &converted, sizeof(float)); - // just zero extended value - return static_cast(conv_repr); - } - else if (size == 2) { // half-precision - float value = static_cast(fp_value); - return static_cast(IEEE754_f32_to_f16(value)); - } - - throw triton::exceptions::Disassembly("AArch64Cpu::disassembly(): Invalid operand."); - }; - - Immediate imm{encode_fp_imm(op->fp), size}; + Immediate imm{op->fp, size, this->getEndianness()}; /* Set Shift type and value */ imm.setShiftType(this->capstoneShiftToTritonShift(op->shift.type)); diff --git a/src/libtriton/arch/immediate.cpp b/src/libtriton/arch/immediate.cpp index 301e25a8f..a3493d2cc 100644 --- a/src/libtriton/arch/immediate.cpp +++ b/src/libtriton/arch/immediate.cpp @@ -5,10 +5,17 @@ ** This program is under the terms of the Apache License 2.0. */ +#include #include #include -#include +#include +#include +#ifdef LITTLE_ENDIAN // provided by CMake +constexpr auto sys_endianness = triton::arch::LE_ENDIANNESS; +#else +constexpr auto sys_endianness = triton::arch::BE_ENDIANNESS; +#endif namespace triton { @@ -23,6 +30,42 @@ namespace triton { this->setValue(value, size); } + Immediate::Immediate(double value, triton::uint32 size /* bytes */, triton::arch::endianness_e platform_endianness) { + triton::uint64 imm_value; + + auto need_swap = sys_endianness != platform_endianness; + + if (size == sizeof(double)) { + static_assert(sizeof(double) == sizeof(triton::uint64), + "Unexpected double type size"); + std::memcpy(&imm_value, &value, sizeof(double)); + if (need_swap) { + imm_value = utils::byteswap(imm_value); + } + } + else if (size == sizeof(float)) { // single-precision + float fvalue = static_cast(value); + triton::uint32 repr; + static_assert(sizeof(float) == sizeof(uint32_t), + "Unexpected float type size"); + std::memcpy(&repr, &fvalue, sizeof(float)); + + imm_value = need_swap ? static_cast(utils::byteswap(repr)) + : static_cast(repr); + } else if (size == 2) { // half-precision + float fvalue = static_cast(value); + triton::uint16 repr = sf::f32_to_f16(fvalue); + imm_value = need_swap ? static_cast(utils::byteswap(repr)) + : static_cast(repr); + + } + else { + throw triton::exceptions::Immediate("Immediate::Immediate(double): Invalid encoding size."); + } + + this->setValue(imm_value, size); + } + Immediate::Immediate(const Immediate& other) : BitsVector(other), diff --git a/src/libtriton/includes/triton/coreUtils.hpp b/src/libtriton/includes/triton/coreUtils.hpp index 9fdec4d62..660523cb9 100644 --- a/src/libtriton/includes/triton/coreUtils.hpp +++ b/src/libtriton/includes/triton/coreUtils.hpp @@ -10,6 +10,7 @@ #include #include +#include #include #include @@ -81,6 +82,19 @@ namespace triton { template <> TRITON_EXPORT triton::uint80 cast(const triton::uint512& value); template <> TRITON_EXPORT triton::uint512 cast(const triton::uint80& value); + template + std::enable_if_t< + std::is_unsigned_v, + T> + byteswap(T value) { + std::array repr; + std::memcpy(&repr, &value, sizeof(value)); + std::reverse(repr.begin(), repr.end()); + T result; + std::memcpy(&result, &repr, sizeof(result)); + return result; + } + /*! @} End of utils namespace */ }; /*! @} End of triton namespace */ diff --git a/src/libtriton/includes/triton/immediate.hpp b/src/libtriton/includes/triton/immediate.hpp index 9db4a9301..29f94a9c5 100644 --- a/src/libtriton/includes/triton/immediate.hpp +++ b/src/libtriton/includes/triton/immediate.hpp @@ -50,6 +50,9 @@ namespace triton { //! Constructor. TRITON_EXPORT Immediate(triton::uint64 value, triton::uint32 size /* bytes*/); + //! Constructor. + TRITON_EXPORT Immediate(double value, triton::uint32 size /* bytes */, triton::arch::endianness_e platform_endianness); + //! Constructor by copy. TRITON_EXPORT Immediate(const Immediate& other); diff --git a/src/libtriton/includes/triton/softfloat.hpp b/src/libtriton/includes/triton/softfloat.hpp new file mode 100644 index 000000000..6c47fa127 --- /dev/null +++ b/src/libtriton/includes/triton/softfloat.hpp @@ -0,0 +1,34 @@ +//! \file +/* +** Copyright (C) - Triton +** +** This program is under the terms of the Apache License 2.0. +*/ + +#ifndef TRITON_SOFTFLOAT_HPP +#define TRITON_SOFTFLOAT_HPP + +#include + +//! The Triton namespace +namespace triton { +/*! + * \addtogroup triton + * @{ + */ + //! The Softfloat namespace + namespace sf { + /*! + * \ingroup triton + * \addtogroup softfloat + * @{ + */ + + //! Cast 32-bit floating point value to 16-bit according to IEEE-754 + auto f32_to_f16(float value) -> uint16_t; + + } + +} + +#endif /* TRITON_SOFTFLOAT_HPP */ \ No newline at end of file diff --git a/src/libtriton/utils/softfloat.cpp b/src/libtriton/utils/softfloat.cpp new file mode 100644 index 000000000..8781b5d4a --- /dev/null +++ b/src/libtriton/utils/softfloat.cpp @@ -0,0 +1,42 @@ +//! \file +/* +** Copyright (C) - Triton +** +** This program is under the terms of the Apache License 2.0. +*/ + +#include + +#include + +namespace triton { + namespace sf { + + auto f32_to_f16(float value) -> uint16_t { + uint32_t f; + static_assert(sizeof(float) == sizeof(uint32_t), + "Unexpected float type size"); + std::memcpy(&f, &value, sizeof(uint32_t)); + uint16_t sign = (f >> 16) & 0x8000; + int16_t exponent = ((f >> 23) & 0xff) - 127 + 15; + uint16_t mantissa = (f >> 13) & 0x3ff; + if (exponent <= 0) { + if (exponent < -10) { + return sign; + } + mantissa = (mantissa | 0x400) >> (1 - exponent); + return sign | mantissa; + } else if (exponent == 0xff - (127 - 15)) { + if (mantissa) { + return sign | 0x7fff; + } else { + return sign | 0x7c00; + } + } else if (exponent > 30) { + return sign | 0x7c00; + } + return sign | (exponent << 10) | mantissa; + } + + } /* sf namespace */ +} /* triton namespace */ \ No newline at end of file