mirror of
https://github.com/nasa/trick.git
synced 2025-01-15 01:09:59 +00:00
7779dc568b
I made a number of changes to Trick to make it compile clean with clang++ -Wdocumentation, which checks doxygen comments. The changes were either to comments or to parameter names in function declarations. None of the changes have any impact on the compiled code.
598 lines
10 KiB
C++
598 lines
10 KiB
C++
/**
|
|
* @if Er7UtilsUseGroups
|
|
* @addtogroup Er7Utils
|
|
* @{
|
|
* @addtogroup Utils
|
|
* @{
|
|
* @endif
|
|
*/
|
|
|
|
/**
|
|
* @file
|
|
* Defines the class Ratio128.
|
|
*/
|
|
|
|
/*
|
|
Purpose: ()
|
|
*/
|
|
|
|
|
|
#ifndef ER7_UTILS_RATIO128_HH
|
|
#define ER7_UTILS_RATIO128_HH
|
|
|
|
// Local includes
|
|
#include "uint128.hh"
|
|
|
|
// Interface includes
|
|
#include "er7_utils/interface/include/er7_class.hh"
|
|
|
|
// System includes
|
|
#include <iosfwd>
|
|
#include <stdint.h>
|
|
#include <string>
|
|
|
|
|
|
namespace er7_utils {
|
|
|
|
|
|
/**
|
|
* Represents a rational number as a sign times the ratio of two unsigned
|
|
* 128 bit integers.
|
|
*/
|
|
class Ratio128 {
|
|
ER7_UTILS_MAKE_SIM_INTERFACES(Ratio128)
|
|
|
|
private:
|
|
// Data members
|
|
|
|
/**
|
|
* The sign of the rational.
|
|
*/
|
|
int sign; //!< trick_units(--)
|
|
|
|
/**
|
|
* The numerator.
|
|
*/
|
|
UInt128 num; //!< trick_units(--)
|
|
|
|
/**
|
|
* The denominator.
|
|
*/
|
|
UInt128 den; //!< trick_units(--)
|
|
|
|
public:
|
|
|
|
/**
|
|
* Default constructor; default value is zero.
|
|
*/
|
|
Ratio128 ()
|
|
:
|
|
sign (0),
|
|
num (0ull),
|
|
den (1ull)
|
|
{ }
|
|
|
|
/**
|
|
* Conversion constructor, unsigned long long to rational,
|
|
* plus optional denominator and sign.
|
|
*/
|
|
Ratio128 (
|
|
unsigned long long num_in,
|
|
unsigned long long den_in=1ull,
|
|
int sign_in=1)
|
|
:
|
|
sign ((num_in == 0ull) ? 0 : ((sign_in < 0) ? -1 : 1)),
|
|
num (num_in),
|
|
den (den_in)
|
|
{
|
|
reduce ();
|
|
}
|
|
|
|
/**
|
|
* Conversion constructor, unsigned long to rational,
|
|
* plus optional denominator and sign.
|
|
*/
|
|
Ratio128 (
|
|
unsigned long num_in,
|
|
unsigned long den_in=1ull,
|
|
int sign_in=1)
|
|
:
|
|
sign ((num_in == 0ul) ? 0 : ((sign_in < 0) ? -1 : 1)),
|
|
num (num_in),
|
|
den (den_in)
|
|
{
|
|
reduce ();
|
|
}
|
|
|
|
/**
|
|
* Conversion constructor, unsigned to rational, plus optional denominator
|
|
* and sign.
|
|
*/
|
|
Ratio128 (
|
|
unsigned int num_in,
|
|
unsigned int den_in=1u,
|
|
int sign_in=1)
|
|
:
|
|
sign ((num_in == 0ull) ? 0 : ((sign_in < 0) ? -1 : 1)),
|
|
num (num_in),
|
|
den (den_in)
|
|
{
|
|
reduce ();
|
|
}
|
|
|
|
/**
|
|
* Conversion constructor, long long to rational, plus optional denominator.
|
|
*/
|
|
Ratio128 (
|
|
long long num_in,
|
|
long long den_in=1LL)
|
|
:
|
|
sign (0),
|
|
num (0ull),
|
|
den (1ull)
|
|
{
|
|
make_ratio (num_in, den_in);
|
|
}
|
|
|
|
/**
|
|
* Conversion constructor, long to rational, plus optional denominator.
|
|
*/
|
|
Ratio128 (
|
|
long num_in,
|
|
long den_in=1L)
|
|
:
|
|
sign (0),
|
|
num (0ull),
|
|
den (1ull)
|
|
{
|
|
make_ratio (num_in, den_in);
|
|
}
|
|
|
|
/**
|
|
* Conversion constructor, int to rational, plus optional denominator.
|
|
*/
|
|
Ratio128 (
|
|
int num_in,
|
|
int den_in=1LL)
|
|
:
|
|
sign (0),
|
|
num (0ull),
|
|
den (1ull)
|
|
{
|
|
make_ratio (num_in, den_in);
|
|
}
|
|
|
|
/**
|
|
* Conversion constructor, double to rational.
|
|
* @throw std::domain_error Input value cannot be represented exactly.
|
|
*/
|
|
explicit Ratio128 (double dval);
|
|
|
|
|
|
// Swap, as a member and as a function.
|
|
void swap (Ratio128& other);
|
|
|
|
friend void swap (Ratio128& a, Ratio128& b)
|
|
{
|
|
a.swap(b);
|
|
}
|
|
|
|
|
|
// Unary operators. Only + and - are defined.
|
|
|
|
/**
|
|
* Unary plus operator; returns copy.
|
|
*/
|
|
friend Ratio128 operator+ (const Ratio128& a)
|
|
{
|
|
return a;
|
|
}
|
|
|
|
/**
|
|
* Unary minus (negation) operator.
|
|
*/
|
|
friend Ratio128 operator- (const Ratio128& a)
|
|
{
|
|
Ratio128 negated (a);
|
|
negated.sign = -a.sign;
|
|
return negated;
|
|
}
|
|
|
|
|
|
// Conversion operators.
|
|
// FIXME (Future) Make these explicit upon transition to C++11.
|
|
|
|
/**
|
|
* Convert to double.
|
|
*/
|
|
operator double () const;
|
|
|
|
/**
|
|
* Convert to std::string, output is "[-]<numerator>/<denominator>".
|
|
*/
|
|
operator std::string () const;
|
|
|
|
|
|
// Math functions.
|
|
|
|
/**
|
|
* Absolute value.
|
|
*/
|
|
Ratio128 abs () const {
|
|
Ratio128 result = *this;
|
|
result.sign = (sign == 0) ? 0 : 1;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Multiplicative inverse (additive inverse is operator-).
|
|
*/
|
|
Ratio128 inverse () const;
|
|
|
|
/**
|
|
* Round to half precision by shifting both the numerator and denominator
|
|
* one bit to the right, adjusting the numerator to best maintain accuracy.
|
|
*/
|
|
void round ();
|
|
|
|
|
|
// Math operators.
|
|
// These are implemented in two parts, the arithmetic assignment operators
|
|
// and the binary arithmetic operators. The former are member functions;
|
|
// the latter are friend functions implemented in terms of the
|
|
// arithmetic assignment operators.
|
|
// Note: There are no increment operators. This is intentional.
|
|
|
|
/**
|
|
* Addition.
|
|
*/
|
|
Ratio128 & operator+= (const Ratio128 & other);
|
|
|
|
/**
|
|
* Subtraction.
|
|
*/
|
|
Ratio128 & operator-= (const Ratio128 & other);
|
|
|
|
/**
|
|
* Multiplication.
|
|
*/
|
|
Ratio128 & operator*= (const Ratio128 & other);
|
|
|
|
/**
|
|
* Division.
|
|
*/
|
|
Ratio128 & operator/= (const Ratio128 & other)
|
|
{
|
|
*this *= other.inverse();
|
|
return *this;
|
|
}
|
|
|
|
|
|
// Arithmetic operators.
|
|
|
|
/**
|
|
* Returns a+b.
|
|
*/
|
|
friend Ratio128 operator+ (const Ratio128 & a, const Ratio128 & b)
|
|
{
|
|
Ratio128 result = a;
|
|
result += b;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Returns a+b.
|
|
*/
|
|
template<typename T>
|
|
friend Ratio128 operator+ (T a, const Ratio128 & b)
|
|
{
|
|
Ratio128 result = Ratio128 (a);
|
|
result += b;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Returns a+b.
|
|
*/
|
|
template<typename T>
|
|
friend Ratio128 operator+ (const Ratio128 & a, T b)
|
|
{
|
|
Ratio128 result = a;
|
|
result += Ratio128(b);
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Returns a-b.
|
|
*/
|
|
friend Ratio128 operator- (const Ratio128 & a, const Ratio128 & b)
|
|
{
|
|
Ratio128 result = a;
|
|
result -= b;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Returns a-b.
|
|
*/
|
|
template<typename T>
|
|
friend Ratio128 operator- (T a, const Ratio128 & b)
|
|
{
|
|
Ratio128 result = Ratio128 (a);
|
|
result -= b;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Returns a-b.
|
|
*/
|
|
template<typename T>
|
|
friend Ratio128 operator- (const Ratio128 & a, T b)
|
|
{
|
|
Ratio128 result = a;
|
|
result -= Ratio128(b);
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Returns a*b.
|
|
*/
|
|
friend Ratio128 operator* (const Ratio128 & a, const Ratio128 & b)
|
|
{
|
|
Ratio128 result = a;
|
|
result *= b;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Returns a*b.
|
|
*/
|
|
template<typename T>
|
|
friend Ratio128 operator* (T a, const Ratio128 & b)
|
|
{
|
|
Ratio128 result = Ratio128 (a);
|
|
result *= b;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Returns a*b.
|
|
*/
|
|
template<typename T>
|
|
friend Ratio128 operator* (const Ratio128 & a, T b)
|
|
{
|
|
Ratio128 result = a;
|
|
result *= Ratio128(b);
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Returns a/b.
|
|
*/
|
|
friend Ratio128 operator/ (const Ratio128 & a, const Ratio128 & b)
|
|
{
|
|
Ratio128 result = a;
|
|
result /= b;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Returns a/b.
|
|
*/
|
|
template<typename T>
|
|
friend Ratio128 operator/ (T a, const Ratio128 & b)
|
|
{
|
|
Ratio128 result = Ratio128 (a);
|
|
result /= b;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Returns a/b.
|
|
*/
|
|
template<typename T>
|
|
friend Ratio128 operator/ (const Ratio128 & a, T b)
|
|
{
|
|
Ratio128 result = a;
|
|
result /= Ratio128(b);
|
|
return result;
|
|
}
|
|
|
|
|
|
// I/O stream operators.
|
|
|
|
/**
|
|
* Stream insertion.
|
|
*/
|
|
friend std::ostream & operator<< (std::ostream& stream, const Ratio128& rat);
|
|
|
|
|
|
// Comparison operators.
|
|
|
|
/**
|
|
* Returns a<b.
|
|
*/
|
|
friend bool operator< (const Ratio128 & a, const Ratio128 & b);
|
|
|
|
/**
|
|
* Returns a<b.
|
|
*/
|
|
template<typename T>
|
|
friend bool operator< (T a, const Ratio128 & b)
|
|
{
|
|
return Ratio128(a) < b;
|
|
}
|
|
|
|
/**
|
|
* Returns a<b.
|
|
*/
|
|
template<typename T>
|
|
friend bool operator< (const Ratio128 & a, T b)
|
|
{
|
|
return a < Ratio128(b);
|
|
}
|
|
|
|
|
|
/**
|
|
* Returns a>=b.
|
|
*/
|
|
friend bool operator>= (const Ratio128 & a, const Ratio128 & b)
|
|
{
|
|
return !(a < b);
|
|
}
|
|
|
|
/**
|
|
* Returns a>=b.
|
|
*/
|
|
template<typename T>
|
|
friend bool operator>= (T a, const Ratio128 & b)
|
|
{
|
|
return Ratio128(a) >= b;
|
|
}
|
|
|
|
/**
|
|
* Returns a>=b.
|
|
*/
|
|
template<typename T>
|
|
friend bool operator>= (const Ratio128 & a, T b)
|
|
{
|
|
return a >= Ratio128(b);
|
|
}
|
|
|
|
|
|
/**
|
|
* Returns a<=b.
|
|
*/
|
|
friend bool operator<= (const Ratio128 & a, const Ratio128 & b);
|
|
|
|
/**
|
|
* Returns a<=b.
|
|
*/
|
|
template<typename T>
|
|
friend bool operator<= (T a, const Ratio128 & b)
|
|
{
|
|
return Ratio128(a) <= b;
|
|
}
|
|
|
|
/**
|
|
* Returns a<=b.
|
|
*/
|
|
template<typename T>
|
|
friend bool operator<= (const Ratio128 & a, T b)
|
|
{
|
|
return a <= Ratio128(b);
|
|
}
|
|
|
|
|
|
/**
|
|
* Returns a>b.
|
|
*/
|
|
friend bool operator> (const Ratio128 & a, const Ratio128 & b)
|
|
{
|
|
return !(a <= b);
|
|
}
|
|
|
|
/**
|
|
* Returns a>b.
|
|
*/
|
|
template<typename T>
|
|
friend bool operator> (T a, const Ratio128 & b)
|
|
{
|
|
return Ratio128(a) > b;
|
|
}
|
|
|
|
/**
|
|
* Returns a>b.
|
|
*/
|
|
template<typename T>
|
|
friend bool operator> (const Ratio128 & a, T b)
|
|
{
|
|
return a > Ratio128(b);
|
|
}
|
|
|
|
|
|
/**
|
|
* Returns a==b.
|
|
* Note: Implementation assumes a and b are normalized (reduced).
|
|
*/
|
|
friend bool operator== (const Ratio128 & a, const Ratio128 & b)
|
|
{
|
|
return (a.sign == b.sign) && (a.num == b.num) && (a.den == b.den);
|
|
}
|
|
|
|
/**
|
|
* Returns a==b.
|
|
*/
|
|
template<typename T>
|
|
friend bool operator== (T a, const Ratio128 & b)
|
|
{
|
|
return Ratio128(a) == b;
|
|
}
|
|
|
|
/**
|
|
* Returns a==b.
|
|
*/
|
|
template<typename T>
|
|
friend bool operator== (const Ratio128 & a, T b)
|
|
{
|
|
return a == Ratio128(b);
|
|
}
|
|
|
|
|
|
/**
|
|
* Returns a!=b.
|
|
*/
|
|
friend bool operator!= (const Ratio128 & a, const Ratio128 & b)
|
|
{
|
|
return !(a == b);
|
|
}
|
|
|
|
/**
|
|
* Returns a!=b.
|
|
*/
|
|
template<typename T>
|
|
friend bool operator!= (T a, const Ratio128 & b)
|
|
{
|
|
return Ratio128(a) != b;
|
|
}
|
|
|
|
/**
|
|
* Returns a!=b.
|
|
*/
|
|
template<typename T>
|
|
friend bool operator!= (const Ratio128 & a, T b)
|
|
{
|
|
return a != Ratio128(b);
|
|
}
|
|
|
|
|
|
private:
|
|
|
|
/**
|
|
* Constructs a rational from two signed integers.
|
|
*/
|
|
void make_ratio (int64_t num_in, int64_t den_in);
|
|
|
|
/**
|
|
* Reduces a rational to canonical form.
|
|
*/
|
|
void reduce ();
|
|
|
|
/**
|
|
* Returns a rough (1.5 ULP) estimate of the double precision equivalent.
|
|
*/
|
|
double to_double () const;
|
|
};
|
|
|
|
|
|
}
|
|
|
|
|
|
#endif
|
|
/**
|
|
* @if Er7UtilsUseGroups
|
|
* @}
|
|
* @}
|
|
* @endif
|
|
*/
|