Annotation of OpenXM_contrib2/asir2000/engine/bf.c, Revision 1.13
1.2 noro 1: /*
1.13 ! noro 2: * $OpenXM: OpenXM_contrib2/asir2000/engine/bf.c,v 1.12 2015/08/20 08:42:07 noro Exp $
1.7 noro 3: */
1.1 noro 4: #include "ca.h"
5: #include "base.h"
6: #include <math.h>
7:
1.11 noro 8: extern int mpfr_roundmode;
9:
1.7 noro 10: Num tobf(Num,int);
1.1 noro 11:
1.7 noro 12: #define BFPREC(a) (((BF)(a))->body->_mpfr_prec)
13:
14: void strtobf(char *s,BF *p)
15: {
16: BF r;
17: NEWBF(r);
18: mpfr_init(r->body);
1.11 noro 19: mpfr_set_str(r->body,s,10,mpfr_roundmode);
1.7 noro 20: *p = r;
21: }
22:
23: double mpfrtodbl(mpfr_t a)
24: {
1.11 noro 25: return mpfr_get_d(a,mpfr_roundmode);
1.7 noro 26: }
27:
28: Num tobf(Num a,int prec)
29: {
30: mpfr_t r;
31: mpz_t z;
32: mpq_t q;
33: BF d;
34: N nm,dn;
1.12 noro 35: C c;
36: Num re,im;
1.7 noro 37: int sgn;
38:
39: if ( !a ) {
1.8 noro 40: prec ? mpfr_init2(r,prec) : mpfr_init(r);
1.7 noro 41: mpfr_set_zero(r,1);
42: MPFRTOBF(r,d);
43: return (Num)d;
44: } else {
45: switch ( NID(a) ) {
46: case N_B:
47: return a;
48: break;
1.8 noro 49: case N_R:
50: prec ? mpfr_init2(r,prec) : mpfr_init(r);
1.11 noro 51: mpfr_init_set_d(r,((Real)a)->body,mpfr_roundmode);
1.7 noro 52: MPFRTOBF(r,d);
53: return (Num)d;
1.8 noro 54: break;
1.7 noro 55: case N_Q:
56: nm = NM((Q)a); dn = DN((Q)a); sgn = SGN((Q)a);
57: if ( INT((Q)a) ) {
58: mpz_init(z);
59: mpz_import(z,PL(nm),-1,sizeof(BD(nm)[0]),0,0,BD(nm));
60: if ( sgn < 0 ) mpz_neg(z,z);
1.11 noro 61: mpfr_init_set_z(r,z,mpfr_roundmode);
1.7 noro 62: } else {
63: mpq_init(q);
64: mpz_import(mpq_numref(q),PL(nm),-1,sizeof(BD(nm)[0]),0,0,BD(nm));
65: mpz_import(mpq_denref(q),PL(dn),-1,sizeof(BD(dn)[0]),0,0,BD(dn));
66: if ( sgn < 0 ) mpq_neg(q,q);
1.11 noro 67: mpfr_init_set_q(r,q,mpfr_roundmode);
1.7 noro 68: }
69: MPFRTOBF(r,d);
70: return (Num)d;
71: break;
1.12 noro 72: case N_C:
73: re = tobf(((C)a)->r,prec); im = tobf(((C)a)->i,prec);
74: NEWC(c); c->r = re; c->i = im;
75: return (Num)c;
76: break;
1.7 noro 77: default:
78: error("tobf : invalid argument");
1.8 noro 79: break;
1.7 noro 80: }
81: }
82: }
83:
84: void addbf(Num a,Num b,Num *c)
85: {
86: mpfr_t r;
87: BF d;
1.8 noro 88: GZ z;
89: GQ q;
1.7 noro 90:
91: if ( !a )
92: *c = b;
93: else if ( !b )
94: *c = a;
95: else if ( (NID(a) <= N_A) && (NID(b) <= N_A ) )
96: (*addnumt[MIN(NID(a),NID(b))])(a,b,c);
1.9 noro 97: else if ( NID(a) == N_B ) {
98: switch ( NID(b) ) {
99: case N_Q:
100: mpfr_init2(r,BFPREC(a));
101: if ( INT((Q)b) ) {
102: z = ztogz((Q)b);
1.11 noro 103: mpfr_add_z(r,((BF)a)->body,z->body,mpfr_roundmode);
1.9 noro 104: } else {
105: q = qtogq((Q)b);
1.11 noro 106: mpfr_add_q(r,((BF)a)->body,q->body,mpfr_roundmode);
1.9 noro 107: }
108: break;
109: case N_R:
110: /* double precision = 53 */
111: mpfr_init2(r,MAX(BFPREC(a),53));
1.11 noro 112: mpfr_add_d(r,((BF)a)->body,((Real)b)->body,mpfr_roundmode);
1.9 noro 113: break;
114: case N_B:
115: mpfr_init2(r,MAX(BFPREC(a),BFPREC(b)));
1.11 noro 116: mpfr_add(r,((BF)a)->body,((BF)b)->body,mpfr_roundmode);
1.9 noro 117: break;
118: }
119: MPFRTOBF(r,d);
120: *c = (Num)d;
121: } else { /* NID(b)==N_B */
122: switch ( NID(a) ) {
123: case N_Q:
124: mpfr_init2(r,BFPREC(b));
125: if ( INT((Q)a) ) {
126: z = ztogz((Q)a);
1.11 noro 127: mpfr_add_z(r,((BF)b)->body,z->body,mpfr_roundmode);
1.9 noro 128: } else {
129: q = qtogq((Q)a);
1.11 noro 130: mpfr_add_q(r,((BF)b)->body,q->body,mpfr_roundmode);
1.8 noro 131: }
1.9 noro 132: break;
133: case N_R:
134: /* double precision = 53 */
135: mpfr_init2(r,MAX(BFPREC(b),53));
1.11 noro 136: mpfr_add_d(r,((BF)b)->body,((Real)a)->body,mpfr_roundmode);
1.9 noro 137: break;
1.8 noro 138: }
1.7 noro 139: MPFRTOBF(r,d);
140: *c = (Num)d;
141: }
1.12 noro 142: if ( !cmpbf(*c,0) ) *c = 0;
1.7 noro 143: }
144:
145: void subbf(Num a,Num b,Num *c)
146: {
1.8 noro 147: mpfr_t r,s;
148: GZ z;
149: GQ q;
1.7 noro 150: BF d;
151:
152: if ( !a )
153: (*chsgnnumt[NID(b)])(b,c);
154: else if ( !b )
155: *c = a;
156: else if ( (NID(a) <= N_A) && (NID(b) <= N_A ) )
157: (*subnumt[MIN(NID(a),NID(b))])(a,b,c);
1.9 noro 158: else if ( NID(a) == N_B ) {
159: switch ( NID(b) ) {
160: case N_Q:
161: mpfr_init2(r,BFPREC(a));
162: if ( INT((Q)b) ) {
163: z = ztogz((Q)b);
1.11 noro 164: mpfr_sub_z(r,((BF)a)->body,z->body,mpfr_roundmode);
1.9 noro 165: } else {
166: q = qtogq((Q)b);
1.11 noro 167: mpfr_sub_q(r,((BF)a)->body,q->body,mpfr_roundmode);
1.9 noro 168: }
169: break;
170: case N_R:
171: /* double precision = 53 */
172: mpfr_init2(r,MAX(BFPREC(a),53));
1.11 noro 173: mpfr_sub_d(r,((BF)a)->body,((Real)b)->body,mpfr_roundmode);
1.9 noro 174: break;
175: case N_B:
176: mpfr_init2(r,MAX(BFPREC(a),BFPREC(b)));
1.11 noro 177: mpfr_sub(r,((BF)a)->body,((BF)b)->body,mpfr_roundmode);
1.9 noro 178: break;
179: }
180: MPFRTOBF(r,d);
181: *c = (Num)d;
182: } else { /* NID(b)==N_B */
183: switch ( NID(a) ) {
184: case N_Q:
185: mpfr_init2(r,BFPREC(b));
186: if ( INT((Q)a) ) {
187: z = ztogz((Q)a);
1.11 noro 188: mpfr_sub_z(r,((BF)b)->body,z->body,mpfr_roundmode);
1.9 noro 189: } else {
190: q = qtogq((Q)a);
1.11 noro 191: mpfr_sub_q(r,((BF)b)->body,q->body,mpfr_roundmode);
1.8 noro 192: }
1.11 noro 193: mpfr_neg(r,r,mpfr_roundmode);
1.9 noro 194: break;
195: case N_R:
196: /* double precision = 53 */
197: mpfr_init2(r,MAX(BFPREC(b),53));
1.11 noro 198: mpfr_d_sub(r,((Real)a)->body,((BF)b)->body,mpfr_roundmode);
1.9 noro 199: break;
1.8 noro 200: }
1.7 noro 201: MPFRTOBF(r,d);
202: *c = (Num)d;
203: }
1.12 noro 204: if ( !cmpbf(*c,0) ) *c = 0;
1.7 noro 205: }
206:
207: void mulbf(Num a,Num b,Num *c)
208: {
209: mpfr_t r;
1.8 noro 210: GZ z;
211: GQ q;
1.7 noro 212: BF d;
213: int prec;
214:
215: if ( !a || !b )
216: *c = 0;
217: else if ( (NID(a) <= N_A) && (NID(b) <= N_A ) )
218: (*mulnumt[MIN(NID(a),NID(b))])(a,b,c);
1.9 noro 219: else if ( NID(a) == N_B ) {
220: switch ( NID(b) ) {
221: case N_Q:
222: mpfr_init2(r,BFPREC(a));
223: if ( INT((Q)b) ) {
224: z = ztogz((Q)b);
1.11 noro 225: mpfr_mul_z(r,((BF)a)->body,z->body,mpfr_roundmode);
1.9 noro 226: } else {
227: q = qtogq((Q)b);
1.11 noro 228: mpfr_mul_q(r,((BF)a)->body,q->body,mpfr_roundmode);
1.9 noro 229: }
230: break;
231: case N_R:
232: /* double precision = 53 */
233: mpfr_init2(r,MAX(BFPREC(a),53));
1.11 noro 234: mpfr_mul_d(r,((BF)a)->body,((Real)b)->body,mpfr_roundmode);
1.9 noro 235: break;
236: case N_B:
237: mpfr_init2(r,MAX(BFPREC(a),BFPREC(b)));
1.11 noro 238: mpfr_mul(r,((BF)a)->body,((BF)b)->body,mpfr_roundmode);
1.9 noro 239: break;
240: }
241: MPFRTOBF(r,d);
242: *c = (Num)d;
243: } else { /* NID(b)==N_B */
244: switch ( NID(a) ) {
245: case N_Q:
246: mpfr_init2(r,BFPREC(b));
247: if ( INT((Q)a) ) {
248: z = ztogz((Q)a);
1.11 noro 249: mpfr_mul_z(r,((BF)b)->body,z->body,mpfr_roundmode);
1.9 noro 250: } else {
251: q = qtogq((Q)a);
1.11 noro 252: mpfr_mul_q(r,((BF)b)->body,q->body,mpfr_roundmode);
1.8 noro 253: }
1.9 noro 254: break;
255: case N_R:
256: /* double precision = 53 */
257: mpfr_init2(r,MAX(BFPREC(b),53));
1.11 noro 258: mpfr_mul_d(r,((BF)b)->body,((Real)a)->body,mpfr_roundmode);
1.9 noro 259: break;
1.8 noro 260: }
1.7 noro 261: MPFRTOBF(r,d);
262: *c = (Num)d;
263: }
1.12 noro 264: if ( !cmpbf(*c,0) ) *c = 0;
1.7 noro 265: }
266:
267: void divbf(Num a,Num b,Num *c)
268: {
1.8 noro 269: mpfr_t s,r;
270: GZ z;
271: GQ q;
1.7 noro 272: BF d;
273:
274: if ( !b )
275: error("divbf : division by 0");
276: else if ( !a )
277: *c = 0;
278: else if ( (NID(a) <= N_A) && (NID(b) <= N_A ) )
279: (*divnumt[MIN(NID(a),NID(b))])(a,b,c);
1.9 noro 280: else if ( NID(a) == N_B ) {
281: switch ( NID(b) ) {
282: case N_Q:
283: mpfr_init2(r,BFPREC(a));
284: if ( INT((Q)b) ) {
285: z = ztogz((Q)b);
1.11 noro 286: mpfr_div_z(r,((BF)a)->body,z->body,mpfr_roundmode);
1.9 noro 287: } else {
288: q = qtogq((Q)b);
1.11 noro 289: mpfr_div_q(r,((BF)a)->body,q->body,mpfr_roundmode);
1.8 noro 290: }
1.9 noro 291: break;
292: case N_R:
293: /* double precision = 53 */
294: mpfr_init2(r,MAX(BFPREC(a),53));
1.11 noro 295: mpfr_div_d(r,((BF)a)->body,((Real)b)->body,mpfr_roundmode);
1.9 noro 296: break;
297: case N_B:
298: mpfr_init2(r,MAX(BFPREC(a),BFPREC(b)));
1.11 noro 299: mpfr_div(r,((BF)a)->body,((BF)b)->body,mpfr_roundmode);
1.9 noro 300: break;
301: }
302: MPFRTOBF(r,d);
303: *c = (Num)d;
304: } else { /* NID(b)==N_B */
305: switch ( NID(a) ) {
306: case N_Q:
307: /* XXX : mpfr_z_div and mpfr_q_div are not implemented */
308: a = tobf(a,BFPREC(b));
309: mpfr_init2(r,BFPREC(b));
1.11 noro 310: mpfr_div(r,((BF)a)->body,((BF)b)->body,mpfr_roundmode);
1.9 noro 311: break;
312: case N_R:
313: /* double precision = 53 */
314: mpfr_init2(r,MAX(BFPREC(b),53));
1.11 noro 315: mpfr_d_div(r,((Real)a)->body,((BF)b)->body,mpfr_roundmode);
1.9 noro 316: break;
1.8 noro 317: }
1.7 noro 318: MPFRTOBF(r,d);
319: *c = (Num)d;
320: }
1.12 noro 321: if ( !cmpbf(*c,0) ) *c = 0;
1.7 noro 322: }
323:
324: void pwrbf(Num a,Num b,Num *c)
325: {
1.8 noro 326: int prec;
1.7 noro 327: mpfr_t r;
1.8 noro 328: GZ z;
1.7 noro 329: BF d;
330:
331: if ( !b )
332: *c = (Num)ONE;
333: else if ( !a )
334: *c = 0;
335: else if ( (NID(a) <= N_A) && (NID(b) <= N_A ) )
336: (*pwrnumt[MIN(NID(a),NID(b))])(a,b,c);
1.9 noro 337: else if ( NID(a) == N_B ) {
338: switch ( NID(b) ) {
339: case N_Q:
340: mpfr_init2(r,BFPREC(a));
341: if ( INT((Q)b) ) {
342: z = ztogz((Q)b);
1.11 noro 343: mpfr_pow_z(r,((BF)a)->body,z->body,mpfr_roundmode);
1.9 noro 344: } else {
345: b = tobf(b,BFPREC(a));
1.11 noro 346: mpfr_pow(r,((BF)a)->body,((BF)b)->body,mpfr_roundmode);
1.8 noro 347: }
1.9 noro 348: break;
349: case N_R:
350: /* double precision = 53 */
351: prec = MAX(BFPREC(a),53);
352: mpfr_init2(r,prec);
353: b = tobf(b,prec);
1.11 noro 354: mpfr_pow(r,((BF)a)->body,((BF)b)->body,mpfr_roundmode);
1.9 noro 355: break;
356: case N_B:
357: mpfr_init2(r,MAX(BFPREC(a),BFPREC(b)));
1.11 noro 358: mpfr_pow(r,((BF)a)->body,((BF)b)->body,mpfr_roundmode);
1.9 noro 359: break;
360: }
361: MPFRTOBF(r,d);
362: *c = (Num)d;
363: } else { /* NID(b)==N_B */
364: switch ( NID(a) ) {
365: case N_Q:
366: mpfr_init2(r,BFPREC(b));
367: a = tobf(a,BFPREC(b));
1.11 noro 368: mpfr_pow(r,((BF)a)->body,((BF)b)->body,mpfr_roundmode);
1.9 noro 369: break;
370: case N_R:
371: /* double precision = 53 */
372: prec = MAX(BFPREC(a),53);
373: mpfr_init2(r,prec);
374: a = tobf(a,prec);
1.11 noro 375: mpfr_pow(r,((BF)a)->body,((BF)b)->body,mpfr_roundmode);
1.9 noro 376: break;
1.8 noro 377: }
1.7 noro 378: MPFRTOBF(r,d);
379: *c = (Num)d;
380: }
1.12 noro 381: if ( !cmpbf(*c,0) ) *c = 0;
1.7 noro 382: }
383:
384: void chsgnbf(Num a,Num *c)
385: {
386: mpfr_t r;
387: BF d;
388:
389: if ( !a )
390: *c = 0;
391: else if ( NID(a) <= N_A )
392: (*chsgnnumt[NID(a)])(a,c);
393: else {
394: mpfr_init2(r,BFPREC(a));
1.11 noro 395: mpfr_neg(r,((BF)a)->body,mpfr_roundmode);
1.7 noro 396: MPFRTOBF(r,d);
397: *c = (Num)d;
398: }
399: }
400:
401: int cmpbf(Num a,Num b)
402: {
1.9 noro 403: int ret;
404: GZ z;
405: GQ q;
406:
1.7 noro 407: if ( !a ) {
1.12 noro 408: if ( !b ) return 0;
409: else if ((NID(b)<=N_A) )
1.7 noro 410: return (*cmpnumt[NID(b)])(a,b);
411: else
1.13 ! noro 412: return -mpfr_sgn(((BF)b)->body);
1.7 noro 413: } else if ( !b ) {
414: if ( !a || (NID(a)<=N_A) )
415: return (*cmpnumt[NID(a)])(a,b);
416: else
417: return mpfr_sgn(((BF)a)->body);
1.9 noro 418: } else if ( NID(a) == N_B ) {
419: switch ( NID(b) ) {
420: case N_Q:
421: if ( INT((Q)b) ) {
422: z = ztogz((Q)b);
423: ret = mpfr_cmp_z(((BF)a)->body,z->body);
424: } else {
425: q = qtogq((Q)b);
426: ret = mpfr_cmp_q(((BF)a)->body,q->body);
427: }
428: break;
429: case N_R:
430: /* double precision = 53 */
431: ret = mpfr_cmp_d(((BF)a)->body,((Real)b)->body);
432: break;
433: case N_B:
434: ret = mpfr_cmp(((BF)a)->body,((BF)b)->body);
435: break;
436: }
437: return ret;
438: } else { /* NID(b)==N_B */
439: switch ( NID(a) ) {
440: case N_Q:
441: if ( INT((Q)a) ) {
442: z = ztogz((Q)a);
443: ret = mpfr_cmp_z(((BF)b)->body,z->body);
444: } else {
445: q = qtogq((Q)a);
446: ret = mpfr_cmp_q(((BF)b)->body,q->body);
447: }
448: break;
449: case N_R:
450: /* double precision = 53 */
451: ret = mpfr_cmp_d(((BF)b)->body,((Real)a)->body);
452: break;
453: }
454: return -ret;
1.7 noro 455: }
1.1 noro 456: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>