mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-15 20:29:48 +00:00
94bd57e154
sequences in SPUDAGToDAGISel.cpp and SPU64InstrInfo.td, killing custom DAG node types as needed. - i64 mul is now a legal instruction, but emits an instruction sequence that stretches tblgen and the imagination, as well as violating laws of several small countries and most southern US states (just kidding, but looking at a function with 80+ parameters is really weird and just plain wrong.) - Update tests as needed. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@62254 91177308-0d34-0410-b5e6-96231b3b80d8
674 lines
20 KiB
C
674 lines
20 KiB
C
#include <stdio.h>
|
|
#include "i64operations.h"
|
|
|
|
int64_t tval_a = 1234567890003LL;
|
|
int64_t tval_b = 2345678901235LL;
|
|
int64_t tval_c = 1234567890001LL;
|
|
int64_t tval_d = 10001LL;
|
|
int64_t tval_e = 10000LL;
|
|
uint64_t tval_f = 0xffffff0750135eb9;
|
|
int64_t tval_g = -1;
|
|
|
|
/* ~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~- */
|
|
|
|
int
|
|
i64_eq(int64_t a, int64_t b)
|
|
{
|
|
return (a == b);
|
|
}
|
|
|
|
int
|
|
i64_neq(int64_t a, int64_t b)
|
|
{
|
|
return (a != b);
|
|
}
|
|
|
|
int
|
|
i64_gt(int64_t a, int64_t b)
|
|
{
|
|
return (a > b);
|
|
}
|
|
|
|
int
|
|
i64_le(int64_t a, int64_t b)
|
|
{
|
|
return (a <= b);
|
|
}
|
|
|
|
int
|
|
i64_ge(int64_t a, int64_t b) {
|
|
return (a >= b);
|
|
}
|
|
|
|
int
|
|
i64_lt(int64_t a, int64_t b) {
|
|
return (a < b);
|
|
}
|
|
|
|
int
|
|
i64_uge(uint64_t a, uint64_t b)
|
|
{
|
|
return (a >= b);
|
|
}
|
|
|
|
int
|
|
i64_ult(uint64_t a, uint64_t b)
|
|
{
|
|
return (a < b);
|
|
}
|
|
|
|
int
|
|
i64_ugt(uint64_t a, uint64_t b)
|
|
{
|
|
return (a > b);
|
|
}
|
|
|
|
int
|
|
i64_ule(uint64_t a, uint64_t b)
|
|
{
|
|
return (a <= b);
|
|
}
|
|
|
|
int64_t
|
|
i64_eq_select(int64_t a, int64_t b, int64_t c, int64_t d)
|
|
{
|
|
return ((a == b) ? c : d);
|
|
}
|
|
|
|
int64_t
|
|
i64_neq_select(int64_t a, int64_t b, int64_t c, int64_t d)
|
|
{
|
|
return ((a != b) ? c : d);
|
|
}
|
|
|
|
int64_t
|
|
i64_gt_select(int64_t a, int64_t b, int64_t c, int64_t d) {
|
|
return ((a > b) ? c : d);
|
|
}
|
|
|
|
int64_t
|
|
i64_le_select(int64_t a, int64_t b, int64_t c, int64_t d) {
|
|
return ((a <= b) ? c : d);
|
|
}
|
|
|
|
int64_t
|
|
i64_ge_select(int64_t a, int64_t b, int64_t c, int64_t d) {
|
|
return ((a >= b) ? c : d);
|
|
}
|
|
|
|
int64_t
|
|
i64_lt_select(int64_t a, int64_t b, int64_t c, int64_t d) {
|
|
return ((a < b) ? c : d);
|
|
}
|
|
|
|
uint64_t
|
|
i64_ugt_select(uint64_t a, uint64_t b, uint64_t c, uint64_t d)
|
|
{
|
|
return ((a > b) ? c : d);
|
|
}
|
|
|
|
uint64_t
|
|
i64_ule_select(uint64_t a, uint64_t b, uint64_t c, uint64_t d)
|
|
{
|
|
return ((a <= b) ? c : d);
|
|
}
|
|
|
|
uint64_t
|
|
i64_uge_select(uint64_t a, uint64_t b, uint64_t c, uint64_t d) {
|
|
return ((a >= b) ? c : d);
|
|
}
|
|
|
|
uint64_t
|
|
i64_ult_select(uint64_t a, uint64_t b, uint64_t c, uint64_t d) {
|
|
return ((a < b) ? c : d);
|
|
}
|
|
|
|
/* ~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~- */
|
|
|
|
struct harness_int64_pred int64_tests_eq[] = {
|
|
{"a %s a", &tval_a, &tval_a, &tval_c, &tval_d, TRUE_VAL, &tval_c},
|
|
{"a %s b", &tval_a, &tval_b, &tval_c, &tval_d, FALSE_VAL, &tval_d},
|
|
{"a %s c", &tval_a, &tval_c, &tval_c, &tval_d, FALSE_VAL, &tval_d},
|
|
{"d %s e", &tval_d, &tval_e, &tval_c, &tval_d, FALSE_VAL, &tval_d},
|
|
{"e %s e", &tval_e, &tval_e, &tval_c, &tval_d, TRUE_VAL, &tval_c}
|
|
};
|
|
|
|
struct harness_int64_pred int64_tests_neq[] = {
|
|
{"a %s a", &tval_a, &tval_a, &tval_c, &tval_d, FALSE_VAL, &tval_d},
|
|
{"a %s b", &tval_a, &tval_b, &tval_c, &tval_d, TRUE_VAL, &tval_c},
|
|
{"a %s c", &tval_a, &tval_c, &tval_c, &tval_d, TRUE_VAL, &tval_c},
|
|
{"d %s e", &tval_d, &tval_e, &tval_c, &tval_d, TRUE_VAL, &tval_c},
|
|
{"e %s e", &tval_e, &tval_e, &tval_c, &tval_d, FALSE_VAL, &tval_d}
|
|
};
|
|
|
|
struct harness_int64_pred int64_tests_sgt[] = {
|
|
{"a %s a", &tval_a, &tval_a, &tval_c, &tval_d, FALSE_VAL, &tval_d},
|
|
{"a %s b", &tval_a, &tval_b, &tval_c, &tval_d, FALSE_VAL, &tval_d},
|
|
{"a %s c", &tval_a, &tval_c, &tval_c, &tval_d, TRUE_VAL, &tval_c},
|
|
{"d %s e", &tval_d, &tval_e, &tval_c, &tval_d, TRUE_VAL, &tval_c},
|
|
{"e %s e", &tval_e, &tval_e, &tval_c, &tval_d, FALSE_VAL, &tval_d}
|
|
};
|
|
|
|
struct harness_int64_pred int64_tests_sle[] = {
|
|
{"a %s a", &tval_a, &tval_a, &tval_c, &tval_d, TRUE_VAL, &tval_c},
|
|
{"a %s b", &tval_a, &tval_b, &tval_c, &tval_d, TRUE_VAL, &tval_c},
|
|
{"a %s c", &tval_a, &tval_c, &tval_c, &tval_d, FALSE_VAL, &tval_d},
|
|
{"d %s e", &tval_d, &tval_e, &tval_c, &tval_d, FALSE_VAL, &tval_d},
|
|
{"e %s e", &tval_e, &tval_e, &tval_c, &tval_d, TRUE_VAL, &tval_c}
|
|
};
|
|
|
|
struct harness_int64_pred int64_tests_sge[] = {
|
|
{"a %s a", &tval_a, &tval_a, &tval_c, &tval_d, TRUE_VAL, &tval_c},
|
|
{"a %s b", &tval_a, &tval_b, &tval_c, &tval_d, FALSE_VAL, &tval_d},
|
|
{"a %s c", &tval_a, &tval_c, &tval_c, &tval_d, TRUE_VAL, &tval_c},
|
|
{"d %s e", &tval_d, &tval_e, &tval_c, &tval_d, TRUE_VAL, &tval_c},
|
|
{"e %s e", &tval_e, &tval_e, &tval_c, &tval_d, TRUE_VAL, &tval_c}
|
|
};
|
|
|
|
struct harness_int64_pred int64_tests_slt[] = {
|
|
{"a %s a", &tval_a, &tval_a, &tval_c, &tval_d, FALSE_VAL, &tval_d},
|
|
{"a %s b", &tval_a, &tval_b, &tval_c, &tval_d, TRUE_VAL, &tval_c},
|
|
{"a %s c", &tval_a, &tval_c, &tval_c, &tval_d, FALSE_VAL, &tval_d},
|
|
{"d %s e", &tval_d, &tval_e, &tval_c, &tval_d, FALSE_VAL, &tval_d},
|
|
{"e %s e", &tval_e, &tval_e, &tval_c, &tval_d, FALSE_VAL, &tval_d}
|
|
};
|
|
|
|
struct int64_pred_s int64_preds[] = {
|
|
{"eq", i64_eq, i64_eq_select,
|
|
int64_tests_eq, ARR_SIZE(int64_tests_eq)},
|
|
{"neq", i64_neq, i64_neq_select,
|
|
int64_tests_neq, ARR_SIZE(int64_tests_neq)},
|
|
{"gt", i64_gt, i64_gt_select,
|
|
int64_tests_sgt, ARR_SIZE(int64_tests_sgt)},
|
|
{"le", i64_le, i64_le_select,
|
|
int64_tests_sle, ARR_SIZE(int64_tests_sle)},
|
|
{"ge", i64_ge, i64_ge_select,
|
|
int64_tests_sge, ARR_SIZE(int64_tests_sge)},
|
|
{"lt", i64_lt, i64_lt_select,
|
|
int64_tests_slt, ARR_SIZE(int64_tests_slt)}
|
|
};
|
|
|
|
/* ~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~- */
|
|
|
|
struct harness_uint64_pred uint64_tests_ugt[] = {
|
|
{"a %s a", (uint64_t *) &tval_a, (uint64_t *) &tval_a, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, FALSE_VAL, (uint64_t *) &tval_d},
|
|
{"a %s b", (uint64_t *) &tval_a, (uint64_t *) &tval_b, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, FALSE_VAL, (uint64_t *) &tval_d },
|
|
{"a %s c", (uint64_t *) &tval_a, (uint64_t *) &tval_c, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, TRUE_VAL, (uint64_t *) &tval_c },
|
|
{"d %s e", (uint64_t *) &tval_d, (uint64_t *) &tval_e, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, TRUE_VAL, (uint64_t *) &tval_c },
|
|
{"e %s e", (uint64_t *) &tval_e, (uint64_t *) &tval_e, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, FALSE_VAL, (uint64_t *) &tval_d }
|
|
};
|
|
|
|
struct harness_uint64_pred uint64_tests_ule[] = {
|
|
{"a %s a", (uint64_t *) &tval_a, (uint64_t *) &tval_a, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, TRUE_VAL, (uint64_t *) &tval_c},
|
|
{"a %s b", (uint64_t *) &tval_a, (uint64_t *) &tval_b, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, TRUE_VAL, (uint64_t *) &tval_c},
|
|
{"a %s c", (uint64_t *) &tval_a, (uint64_t *) &tval_c, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, FALSE_VAL, (uint64_t *) &tval_d},
|
|
{"d %s e", (uint64_t *) &tval_d, (uint64_t *) &tval_e, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, FALSE_VAL, (uint64_t *) &tval_d},
|
|
{"e %s e", (uint64_t *) &tval_e, (uint64_t *) &tval_e, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, TRUE_VAL, (uint64_t *) &tval_c}
|
|
};
|
|
|
|
struct harness_uint64_pred uint64_tests_uge[] = {
|
|
{"a %s a", (uint64_t *) &tval_a, (uint64_t *) &tval_a, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, TRUE_VAL, (uint64_t *) &tval_c},
|
|
{"a %s b", (uint64_t *) &tval_a, (uint64_t *) &tval_b, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, FALSE_VAL, (uint64_t *) &tval_d},
|
|
{"a %s c", (uint64_t *) &tval_a, (uint64_t *) &tval_c, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, TRUE_VAL, (uint64_t *) &tval_c},
|
|
{"d %s e", (uint64_t *) &tval_d, (uint64_t *) &tval_e, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, TRUE_VAL, (uint64_t *) &tval_c},
|
|
{"e %s e", (uint64_t *) &tval_e, (uint64_t *) &tval_e, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, TRUE_VAL, (uint64_t *) &tval_c}
|
|
};
|
|
|
|
struct harness_uint64_pred uint64_tests_ult[] = {
|
|
{"a %s a", (uint64_t *) &tval_a, (uint64_t *) &tval_a, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, FALSE_VAL, (uint64_t *) &tval_d},
|
|
{"a %s b", (uint64_t *) &tval_a, (uint64_t *) &tval_b, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, TRUE_VAL, (uint64_t *) &tval_c},
|
|
{"a %s c", (uint64_t *) &tval_a, (uint64_t *) &tval_c, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, FALSE_VAL, (uint64_t *) &tval_d},
|
|
{"d %s e", (uint64_t *) &tval_d, (uint64_t *) &tval_e, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, FALSE_VAL, (uint64_t *) &tval_d},
|
|
{"e %s e", (uint64_t *) &tval_e, (uint64_t *) &tval_e, (uint64_t *) &tval_c,
|
|
(uint64_t *) &tval_d, FALSE_VAL, (uint64_t *) &tval_d}
|
|
};
|
|
|
|
struct uint64_pred_s uint64_preds[] = {
|
|
{"ugt", i64_ugt, i64_ugt_select,
|
|
uint64_tests_ugt, ARR_SIZE(uint64_tests_ugt)},
|
|
{"ule", i64_ule, i64_ule_select,
|
|
uint64_tests_ule, ARR_SIZE(uint64_tests_ule)},
|
|
{"uge", i64_uge, i64_uge_select,
|
|
uint64_tests_uge, ARR_SIZE(uint64_tests_uge)},
|
|
{"ult", i64_ult, i64_ult_select,
|
|
uint64_tests_ult, ARR_SIZE(uint64_tests_ult)}
|
|
};
|
|
|
|
int
|
|
compare_expect_int64(const struct int64_pred_s * pred)
|
|
{
|
|
int j, failed = 0;
|
|
|
|
for (j = 0; j < pred->n_tests; ++j) {
|
|
int pred_result;
|
|
|
|
pred_result = (*pred->predfunc) (*pred->tests[j].lhs, *pred->tests[j].rhs);
|
|
|
|
if (pred_result != pred->tests[j].expected) {
|
|
char str[64];
|
|
|
|
sprintf(str, pred->tests[j].fmt_string, pred->name);
|
|
printf("%s: returned value is %d, expecting %d\n", str,
|
|
pred_result, pred->tests[j].expected);
|
|
printf(" lhs = %19lld (0x%016llx)\n", *pred->tests[j].lhs,
|
|
*pred->tests[j].lhs);
|
|
printf(" rhs = %19lld (0x%016llx)\n", *pred->tests[j].rhs,
|
|
*pred->tests[j].rhs);
|
|
++failed;
|
|
} else {
|
|
int64_t selresult;
|
|
|
|
selresult = (pred->selfunc) (*pred->tests[j].lhs, *pred->tests[j].rhs,
|
|
*pred->tests[j].select_a,
|
|
*pred->tests[j].select_b);
|
|
|
|
if (selresult != *pred->tests[j].select_expected) {
|
|
char str[64];
|
|
|
|
sprintf(str, pred->tests[j].fmt_string, pred->name);
|
|
printf("%s select: returned value is %d, expecting %d\n", str,
|
|
pred_result, pred->tests[j].expected);
|
|
printf(" lhs = %19lld (0x%016llx)\n", *pred->tests[j].lhs,
|
|
*pred->tests[j].lhs);
|
|
printf(" rhs = %19lld (0x%016llx)\n", *pred->tests[j].rhs,
|
|
*pred->tests[j].rhs);
|
|
printf(" true = %19lld (0x%016llx)\n", *pred->tests[j].select_a,
|
|
*pred->tests[j].select_a);
|
|
printf(" false = %19lld (0x%016llx)\n", *pred->tests[j].select_b,
|
|
*pred->tests[j].select_b);
|
|
++failed;
|
|
}
|
|
}
|
|
}
|
|
|
|
printf(" %d tests performed, should be %d.\n", j, pred->n_tests);
|
|
|
|
return failed;
|
|
}
|
|
|
|
int
|
|
compare_expect_uint64(const struct uint64_pred_s * pred)
|
|
{
|
|
int j, failed = 0;
|
|
|
|
for (j = 0; j < pred->n_tests; ++j) {
|
|
int pred_result;
|
|
|
|
pred_result = (*pred->predfunc) (*pred->tests[j].lhs, *pred->tests[j].rhs);
|
|
if (pred_result != pred->tests[j].expected) {
|
|
char str[64];
|
|
|
|
sprintf(str, pred->tests[j].fmt_string, pred->name);
|
|
printf("%s: returned value is %d, expecting %d\n", str,
|
|
pred_result, pred->tests[j].expected);
|
|
printf(" lhs = %19llu (0x%016llx)\n", *pred->tests[j].lhs,
|
|
*pred->tests[j].lhs);
|
|
printf(" rhs = %19llu (0x%016llx)\n", *pred->tests[j].rhs,
|
|
*pred->tests[j].rhs);
|
|
++failed;
|
|
} else {
|
|
uint64_t selresult;
|
|
|
|
selresult = (pred->selfunc) (*pred->tests[j].lhs, *pred->tests[j].rhs,
|
|
*pred->tests[j].select_a,
|
|
*pred->tests[j].select_b);
|
|
if (selresult != *pred->tests[j].select_expected) {
|
|
char str[64];
|
|
|
|
sprintf(str, pred->tests[j].fmt_string, pred->name);
|
|
printf("%s select: returned value is %d, expecting %d\n", str,
|
|
pred_result, pred->tests[j].expected);
|
|
printf(" lhs = %19llu (0x%016llx)\n", *pred->tests[j].lhs,
|
|
*pred->tests[j].lhs);
|
|
printf(" rhs = %19llu (0x%016llx)\n", *pred->tests[j].rhs,
|
|
*pred->tests[j].rhs);
|
|
printf(" true = %19llu (0x%016llx)\n", *pred->tests[j].select_a,
|
|
*pred->tests[j].select_a);
|
|
printf(" false = %19llu (0x%016llx)\n", *pred->tests[j].select_b,
|
|
*pred->tests[j].select_b);
|
|
++failed;
|
|
}
|
|
}
|
|
}
|
|
|
|
printf(" %d tests performed, should be %d.\n", j, pred->n_tests);
|
|
|
|
return failed;
|
|
}
|
|
|
|
/* ~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~- */
|
|
|
|
int
|
|
test_i64_sext_i32(int in, int64_t expected) {
|
|
int64_t result = (int64_t) in;
|
|
|
|
if (result != expected) {
|
|
char str[64];
|
|
sprintf(str, "i64_sext_i32(%d) returns %lld\n", in, result);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
test_i64_sext_i16(short in, int64_t expected) {
|
|
int64_t result = (int64_t) in;
|
|
|
|
if (result != expected) {
|
|
char str[64];
|
|
sprintf(str, "i64_sext_i16(%hd) returns %lld\n", in, result);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
test_i64_sext_i8(signed char in, int64_t expected) {
|
|
int64_t result = (int64_t) in;
|
|
|
|
if (result != expected) {
|
|
char str[64];
|
|
sprintf(str, "i64_sext_i8(%d) returns %lld\n", in, result);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
test_i64_zext_i32(unsigned int in, uint64_t expected) {
|
|
uint64_t result = (uint64_t) in;
|
|
|
|
if (result != expected) {
|
|
char str[64];
|
|
sprintf(str, "i64_zext_i32(%u) returns %llu\n", in, result);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
test_i64_zext_i16(unsigned short in, uint64_t expected) {
|
|
uint64_t result = (uint64_t) in;
|
|
|
|
if (result != expected) {
|
|
char str[64];
|
|
sprintf(str, "i64_zext_i16(%hu) returns %llu\n", in, result);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
test_i64_zext_i8(unsigned char in, uint64_t expected) {
|
|
uint64_t result = (uint64_t) in;
|
|
|
|
if (result != expected) {
|
|
char str[64];
|
|
sprintf(str, "i64_zext_i8(%u) returns %llu\n", in, result);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~- */
|
|
|
|
int64_t
|
|
i64_shl_const(int64_t a) {
|
|
return a << 10;
|
|
}
|
|
|
|
int64_t
|
|
i64_shl(int64_t a, int amt) {
|
|
return a << amt;
|
|
}
|
|
|
|
uint64_t
|
|
u64_shl_const(uint64_t a) {
|
|
return a << 10;
|
|
}
|
|
|
|
uint64_t
|
|
u64_shl(uint64_t a, int amt) {
|
|
return a << amt;
|
|
}
|
|
|
|
int64_t
|
|
i64_srl_const(int64_t a) {
|
|
return a >> 10;
|
|
}
|
|
|
|
int64_t
|
|
i64_srl(int64_t a, int amt) {
|
|
return a >> amt;
|
|
}
|
|
|
|
uint64_t
|
|
u64_srl_const(uint64_t a) {
|
|
return a >> 10;
|
|
}
|
|
|
|
uint64_t
|
|
u64_srl(uint64_t a, int amt) {
|
|
return a >> amt;
|
|
}
|
|
|
|
int64_t
|
|
i64_sra_const(int64_t a) {
|
|
return a >> 10;
|
|
}
|
|
|
|
int64_t
|
|
i64_sra(int64_t a, int amt) {
|
|
return a >> amt;
|
|
}
|
|
|
|
uint64_t
|
|
u64_sra_const(uint64_t a) {
|
|
return a >> 10;
|
|
}
|
|
|
|
uint64_t
|
|
u64_sra(uint64_t a, int amt) {
|
|
return a >> amt;
|
|
}
|
|
|
|
int
|
|
test_u64_constant_shift(const char *func_name, uint64_t (*func)(uint64_t), uint64_t a, uint64_t expected) {
|
|
uint64_t result = (*func)(a);
|
|
|
|
if (result != expected) {
|
|
printf("%s(0x%016llx) returns 0x%016llx, expected 0x%016llx\n", func_name, a, result, expected);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
test_i64_constant_shift(const char *func_name, int64_t (*func)(int64_t), int64_t a, int64_t expected) {
|
|
int64_t result = (*func)(a);
|
|
|
|
if (result != expected) {
|
|
printf("%s(0x%016llx) returns 0x%016llx, expected 0x%016llx\n", func_name, a, result, expected);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
test_u64_variable_shift(const char *func_name, uint64_t (*func)(uint64_t, int), uint64_t a, unsigned int b, uint64_t expected) {
|
|
uint64_t result = (*func)(a, b);
|
|
|
|
if (result != expected) {
|
|
printf("%s(0x%016llx, %d) returns 0x%016llx, expected 0x%016llx\n", func_name, a, b, result, expected);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
test_i64_variable_shift(const char *func_name, int64_t (*func)(int64_t, int), int64_t a, unsigned int b, int64_t expected) {
|
|
int64_t result = (*func)(a, b);
|
|
|
|
if (result != expected) {
|
|
printf("%s(0x%016llx, %d) returns 0x%016llx, expected 0x%016llx\n", func_name, a, b, result, expected);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~- */
|
|
|
|
int64_t i64_mul(int64_t a, int64_t b) {
|
|
return a * b;
|
|
}
|
|
|
|
/* ~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~- */
|
|
|
|
int
|
|
main(void)
|
|
{
|
|
int i, j, failed = 0;
|
|
const char *something_failed = " %d tests failed.\n";
|
|
const char *all_tests_passed = " All tests passed.\n";
|
|
|
|
printf("tval_a = %20lld (0x%016llx)\n", tval_a, tval_a);
|
|
printf("tval_b = %20lld (0x%016llx)\n", tval_b, tval_b);
|
|
printf("tval_c = %20lld (0x%016llx)\n", tval_c, tval_c);
|
|
printf("tval_d = %20lld (0x%016llx)\n", tval_d, tval_d);
|
|
printf("tval_e = %20lld (0x%016llx)\n", tval_e, tval_e);
|
|
printf("tval_f = %20llu (0x%016llx)\n", tval_f, tval_f);
|
|
printf("tval_g = %20llu (0x%016llx)\n", tval_g, tval_g);
|
|
printf("----------------------------------------\n");
|
|
|
|
for (i = 0; i < ARR_SIZE(int64_preds); ++i) {
|
|
printf("%s series:\n", int64_preds[i].name);
|
|
if ((failed = compare_expect_int64(int64_preds + i)) > 0) {
|
|
printf(something_failed, failed);
|
|
} else {
|
|
printf(all_tests_passed);
|
|
}
|
|
|
|
printf("----------------------------------------\n");
|
|
}
|
|
|
|
for (i = 0; i < ARR_SIZE(uint64_preds); ++i) {
|
|
printf("%s series:\n", uint64_preds[i].name);
|
|
if ((failed = compare_expect_uint64(uint64_preds + i)) > 0) {
|
|
printf(something_failed, failed);
|
|
} else {
|
|
printf(all_tests_passed);
|
|
}
|
|
|
|
printf("----------------------------------------\n");
|
|
}
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
puts("signed/zero-extend tests:");
|
|
|
|
failed = 0;
|
|
failed += test_i64_sext_i32(-1, -1LL);
|
|
failed += test_i64_sext_i32(10, 10LL);
|
|
failed += test_i64_sext_i32(0x7fffffff, 0x7fffffffLL);
|
|
failed += test_i64_sext_i16(-1, -1LL);
|
|
failed += test_i64_sext_i16(10, 10LL);
|
|
failed += test_i64_sext_i16(0x7fff, 0x7fffLL);
|
|
failed += test_i64_sext_i8(-1, -1LL);
|
|
failed += test_i64_sext_i8(10, 10LL);
|
|
failed += test_i64_sext_i8(0x7f, 0x7fLL);
|
|
|
|
failed += test_i64_zext_i32(0xffffffff, 0x00000000ffffffffLLU);
|
|
failed += test_i64_zext_i32(0x01234567, 0x0000000001234567LLU);
|
|
failed += test_i64_zext_i16(0xffff, 0x000000000000ffffLLU);
|
|
failed += test_i64_zext_i16(0x569a, 0x000000000000569aLLU);
|
|
failed += test_i64_zext_i8(0xff, 0x00000000000000ffLLU);
|
|
failed += test_i64_zext_i8(0xa0, 0x00000000000000a0LLU);
|
|
|
|
if (failed > 0) {
|
|
printf(" %d tests failed.\n", failed);
|
|
} else {
|
|
printf(" All tests passed.\n");
|
|
}
|
|
|
|
printf("----------------------------------------\n");
|
|
|
|
failed = 0;
|
|
puts("signed left/right shift tests:");
|
|
failed += test_i64_constant_shift("i64_shl_const", i64_shl_const, tval_a, 0x00047dc7ec114c00LL);
|
|
failed += test_i64_variable_shift("i64_shl", i64_shl, tval_a, 10, 0x00047dc7ec114c00LL);
|
|
failed += test_i64_constant_shift("i64_srl_const", i64_srl_const, tval_a, 0x0000000047dc7ec1LL);
|
|
failed += test_i64_variable_shift("i64_srl", i64_srl, tval_a, 10, 0x0000000047dc7ec1LL);
|
|
failed += test_i64_constant_shift("i64_sra_const", i64_sra_const, tval_a, 0x0000000047dc7ec1LL);
|
|
failed += test_i64_variable_shift("i64_sra", i64_sra, tval_a, 10, 0x0000000047dc7ec1LL);
|
|
|
|
if (failed > 0) {
|
|
printf(" %d tests ailed.\n", failed);
|
|
} else {
|
|
printf(" All tests passed.\n");
|
|
}
|
|
|
|
printf("----------------------------------------\n");
|
|
|
|
failed = 0;
|
|
puts("unsigned left/right shift tests:");
|
|
failed += test_u64_constant_shift("u64_shl_const", u64_shl_const, tval_f, 0xfffc1d404d7ae400LL);
|
|
failed += test_u64_variable_shift("u64_shl", u64_shl, tval_f, 10, 0xfffc1d404d7ae400LL);
|
|
failed += test_u64_constant_shift("u64_srl_const", u64_srl_const, tval_f, 0x003fffffc1d404d7LL);
|
|
failed += test_u64_variable_shift("u64_srl", u64_srl, tval_f, 10, 0x003fffffc1d404d7LL);
|
|
failed += test_i64_constant_shift("i64_sra_const", i64_sra_const, tval_f, 0xffffffffc1d404d7LL);
|
|
failed += test_i64_variable_shift("i64_sra", i64_sra, tval_f, 10, 0xffffffffc1d404d7LL);
|
|
failed += test_u64_constant_shift("u64_sra_const", u64_sra_const, tval_f, 0x003fffffc1d404d7LL);
|
|
failed += test_u64_variable_shift("u64_sra", u64_sra, tval_f, 10, 0x003fffffc1d404d7LL);
|
|
|
|
if (failed > 0) {
|
|
printf(" %d tests ailed.\n", failed);
|
|
} else {
|
|
printf(" All tests passed.\n");
|
|
}
|
|
|
|
printf("----------------------------------------\n");
|
|
|
|
int64_t result;
|
|
|
|
result = i64_mul(tval_g, tval_g);
|
|
printf("%20lld * %20lld = %20lld (0x%016llx)\n", tval_g, tval_g, result, result);
|
|
result = i64_mul(tval_d, tval_e);
|
|
printf("%20lld * %20lld = %20lld (0x%016llx)\n", tval_d, tval_e, result, result);
|
|
/* 0xba7a664f13077c9 */
|
|
result = i64_mul(tval_a, tval_b);
|
|
printf("%20lld * %20lld = %20lld (0x%016llx)\n", tval_a, tval_b, result, result);
|
|
|
|
printf("----------------------------------------\n");
|
|
|
|
return 0;
|
|
}
|