do all trig calculations in degrees
This commit is contained in:
parent
1e6d786482
commit
5bcfb8a371
@ -85,9 +85,9 @@ const dec80 DECN_LN_10 = {
|
|||||||
0, {23, 2, 58, 50, 92, 99, 40, 45, 68}
|
0, {23, 2, 58, 50, 92, 99, 40, 45, 68}
|
||||||
};
|
};
|
||||||
|
|
||||||
// 2 pi
|
// pi
|
||||||
const dec80 DECN_2PI = {
|
const dec80 DECN_PI = {
|
||||||
0, {62, 83, 18, 53, 7, 17, 95, 86, 48}
|
0, {31, 41, 59, 26, 53, 58, 97, 93, 24}
|
||||||
};
|
};
|
||||||
|
|
||||||
// pi/2
|
// pi/2
|
||||||
@ -1426,8 +1426,8 @@ void sqrt_decn(void){
|
|||||||
#endif //USE_POW_SQRT_IMPL
|
#endif //USE_POW_SQRT_IMPL
|
||||||
|
|
||||||
|
|
||||||
// see W.E. Egbert, "Personal Calculator Algorithms II: Trigonometric functions"
|
// normal angle to between 0 and 360 degrees
|
||||||
void project_decn_into_0_2pi(void) {
|
void normalize_0_360(void) {
|
||||||
const uint8_t is_negative = (AccDecn.exponent < 0);
|
const uint8_t is_negative = (AccDecn.exponent < 0);
|
||||||
exp_t exponent;
|
exp_t exponent;
|
||||||
|
|
||||||
@ -1436,11 +1436,14 @@ void project_decn_into_0_2pi(void) {
|
|||||||
negate_decn(&AccDecn);
|
negate_decn(&AccDecn);
|
||||||
}
|
}
|
||||||
exponent = get_exponent(&AccDecn);
|
exponent = get_exponent(&AccDecn);
|
||||||
copy_decn(&BDecn, &DECN_2PI);
|
//B = 360
|
||||||
|
set_dec80_zero(&BDecn);
|
||||||
|
BDecn.lsu[0] = 36;
|
||||||
|
BDecn.exponent = 2;
|
||||||
if (compare_magn() > 0) {
|
if (compare_magn() > 0) {
|
||||||
do {
|
do {
|
||||||
do {
|
do {
|
||||||
copy_decn(&BDecn, &DECN_2PI);
|
//B = 3.6e...
|
||||||
BDecn.exponent = exponent;
|
BDecn.exponent = exponent;
|
||||||
if (compare_magn() >= 0) {
|
if (compare_magn() >= 0) {
|
||||||
negate_decn(&BDecn);
|
negate_decn(&BDecn);
|
||||||
@ -1450,12 +1453,13 @@ void project_decn_into_0_2pi(void) {
|
|||||||
}
|
}
|
||||||
} while (1);
|
} while (1);
|
||||||
exponent--;
|
exponent--;
|
||||||
} while (exponent >= 0);
|
} while (exponent >= 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (is_negative) {
|
if (is_negative) {
|
||||||
negate_decn(&AccDecn);
|
negate_decn(&AccDecn);
|
||||||
copy_decn(&BDecn, &DECN_2PI);
|
//B = 360
|
||||||
|
BDecn.exponent = 2;
|
||||||
add_decn();
|
add_decn();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1467,13 +1471,14 @@ void project_decn_into_0_2pi(void) {
|
|||||||
#define THETA Tmp4Decn
|
#define THETA Tmp4Decn
|
||||||
void sincos_decn(const uint8_t sincos_arctan) {
|
void sincos_decn(const uint8_t sincos_arctan) {
|
||||||
const uint8_t is_negative = AccDecn.exponent < 0;
|
const uint8_t is_negative = AccDecn.exponent < 0;
|
||||||
if (sincos_arctan) {
|
if (sincos_arctan) { //calculate arctan
|
||||||
set_dec80_zero(&THETA);
|
set_dec80_zero(&THETA);
|
||||||
if (is_negative) negate_decn(&AccDecn);
|
if (is_negative) negate_decn(&AccDecn);
|
||||||
copy_decn(&COS, &AccDecn);
|
copy_decn(&COS, &AccDecn);
|
||||||
set_decn_one(&SIN);
|
set_decn_one(&SIN);
|
||||||
} else {
|
} else { //calculate sin/cos
|
||||||
project_decn_into_0_2pi();
|
normalize_0_360();
|
||||||
|
to_radian_decn();
|
||||||
copy_decn(&THETA, &AccDecn);
|
copy_decn(&THETA, &AccDecn);
|
||||||
set_decn_one(&COS);
|
set_decn_one(&COS);
|
||||||
set_dec80_zero(&SIN);
|
set_dec80_zero(&SIN);
|
||||||
@ -1482,13 +1487,13 @@ void sincos_decn(const uint8_t sincos_arctan) {
|
|||||||
negate_decn(&SIN);
|
negate_decn(&SIN);
|
||||||
}
|
}
|
||||||
do {
|
do {
|
||||||
if (sincos_arctan) {
|
if (sincos_arctan) { //calculate arctan
|
||||||
// THETA is in AccDecn from previous iteration
|
// THETA is in AccDecn from previous iteration
|
||||||
if (COS.exponent < 0) {
|
if (COS.exponent < 0) {
|
||||||
if (is_negative) negate_decn(&AccDecn);
|
if (is_negative) negate_decn(&AccDecn);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
} else {
|
} else { //calculate sin/cos
|
||||||
if (THETA.exponent < 0) {
|
if (THETA.exponent < 0) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -1539,10 +1544,11 @@ void tan_decn(void) {
|
|||||||
|
|
||||||
void arctan_decn(void) {
|
void arctan_decn(void) {
|
||||||
sincos_decn(1);
|
sincos_decn(1);
|
||||||
|
to_degree_decn();
|
||||||
}
|
}
|
||||||
|
|
||||||
// see W.E. Egbert, "Personal Calculator Algorithms III: Inverse Trigonometric Functions"
|
// see W.E. Egbert, "Personal Calculator Algorithms III: Inverse Trigonometric Functions"
|
||||||
void arcsin_decn(void) {
|
void arcsin_decn_rad(void) {
|
||||||
st_push_decn(&AccDecn);
|
st_push_decn(&AccDecn);
|
||||||
copy_decn(&BDecn, &AccDecn);
|
copy_decn(&BDecn, &AccDecn);
|
||||||
mult_decn();
|
mult_decn();
|
||||||
@ -1553,15 +1559,20 @@ void arcsin_decn(void) {
|
|||||||
recip_decn();
|
recip_decn();
|
||||||
st_pop_decn(&BDecn);
|
st_pop_decn(&BDecn);
|
||||||
mult_decn();
|
mult_decn();
|
||||||
|
|
||||||
sincos_decn(1);
|
sincos_decn(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void arcsin_decn(void) {
|
||||||
|
arcsin_decn_rad();
|
||||||
|
to_degree_decn();
|
||||||
|
}
|
||||||
|
|
||||||
void arccos_decn(void) {
|
void arccos_decn(void) {
|
||||||
arcsin_decn();
|
arcsin_decn_rad();
|
||||||
negate_decn(&AccDecn);
|
negate_decn(&AccDecn);
|
||||||
copy_decn(&BDecn, &DECN_PI2);
|
copy_decn(&BDecn, &DECN_PI2);
|
||||||
add_decn();
|
add_decn();
|
||||||
|
to_degree_decn();
|
||||||
}
|
}
|
||||||
#undef SIN
|
#undef SIN
|
||||||
#undef COS
|
#undef COS
|
||||||
@ -1579,8 +1590,7 @@ void to_radian_decn(void) {
|
|||||||
|
|
||||||
void pi_decn(void) {
|
void pi_decn(void) {
|
||||||
set_dec80_zero(&BDecn);
|
set_dec80_zero(&BDecn);
|
||||||
BDecn.lsu[0] = 5; // 0.5 00 ..
|
copy_decn(&AccDecn, &DECN_PI);
|
||||||
copy_decn(&AccDecn, &DECN_2PI);
|
|
||||||
mult_decn();
|
mult_decn();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -22,6 +22,8 @@
|
|||||||
namespace bmp = boost::multiprecision;
|
namespace bmp = boost::multiprecision;
|
||||||
using Catch::Matchers::Equals;
|
using Catch::Matchers::Equals;
|
||||||
|
|
||||||
|
static const bmp::mpfr_float mPI = boost::math::constants::pi<bmp::mpfr_float>();
|
||||||
|
|
||||||
|
|
||||||
static void trig_test(void (*operation)(void), bmp::mpfr_float (*mpfr_operation)(bmp::mpfr_float x),
|
static void trig_test(void (*operation)(void), bmp::mpfr_float (*mpfr_operation)(bmp::mpfr_float x),
|
||||||
double rtol, double atol)
|
double rtol, double atol)
|
||||||
@ -50,10 +52,10 @@ static void trig_test(void (*operation)(void), bmp::mpfr_float (*mpfr_operation)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void sin_test(double rtol=5e-3, double atol=1e-3)
|
static void sin_test(double rtol=6e-3, double atol=1e-3)
|
||||||
{
|
{
|
||||||
CAPTURE("sin test");
|
CAPTURE("sin test");
|
||||||
trig_test(sin_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return sin(x);}, rtol, atol);
|
trig_test(sin_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return sin(x * mPI / 180);}, rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void sin_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
static void sin_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
||||||
@ -64,10 +66,10 @@ static void sin_test(const char* a_str, int a_exp, double rtol=5e-3, double atol
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void cos_test(double rtol=5e-3, double atol=1e-3)
|
static void cos_test(double rtol=6e-3, double atol=1e-3)
|
||||||
{
|
{
|
||||||
CAPTURE("cos test");
|
CAPTURE("cos test");
|
||||||
trig_test(cos_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return cos(x);}, rtol, atol);
|
trig_test(cos_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return cos(x * mPI / 180);}, rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void cos_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
static void cos_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
||||||
@ -80,7 +82,7 @@ static void cos_test(const char* a_str, int a_exp, double rtol=5e-3, double atol
|
|||||||
|
|
||||||
static void tan_test(double rtol=5e-3, double atol=1e-3)
|
static void tan_test(double rtol=5e-3, double atol=1e-3)
|
||||||
{
|
{
|
||||||
trig_test(tan_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return tan(x);}, rtol, atol);
|
trig_test(tan_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return tan(x * mPI / 180);}, rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tan_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
static void tan_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
||||||
@ -93,7 +95,7 @@ static void tan_test(const char* a_str, int a_exp, double rtol=5e-3, double atol
|
|||||||
|
|
||||||
static void atan_test(double rtol=5e-3, double atol=1e-3)
|
static void atan_test(double rtol=5e-3, double atol=1e-3)
|
||||||
{
|
{
|
||||||
trig_test(arctan_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return atan(x);}, rtol, atol);
|
trig_test(arctan_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return atan(x)*180/mPI;}, rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void atan_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
static void atan_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
||||||
@ -106,7 +108,7 @@ static void atan_test(const char* a_str, int a_exp, double rtol=5e-3, double ato
|
|||||||
|
|
||||||
static void asin_test(double rtol=5e-3, double atol=1e-3)
|
static void asin_test(double rtol=5e-3, double atol=1e-3)
|
||||||
{
|
{
|
||||||
trig_test(arcsin_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return asin(x);}, rtol, atol);
|
trig_test(arcsin_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return asin(x)*180/mPI;}, rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void asin_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
static void asin_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
||||||
@ -119,7 +121,7 @@ static void asin_test(const char* a_str, int a_exp, double rtol=5e-3, double ato
|
|||||||
|
|
||||||
static void acos_test(double rtol=5e-3, double atol=1e-3)
|
static void acos_test(double rtol=5e-3, double atol=1e-3)
|
||||||
{
|
{
|
||||||
trig_test(arccos_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return acos(x);}, rtol, atol);
|
trig_test(arccos_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return acos(x)*180/mPI;}, rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void acos_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
static void acos_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
||||||
@ -129,101 +131,175 @@ static void acos_test(const char* a_str, int a_exp, double rtol=5e-3, double ato
|
|||||||
acos_test(rtol, atol);
|
acos_test(rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
const char * const pi = "3.141592653589793239";
|
// const char * const pi = "3.141592653589793239";
|
||||||
const char * const pi_threequarters = "2.356194490192344929";
|
// const char * const pi_threequarters = "2.356194490192344929";
|
||||||
const char * const pi_halved = "1.570796326794896619";
|
// const char * const pi_halved = "1.570796326794896619";
|
||||||
const char * const pi_quarter = ".7853981633974483096";
|
// const char * const pi_quarter = ".7853981633974483096";
|
||||||
|
|
||||||
|
|
||||||
TEST_CASE("sin") {
|
TEST_CASE("sin") {
|
||||||
sin_test("0.1", 0);
|
sin_test("0.1", 0, 0.2);
|
||||||
sin_test("0.05", 0, 1e-2);
|
|
||||||
sin_test("0.01", 0, -1);
|
|
||||||
sin_test("0.001", 0, -1);
|
|
||||||
sin_test("0.0001", 0, -1);
|
|
||||||
sin_test("0.00001", 0, -1);
|
|
||||||
sin_test("0.000001", 0, -1);
|
|
||||||
sin_test("0.0", 0, -1);
|
sin_test("0.0", 0, -1);
|
||||||
sin_test("0.2", 0);
|
sin_test("1.5", 0, 0.02);
|
||||||
sin_test("0.3", 0);
|
sin_test("2.0", 0, 0.02);
|
||||||
sin_test("0.4", 0);
|
sin_test("2.5", 0, 0.02);
|
||||||
sin_test("0.9", 0);
|
sin_test("3.0", 0, 0.02);
|
||||||
sin_test("1.5", 0);
|
sin_test("10", 0);
|
||||||
sin_test("2.0", 0);
|
sin_test("20", 0);
|
||||||
sin_test("2.5", 0);
|
sin_test("30", 0);
|
||||||
sin_test("3.0", 0);
|
sin_test("40", 0);
|
||||||
sin_test(pi, 0, -1);
|
sin_test("80", 0);
|
||||||
sin_test(pi_quarter, 0);
|
sin_test("120", 0);
|
||||||
sin_test(pi_halved, 0);
|
sin_test("160", 0);
|
||||||
sin_test(pi_threequarters, 0);
|
sin_test("200", 0);
|
||||||
|
sin_test("240", 0);
|
||||||
|
sin_test("280", 0);
|
||||||
|
sin_test("320", 0);
|
||||||
|
sin_test("359", 0, 0.02);
|
||||||
|
sin_test("360", 0, -1, 0.001);
|
||||||
|
sin_test("361", 0, 0.02);
|
||||||
|
sin_test("400", 0);
|
||||||
|
// sin_test(pi, 0, -1);
|
||||||
|
// sin_test(pi_quarter, 0);
|
||||||
|
// sin_test(pi_halved, 0);
|
||||||
|
// sin_test(pi_threequarters, 0);
|
||||||
|
sin_test("180.0", 0, -1);
|
||||||
|
sin_test("45.0", 0);
|
||||||
|
sin_test("90.0", 0);
|
||||||
|
sin_test("135.0", 0);
|
||||||
sin_test("1000.0", 0);
|
sin_test("1000.0", 0);
|
||||||
sin_test("-0.5", 0);
|
sin_test("-0.5", 0, 0.2);
|
||||||
sin_test("-1.5", 0);
|
sin_test("-1.5", 0, 0.02);
|
||||||
sin_test("-2.0", 0);
|
sin_test("-2.0", 0, 0.02);
|
||||||
sin_test("-2.5", 0);
|
sin_test("-2.5", 0, 0.02);
|
||||||
sin_test("-3.0", 0);
|
sin_test("-3.0", 0, 0.02);
|
||||||
sin_test("-9.0", 0);
|
sin_test("-9.0", 0);
|
||||||
sin_test("-18.0", 0);
|
sin_test("-18.0", 0);
|
||||||
sin_test("-27.0", 0);
|
sin_test("-27.0", 0);
|
||||||
sin_test("-1000.0", 0);
|
sin_test("-1000.0", 0);
|
||||||
|
sin_test("-30", 0);
|
||||||
|
sin_test("-40", 0);
|
||||||
|
sin_test("-80", 0);
|
||||||
|
sin_test("-120", 0);
|
||||||
|
sin_test("-160", 0);
|
||||||
|
sin_test("-200", 0);
|
||||||
|
sin_test("-240", 0);
|
||||||
|
sin_test("-280", 0);
|
||||||
|
sin_test("-320", 0);
|
||||||
|
sin_test("-360", 0, -1, 0.001);
|
||||||
|
sin_test("-400", 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("cos") {
|
TEST_CASE("cos") {
|
||||||
cos_test("0.1", 0);
|
cos_test("0.1", 0, 0.2);
|
||||||
cos_test("0.05", 0);
|
cos_test("0.0", 0, -1);
|
||||||
cos_test("0.01", 0);
|
cos_test("1.5", 0, 0.02);
|
||||||
cos_test("0.001", 0);
|
cos_test("2.0", 0, 0.02);
|
||||||
cos_test("0.0001", 0);
|
cos_test("2.5", 0, 0.02);
|
||||||
cos_test("0.00001", 0);
|
cos_test("3.0", 0, 0.02);
|
||||||
cos_test("0.000001", 0);
|
cos_test("10", 0);
|
||||||
cos_test("0.0", 0);
|
cos_test("20", 0);
|
||||||
cos_test("0.2", 0);
|
cos_test("30", 0);
|
||||||
cos_test("0.3", 0);
|
cos_test("40", 0);
|
||||||
cos_test("0.4", 0);
|
cos_test("80", 0);
|
||||||
cos_test("0.9", 0);
|
cos_test("120", 0);
|
||||||
cos_test("1.5", 0);
|
cos_test("160", 0);
|
||||||
cos_test("2.0", 0);
|
cos_test("200", 0);
|
||||||
cos_test("2.5", 0);
|
cos_test("240", 0);
|
||||||
cos_test("3.0", 0);
|
cos_test("280", 0, 0.006);
|
||||||
cos_test(pi, 0);
|
cos_test("320", 0);
|
||||||
cos_test(pi_quarter, 0);
|
cos_test("359", 0, 0.02);
|
||||||
cos_test(pi_halved, 0, -1);
|
cos_test("360", 0, -1, 0.001);
|
||||||
cos_test(pi_threequarters, 0);
|
cos_test("361", 0, 0.02);
|
||||||
cos_test("1000.0", 0);
|
cos_test("400", 0);
|
||||||
cos_test("-0.5", 0);
|
// cos_test(pi, 0, -1);
|
||||||
cos_test("-1.5", 0);
|
// cos_test(pi_quarter, 0);
|
||||||
cos_test("-2.0", 0);
|
// cos_test(pi_halved, 0);
|
||||||
cos_test("-2.5", 0);
|
// cos_test(pi_threequarters, 0);
|
||||||
cos_test("-3.0", 0);
|
cos_test("180.0", 0, -1);
|
||||||
|
cos_test("45.0", 0);
|
||||||
|
cos_test("90.0", 0, -1, 0.001);
|
||||||
|
cos_test("135.0", 0);
|
||||||
|
cos_test("1000.0", 0, 0.006);
|
||||||
|
cos_test("-0.5", 0, 0.2);
|
||||||
|
cos_test("-1.5", 0, 0.02);
|
||||||
|
cos_test("-2.0", 0, 0.02);
|
||||||
|
cos_test("-2.5", 0, 0.02);
|
||||||
|
cos_test("-3.0", 0, 0.02);
|
||||||
cos_test("-9.0", 0);
|
cos_test("-9.0", 0);
|
||||||
cos_test("-18.0", 0);
|
cos_test("-18.0", 0);
|
||||||
cos_test("-27.0", 0);
|
cos_test("-27.0", 0);
|
||||||
cos_test("-1000.0", 0);
|
cos_test("-1000.0", 0);
|
||||||
|
cos_test("-30", 0);
|
||||||
|
cos_test("-40", 0);
|
||||||
|
cos_test("-80", 0, 0.006);
|
||||||
|
cos_test("-120", 0);
|
||||||
|
cos_test("-160", 0);
|
||||||
|
cos_test("-200", 0);
|
||||||
|
cos_test("-240", 0);
|
||||||
|
cos_test("-280", 0);
|
||||||
|
cos_test("-320", 0);
|
||||||
|
cos_test("-360", 0, -1, 0.001);
|
||||||
|
cos_test("-400", 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
TEST_CASE("tan") {
|
TEST_CASE("tan") {
|
||||||
tan_test("0.1", 0);
|
tan_test("0.1", 0, 0.2);
|
||||||
tan_test("0.05", 0, 1e-2);
|
|
||||||
tan_test("0.01", 0, 5e-2);
|
|
||||||
tan_test("0.001", 0, -1);
|
|
||||||
tan_test("0.0001", 0, -1);
|
|
||||||
tan_test("0.00001", 0, -1);
|
|
||||||
tan_test("0.000001", 0, -1);
|
|
||||||
tan_test("0.0", 0, -1);
|
tan_test("0.0", 0, -1);
|
||||||
tan_test("0.2", 0);
|
tan_test("1.5", 0, 0.02);
|
||||||
tan_test("0.3", 0);
|
tan_test("2.0", 0, 0.02);
|
||||||
tan_test("0.4", 0);
|
tan_test("2.5", 0, 0.02);
|
||||||
tan_test("0.9", 0);
|
tan_test("3.0", 0, 0.02);
|
||||||
tan_test("1.5", 0);
|
tan_test("10", 0);
|
||||||
tan_test("2.0", 0);
|
tan_test("20", 0);
|
||||||
tan_test("2.5", 0);
|
tan_test("30", 0);
|
||||||
tan_test("3.0", 0);
|
tan_test("40", 0);
|
||||||
|
tan_test("80", 0);
|
||||||
|
tan_test("120", 0);
|
||||||
|
tan_test("160", 0);
|
||||||
|
tan_test("200", 0);
|
||||||
|
tan_test("240", 0);
|
||||||
|
tan_test("280", 0, 0.006);
|
||||||
|
tan_test("320", 0);
|
||||||
|
tan_test("359", 0, 0.02);
|
||||||
|
tan_test("360", 0, -1, 0.001);
|
||||||
|
tan_test("361", 0, 0.02);
|
||||||
|
tan_test("400", 0);
|
||||||
|
// tan_test(pi, 0, -1);
|
||||||
|
// tan_test(pi_quarter, 0);
|
||||||
|
// tan_test(pi_halved, 0);
|
||||||
|
// tan_test(pi_threequarters, 0);
|
||||||
|
tan_test("180.0", 0, -1);
|
||||||
|
tan_test("45.0", 0);
|
||||||
|
tan_test("90.0", 0, 2);
|
||||||
|
tan_test("135.0", 0);
|
||||||
|
tan_test("1000.0", 0, 0.006);
|
||||||
|
tan_test("-0.5", 0, 0.2);
|
||||||
|
tan_test("-1.5", 0, 0.02);
|
||||||
|
tan_test("-2.0", 0, 0.02);
|
||||||
|
tan_test("-2.5", 0, 0.02);
|
||||||
|
tan_test("-3.0", 0, 0.02);
|
||||||
|
tan_test("-9.0", 0);
|
||||||
|
tan_test("-18.0", 0);
|
||||||
|
tan_test("-27.0", 0);
|
||||||
|
tan_test("-1000.0", 0);
|
||||||
|
tan_test("-30", 0);
|
||||||
|
tan_test("-40", 0);
|
||||||
|
tan_test("-80", 0, 0.006);
|
||||||
|
tan_test("-120", 0);
|
||||||
|
tan_test("-160", 0);
|
||||||
|
tan_test("-200", 0);
|
||||||
|
tan_test("-240", 0);
|
||||||
|
tan_test("-280", 0);
|
||||||
|
tan_test("-320", 0);
|
||||||
|
tan_test("-360", 0, -1, 0.001);
|
||||||
|
tan_test("-400", 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("arctan") {
|
TEST_CASE("arctan") {
|
||||||
atan_test("0.001", 0, -1, 2e-3);
|
atan_test("0.001", 0, -1, 0.06);
|
||||||
atan_test("-0.001", 0, -1, 2e-3);
|
atan_test("-0.001", 0, -1, 0.06);
|
||||||
atan_test("0.7", 0);
|
atan_test("0.7", 0);
|
||||||
atan_test("-0.7", 0);
|
atan_test("-0.7", 0);
|
||||||
atan_test("0.1", 0);
|
atan_test("0.1", 0);
|
||||||
@ -234,12 +310,12 @@ TEST_CASE("arctan") {
|
|||||||
atan_test("-2.0", 0);
|
atan_test("-2.0", 0);
|
||||||
atan_test("3.0", 0);
|
atan_test("3.0", 0);
|
||||||
atan_test("-3.0", 0);
|
atan_test("-3.0", 0);
|
||||||
atan_test("0", 0, -1);
|
atan_test("0", 0, -1, 0.06);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("arcsin") {
|
TEST_CASE("arcsin") {
|
||||||
asin_test("0.001", 0, -1);
|
asin_test("0.001", 0, -1, 0.06);
|
||||||
asin_test("-0.001", 0, -1);
|
asin_test("-0.001", 0, -1, 0.06);
|
||||||
asin_test("0.7", 0);
|
asin_test("0.7", 0);
|
||||||
asin_test("-0.7", 0);
|
asin_test("-0.7", 0);
|
||||||
asin_test("0.1", 0, 1e-2);
|
asin_test("0.1", 0, 1e-2);
|
||||||
@ -265,7 +341,7 @@ static const int NUM_RAND_TRIG_TESTS = 4321; //trig tests are slow
|
|||||||
TEST_CASE("sin random"){
|
TEST_CASE("sin random"){
|
||||||
std::default_random_engine gen;
|
std::default_random_engine gen;
|
||||||
std::uniform_int_distribution<int> distrib(0,99);
|
std::uniform_int_distribution<int> distrib(0,99);
|
||||||
std::uniform_int_distribution<int> exp_distrib(-1,0); //restrict range for now
|
std::uniform_int_distribution<int> exp_distrib(0, 2); //restrict range for now
|
||||||
std::uniform_int_distribution<int> sign_distrib(0,1);
|
std::uniform_int_distribution<int> sign_distrib(0,1);
|
||||||
for (int j = 0; j < NUM_RAND_TRIG_TESTS; j++){
|
for (int j = 0; j < NUM_RAND_TRIG_TESTS; j++){
|
||||||
for (int i = 0; i < DEC80_NUM_LSU; i++){
|
for (int i = 0; i < DEC80_NUM_LSU; i++){
|
||||||
@ -274,25 +350,27 @@ TEST_CASE("sin random"){
|
|||||||
int exp = exp_distrib(gen);
|
int exp = exp_distrib(gen);
|
||||||
int sign = sign_distrib(gen);
|
int sign = sign_distrib(gen);
|
||||||
set_exponent(&AccDecn, exp, sign);
|
set_exponent(&AccDecn, exp, sign);
|
||||||
|
remove_leading_zeros(&AccDecn);
|
||||||
int lsu0 = AccDecn.lsu[0];
|
int lsu0 = AccDecn.lsu[0];
|
||||||
|
exp = get_exponent(&AccDecn);
|
||||||
CAPTURE(lsu0);
|
CAPTURE(lsu0);
|
||||||
CAPTURE(exp);
|
CAPTURE(exp);
|
||||||
CAPTURE(sign);
|
CAPTURE(sign);
|
||||||
if (exp == -1 && lsu0 == 0){
|
if (exp <= -1){
|
||||||
//very small
|
//very small
|
||||||
sin_test(40);
|
sin_test(4000);
|
||||||
} else if ((exp == -1 && lsu0 < 10) || (exp == 0 && lsu0 == 0)){
|
} else if (exp == 0 && lsu0 < 50){
|
||||||
//small
|
//small
|
||||||
sin_test(0.4);
|
sin_test(0.4);
|
||||||
} else if ((exp == 0 && lsu0 == 31)){
|
} else if (exp == 2 && lsu0 >= 17 && lsu0 <= 19){
|
||||||
//near pi
|
//near 180
|
||||||
sin_test(0.2);
|
sin_test(3);
|
||||||
} else if ((exp == 0 && lsu0 == 62)){
|
} else if (exp == 2 && lsu0 >= 35 && lsu0 <= 36){
|
||||||
//near 2pi
|
//near 360
|
||||||
sin_test(0.2);
|
sin_test(12);
|
||||||
} else if ((exp == 0 && lsu0 > 62)){
|
} else if (exp == 2 && lsu0 > 50){
|
||||||
//large
|
//large
|
||||||
sin_test(0.1);
|
sin_test(35);
|
||||||
} else {
|
} else {
|
||||||
sin_test(0.02);
|
sin_test(0.02);
|
||||||
}
|
}
|
||||||
@ -302,7 +380,7 @@ TEST_CASE("sin random"){
|
|||||||
TEST_CASE("cos random"){
|
TEST_CASE("cos random"){
|
||||||
std::default_random_engine gen;
|
std::default_random_engine gen;
|
||||||
std::uniform_int_distribution<int> distrib(0,99);
|
std::uniform_int_distribution<int> distrib(0,99);
|
||||||
std::uniform_int_distribution<int> exp_distrib(-1,0); //restrict range for now
|
std::uniform_int_distribution<int> exp_distrib(0, 2); //restrict range for now
|
||||||
std::uniform_int_distribution<int> sign_distrib(0,1);
|
std::uniform_int_distribution<int> sign_distrib(0,1);
|
||||||
for (int j = 0; j < NUM_RAND_TRIG_TESTS; j++){
|
for (int j = 0; j < NUM_RAND_TRIG_TESTS; j++){
|
||||||
for (int i = 0; i < DEC80_NUM_LSU; i++){
|
for (int i = 0; i < DEC80_NUM_LSU; i++){
|
||||||
@ -311,20 +389,24 @@ TEST_CASE("cos random"){
|
|||||||
int exp = exp_distrib(gen);
|
int exp = exp_distrib(gen);
|
||||||
int sign = sign_distrib(gen);
|
int sign = sign_distrib(gen);
|
||||||
set_exponent(&AccDecn, exp, sign);
|
set_exponent(&AccDecn, exp, sign);
|
||||||
|
remove_leading_zeros(&AccDecn);
|
||||||
int lsu0 = AccDecn.lsu[0];
|
int lsu0 = AccDecn.lsu[0];
|
||||||
|
exp = get_exponent(&AccDecn);
|
||||||
CAPTURE(lsu0);
|
CAPTURE(lsu0);
|
||||||
CAPTURE(exp);
|
CAPTURE(exp);
|
||||||
CAPTURE(sign);
|
CAPTURE(sign);
|
||||||
if (exp == 0 && lsu0 == 15){
|
if (exp == 1 && lsu0 >= 89 && lsu0 <= 90){
|
||||||
//near pi/2
|
//very near 90
|
||||||
cos_test(0.4);
|
cos_test(500);
|
||||||
} else if (exp == 0 && lsu0 == 47){
|
} else if (exp == 1 && lsu0 >= 87 && lsu0 <= 92){
|
||||||
//near 3/2 * pi
|
//near 90
|
||||||
cos_test(0.4);
|
cos_test(2);
|
||||||
} else if (exp == 0 && lsu0 == 78){
|
} else if (exp == 2 && lsu0 >= 26 && lsu0 <= 27){
|
||||||
//near 5/2 * pi
|
//near 270
|
||||||
// cos_test(0.4);
|
cos_test(500);
|
||||||
cos_test(1.1); //actual rtol is much worse than 0.4, random test happens to hit a bad one
|
} else if (exp == 2 && lsu0 >= 44){
|
||||||
|
//large
|
||||||
|
cos_test(20);
|
||||||
} else {
|
} else {
|
||||||
cos_test(0.02);
|
cos_test(0.02);
|
||||||
}
|
}
|
||||||
@ -334,7 +416,7 @@ TEST_CASE("cos random"){
|
|||||||
TEST_CASE("tan random"){
|
TEST_CASE("tan random"){
|
||||||
std::default_random_engine gen;
|
std::default_random_engine gen;
|
||||||
std::uniform_int_distribution<int> distrib(0,99);
|
std::uniform_int_distribution<int> distrib(0,99);
|
||||||
std::uniform_int_distribution<int> exp_distrib(-1,0); //restrict range for now
|
std::uniform_int_distribution<int> exp_distrib(0, 2); //restrict range for now
|
||||||
std::uniform_int_distribution<int> sign_distrib(0,1);
|
std::uniform_int_distribution<int> sign_distrib(0,1);
|
||||||
for (int j = 0; j < NUM_RAND_TRIG_TESTS; j++){
|
for (int j = 0; j < NUM_RAND_TRIG_TESTS; j++){
|
||||||
for (int i = 0; i < DEC80_NUM_LSU; i++){
|
for (int i = 0; i < DEC80_NUM_LSU; i++){
|
||||||
@ -343,35 +425,39 @@ TEST_CASE("tan random"){
|
|||||||
int exp = exp_distrib(gen);
|
int exp = exp_distrib(gen);
|
||||||
int sign = sign_distrib(gen);
|
int sign = sign_distrib(gen);
|
||||||
set_exponent(&AccDecn, exp, sign);
|
set_exponent(&AccDecn, exp, sign);
|
||||||
|
remove_leading_zeros(&AccDecn);
|
||||||
int lsu0 = AccDecn.lsu[0];
|
int lsu0 = AccDecn.lsu[0];
|
||||||
|
exp = get_exponent(&AccDecn);
|
||||||
CAPTURE(lsu0);
|
CAPTURE(lsu0);
|
||||||
CAPTURE(exp);
|
CAPTURE(exp);
|
||||||
CAPTURE(sign);
|
CAPTURE(sign);
|
||||||
if (exp == -1 && lsu0 == 0){
|
if (exp <= -3){
|
||||||
|
//extremely small
|
||||||
|
tan_test(5000);
|
||||||
|
} if (exp <= -1){
|
||||||
//very small
|
//very small
|
||||||
tan_test(40);
|
tan_test(400);
|
||||||
} else if ((exp == -1 && lsu0 < 10) || (exp == 0 && lsu0 == 0)){
|
} else if (exp == 0 && lsu0 < 50){
|
||||||
//small
|
//small
|
||||||
tan_test(0.5);
|
tan_test(1);
|
||||||
} else if (exp == 0 && lsu0 == 15){
|
} else if (exp == 1 && lsu0 >= 89 && lsu0 <= 90){
|
||||||
//near pi/2
|
//very near 90
|
||||||
tan_test(0.5);
|
tan_test(5);
|
||||||
} else if ((exp == 0 && lsu0 == 31)){
|
} else if (exp == 1 && lsu0 >= 87 && lsu0 <= 92){
|
||||||
//near pi
|
//near 90
|
||||||
tan_test(0.2);
|
tan_test(1);
|
||||||
} else if (exp == 0 && lsu0 == 47){
|
} else if (exp == 2 && lsu0 >= 17 && lsu0 <= 19){
|
||||||
//near 3/2 * pi
|
//near 180
|
||||||
tan_test(0.5);
|
tan_test(3);
|
||||||
} else if ((exp == 0 && lsu0 == 62)){
|
} else if (exp == 2 && lsu0 >= 26 && lsu0 <= 27){
|
||||||
//near 2pi
|
//near 270
|
||||||
tan_test(0.2);
|
tan_test(5);
|
||||||
} else if (exp == 0 && lsu0 == 78){
|
} else if (exp == 2 && lsu0 >= 35 && lsu0 <= 37){
|
||||||
//near 5/2 * pi
|
//near 360
|
||||||
// tan_test(0.5);
|
tan_test(20);
|
||||||
tan_test(0.6); //actual rtol is much worse than 0.4, random test happens to hit a bad one
|
} else if (exp == 2 && lsu0 >= 44){
|
||||||
} else if ((exp == 0 && lsu0 > 62)){
|
|
||||||
//large
|
//large
|
||||||
tan_test(0.1);
|
tan_test(50);
|
||||||
} else {
|
} else {
|
||||||
tan_test(0.02);
|
tan_test(0.02);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user