/* Miscellaneous routines making it easier to use GMP within GDB's framework. Copyright (C) 2019-2021 Free Software Foundation, Inc. This file is part of GDB. This program is free 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. 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/>. */ #ifndef GMP_UTILS_H #define GMP_UTILS_H #include "defs.h" /* Include <stdio.h> and <stdarg.h> ahead of <gmp.h>, so as to get access to GMP's various formatting functions. */ #include <stdio.h> #include <stdarg.h> #include <gmp.h> #include "gdbsupport/traits.h" /* Same as gmp_asprintf, but returning an std::string. */ std::string gmp_string_printf (const char *fmt, ...); /* A class to make it easier to use GMP's mpz_t values within GDB. */ struct gdb_mpz { mpz_t val; /* Constructors. */ gdb_mpz () { mpz_init (val); } explicit gdb_mpz (const mpz_t &from_val) { mpz_init (val); mpz_set (val, from_val); } gdb_mpz (const gdb_mpz &from) { mpz_init (val); mpz_set (val, from.val); } /* Initialize using the given integral value. The main advantage of this method is that it handles both signed and unsigned types, with no size restriction. */ template<typename T, typename = gdb::Requires<std::is_integral<T>>> explicit gdb_mpz (T src) { mpz_init (val); set (src); } explicit gdb_mpz (gdb_mpz &&from) { mpz_init (val); mpz_swap (val, from.val); } gdb_mpz &operator= (const gdb_mpz &from) { mpz_set (val, from.val); return *this; } gdb_mpz &operator= (gdb_mpz &&other) { mpz_swap (val, other.val); return *this; } template<typename T, typename = gdb::Requires<std::is_integral<T>>> gdb_mpz &operator= (T src) { set (src); return *this; } /* Convert VAL to an integer of the given type. The return type can signed or unsigned, with no size restriction. */ template<typename T> T as_integer () const; /* Set VAL by importing the number stored in the byte array (BUF), using the given BYTE_ORDER. The size of the data to read is the byte array's size. UNSIGNED_P indicates whether the number has an unsigned type. */ void read (gdb::array_view<const gdb_byte> buf, enum bfd_endian byte_order, bool unsigned_p); /* Write VAL into BUF as a number whose byte size is the size of BUF, using the given BYTE_ORDER. UNSIGNED_P indicates whether the number has an unsigned type. */ void write (gdb::array_view<gdb_byte> buf, enum bfd_endian byte_order, bool unsigned_p) const; /* Return a string containing VAL. */ std::string str () const { return gmp_string_printf ("%Zd", val); } /* The destructor. */ ~gdb_mpz () { mpz_clear (val); } private: /* Helper template for constructor and operator=. */ template<typename T> void set (T src); /* Low-level function to export VAL into BUF as a number whose byte size is the size of BUF. If UNSIGNED_P is true, then export VAL into BUF as an unsigned value. Otherwise, export it as a signed value. The API is inspired from GMP's mpz_export, hence the naming and types of the following parameter: - ENDIAN should be: . 1 for most significant byte first; or . -1 for least significant byte first; or . 0 for native endianness. An error is raised if BUF is not large enough to contain the value being exported. */ void safe_export (gdb::array_view<gdb_byte> buf, int endian, bool unsigned_p) const; }; /* A class to make it easier to use GMP's mpq_t values within GDB. */ struct gdb_mpq { mpq_t val; /* Constructors. */ gdb_mpq () { mpq_init (val); } explicit gdb_mpq (const mpq_t &from_val) { mpq_init (val); mpq_set (val, from_val); } gdb_mpq (const gdb_mpq &from) { mpq_init (val); mpq_set (val, from.val); } explicit gdb_mpq (gdb_mpq &&from) { mpq_init (val); mpq_swap (val, from.val); } /* Copy assignment operator. */ gdb_mpq &operator= (const gdb_mpq &from) { mpq_set (val, from.val); return *this; } gdb_mpq &operator= (gdb_mpq &&from) { mpq_swap (val, from.val); return *this; } /* Return a string representing VAL as "<numerator> / <denominator>". */ std::string str () const { return gmp_string_printf ("%Qd", val); } /* Return VAL rounded to the nearest integer. */ gdb_mpz get_rounded () const; /* Set VAL from the contents of the given byte array (BUF), which contains the unscaled value of a fixed point type object. The byte size of the data is the size of BUF. BYTE_ORDER provides the byte_order to use when reading the data. UNSIGNED_P indicates whether the number has an unsigned type. SCALING_FACTOR is the scaling factor to apply after having read the unscaled value from our buffer. */ void read_fixed_point (gdb::array_view<const gdb_byte> buf, enum bfd_endian byte_order, bool unsigned_p, const gdb_mpq &scaling_factor); /* Write VAL into BUF as fixed point value following the given BYTE_ORDER. The size of BUF is used as the length to write the value into. UNSIGNED_P indicates whether the number has an unsigned type. SCALING_FACTOR is the scaling factor to apply before writing the unscaled value to our buffer. */ void write_fixed_point (gdb::array_view<gdb_byte> buf, enum bfd_endian byte_order, bool unsigned_p, const gdb_mpq &scaling_factor) const; /* The destructor. */ ~gdb_mpq () { mpq_clear (val); } }; /* A class to make it easier to use GMP's mpf_t values within GDB. Should MPFR become a required dependency, we should probably drop this class in favor of using MPFR. */ struct gdb_mpf { mpf_t val; /* Constructors. */ gdb_mpf () { mpf_init (val); } DISABLE_COPY_AND_ASSIGN (gdb_mpf); /* Set VAL from the contents of the given buffer (BUF), which contains the unscaled value of a fixed point type object with the given size (LEN) and byte order (BYTE_ORDER). UNSIGNED_P indicates whether the number has an unsigned type. SCALING_FACTOR is the scaling factor to apply after having read the unscaled value from our buffer. */ void read_fixed_point (gdb::array_view<const gdb_byte> buf, enum bfd_endian byte_order, bool unsigned_p, const gdb_mpq &scaling_factor) { gdb_mpq tmp_q; tmp_q.read_fixed_point (buf, byte_order, unsigned_p, scaling_factor); mpf_set_q (val, tmp_q.val); } /* The destructor. */ ~gdb_mpf () { mpf_clear (val); } }; /* See declaration above. */ template<typename T> void gdb_mpz::set (T src) { mpz_import (val, 1 /* count */, -1 /* order */, sizeof (T) /* size */, 0 /* endian (0 = native) */, 0 /* nails */, &src /* op */); if (std::is_signed<T>::value && src < 0) { /* mpz_import does not handle the sign, so our value was imported as an unsigned. Adjust that imported value so as to make it the correct negative value. */ gdb_mpz neg_offset; mpz_ui_pow_ui (neg_offset.val, 2, sizeof (T) * HOST_CHAR_BIT); mpz_sub (val, val, neg_offset.val); } } /* See declaration above. */ template<typename T> T gdb_mpz::as_integer () const { T result; this->safe_export ({(gdb_byte *) &result, sizeof (result)}, 0 /* endian (0 = native) */, !std::is_signed<T>::value /* unsigned_p */); return result; } #endif