ref: c3f004633be7e8a15fe8e1e158dde24c2af2f765
parent: 467826415fe2cc2846097dfd37ec879b9d62cf62
author: Sigrid Solveig Haflínudóttir <sigrid@ftrv.se>
date: Sat Nov 16 22:05:28 EST 2024
import the rest of libmp tests from 9front
--- a/3rd/mp/test.c
+++ b/3rd/mp/test.c
@@ -436,7 +436,6 @@
if(argc == 3 && strcmp(argv[1], "-n") == 0)
loops = atoi(argv[2]);
- memset(str, 0, sizeof(str));
D_PNAN = D_NNAN = strtod("+NaN", nil);
D_PINF = D_NINF = strtod("+Inf", nil);
--- /dev/null
+++ b/3rd/mp/test/convtest.c
@@ -1,0 +1,126 @@
+#include "platform.h"
+#include "mp.h"
+#include "dat.h"
+#include "fns.h"
+
+/* these tests suck but better than nothing... but really should test more values than just 1<<i */
+
+#define MPTOX(_name,_type,_func) \
+static void \
+_name(void) \
+{ \
+ mpint *m; \
+ int i, sign, mag; \
+ _type v, e; \
+ int fail; \
+ \
+ fail = 0; \
+ m = mpnew(0); \
+ for(i = 0; i < 256; i++){ \
+ sign = i >= 128 ? -1 : 1; \
+ mag = i % 128; \
+ itomp(sign, m); \
+ mpleft(m, mag, m); \
+ v = _func(m); \
+ e = 0xcafebabe; USED(e);
+#define MPTOX_END(_func,_format) \
+ if(v != e){ \
+ fprintf(stderr, "FAIL: "#_func"(%s): return value: got "_format", expected "_format"\n", MFMT(m), v, e); \
+ fail=1; \
+ } \
+ } \
+ mpfree(m); \
+ if(!fail) \
+ fprintf(stderr, #_func": passed\n"); \
+}
+
+#define XTOMP(_name,_type,_func) \
+static void \
+_name(void) \
+{ \
+ mpint *m, *r; \
+ int i, sign, mag; \
+ _type v; \
+ int fail; \
+ \
+ fail = 0; \
+ m = mpnew(0); \
+ r = mpnew(0); \
+ for(i = 0; i < 256; i++){ \
+ sign = i >= 128 ? -1 : 1; \
+ mag = i % 128; \
+ itomp(sign, r); \
+ mpleft(r, mag, r);
+#define XTOMP_END(_func,_type,_format) \
+ _func(sign * ((_type)1<<mag), m); \
+ if(mpcmp(r, m) != 0){ \
+ fprintf(stderr, "FAIL: "#_func"("_format"): return value: got %s, expected %s\n", sign * ((_type)1<<mag), MFMT(m), MFMT(r)); \
+ fail=1; \
+ } \
+ } \
+ mpfree(m); \
+ mpfree(r); \
+ if(!fail) \
+ fprintf(stderr, #_func": passed\n"); \
+ USED(v); \
+}
+
+MPTOX(test_mptoi, int, mptoi)
+ if(mag < 31)
+ e = sign*(1<<mag);
+ else
+ e = sign > 0 ? (1U<<31)-1 : 1U<<31;
+MPTOX_END(mptoi, "%#x")
+
+MPTOX(test_mptoui, uint32_t, mptoui)
+ if(mag < 32 && sign > 0)
+ e = 1<<mag;
+ else
+ e = sign > 0 ? -1 : 0;
+MPTOX_END(mptoui, "%#x")
+
+
+MPTOX(test_mptov, int64_t, mptov)
+ if(mag < 63)
+ e = sign*(1LL<<mag);
+ else
+ e = sign > 0 ? (1ULL<<63)-1 : 1ULL<<63;
+MPTOX_END(mptov, "%#"PRIx64)
+
+MPTOX(test_mptouv, uint64_t, mptouv)
+ if(mag < 64 && sign > 0)
+ e = 1LL<<mag;
+ else
+ e = sign > 0 ? -1ULL : 0;
+MPTOX_END(mptouv, "%#"PRIx64)
+
+XTOMP(test_itomp, int, itomp)
+ if(mag >= 31) continue;
+XTOMP_END(vtomp, int64_t, "%"PRId64)
+
+XTOMP(test_uitomp, uint32_t, uitomp)
+ if(mag >= 32 || sign < 0) continue;
+XTOMP_END(uitomp, uint32_t, "%"PRIu32)
+
+XTOMP(test_vtomp, int64_t, vtomp)
+ if(mag >= 63) continue;
+XTOMP_END(vtomp, int64_t, "%"PRId64)
+
+XTOMP(test_uvtomp, uint64_t, uvtomp)
+ if(mag >= 64 || sign < 0) continue;
+XTOMP_END(uvtomp, int64_t, "%"PRId64)
+
+
+void
+convtests(void)
+{
+ test_mptoi();
+ test_mptoui();
+ test_mptov();
+ test_mptouv();
+ test_itomp();
+ test_uitomp();
+ test_vtomp();
+ test_uvtomp();
+}
+
--- /dev/null
+++ b/3rd/mp/test/dat.h
@@ -1,0 +1,8 @@
+typedef struct ldint ldint;
+
+struct ldint {
+ uint32_t n;
+ uint8_t *b;
+};
+
+enum {NTEST = 2 * 257 + 32};
--- /dev/null
+++ b/3rd/mp/test/fns.h
@@ -1,0 +1,28 @@
+ldint* ldnew(int);
+int ldcmp(ldint *, ldint *);
+int ldmagcmp(ldint *, ldint *);
+void ldadd(ldint *, ldint *, ldint *);
+void ldsub(ldint *, ldint *, ldint *);
+void ldmagadd(ldint *, ldint *, ldint *);
+void ldmagsub(ldint *, ldint *, ldint *);
+void ldand(ldint *, ldint *, ldint *);
+void ldbic(ldint *, ldint *, ldint *);
+void ldor(ldint *, ldint *, ldint *);
+void ldxor(ldint *, ldint *, ldint *);
+void ldnot(ldint *, ldint *);
+void ldleft(ldint *, int, ldint *);
+void ldright(ldint *, int, ldint *);
+void ldasr(ldint *, int, ldint *);
+void lddiv_(ldint *, ldint *, ldint *, ldint *);
+void ldfree(ldint *);
+void testgen(int, ldint *);
+int ldmpeq(ldint *, mpint *);
+mpint* ldtomp(ldint *, mpint *);
+void mptarget(mpint *);
+void tests(void);
+void mpdiv_(mpint *, mpint *, mpint *, mpint *);
+void convtests(void);
+
+char *LFMT(ldint *);
+char *MFMT(mpint *);
+void prng(uint8_t *p, int n);
--- /dev/null
+++ b/3rd/mp/test/gen.tab.c
@@ -1,0 +1,1170 @@
+#include "platform.h"
+#include "mp.h"
+#include "dat.h"
+#include "fns.h"
+static void
+testcmp(void)
+{
+ int fail=0;
+ int a, b;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ ldint *l3 = ldnew(0);
+ mpint *m3 = mpnew(0);
+ int i3;
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=0;i3<NTEST;i3++){
+ testgen(i3, l3);
+ a=ldcmp(l2, l3);
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ b=mpcmp(m2, m3);
+ if(b<0) b=-1;
+ if(b>0) b=1;
+ if(a != b){
+ fprintf(stderr, "FAIL: mpcmp(%s, %s): return value: got %d, expected %d\n", LFMT(l2), LFMT(l3), b, a);
+ fail=1;
+ }
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpcmp(%s, %s): arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpcmp(%s, %s): arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l3);
+ mpfree(m3);
+ if(!fail) fprintf(stderr, "mpcmp: passed\n");
+}
+static void
+testmagcmp(void)
+{
+ int fail=0;
+ int a, b;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ ldint *l3 = ldnew(0);
+ mpint *m3 = mpnew(0);
+ int i3;
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=0;i3<NTEST;i3++){
+ testgen(i3, l3);
+ a=ldmagcmp(l2, l3);
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ b=mpmagcmp(m2, m3);
+ if(b<0) b=-1;
+ if(b>0) b=1;
+ if(a != b){
+ fprintf(stderr, "FAIL: mpmagcmp(%s, %s): return value: got %d, expected %d\n", LFMT(l2), LFMT(l3), b, a);
+ fail=1;
+ }
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpmagcmp(%s, %s): arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpmagcmp(%s, %s): arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l3);
+ mpfree(m3);
+ if(!fail) fprintf(stderr, "mpmagcmp: passed\n");
+}
+static void
+testadd(void)
+{
+ int fail=0;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ ldint *l3 = ldnew(0);
+ mpint *m3 = mpnew(0);
+ int i3;
+ ldint *l4 = ldnew(0);
+ mpint *m4 = mpnew(0);
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=0;i3<NTEST;i3++){
+ testgen(i3, l3);
+ ldadd(l2, l3, l4);
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpadd(m2, m3, m4);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpadd(%s, %s, out): arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpadd(%s, %s, out): arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpadd(%s, %s, out): arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpadd(%s, %s, out): arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpadd(%s, %s, out): arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpadd(m2, m3, m2);
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpadd(%s, %s, out): alias 3 and 1: arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpadd(%s, %s, out): alias 3 and 1: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpadd(%s, %s, out): alias 3 and 1: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m2)){
+ fprintf(stderr, "FAIL: mpadd(%s, %s, out): alias 3 and 1: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m2), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpadd(m2, m3, m3);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpadd(%s, %s, out): alias 3 and 2: arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m3->top == 0 && m3->sign < 0){
+ fprintf(stderr, "FAIL: mpadd(%s, %s, out): alias 3 and 2: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m3->top != 0 && m3->p[m3->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpadd(%s, %s, out): alias 3 and 2: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m3)){
+ fprintf(stderr, "FAIL: mpadd(%s, %s, out): alias 3 and 2: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m3), LFMT(l4));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l3);
+ mpfree(m3);
+ ldfree(l4);
+ mpfree(m4);
+ if(!fail) fprintf(stderr, "mpadd: passed\n");
+}
+static void
+testsub(void)
+{
+ int fail=0;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ ldint *l3 = ldnew(0);
+ mpint *m3 = mpnew(0);
+ int i3;
+ ldint *l4 = ldnew(0);
+ mpint *m4 = mpnew(0);
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=0;i3<NTEST;i3++){
+ testgen(i3, l3);
+ ldsub(l2, l3, l4);
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpsub(m2, m3, m4);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpsub(%s, %s, out): arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpsub(%s, %s, out): arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpsub(%s, %s, out): arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpsub(%s, %s, out): arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpsub(%s, %s, out): arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpsub(m2, m3, m2);
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpsub(%s, %s, out): alias 3 and 1: arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpsub(%s, %s, out): alias 3 and 1: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpsub(%s, %s, out): alias 3 and 1: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m2)){
+ fprintf(stderr, "FAIL: mpsub(%s, %s, out): alias 3 and 1: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m2), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpsub(m2, m3, m3);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpsub(%s, %s, out): alias 3 and 2: arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m3->top == 0 && m3->sign < 0){
+ fprintf(stderr, "FAIL: mpsub(%s, %s, out): alias 3 and 2: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m3->top != 0 && m3->p[m3->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpsub(%s, %s, out): alias 3 and 2: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m3)){
+ fprintf(stderr, "FAIL: mpsub(%s, %s, out): alias 3 and 2: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m3), LFMT(l4));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l3);
+ mpfree(m3);
+ ldfree(l4);
+ mpfree(m4);
+ if(!fail) fprintf(stderr, "mpsub: passed\n");
+}
+static void
+testmagadd(void)
+{
+ int fail=0;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ ldint *l3 = ldnew(0);
+ mpint *m3 = mpnew(0);
+ int i3;
+ ldint *l4 = ldnew(0);
+ mpint *m4 = mpnew(0);
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=0;i3<NTEST;i3++){
+ testgen(i3, l3);
+ ldmagadd(l2, l3, l4);
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpmagadd(m2, m3, m4);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpmagadd(%s, %s, out): arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpmagadd(%s, %s, out): arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpmagadd(%s, %s, out): arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpmagadd(%s, %s, out): arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpmagadd(%s, %s, out): arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpmagadd(m2, m3, m2);
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpmagadd(%s, %s, out): alias 3 and 1: arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpmagadd(%s, %s, out): alias 3 and 1: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpmagadd(%s, %s, out): alias 3 and 1: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m2)){
+ fprintf(stderr, "FAIL: mpmagadd(%s, %s, out): alias 3 and 1: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m2), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpmagadd(m2, m3, m3);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpmagadd(%s, %s, out): alias 3 and 2: arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m3->top == 0 && m3->sign < 0){
+ fprintf(stderr, "FAIL: mpmagadd(%s, %s, out): alias 3 and 2: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m3->top != 0 && m3->p[m3->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpmagadd(%s, %s, out): alias 3 and 2: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m3)){
+ fprintf(stderr, "FAIL: mpmagadd(%s, %s, out): alias 3 and 2: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m3), LFMT(l4));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l3);
+ mpfree(m3);
+ ldfree(l4);
+ mpfree(m4);
+ if(!fail) fprintf(stderr, "mpmagadd: passed\n");
+}
+static void
+testmagsub(void)
+{
+ int fail=0;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ ldint *l3 = ldnew(0);
+ mpint *m3 = mpnew(0);
+ int i3;
+ ldint *l4 = ldnew(0);
+ mpint *m4 = mpnew(0);
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=0;i3<NTEST;i3++){
+ testgen(i3, l3);
+ ldmagsub(l2, l3, l4);
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpmagsub(m2, m3, m4);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpmagsub(%s, %s, out): arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpmagsub(%s, %s, out): arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpmagsub(%s, %s, out): arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpmagsub(%s, %s, out): arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpmagsub(%s, %s, out): arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpmagsub(m2, m3, m2);
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpmagsub(%s, %s, out): alias 3 and 1: arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpmagsub(%s, %s, out): alias 3 and 1: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpmagsub(%s, %s, out): alias 3 and 1: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m2)){
+ fprintf(stderr, "FAIL: mpmagsub(%s, %s, out): alias 3 and 1: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m2), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpmagsub(m2, m3, m3);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpmagsub(%s, %s, out): alias 3 and 2: arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m3->top == 0 && m3->sign < 0){
+ fprintf(stderr, "FAIL: mpmagsub(%s, %s, out): alias 3 and 2: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m3->top != 0 && m3->p[m3->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpmagsub(%s, %s, out): alias 3 and 2: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m3)){
+ fprintf(stderr, "FAIL: mpmagsub(%s, %s, out): alias 3 and 2: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m3), LFMT(l4));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l3);
+ mpfree(m3);
+ ldfree(l4);
+ mpfree(m4);
+ if(!fail) fprintf(stderr, "mpmagsub: passed\n");
+}
+static void
+testand(void)
+{
+ int fail=0;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ ldint *l3 = ldnew(0);
+ mpint *m3 = mpnew(0);
+ int i3;
+ ldint *l4 = ldnew(0);
+ mpint *m4 = mpnew(0);
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=0;i3<NTEST;i3++){
+ testgen(i3, l3);
+ ldand(l2, l3, l4);
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpand(m2, m3, m4);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpand(%s, %s, out): arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpand(%s, %s, out): arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpand(%s, %s, out): arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpand(%s, %s, out): arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpand(%s, %s, out): arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpand(m2, m3, m2);
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpand(%s, %s, out): alias 3 and 1: arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpand(%s, %s, out): alias 3 and 1: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpand(%s, %s, out): alias 3 and 1: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m2)){
+ fprintf(stderr, "FAIL: mpand(%s, %s, out): alias 3 and 1: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m2), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpand(m2, m3, m3);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpand(%s, %s, out): alias 3 and 2: arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m3->top == 0 && m3->sign < 0){
+ fprintf(stderr, "FAIL: mpand(%s, %s, out): alias 3 and 2: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m3->top != 0 && m3->p[m3->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpand(%s, %s, out): alias 3 and 2: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m3)){
+ fprintf(stderr, "FAIL: mpand(%s, %s, out): alias 3 and 2: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m3), LFMT(l4));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l3);
+ mpfree(m3);
+ ldfree(l4);
+ mpfree(m4);
+ if(!fail) fprintf(stderr, "mpand: passed\n");
+}
+static void
+testbic(void)
+{
+ int fail=0;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ ldint *l3 = ldnew(0);
+ mpint *m3 = mpnew(0);
+ int i3;
+ ldint *l4 = ldnew(0);
+ mpint *m4 = mpnew(0);
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=0;i3<NTEST;i3++){
+ testgen(i3, l3);
+ ldbic(l2, l3, l4);
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpbic(m2, m3, m4);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpbic(%s, %s, out): arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpbic(%s, %s, out): arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpbic(%s, %s, out): arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpbic(%s, %s, out): arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpbic(%s, %s, out): arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpbic(m2, m3, m2);
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpbic(%s, %s, out): alias 3 and 1: arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpbic(%s, %s, out): alias 3 and 1: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpbic(%s, %s, out): alias 3 and 1: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m2)){
+ fprintf(stderr, "FAIL: mpbic(%s, %s, out): alias 3 and 1: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m2), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpbic(m2, m3, m3);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpbic(%s, %s, out): alias 3 and 2: arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m3->top == 0 && m3->sign < 0){
+ fprintf(stderr, "FAIL: mpbic(%s, %s, out): alias 3 and 2: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m3->top != 0 && m3->p[m3->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpbic(%s, %s, out): alias 3 and 2: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m3)){
+ fprintf(stderr, "FAIL: mpbic(%s, %s, out): alias 3 and 2: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m3), LFMT(l4));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l3);
+ mpfree(m3);
+ ldfree(l4);
+ mpfree(m4);
+ if(!fail) fprintf(stderr, "mpbic: passed\n");
+}
+static void
+testor(void)
+{
+ int fail=0;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ ldint *l3 = ldnew(0);
+ mpint *m3 = mpnew(0);
+ int i3;
+ ldint *l4 = ldnew(0);
+ mpint *m4 = mpnew(0);
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=0;i3<NTEST;i3++){
+ testgen(i3, l3);
+ ldor(l2, l3, l4);
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpor(m2, m3, m4);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpor(%s, %s, out): arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpor(%s, %s, out): arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpor(%s, %s, out): arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpor(%s, %s, out): arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpor(%s, %s, out): arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpor(m2, m3, m2);
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpor(%s, %s, out): alias 3 and 1: arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpor(%s, %s, out): alias 3 and 1: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpor(%s, %s, out): alias 3 and 1: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m2)){
+ fprintf(stderr, "FAIL: mpor(%s, %s, out): alias 3 and 1: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m2), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpor(m2, m3, m3);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpor(%s, %s, out): alias 3 and 2: arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m3->top == 0 && m3->sign < 0){
+ fprintf(stderr, "FAIL: mpor(%s, %s, out): alias 3 and 2: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m3->top != 0 && m3->p[m3->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpor(%s, %s, out): alias 3 and 2: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m3)){
+ fprintf(stderr, "FAIL: mpor(%s, %s, out): alias 3 and 2: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m3), LFMT(l4));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l3);
+ mpfree(m3);
+ ldfree(l4);
+ mpfree(m4);
+ if(!fail) fprintf(stderr, "mpor: passed\n");
+}
+static void
+testxor(void)
+{
+ int fail=0;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ ldint *l3 = ldnew(0);
+ mpint *m3 = mpnew(0);
+ int i3;
+ ldint *l4 = ldnew(0);
+ mpint *m4 = mpnew(0);
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=0;i3<NTEST;i3++){
+ testgen(i3, l3);
+ ldxor(l2, l3, l4);
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpxor(m2, m3, m4);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpxor(%s, %s, out): arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpxor(%s, %s, out): arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpxor(%s, %s, out): arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpxor(%s, %s, out): arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpxor(%s, %s, out): arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpxor(m2, m3, m2);
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpxor(%s, %s, out): alias 3 and 1: arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpxor(%s, %s, out): alias 3 and 1: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpxor(%s, %s, out): alias 3 and 1: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m2)){
+ fprintf(stderr, "FAIL: mpxor(%s, %s, out): alias 3 and 1: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m2), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mpxor(m2, m3, m3);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpxor(%s, %s, out): alias 3 and 2: arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m3->top == 0 && m3->sign < 0){
+ fprintf(stderr, "FAIL: mpxor(%s, %s, out): alias 3 and 2: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m3->top != 0 && m3->p[m3->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpxor(%s, %s, out): alias 3 and 2: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m3)){
+ fprintf(stderr, "FAIL: mpxor(%s, %s, out): alias 3 and 2: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m3), LFMT(l4));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l3);
+ mpfree(m3);
+ ldfree(l4);
+ mpfree(m4);
+ if(!fail) fprintf(stderr, "mpxor: passed\n");
+}
+static void
+testnot(void)
+{
+ int fail=0;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ ldint *l3 = ldnew(0);
+ mpint *m3 = mpnew(0);
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ ldnot(l2, l3);
+ ldtomp(l2, m2);
+ mptarget(m3);
+ mpnot(m2, m3);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpnot(%s, out): arg 1: input corrupted\n", LFMT(l2));
+ fail=1;
+ }
+ if(m3->top == 0 && m3->sign < 0){
+ fprintf(stderr, "FAIL: mpnot(%s, out): arg 2: got -0, shouldn't happen\n", LFMT(l2));
+ fail=1;
+ }else if(m3->top != 0 && m3->p[m3->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpnot(%s, out): arg 2: got denormalised result\n", LFMT(l2));
+ fail=1;
+ }else if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpnot(%s, out): arg 2: got %s, expected %s\n", LFMT(l2), MFMT(m3), LFMT(l3));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ mptarget(m3);
+ mpnot(m2, m2);
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpnot(%s, out): alias 2 and 1: arg 2: got -0, shouldn't happen\n", LFMT(l2));
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpnot(%s, out): alias 2 and 1: arg 2: got denormalised result\n", LFMT(l2));
+ fail=1;
+ }else if(!ldmpeq(l3, m2)){
+ fprintf(stderr, "FAIL: mpnot(%s, out): alias 2 and 1: arg 2: got %s, expected %s\n", LFMT(l2), MFMT(m2), LFMT(l3));
+ fail=1;
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l3);
+ mpfree(m3);
+ if(!fail) fprintf(stderr, "mpnot: passed\n");
+}
+static void
+testleft(void)
+{
+ int fail=0;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ int i3;
+ ldint *l4 = ldnew(0);
+ mpint *m4 = mpnew(0);
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=-128;i3<=128;i3++){
+ ldleft(l2, i3, l4);
+ ldtomp(l2, m2);
+ mptarget(m4);
+ mpleft(m2, i3, m4);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpleft(%s, %d, out): arg 1: input corrupted\n", LFMT(l2), i3);
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpleft(%s, %d, out): arg 3: got -0, shouldn't happen\n", LFMT(l2), i3);
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpleft(%s, %d, out): arg 3: got denormalised result\n", LFMT(l2), i3);
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpleft(%s, %d, out): arg 3: got %s, expected %s\n", LFMT(l2), i3, MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ mptarget(m4);
+ mpleft(m2, i3, m2);
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpleft(%s, %d, out): alias 3 and 1: arg 3: got -0, shouldn't happen\n", LFMT(l2), i3);
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpleft(%s, %d, out): alias 3 and 1: arg 3: got denormalised result\n", LFMT(l2), i3);
+ fail=1;
+ }else if(!ldmpeq(l4, m2)){
+ fprintf(stderr, "FAIL: mpleft(%s, %d, out): alias 3 and 1: arg 3: got %s, expected %s\n", LFMT(l2), i3, MFMT(m2), LFMT(l4));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l4);
+ mpfree(m4);
+ if(!fail) fprintf(stderr, "mpleft: passed\n");
+}
+static void
+testright(void)
+{
+ int fail=0;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ int i3;
+ ldint *l4 = ldnew(0);
+ mpint *m4 = mpnew(0);
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=-128;i3<=128;i3++){
+ ldright(l2, i3, l4);
+ ldtomp(l2, m2);
+ mptarget(m4);
+ mpright(m2, i3, m4);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpright(%s, %d, out): arg 1: input corrupted\n", LFMT(l2), i3);
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpright(%s, %d, out): arg 3: got -0, shouldn't happen\n", LFMT(l2), i3);
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpright(%s, %d, out): arg 3: got denormalised result\n", LFMT(l2), i3);
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpright(%s, %d, out): arg 3: got %s, expected %s\n", LFMT(l2), i3, MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ mptarget(m4);
+ mpright(m2, i3, m2);
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpright(%s, %d, out): alias 3 and 1: arg 3: got -0, shouldn't happen\n", LFMT(l2), i3);
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpright(%s, %d, out): alias 3 and 1: arg 3: got denormalised result\n", LFMT(l2), i3);
+ fail=1;
+ }else if(!ldmpeq(l4, m2)){
+ fprintf(stderr, "FAIL: mpright(%s, %d, out): alias 3 and 1: arg 3: got %s, expected %s\n", LFMT(l2), i3, MFMT(m2), LFMT(l4));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l4);
+ mpfree(m4);
+ if(!fail) fprintf(stderr, "mpright: passed\n");
+}
+static void
+testasr(void)
+{
+ int fail=0;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ int i3;
+ ldint *l4 = ldnew(0);
+ mpint *m4 = mpnew(0);
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=-128;i3<=128;i3++){
+ ldasr(l2, i3, l4);
+ ldtomp(l2, m2);
+ mptarget(m4);
+ mpasr(m2, i3, m4);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpasr(%s, %d, out): arg 1: input corrupted\n", LFMT(l2), i3);
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpasr(%s, %d, out): arg 3: got -0, shouldn't happen\n", LFMT(l2), i3);
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpasr(%s, %d, out): arg 3: got denormalised result\n", LFMT(l2), i3);
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpasr(%s, %d, out): arg 3: got %s, expected %s\n", LFMT(l2), i3, MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ mptarget(m4);
+ mpasr(m2, i3, m2);
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpasr(%s, %d, out): alias 3 and 1: arg 3: got -0, shouldn't happen\n", LFMT(l2), i3);
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpasr(%s, %d, out): alias 3 and 1: arg 3: got denormalised result\n", LFMT(l2), i3);
+ fail=1;
+ }else if(!ldmpeq(l4, m2)){
+ fprintf(stderr, "FAIL: mpasr(%s, %d, out): alias 3 and 1: arg 3: got %s, expected %s\n", LFMT(l2), i3, MFMT(m2), LFMT(l4));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l4);
+ mpfree(m4);
+ if(!fail) fprintf(stderr, "mpasr: passed\n");
+}
+static void
+testdiv_(void)
+{
+ int fail=0;
+ ldint *l2 = ldnew(0);
+ mpint *m2 = mpnew(0);
+ int i2;
+ ldint *l3 = ldnew(0);
+ mpint *m3 = mpnew(0);
+ int i3;
+ ldint *l4 = ldnew(0);
+ mpint *m4 = mpnew(0);
+ ldint *l5 = ldnew(0);
+ mpint *m5 = mpnew(0);
+ for(i2=0;i2<NTEST;i2++){
+ testgen(i2, l2);
+ for(i3=0;i3<NTEST;i3++){
+ testgen(i3, l3);
+ lddiv_(l2, l3, l4, l5);
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mptarget(m5);
+ mpdiv_(m2, m3, m4, m5);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ if(m5->top == 0 && m5->sign < 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): arg 4: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m5->top != 0 && m5->p[m5->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): arg 4: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l5, m5)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): arg 4: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m5), LFMT(l5));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mptarget(m5);
+ mpdiv_(m2, m3, m2, m5);
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 1: arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m5->top == 0 && m5->sign < 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 1: arg 4: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m5->top != 0 && m5->p[m5->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 1: arg 4: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l5, m5)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 1: arg 4: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m5), LFMT(l5));
+ fail=1;
+ }
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 1: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 1: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m2)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 1: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m2), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mptarget(m5);
+ mpdiv_(m2, m3, m3, m5);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 2: arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m5->top == 0 && m5->sign < 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 2: arg 4: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m5->top != 0 && m5->p[m5->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 2: arg 4: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l5, m5)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 2: arg 4: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m5), LFMT(l5));
+ fail=1;
+ }
+ if(m3->top == 0 && m3->sign < 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 2: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m3->top != 0 && m3->p[m3->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 2: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m3)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 3 and 2: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m3), LFMT(l4));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mptarget(m5);
+ mpdiv_(m2, m3, m4, m2);
+ if(!ldmpeq(l3, m3)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 1: arg 2: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 1: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 1: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 1: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ if(m2->top == 0 && m2->sign < 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 1: arg 4: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m2->top != 0 && m2->p[m2->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 1: arg 4: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l5, m2)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 1: arg 4: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m2), LFMT(l5));
+ fail=1;
+ }
+ ldtomp(l2, m2);
+ ldtomp(l3, m3);
+ mptarget(m4);
+ mptarget(m5);
+ mpdiv_(m2, m3, m4, m3);
+ if(!ldmpeq(l2, m2)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 2: arg 1: input corrupted\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }
+ if(m4->top == 0 && m4->sign < 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 2: arg 3: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m4->top != 0 && m4->p[m4->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 2: arg 3: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l4, m4)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 2: arg 3: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m4), LFMT(l4));
+ fail=1;
+ }
+ if(m3->top == 0 && m3->sign < 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 2: arg 4: got -0, shouldn't happen\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(m3->top != 0 && m3->p[m3->top - 1] == 0){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 2: arg 4: got denormalised result\n", LFMT(l2), LFMT(l3));
+ fail=1;
+ }else if(!ldmpeq(l5, m3)){
+ fprintf(stderr, "FAIL: mpdiv_(%s, %s, out, out): alias 4 and 2: arg 4: got %s, expected %s\n", LFMT(l2), LFMT(l3), MFMT(m3), LFMT(l5));
+ fail=1;
+ }
+ }
+ }
+ ldfree(l2);
+ mpfree(m2);
+ ldfree(l3);
+ mpfree(m3);
+ ldfree(l4);
+ mpfree(m4);
+ ldfree(l5);
+ mpfree(m5);
+ if(!fail) fprintf(stderr, "mpdiv_: passed\n");
+}
+void
+tests(void)
+{
+ testcmp();
+ testmagcmp();
+ testadd();
+ testsub();
+ testmagadd();
+ testmagsub();
+ testand();
+ testbic();
+ testor();
+ testxor();
+ testnot();
+ testleft();
+ testright();
+ testasr();
+ testdiv_();
+}
--- /dev/null
+++ b/3rd/mp/test/ld.c
@@ -1,0 +1,560 @@
+#include "platform.h"
+#include "mp.h"
+#include "dat.h"
+#include "fns.h"
+
+static int
+ldget(ldint *a, uint32_t n)
+{
+ if(n < 0) return 0;
+ if(n >= a->n) return a->b[a->n - 1]&1;
+ return a->b[n]&1;
+}
+
+static void
+ldbits(ldint *a, int n)
+{
+ a->b = realloc(a->b, n);
+ a->n = n;
+}
+
+static ldint *
+ldnorm(ldint *a)
+{
+ int i;
+
+ if(a->n > 0){
+ for(i = a->n - 2; i >= 0; i--)
+ if(a->b[i] != a->b[a->n-1])
+ break;
+ ldbits(a, i + 2);
+ }else{
+ ldbits(a, 1);
+ a->b[0] = 0;
+ }
+ return a;
+}
+
+static void
+ldneg(ldint *a)
+{
+ int c, s, z;
+ uint32_t i;
+
+ c = 1;
+ s = a->b[a->n - 1];
+ z = 1;
+ for(i = 0; i < a->n; i++){
+ if(a->b[i]) z = 0;
+ c += 1 ^ a->b[i] & 1;
+ a->b[i] = c & 1;
+ c >>= 1;
+ }
+ if(!z && s == a->b[a->n - 1]){
+ ldbits(a, a->n + 1);
+ a->b[a->n - 1] = !s;
+ }
+}
+
+static int
+max(int a, int b)
+{
+ return a>b? a : b;
+}
+
+ldint *
+ldnew(int n)
+{
+ ldint *a;
+
+ a = malloc(sizeof(ldint));
+ if(n <= 0) n = 1;
+ a->b = malloc(n);
+ a->n = n;
+ return a;
+}
+
+void
+ldfree(ldint *a)
+{
+ if(a == nil) return;
+ free(a->b);
+ free(a);
+}
+
+mpint *
+ldtomp(ldint *a, mpint *b)
+{
+ int s, c;
+ uint32_t i;
+
+ if(b == nil)
+ b = mpnew(0);
+ mpbits(b, a->n);
+ s = a->b[a->n - 1] & 1;
+ b->sign = 1 - 2 * s;
+ c = s;
+ memset(b->p, 0, (a->n + Dbits - 1) / Dbits * Dbytes);
+ for(i = 0; i < a->n; i++){
+ c += s ^ a->b[i] & 1;
+ b->p[i / Dbits] |= (mpdigit)(c & 1) << (i & Dbits - 1);
+ c >>= 1;
+ }
+ b->top = (a->n + Dbits - 1) / Dbits;
+ mpnorm(b);
+ return b;
+}
+
+static ldint *
+itold(int n, ldint *a)
+{
+ uint32_t i;
+
+ if(a == nil)
+ a = ldnew(sizeof(n)*8);
+ else
+ ldbits(a, sizeof(n)*8);
+ for(i = 0; i < sizeof(n)*8; i++)
+ a->b[i] = n >> i & 1;
+ ldnorm(a);
+ return a;
+}
+
+static ldint *
+pow2told(int n, ldint *a)
+{
+ int k;
+
+ k = abs(n);
+ if(a == nil)
+ a = ldnew(k+2);
+ else
+ ldbits(a, k+2);
+ memset(a->b, 0, k+2);
+ a->b[k] = 1;
+ if(n < 0) ldneg(a);
+ ldnorm(a);
+ return a;
+}
+
+static char str[16][8192];
+static int istr = -1;
+
+char *
+LFMT(ldint *a)
+{
+ char *b, *p;
+ int s, c;
+ uint32_t i, d;
+
+ istr = (istr+1) % nelem(str);
+ b = str[istr] + 3;
+ d = (a->n + 3) / 4;
+ c = s = a->b[a->n - 1];
+ for(i = 0; i < a->n; i++){
+ c += s^ldget(a, i);
+ b[d - 1 - (i >> 2)] |= (c & 1) << (i & 3);
+ c >>= 1;
+ }
+ for(i = 0; i < d; i++)
+ b[i] = "0123456789ABCDEF"[(int)b[i]];
+ p = b;
+ while(*p == '0' && p[1] != 0) p++;
+ *--p = 'x';
+ *--p = '0';
+ if(a->b[a->n - 1])
+ *--p = '-';
+ return p;
+}
+
+char *
+MFMT(mpint *m)
+{
+ char *b;
+ istr = (istr+1) % nelem(str);
+ b = str[istr];
+ return mptoa(m, 16, b, sizeof(str[istr]));
+}
+
+int
+ldcmp(ldint *a, ldint *b)
+{
+ int x, y;
+ int i, r;
+
+ r = max(a->n, b->n);
+ if(a->b[a->n-1] != b->b[b->n-1])
+ return b->b[b->n - 1] - a->b[a->n - 1];
+ for(i = r - 1; --i >= 0; ){
+ x = ldget(a, i);
+ y = ldget(b, i);
+ if(x != y)
+ return x - y;
+ }
+ return 0;
+}
+
+int
+ldmagcmp(ldint *a, ldint *b)
+{
+ int s1, s2, r;
+
+ s1 = a->b[a->n - 1];
+ s2 = b->b[b->n - 1];
+ if(s1) ldneg(a);
+ if(s2) ldneg(b);
+ r = ldcmp(a, b);
+ if(s1) ldneg(a);
+ if(s2) ldneg(b);
+ return r;
+}
+
+int
+ldmpeq(ldint *a, mpint *b)
+{
+ uint32_t i, c;
+
+ if(b->sign > 0){
+ for(i = 0; i < b->top * Dbits; i++)
+ if(ldget(a, i) != (b->p[i / Dbits] >> (i & Dbits - 1) & 1))
+ return 0;
+ for(; i < a->n; i++)
+ if(a->b[i] != 0)
+ return 0;
+ return 1;
+ }else{
+ c = 1;
+ for(i = 0; i < b->top * Dbits; i++){
+ c += !ldget(a, i);
+ if((c & 1) != (b->p[i / Dbits] >> (i & Dbits - 1) & 1))
+ return 0;
+ c >>= 1;
+ }
+ for(; i < a->n; i++)
+ if(a->b[i] != 1)
+ return 0;
+ return 1;
+ }
+}
+
+void
+mptarget(mpint *r)
+{
+ int n;
+
+ n = rand() & 15;
+ mpbits(r, n * Dbits);
+ r->top = n;
+ prng((void *) r->p, n * Dbytes);
+ r->sign = 1 - 2 * (rand() & 1);
+}
+
+void
+ldadd(ldint *a, ldint *b, ldint *q)
+{
+ int r, i, c;
+
+ r = max(a->n, b->n) + 1;
+ ldbits(q, r);
+ c = 0;
+ for(i = 0; i < r; i++){
+ c += ldget(a, i) + ldget(b, i);
+ q->b[i] = c & 1;
+ c >>= 1;
+ }
+ ldnorm(q);
+}
+
+void
+ldmagadd(ldint *a, ldint *b, ldint *q)
+{
+ int i, r, s1, s2, c1, c2, co;
+
+ r = max(a->n, b->n) + 2;
+ ldbits(q, r);
+ co = 0;
+ s1 = c1 = a->b[a->n - 1] & 1;
+ s2 = c2 = b->b[b->n - 1] & 1;
+ for(i = 0; i < r; i++){
+ c1 += s1 ^ ldget(a, i) & 1;
+ c2 += s2 ^ ldget(b, i) & 1;
+ co += (c1 & 1) + (c2 & 1);
+ q->b[i] = co & 1;
+ co >>= 1;
+ c1 >>= 1;
+ c2 >>= 1;
+ }
+ ldnorm(q);
+}
+
+void
+ldmagsub(ldint *a, ldint *b, ldint *q)
+{
+ int i, r, s1, s2, c1, c2, co;
+
+ r = max(a->n, b->n) + 2;
+ ldbits(q, r);
+ co = 0;
+ s1 = c1 = a->b[a->n - 1] & 1;
+ s2 = c2 = 1 ^ b->b[b->n - 1] & 1;
+ for(i = 0; i < r; i++){
+ c1 += s1 ^ ldget(a, i) & 1;
+ c2 += s2 ^ ldget(b, i) & 1;
+ co += (c1 & 1) + (c2 & 1);
+ q->b[i] = co & 1;
+ co >>= 1;
+ c1 >>= 1;
+ c2 >>= 1;
+ }
+ ldnorm(q);
+}
+
+void
+ldsub(ldint *a, ldint *b, ldint *q)
+{
+ int r, i, c;
+
+ r = max(a->n, b->n) + 1;
+ ldbits(q, r);
+ c = 1;
+ for(i = 0; i < r; i++){
+ c += ldget(a, i) + (1^ldget(b, i));
+ q->b[i] = c & 1;
+ c >>= 1;
+ }
+ ldnorm(q);
+}
+
+static void
+lddiv(ldint *a, ldint *b, ldint *q, ldint *r)
+{
+ int n, i, j, c, s;
+
+ n = max(a->n, b->n) + 1;
+ ldbits(q, n);
+ ldbits(r, n);
+ memset(r->b, 0, n);
+ c = s = a->b[a->n-1];
+ for(i = 0; i < n; i++){
+ c += s ^ ldget(a, i);
+ q->b[i] = c & 1;
+ c >>= 1;
+ }
+ for(i = 0; i < n; i++){
+ for(j = n-1; --j >= 0; )
+ r->b[j + 1] = r->b[j];
+ r->b[0] = q->b[n - 1];
+ for(j = n-1; --j >= 0; )
+ q->b[j + 1] = q->b[j];
+ q->b[0] = !r->b[n - 1];
+ c = s = r->b[n - 1] == b->b[b->n - 1];
+ for(j = 0; j < n; j++){
+ c += r->b[j] + (s ^ ldget(b, j));
+ r->b[j] = c & 1;
+ c >>= 1;
+ }
+ }
+ for(j = n-1; --j >= 0; )
+ q->b[j + 1] = q->b[j];
+ q->b[0] = 1;
+ if(r->b[r->n - 1]){
+ c = 0;
+ for(j = 0; j < n; j++){
+ c += 1 + q->b[j];
+ q->b[j] = c & 1;
+ c >>= 1;
+ }
+ c = s = b->b[b->n - 1];
+ for(j = 0; j < n; j++){
+ c += r->b[j] + (s ^ ldget(b, j));
+ r->b[j] = c & 1;
+ c >>= 1;
+ }
+ }
+ c = s = a->b[a->n-1] ^ b->b[b->n-1];
+ for(j = 0; j < n; j++){
+ c += s ^ q->b[j];
+ q->b[j] = c & 1;
+ c >>= 1;
+ }
+ c = s = a->b[a->n-1];
+ for(j = 0; j < n; j++){
+ c += s ^ r->b[j];
+ r->b[j] = c & 1;
+ c >>= 1;
+ }
+ ldnorm(q);
+ ldnorm(r);
+}
+
+void
+lddiv_(ldint *a, ldint *b, ldint *q, ldint *r)
+{
+ if(ldmpeq(b, mpzero)){
+ memset(q->b, 0, q->n);
+ memset(r->b, 0, r->n);
+ return;
+ }
+ lddiv(a, b, q, r);
+}
+
+void
+mpdiv_(mpint *a, mpint *b, mpint *q, mpint *r)
+{
+ if(mpcmp(b, mpzero) == 0){
+ mpassign(mpzero, q);
+ mpassign(mpzero, r);
+ return;
+ }
+ mpdiv(a, b, q, r);
+}
+
+void
+ldand(ldint *a, ldint *b, ldint *q)
+{
+ int r, i;
+
+ r = max(a->n, b->n);
+ ldbits(q, r);
+ for(i = 0; i < r; i++)
+ q->b[i] = ldget(a, i) & ldget(b, i);
+ ldnorm(q);
+}
+
+void
+ldbic(ldint *a, ldint *b, ldint *q)
+{
+ int r, i;
+
+ r = max(a->n, b->n);
+ ldbits(q, r);
+ for(i = 0; i < r; i++)
+ q->b[i] = ldget(a, i) & ~ldget(b, i);
+ ldnorm(q);
+}
+
+void
+ldor(ldint *a, ldint *b, ldint *q)
+{
+ int r, i;
+
+ r = max(a->n, b->n);
+ ldbits(q, r);
+ for(i = 0; i < r; i++)
+ q->b[i] = ldget(a, i) | ldget(b, i);
+ ldnorm(q);
+}
+
+void
+ldxor(ldint *a, ldint *b, ldint *q)
+{
+ int r, i;
+
+ r = max(a->n, b->n);
+ ldbits(q, r);
+ for(i = 0; i < r; i++)
+ q->b[i] = ldget(a, i) ^ ldget(b, i);
+ ldnorm(q);
+}
+
+void
+ldleft(ldint *a, int n, ldint *b)
+{
+ int c;
+ uint32_t i;
+
+ if(n < 0){
+ if(a->n <= (uint32_t)-n){
+ b->n = 0;
+ ldnorm(b);
+ return;
+ }
+ c = 0;
+ if(a->b[a->n - 1])
+ for(i = 0; i < (uint32_t)-n; i++)
+ if(a->b[i]){
+ c = 1;
+ break;
+ }
+ ldbits(b, a->n + n);
+ for(i = 0; i < a->n + n; i++){
+ c += a->b[i - n] & 1;
+ b->b[i] = c & 1;
+ c >>= 1;
+ }
+ }else{
+ ldbits(b, a->n + n);
+ memmove(b->b + n, a->b, a->n);
+ memset(b->b, 0, n);
+ }
+ ldnorm(b);
+}
+
+void
+ldright(ldint *a, int n, ldint *b)
+{
+ ldleft(a, -n, b);
+}
+
+void
+ldasr(ldint *a, int n, ldint *b)
+{
+ if(n < 0){
+ ldleft(a, -n, b);
+ return;
+ }
+ if(a->n <= (uint32_t)n){
+ ldbits(b, 1);
+ b->b[0] = a->b[a->n - 1];
+ return;
+ }
+ ldbits(b, a->n - n);
+ memmove(b->b, a->b + n, a->n - n);
+ ldnorm(b);
+}
+
+void
+ldnot(ldint *a, ldint *b)
+{
+ uint32_t i;
+
+ ldbits(b, a->n);
+ for(i = 0; i < a->n; i++)
+ b->b[i] = a->b[i] ^ 1;
+}
+
+static uint32_t
+xorshift(uint32_t *state)
+{
+ uint32_t x = *state;
+ x ^= x << 13;
+ x ^= x >> 17;
+ x ^= x << 5;
+ *state = x;
+ return x;
+}
+
+void
+testgen(int i, ldint *a)
+{
+ uint32_t j, state;
+ uint32_t r = 0;
+
+ if(i < 257)
+ itold(i-128, a);
+ else if(i < 514)
+ pow2told(i-385, a);
+ else{
+ state = i;
+ xorshift(&state);
+ xorshift(&state);
+ xorshift(&state);
+ ldbits(a, Dbits * (1 + (xorshift(&state) & 15)));
+ for(j = 0; j < a->n; j++){
+ if((j & 31) == 0)
+ r = xorshift(&state);
+ a->b[j] = r & 1;
+ r >>= 1;
+ }
+ }
+}
--- /dev/null
+++ b/3rd/mp/test/main.c
@@ -1,0 +1,32 @@
+#include "platform.h"
+#include "mp.h"
+#include "dat.h"
+#include "fns.h"
+#include "ieee754.h"
+
+double D_PNAN, D_NNAN, D_PINF, D_NINF;
+float F_PNAN, F_NNAN, F_PINF, F_NINF;
+
+void
+prng(uint8_t *p, int n)
+{
+ while(n-- > 0)
+ *p++ = rand();
+}
+
+int
+main()
+{
+ D_PNAN = D_NNAN = strtod("+NaN", nil);
+ D_PINF = D_NINF = strtod("+Inf", nil);
+
+ union ieee754_double *d;
+ d = (union ieee754_double *)&D_NNAN;
+ d->ieee.negative = 1;
+ d = (union ieee754_double *)&D_NINF;
+ d->ieee.negative = 1;
+
+ convtests();
+ tests();
+ return 0;
+}
--- a/meson.build
+++ b/meson.build
@@ -25,40 +25,6 @@
language: 'c',
)
-mp = [
- '3rd/mp/mpadd.c',
- '3rd/mp/mpaux.c',
- '3rd/mp/mpcmp.c',
- '3rd/mp/mpdigdiv.c',
- '3rd/mp/mpdiv.c',
- '3rd/mp/mpextendedgcd.c',
- '3rd/mp/mpexp.c',
- '3rd/mp/mpfmt.c',
- '3rd/mp/mpinvert.c',
- '3rd/mp/mpleft.c',
- '3rd/mp/mplogic.c',
- '3rd/mp/mpmod.c',
- '3rd/mp/mpmodop.c',
- '3rd/mp/mpmul.c',
- '3rd/mp/mprand.c',
- '3rd/mp/mpright.c',
- '3rd/mp/mpsub.c',
- '3rd/mp/mptobe.c',
- '3rd/mp/mptober.c',
- '3rd/mp/mptod.c',
- '3rd/mp/mptoi.c',
- '3rd/mp/mptoui.c',
- '3rd/mp/mptouv.c',
- '3rd/mp/mptov.c',
- '3rd/mp/mpvecadd.c',
- '3rd/mp/mpveccmp.c',
- '3rd/mp/mpvecdigmuladd.c',
- '3rd/mp/mpvecsub.c',
- '3rd/mp/mpvectscmp.c',
- '3rd/mp/strtomp.c',
- '3rd/mp/u16.c',
-]
-
utf = [
'3rd/utf/rune.c',
'3rd/utf/runeistype.c',
@@ -153,11 +119,51 @@
],
)
+mp = static_library(
+ 'mp',
+ [
+ '3rd/mp/mpadd.c',
+ '3rd/mp/mpaux.c',
+ '3rd/mp/mpcmp.c',
+ '3rd/mp/mpdigdiv.c',
+ '3rd/mp/mpdiv.c',
+ '3rd/mp/mpexp.c',
+ '3rd/mp/mpextendedgcd.c',
+ '3rd/mp/mpfmt.c',
+ '3rd/mp/mpinvert.c',
+ '3rd/mp/mpleft.c',
+ '3rd/mp/mplogic.c',
+ '3rd/mp/mpmod.c',
+ '3rd/mp/mpmodop.c',
+ '3rd/mp/mpmul.c',
+ '3rd/mp/mprand.c',
+ '3rd/mp/mpright.c',
+ '3rd/mp/mpsub.c',
+ '3rd/mp/mptobe.c',
+ '3rd/mp/mptober.c',
+ '3rd/mp/mptod.c',
+ '3rd/mp/mptoi.c',
+ '3rd/mp/mptoui.c',
+ '3rd/mp/mptouv.c',
+ '3rd/mp/mptov.c',
+ '3rd/mp/mpvecadd.c',
+ '3rd/mp/mpveccmp.c',
+ '3rd/mp/mpvecdigmuladd.c',
+ '3rd/mp/mpvecsub.c',
+ '3rd/mp/mpvectscmp.c',
+ '3rd/mp/strtomp.c',
+ '3rd/mp/u16.c',
+ ],
+ include_directories: include_directories(
+ '3rd/mp',
+ 'posix',
+ ),
+)
+
flisp = executable(
'flisp',
sources: [
src,
- mp,
utf,
boot,
builtins,
@@ -171,6 +177,9 @@
'3rd/utf',
'posix',
),
+ link_with: [
+ mp,
+ ],
)
mptest = executable(
@@ -177,22 +186,40 @@
'mptest',
sources: [
'3rd/mp/test.c',
- mp,
],
include_directories: include_directories(
- '3rd',
- '3rd/mp',
'posix',
),
+ link_with: [
+ mp,
+ ],
)
test('mp', mptest)
+mptest2 = executable(
+ 'mptest2',
+ sources: [
+ '3rd/mp/test/convtest.c',
+ '3rd/mp/test/gen.tab.c',
+ '3rd/mp/test/ld.c',
+ '3rd/mp/test/main.c',
+ ],
+ include_directories: include_directories(
+ '3rd/mp/test',
+ 'posix',
+ ),
+ link_with: [
+ mp,
+ ],
+)
+test('mp2', mptest2, timeout: -1)
+
tests_dir = join_paths(meson.current_source_dir(), 'test')
test('argv', flisp, args: ['argv.lsp'], workdir: tests_dir)
test('hash', flisp, args: ['hashtest.lsp'], workdir: tests_dir)
-test('perf', flisp, args: ['perf.lsp'], workdir: tests_dir, timeout: 60)
+test('perf', flisp, args: ['perf.lsp'], workdir: tests_dir, timeout: -1)
test('tme', flisp, args: ['tme.lsp'], workdir: tests_dir)
-test('torture', flisp, args: ['torture.scm'], workdir: tests_dir, timeout: 60)
+test('torture', flisp, args: ['torture.scm'], workdir: tests_dir, timeout: -1)
test('torus', flisp, args: ['torus.lsp'], workdir: tests_dir)
test('unit', flisp, args: ['unittest.lsp'], workdir: tests_dir)