Annotation of freem/src/operator.c, revision 1.1
1.1 ! snw 1: /*
! 2: * *
! 3: * * *
! 4: * * *
! 5: * ***************
! 6: * * * * *
! 7: * * MUMPS *
! 8: * * * * *
! 9: * ***************
! 10: * * *
! 11: * * *
! 12: * *
! 13: *
! 14: * operator.c
! 15: * operators pattern-match, divide, multiply, add, power
! 16: *
! 17: *
! 18: * Author: Serena Willis <jpw@coherent-logic.com>
! 19: * Copyright (C) 1998 MUG Deutschland
! 20: * Copyright (C) 2020 Coherent Logic Development LLC
! 21: *
! 22: *
! 23: * This file is part of FreeM.
! 24: *
! 25: * FreeM is free software: you can redistribute it and/or modify
! 26: * it under the terms of the GNU Affero Public License as published by
! 27: * the Free Software Foundation, either version 3 of the License, or
! 28: * (at your option) any later version.
! 29: *
! 30: * FreeM is distributed in the hope that it will be useful,
! 31: * but WITHOUT ANY WARRANTY; without even the implied warranty of
! 32: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
! 33: * GNU Affero Public License for more details.
! 34: *
! 35: * You should have received a copy of the GNU Affero Public License
! 36: * along with FreeM. If not, see <https://www.gnu.org/licenses/>.
! 37: *
! 38: **/
! 39:
! 40: #include "mpsdef.h"
! 41: #include <stdlib.h>
! 42: #include <math.h>
! 43:
! 44: void root ();
! 45: void mround ();
! 46: int unit (char *str);
! 47: extern void cond_round (char *a, int digits); /* defined in expr.c */
! 48:
! 49:
! 50:
! 51: #define PLUS '+'
! 52: #define MINUS '-'
! 53: #define POINT '.'
! 54: #define point (POINT-ZERO)
! 55: #define ZERO '0'
! 56: #define ONE (ZERO+1)
! 57: #define TWO (ZERO+2)
! 58: #define THREE (ZERO+3)
! 59: #define FIVE (ZERO+(NUMBASE/2))
! 60: #define NINE (ZERO+NUMBASE-1)
! 61: #define NUMBASE 10
! 62:
! 63: short int
! 64: pattern (a, b) /* evaluates a ? b */
! 65: char *a;
! 66: char *b;
! 67:
! 68: {
! 69: short levels; /* depth of stack */
! 70: register int patx; /* match stack pointer */
! 71: short notpatclass; /* pattern class negation */
! 72: char *ptrpcd[PATDEPTH], /* pointers to patcode */
! 73: *position[PATDEPTH]; /* position of matching substring */
! 74: short mincnt[PATDEPTH], /* minimum number of matches */
! 75: maxcnt[PATDEPTH], /* maximum number of matches */
! 76: actcnt[PATDEPTH]; /* actual count of matches */
! 77: short Pflag,
! 78: Pchar; /* status in pattern alternation */
! 79: short altc; /* alternation counter */
! 80: short altcnt[PATDEPTH]; /* gr.pat.alternation counters */
! 81:
! 82: unsigned char gpmin[PATDEPTH][PATDEPTH][255]; /* grouped pattern minimum lengthes */
! 83: char *gp_position[PATDEPTH][PATDEPTH]; /* grouped patt.pos.of substr */
! 84:
! 85: char *ptrtom; /* pointer to match code */
! 86: char patcode;
! 87: int ch;
! 88: int i;
! 89:
! 90: pattrnflag = Pflag = FALSE; /* incomplete match flag */
! 91: pattrnchar = Pchar = EOL; /* incomplete match supplement */
! 92: notpatclass = FALSE; /* pattern class negation */
! 93: patx = 0;
! 94: while (*b != EOL) { /* get minimum repeat count */
! 95: mincnt[patx] = 0;
! 96: maxcnt[patx] = 255;
! 97: altcnt[patx] = (-1);
! 98: if (*b != '.') {
! 99: ch = (*b++) - '0';
! 100: while (*b >= '0' && *b <= '9') {
! 101: ch *= 10;
! 102: ch += (*b++) - '0';
! 103: }
! 104: mincnt[patx] = ch;
! 105: if (*b != '.')
! 106: maxcnt[patx] = ch;
! 107: }
! 108: /* get maximum repeat count */
! 109: if (*b == '.') {
! 110: b++;
! 111: if (*b >= '0' && *b <= '9') {
! 112: ch = (*b++) - '0';
! 113: while (*b >= '0' && *b <= '9') {
! 114: ch *= 10;
! 115: ch += (*b++) - '0';
! 116: }
! 117: maxcnt[patx] = ch;
! 118: }
! 119: }
! 120: if (maxcnt[patx] < mincnt[patx])
! 121: return '2'; /* just impossible! */
! 122: ptrpcd[patx] = b;
! 123: actcnt[patx] = 0;
! 124: position[patx] = 0;
! 125: /* scan strlit, ignore it when empty */
! 126: if (*b == '"' || *b == 'z' || (*b == '\'' && *(b + 1) == '"')) {
! 127: if (*(++b) == DELIM) {
! 128: b++;
! 129: continue;
! 130: }
! 131: while (*(++b) != DELIM) ;
! 132: b++;
! 133: } else if (*b == '(') {
! 134: i = 1;
! 135: b++;
! 136: while ((ch = *b) != EOL) {
! 137: b++;
! 138: if (ch == '"') {
! 139: while (*(++b) != DELIM) ;
! 140: }
! 141: if (ch == '(') {
! 142: i++;
! 143: continue;
! 144: }
! 145: if (ch == ')') {
! 146: i--;
! 147: if (i < 1)
! 148: break;
! 149: }
! 150: }
! 151: } else
! 152: while (*(++b) >= 'A') ;
! 153: if (++patx >= (PATDEPTH - 1))
! 154: return '3'; /* stack overflow */
! 155: }
! 156: levels = patx;
! 157: if (*(b - 1) == 'e' && mincnt[levels - 1] == 0 && maxcnt[levels - 1] == 255)
! 158: *(b - 1) = '~'; /* frequent special case: last pattern is '.E' */
! 159: mincnt[levels] = maxcnt[levels] = 1; /* sentinel, does never match */
! 160: actcnt[levels] = 0;
! 161: ptrpcd[levels] = b; /* (*b==EOL) */
! 162: patx = 0;
! 163: while (patx <= levels) {
! 164: while (actcnt[patx] < mincnt[patx]) {
! 165: actcnt[patx]++;
! 166: if (*a == EOL) {
! 167: pattrnflag = TRUE; /* incomplete match flag */
! 168: if (patx >= levels) {
! 169: pattrnchar = EOL;
! 170: return '1';
! 171: }
! 172: if (patx > 0) {
! 173: if (actcnt[patx - 1] != maxcnt[patx - 1])
! 174: return '0';
! 175: /* after alternation we are not sure about */
! 176: /* that supplement character */
! 177: if (*(ptrpcd[patx - 1]) == '(') {
! 178: pattrnchar = EOL;
! 179: return '0';
! 180: }
! 181: }
! 182: if (*(ptrpcd[patx]) == '"')
! 183: pattrnchar = *(ptrpcd[patx] + 1);
! 184: return '0';
! 185: }
! 186: for (;;)
! 187: {
! 188: /***begin section: does that char match current pattern code ***/
! 189: ptrtom = ptrpcd[patx];
! 190: ch = (*a);
! 191: for (;;)
! 192: {
! 193: patcode = (*ptrtom++);
! 194: if ((notpatclass = (patcode == '\'')))
! 195: patcode = (*ptrtom++);
! 196: switch (patcode) { /* we live in an ASCII/ISO world !! */
! 197: case 'c':
! 198: if (((ch < SP && ch >= NUL)
! 199: || ch == DEL) != notpatclass)
! 200: goto match;
! 201: break;
! 202: case 'n':
! 203: if ((ch <= '9' && ch >= '0') != notpatclass)
! 204: goto match;
! 205: break;
! 206: case 'p':
! 207: if (((ch >= SP && ch <= '/') ||
! 208: (ch >= ':' && ch <= '@') ||
! 209: (ch >= '[' && ch <= '`') ||
! 210: (ch >= '{' && ch <= '~') ||
! 211: (ch == '\200')) != notpatclass)
! 212: goto match;
! 213: break;
! 214: case 'a':
! 215: if (((ch >= 'A' && ch <= 'Z') ||
! 216: (ch >= 'a' && ch <= 'z')) != notpatclass)
! 217: goto match;
! 218: break;
! 219: case 'l':
! 220: if ((ch >= 'a' && ch <= 'z') != notpatclass)
! 221: goto match;
! 222: break;
! 223: case 'u':
! 224: if ((ch >= 'A' && ch <= 'Z') != notpatclass)
! 225: goto match;
! 226: break;
! 227: case 'e':
! 228: if (!notpatclass)
! 229: goto match;
! 230: break;
! 231: case '"':
! 232: i = 0;
! 233: while (a[i++] == (*ptrtom++)) ;
! 234: if ((*--ptrtom) == DELIM) {
! 235: if (notpatclass)
! 236: goto nomatch;
! 237: b = ptrpcd[patx] + 1;
! 238: while (*b++ != DELIM)
! 239: a++;
! 240: goto match0;
! 241: }
! 242: if (notpatclass) {
! 243: i--;
! 244: while (*ptrtom++ != DELIM) {
! 245: if (a[i++] == EOL)
! 246: goto nomatch;
! 247: }
! 248: b = ptrpcd[patx] + 2;
! 249: while (*b++ != DELIM)
! 250: a++;
! 251: goto match0;
! 252: }
! 253: if (a[i - 1] == EOL) {
! 254: pattrnflag = TRUE;
! 255: pattrnchar = *ptrtom;
! 256: }
! 257: goto nomatch;
! 258:
! 259: case '~':{
! 260: pattrnchar = EOL; /* '.E' as last pat_atom */
! 261: pattrnflag = TRUE;
! 262: return '1';
! 263: }
! 264: /* grouped pattern match */
! 265: case '(':{
! 266: char aa[256];
! 267: char bb[256];
! 268: int i1,
! 269: min,
! 270: max;
! 271: short pflag;
! 272: short pchar;
! 273:
! 274: if (Pflag) {
! 275: pflag = Pflag;
! 276: pchar = Pchar;
! 277: } else {
! 278: pflag = FALSE;
! 279: pchar = EOL;
! 280: }
! 281: if (altcnt[patx] < 0) {
! 282: for (altc = 0; altc < PATDEPTH; altc++)
! 283: gpmin[patx][altc][1] = 0;
! 284: }
! 285: altcnt[patx] = 0;
! 286: alternation:;
! 287: i = 0;
! 288: i1 = 1;
! 289: while (i1) {
! 290: bb[i] = *ptrtom++;
! 291: if (bb[i] == '"') {
! 292: while ((bb[++i] = (*ptrtom++)) != DELIM) ;
! 293: }
! 294: if (bb[i] == '(')
! 295: i1++;
! 296: if (bb[i] == ')')
! 297: i1--;
! 298: if (bb[i] == ',' && i1 == 1)
! 299: i1--;
! 300: i++;
! 301: }
! 302: bb[--i] = EOL;
! 303: pminmax (bb, &min, &max);
! 304:
! 305: if ((i1 = gpmin[patx][altcnt[patx]][actcnt[patx]]) < min)
! 306: gpmin[patx][altcnt[patx]][actcnt[patx]] = i1 = min;
! 307: gpmin[patx][altcnt[patx]][actcnt[patx] + 1] = 0;
! 308: /* too much charaters to get a match! */
! 309: if (i1 > max) {
! 310: if (*(ptrtom - 1) == ',') {
! 311: altcnt[patx]++;
! 312: goto alternation;
! 313: }
! 314: pattrnflag = pflag;
! 315: pattrnchar = pchar;
! 316: goto nomatch;
! 317: }
! 318: /* if number of chars too small, try anyway */
! 319: /* to get info for "incomplete" match */
! 320: for (i = 0; i < i1; i++)
! 321: if ((aa[i] = a[i]) == EOL)
! 322: break;
! 323: gp_position[patx][actcnt[patx]] = a;
! 324:
! 325: for (;;)
! 326: {
! 327: aa[i] = EOL;
! 328: i1 = pattern (aa, bb);
! 329: if (i1 == '1') {
! 330: gpmin[patx][altcnt[patx]][actcnt[patx]] = i;
! 331: a += i;
! 332: goto match0;
! 333: }
! 334: if (i1 != '0')
! 335: return i1;
! 336: if (pattrnflag) {
! 337: if (pflag == FALSE)
! 338: pchar = pattrnchar;
! 339: else if (pchar != pattrnchar)
! 340: pchar = EOL;
! 341: pflag = TRUE;
! 342: }
! 343: if (!pattrnflag) {
! 344: if (*(ptrtom - 1) == ',') {
! 345: altcnt[patx]++;
! 346: goto alternation;
! 347: }
! 348: pattrnflag = pflag;
! 349: pattrnchar = pchar;
! 350: goto nomatch;
! 351: }
! 352: if (a[i] == EOL) {
! 353: Pflag = pflag;
! 354: Pchar = pchar;
! 355: if (*(ptrtom - 1) == ',') {
! 356: altcnt[patx]++;
! 357: goto alternation;
! 358: }
! 359: pattrnflag = pflag;
! 360: pattrnchar = pchar;
! 361: return '0';
! 362: }
! 363: aa[i] = a[i];
! 364: i++;
! 365: }
! 366: }
! 367: /* match one of listed characters ?1Z"string" */
! 368: case 'z':
! 369: for (;;)
! 370: {
! 371: if ((*++ptrtom) == DELIM) {
! 372: if (notpatclass)
! 373: goto match;
! 374: goto nomatch;
! 375: }
! 376: if (ch != *ptrtom) {
! 377: if (*(ptrtom + 1) == '.' && *(ptrtom + 2) == '.') {
! 378: if (ch < *ptrtom ||
! 379: (ch > *(ptrtom + 3) && *(ptrtom + 3) != DELIM)) {
! 380: ptrtom += 2;
! 381: continue;
! 382: }
! 383: } else
! 384: continue;
! 385: }
! 386: while (*++ptrtom != DELIM) ;
! 387: if (notpatclass)
! 388: goto nomatch;
! 389: goto match;
! 390: }
! 391:
! 392: /* loadable matches */
! 393: case 'C':
! 394: i = 0;
! 395: while (zmc[i] != EOL) {
! 396: if (zmc[i] == ch && !notpatclass)
! 397: goto match;
! 398: i++;
! 399: }
! 400: if (notpatclass)
! 401: goto match;
! 402: break;
! 403: case 'N':
! 404: i = 0;
! 405: while (zmn[i] != EOL) {
! 406: if (zmn[i] == ch && !notpatclass)
! 407: goto match;
! 408: i++;
! 409: }
! 410: if (notpatclass)
! 411: goto match;
! 412: break;
! 413: case 'P':
! 414: i = 0;
! 415: while (zmp[i] != EOL) {
! 416: if (zmp[i] == ch && !notpatclass)
! 417: goto match;
! 418: i++;
! 419: }
! 420: if (notpatclass)
! 421: goto match;
! 422: break;
! 423: case 'A':
! 424: i = 0;
! 425: while (zmu[i] != EOL) {
! 426: if (zmu[i] == ch && !notpatclass)
! 427: goto match;
! 428: i++;
! 429: }
! 430: case 'L':
! 431: i = 0;
! 432: while (zml[i] != EOL) {
! 433: if (zml[i] == ch && !notpatclass)
! 434: goto match;
! 435: i++;
! 436: }
! 437: if (notpatclass)
! 438: goto match;
! 439: break;
! 440: case 'U':
! 441: i = 0;
! 442: while (zmu[i] != EOL) {
! 443: if (zmu[i] == ch && !notpatclass)
! 444: goto match;
! 445: i++;
! 446: }
! 447: if (notpatclass)
! 448: goto match;
! 449: break;
! 450: default:
! 451: goto nomatch;
! 452: } /* end_switch */
! 453: } /* end repeat */
! 454: /*** end section: does that char match current pattern atom ***/
! 455: nomatch:;
! 456: if (patcode == '(') {
! 457: for (altc = 0; altc <= altcnt[patx]; altc++)
! 458: gpmin[patx][altc][actcnt[patx]] = 0;
! 459: if (--actcnt[patx] > 0) {
! 460: for (altc = 0; altc <= altcnt[patx]; altc++)
! 461: gpmin[patx][altc][actcnt[patx]]++;
! 462: a = gp_position[patx][actcnt[patx]]; /* try previous patterns again */
! 463: continue;
! 464: }
! 465: }
! 466: do {
! 467: actcnt[patx] = 0;
! 468: if (--patx < 0)
! 469: return '0'; /* stack exhausted */
! 470: if (*(ptrpcd[patx]) == '(') {
! 471: if (actcnt[patx] >= maxcnt[patx]) {
! 472: ++actcnt[patx];
! 473: patcode = '(';
! 474: goto nomatch;
! 475: }
! 476: }
! 477: } while (++actcnt[patx] > maxcnt[patx]);
! 478: a = position[patx]; /* try previous patterns again */
! 479:
! 480: } /* end repeat */
! 481: match:;
! 482: a++;
! 483: match0:;
! 484: }
! 485: position[patx++] = a; /* pos after last match */
! 486: }
! 487: return '0';
! 488: } /* end of pattern */
! 489: /******************************************************************************/
! 490: void
! 491: pminmax (str, min, max) /* auxiliary function for grouped pattern match */
! 492:
! 493: /* determines */
! 494: char *str; /* of a pattern 'str' */
! 495: int *min,
! 496: *max; /* the minimum and maximum possible length */
! 497:
! 498: {
! 499: int mininc,
! 500: maxinc,
! 501: i,
! 502: ch;
! 503:
! 504: *min = 0;
! 505: *max = 0;
! 506: mininc = 0;
! 507: maxinc = 0;
! 508: i = 0;
! 509: ch = 0;
! 510: while (str[i] != EOL) {
! 511: if (str[i] != '.') { /* scan minimum repeat count */
! 512: ch = (str[i++]) - '0';
! 513: while (str[i] >= '0' && str[i] <= '9') {
! 514: ch *= 10;
! 515: ch += (str[i++]) - '0';
! 516: }
! 517: mininc = ch;
! 518: maxinc = ch;
! 519: } else {
! 520: mininc = 0;
! 521: maxinc = 255;
! 522: }
! 523: if (str[i] == '.') { /* scan maximum repeat count */
! 524: i++;
! 525: if (str[i] >= '0' && str[i] <= '9') {
! 526: ch = (str[i++]) - '0';
! 527: while (str[i] >= '0' && str[i] <= '9') {
! 528: ch *= 10;
! 529: ch += (str[i]++) - '0';
! 530: }
! 531: } else
! 532: ch = 255;
! 533: maxinc = ch;
! 534: }
! 535: /* skip pattern codes */
! 536: if (str[i] == '"') {
! 537: int cnt;
! 538:
! 539: cnt = 0;
! 540: while (str[++i] != DELIM)
! 541: cnt++;
! 542: mininc = mininc * cnt;
! 543: maxinc = maxinc * cnt;
! 544: }
! 545: if (str[i] == 'z' || str[i] == '"') {
! 546: while (str[++i] != DELIM) ;
! 547: i++;
! 548: } else if (str[i] == '(') {
! 549: char tmp[256];
! 550: char *tcur;
! 551: int tmin,
! 552: tmax,
! 553: Tmin,
! 554: Tmax,
! 555: i1;
! 556:
! 557: tmin = 255;
! 558: tmax = 0;
! 559: alternation:;
! 560: i1 = 1;
! 561: tcur = tmp;
! 562: while (i1) {
! 563: ch = str[++i];
! 564: *tcur++ = ch;
! 565: if (ch == '"')
! 566: while ((*tcur++ = str[++i]) != DELIM) ;
! 567: if (ch == '(')
! 568: i1++;
! 569: if (ch == ')')
! 570: i1--;
! 571: if (ch == ',' && i1 == 1)
! 572: i1--;
! 573: }
! 574: *(--tcur) = EOL;
! 575: pminmax (tmp, &Tmin, &Tmax);
! 576: if (Tmin < tmin)
! 577: tmin = Tmin;
! 578: if (Tmax > tmax)
! 579: tmax = Tmax;
! 580: if (str[i] == ',')
! 581: goto alternation;
! 582: i++;
! 583: mininc *= tmin;
! 584: maxinc *= tmax;
! 585: } else
! 586: while (str[++i] >= 'A') ;
! 587: *min += mininc;
! 588: *max += maxinc;
! 589: }
! 590: if (*max > 255)
! 591: *max = 255;
! 592: return;
! 593: } /* end pminmax() */
! 594: /******************************************************************************/
! 595: void
! 596: add (a, b) /* string arithmetic a+=b; */
! 597: char *a,
! 598: *b;
! 599:
! 600: {
! 601:
! 602: if (b[0] == ZERO)
! 603: return;
! 604: if (a[0] == ZERO) {
! 605: stcpy (a, b);
! 606: return;
! 607: } {
! 608: long dpa, /* decimal point of 'a' */
! 609: dpb, /* decimal point of 'b' */
! 610: lena, /* length of 'a' */
! 611: lenb; /* length of 'b' */
! 612: char mi; /* minus flag */
! 613: short sign; /* sign flag if a<0<b sign=-1; */
! 614:
! 615: /* if a>0>b sign=1; */
! 616: /* else sign=0; */
! 617:
! 618: register int i;
! 619: register int ch;
! 620: register int j;
! 621: register int carry;
! 622:
! 623: #if !defined(_AIX)
! 624: if (fp_mode) {
! 625: double fp_a;
! 626: double fp_b;
! 627:
! 628: stcnv_m2c (a);
! 629: stcnv_m2c (b);
! 630:
! 631: fp_a = atof (a);
! 632: fp_b = atof (b);
! 633:
! 634: snprintf (a, STRLEN - 1, fp_conversion, fp_a + fp_b);
! 635: trim_decimal (a);
! 636:
! 637: return;
! 638: }
! 639: #endif
! 640:
! 641:
! 642:
! 643: /* look at the signs */
! 644: mi = 0;
! 645: sign = 0;
! 646: if (a[0] == b[0] && a[0] == MINUS) {
! 647: mi++;
! 648: a[0] = b[0] = ZERO;
! 649: } else if (a[0] == MINUS) {
! 650: sign--;
! 651: a[0] = NINE;
! 652: i = 0;
! 653: while ((ch = a[++i]) != EOL)
! 654: if (ch != POINT)
! 655: a[i] = ZERO + NINE - ch;
! 656: a[--i]++;
! 657: } else if (b[0] == MINUS) {
! 658: sign++;
! 659: b[0] = NINE;
! 660: i = 0;
! 661: while ((ch = b[++i]) != EOL)
! 662: if (ch != POINT)
! 663: b[i] = ZERO + NINE - ch;
! 664: b[--i]++;
! 665: }
! 666: /* search decimal points and length */
! 667: dpa = dpb = (-1);
! 668: i = 0;
! 669: while (a[i] != EOL) {
! 670: if (a[i] == POINT)
! 671: dpa = i;
! 672: i++;
! 673: }
! 674: lena = i;
! 675: if (dpa < 0)
! 676: dpa = i;
! 677: again:;
! 678: i = 0;
! 679: while (b[i] != EOL) {
! 680: if (b[i] == POINT)
! 681: dpb = i;
! 682: i++;
! 683: }
! 684: lenb = i;
! 685: if (dpb < 0)
! 686: dpb = i;
! 687: if (i == 1) {
! 688: if (b[0] == ONE && sign == 0 && dpa > 0) { /* frequent special case: add 1 */
! 689: i = dpa - 1;
! 690: while (++a[i] > NINE) {
! 691: a[i--] = ZERO;
! 692: if (i < 0) {
! 693: i = lena;
! 694: while (i >= 0) {
! 695: a[i + 1] = a[i];
! 696: i--;
! 697: }
! 698: a[0] = ONE;
! 699: return;
! 700: }
! 701: }
! 702: return;
! 703: }
! 704: }
! 705: /* copy additional trailing digits from b to a */
! 706: if (lenb - dpb > lena - dpa) {
! 707: j = dpa - dpb;
! 708: if (lenb + j > STRLEN) { /* round off that monster ! */
! 709: i = STRLEN - j;
! 710: if (b[i] < FIVE) {
! 711: b[i] = EOL;
! 712: lenb--;
! 713: while (b[--i] == ZERO) {
! 714: b[i] = EOL;
! 715: lenb--;
! 716: }
! 717: } else {
! 718: for (;;)
! 719: {
! 720: if (i >= dpb) {
! 721: b[i] = EOL;
! 722: lenb--;
! 723: } else
! 724: b[i] = ZERO;
! 725: if (--i < 0) {
! 726: for (i = lenb; i >= 0; i--)
! 727: b[i + 1] = b[i];
! 728: b[0] = ONE;
! 729: dpb = ++lenb;
! 730: break;
! 731: }
! 732: if ((ch = b[i]) == POINT) {
! 733: dpb = i;
! 734: continue;
! 735: }
! 736: if (ch < NINE && ch >= ZERO) {
! 737: b[i] = ++ch;
! 738: break;
! 739: }
! 740: }
! 741: }
! 742: goto again; /* look what's left from b */
! 743: }
! 744: lenb = i = lena - dpa + dpb;
! 745: j = lena;
! 746: while ((a[j++] = b[i++]) != EOL) ;
! 747: lena = (--j);
! 748: b[lenb] = EOL;
! 749: }
! 750: /* $justify a or b */
! 751: i = dpa - dpb;
! 752: if (i < 0) {
! 753: j = lena;
! 754: if ((i = (lena -= i)) > (STRLEN - 2) /*was 253*/) {
! 755: merr_raise (M75);
! 756: return;
! 757: }
! 758: ch = (sign >= 0 ? ZERO : NINE);
! 759: while (j >= 0)
! 760: a[i--] = a[j--];
! 761: while (i >= 0)
! 762: a[i--] = ch;
! 763: dpa = dpb;
! 764: } else if (i > 0) {
! 765: j = lenb;
! 766: if ((lenb = (i += lenb)) > (STRLEN - 2)/*was 253*/) {
! 767: merr_raise (M75);
! 768: return;
! 769: }
! 770: ch = (sign <= 0 ? ZERO : NINE);
! 771: while (j >= 0)
! 772: b[i--] = b[j--];
! 773: while (i >= 0)
! 774: b[i--] = ch;
! 775: dpb = dpa;
! 776: }
! 777: /* now add */
! 778: carry = 0;
! 779: for (i = lenb - 1; i >= 0; i--) {
! 780: if ((ch = a[i]) == POINT)
! 781: continue;
! 782: ch += b[i] - ZERO + carry;
! 783: if ((carry = (ch > NINE)))
! 784: ch -= NUMBASE;
! 785: a[i] = ch;
! 786: }
! 787: while (a[lena] != EOL)
! 788: lena++;
! 789: if (carry) {
! 790: if ((i = (++lena)) > (STRLEN - 2)/*was 253*/) {
! 791: merr_raise (M75);
! 792: return;
! 793: }
! 794: while (i > 0) {
! 795: a[i] = a[i - 1];
! 796: i--;
! 797: }
! 798: a[0] = ONE;
! 799: }
! 800: if (sign) {
! 801: if (a[0] == ONE) {
! 802: a[0] = ZERO;
! 803: } else {
! 804: i = 0;
! 805: carry = 0;
! 806: while ((ch = a[++i]) != EOL)
! 807: if (ch != POINT)
! 808: a[i] = ZERO + NINE - ch;
! 809: while (--i > 0) {
! 810: if (a[i] != POINT) {
! 811: if (++a[i] <= NINE)
! 812: break;
! 813: a[i] = ZERO;
! 814: }
! 815: }
! 816: mi = 1;
! 817: a[0] = ZERO;
! 818: }
! 819: while (a[mi] == ZERO) {
! 820: stcpy (&a[mi], &a[mi + 1]);
! 821: dpa--;
! 822: }
! 823: if (dpa < 0)
! 824: dpa = 0;
! 825: }
! 826: /* remove trailing zeroes */
! 827: i = dpa;
! 828: while (a[i] != EOL)
! 829: i++;
! 830: if (--i > dpa) {
! 831: while (a[i] == ZERO)
! 832: a[i--] = EOL;
! 833: }
! 834: /* remove trailing point */
! 835: if (a[i] == POINT)
! 836: a[i] = EOL;
! 837: if (mi) {
! 838: if (a[0] != ZERO) {
! 839: i = 0;
! 840: while (a[i++] != EOL) ;
! 841: while (i > 0) {
! 842: a[i] = a[i - 1];
! 843: i--;
! 844: }
! 845: }
! 846: a[0] = MINUS;
! 847: }
! 848: if (a[mi] == EOL) {
! 849: a[0] = ZERO;
! 850: a[1] = EOL;
! 851: }
! 852: return;
! 853: }
! 854: }
! 855: /******************************************************************************/
! 856: void
! 857: mul (a, b) /* string arithmetic a=a*b */
! 858: char *a,
! 859: *b;
! 860: {
! 861: char c[2*(STRLEN+1) /*was 512*/];
! 862: short alen,
! 863: blen,
! 864: clen,
! 865: mi,
! 866: tmpx;
! 867: register int acur;
! 868: register int bcur;
! 869: register int ccur;
! 870: register int carry;
! 871:
! 872: #if !defined(_AIX)
! 873: if (fp_mode) {
! 874: double fp_a;
! 875: double fp_b;
! 876:
! 877: stcnv_m2c (a);
! 878: stcnv_m2c (b);
! 879:
! 880: fp_a = atof (a);
! 881: fp_b = atof (b);
! 882:
! 883: snprintf (a, STRLEN - 1, fp_conversion, fp_a * fp_b);
! 884: trim_decimal (a);
! 885:
! 886: return;
! 887: }
! 888: #endif
! 889:
! 890:
! 891: if (merr () > OK)
! 892: return; /* avoid nonsense in recursion */
! 893: /* if zero or one there's not much to do */
! 894: if (b[1] == EOL) {
! 895: if (b[0] == ZERO) {
! 896: a[0] = ZERO;
! 897: a[1] = EOL;
! 898: return;
! 899: }
! 900: if (b[0] <= ONE)
! 901: return;
! 902: if (b[0] == TWO) {
! 903: multwo:acur = 0;
! 904: while (a[++acur] != EOL) ;
! 905: mi = (a[acur - 1] == FIVE);
! 906: carry = 0;
! 907: ccur = acur;
! 908: while (acur >= 0) {
! 909: if ((bcur = a[--acur]) < ZERO)
! 910: continue;
! 911: bcur = bcur * 2 - ZERO + carry;
! 912: carry = 0;
! 913: if (bcur > NINE) {
! 914: carry = 1;
! 915: bcur -= NUMBASE;
! 916: }
! 917: a[acur] = bcur;
! 918: }
! 919: if (carry) {
! 920: acur = ccur;
! 921: if (acur > (STRLEN - 1)/*was 254*/) {
! 922: merr_raise (M75);
! 923: return;
! 924: }
! 925: while (acur >= 0) {
! 926: a[acur + 1] = a[acur];
! 927: acur--;
! 928: }
! 929: a[a[0] == MINUS] = ONE;
! 930: }
! 931: if (mi) {
! 932: if (carry)
! 933: ccur++;
! 934: acur = 0;
! 935: while (acur < ccur)
! 936: if (a[acur++] == POINT) {
! 937: a[--ccur] = EOL;
! 938: if (acur == ccur)
! 939: a[--ccur] = EOL;
! 940: return;
! 941: }
! 942: }
! 943: return;
! 944: }
! 945: }
! 946: if (a[1] == EOL) {
! 947: if (a[0] == ZERO) {
! 948: return;
! 949: }
! 950: if (a[0] <= ONE) {
! 951: stcpy (a, b);
! 952: return;
! 953: }
! 954: if (a[0] == TWO) {
! 955: stcpy (a, b);
! 956: goto multwo;
! 957: }
! 958: }
! 959: /* get length of strings and convert ASCII to decimal */
! 960: /* have a look at the signs */
! 961: if ((mi = (a[0] == MINUS))) {
! 962: a[0] = ZERO;
! 963: }
! 964: if (b[0] == MINUS) {
! 965: b[0] = ZERO;
! 966: toggle (mi);
! 967: }
! 968: carry = 0;
! 969: alen = 0;
! 970: while (a[alen] != EOL) {
! 971: a[alen] -= ZERO;
! 972: if (a[alen++] == point)
! 973: carry = alen;
! 974: }
! 975: /* append a point on the right side if there was none */
! 976: if (--carry < 0) {
! 977: carry = alen;
! 978: a[alen++] = point;
! 979: a[alen] = 0;
! 980: }
! 981: ccur = 0;
! 982: blen = 0;
! 983: while (b[blen] != EOL) {
! 984: b[blen] -= ZERO;
! 985: if (b[blen++] == point)
! 986: ccur = blen;
! 987: }
! 988: if (--ccur < 0) {
! 989: ccur = blen;
! 990: b[blen++] = point;
! 991: b[blen] = 0;
! 992: }
! 993: carry += ccur;
! 994: if (carry > (STRLEN - 3) /*was 252*/) {
! 995: a[0] = EOL;
! 996: merr_raise (M75);
! 997: return;
! 998: }
! 999: ccur = clen = alen + blen;
! 1000: /* init c to zero */
! 1001: while (ccur >= 0)
! 1002: c[ccur--] = 0;
! 1003: c[carry] = point;
! 1004:
! 1005: bcur = blen;
! 1006: clen = alen + blen - 1;
! 1007: carry = 0;
! 1008: while (bcur > 0) {
! 1009: if (b[--bcur] == point) {
! 1010: continue;
! 1011: }
! 1012: if (c[clen] == point)
! 1013: clen--;
! 1014: acur = alen;
! 1015: ccur = clen--;
! 1016: while (acur > 0) {
! 1017: if (a[--acur] == point)
! 1018: continue;
! 1019: if (c[--ccur] == point)
! 1020: --ccur;
! 1021: tmpx = a[acur] * b[bcur] + c[ccur] + carry;
! 1022: carry = tmpx / NUMBASE;
! 1023: c[ccur] = tmpx % NUMBASE;
! 1024: }
! 1025: while (carry) {
! 1026: if (c[--ccur] == point)
! 1027: ccur--;
! 1028: if ((c[ccur] += carry) >= NUMBASE) {
! 1029: c[ccur] -= NUMBASE;
! 1030: carry = 1;
! 1031: } else
! 1032: carry = 0;
! 1033: }
! 1034: }
! 1035: /* copy result to a and convert it */
! 1036: a[ccur = clen = acur = (alen += blen)] = EOL;
! 1037: while (--ccur >= 0) {
! 1038: if (c[ccur] < NUMBASE)
! 1039: a[ccur] = c[ccur] + ZERO;
! 1040: else
! 1041: a[alen = ccur] = POINT;
! 1042: }
! 1043: /* oversize string */
! 1044: if (acur > STRLEN) {
! 1045: if (a[acur = STRLEN] >= FIVE) {
! 1046: int l1;
! 1047:
! 1048: l1 = STRLEN;
! 1049: if (a[l1] >= FIVE) {
! 1050: for (;;)
! 1051: {
! 1052: if (a[--l1] == POINT)
! 1053: l1--;
! 1054: if (l1 < (a[0] == MINUS)) {
! 1055: for (l1 = STRLEN; l1 > 0; l1--)
! 1056: a[l1] = a[l1 - 1];
! 1057: a[a[0] == MINUS] = ONE;
! 1058: break;
! 1059: }
! 1060: if ((++a[l1]) == (NINE + 1))
! 1061: a[l1] = ZERO;
! 1062: else
! 1063: break;
! 1064: }
! 1065: }
! 1066: }
! 1067: a[acur] = EOL;
! 1068: }
! 1069: /* remove trailing zeroes */
! 1070: if (acur >= alen) {
! 1071: while (a[--acur] == ZERO)
! 1072: a[acur] = EOL;
! 1073: }
! 1074: /* remove trailing point */
! 1075: if (a[acur] == POINT)
! 1076: a[acur] = EOL;
! 1077: /* remove leading zeroes */
! 1078: while (a[mi] == ZERO) {
! 1079: acur = mi;
! 1080: while ((a[acur] = a[acur + 1]) != EOL)
! 1081: acur++;
! 1082: }
! 1083: if (a[0] == EOL) {
! 1084: a[0] = ZERO;
! 1085: a[1] = EOL;
! 1086: mi = 0;
! 1087: }
! 1088: if (mi) {
! 1089: if (a[0] != ZERO) {
! 1090: acur = clen;
! 1091: while (acur > 0) {
! 1092: a[acur] = a[acur - 1];
! 1093: acur--;
! 1094: }
! 1095: }
! 1096: a[0] = MINUS;
! 1097: }
! 1098: return;
! 1099: }
! 1100: /******************************************************************************
! 1101: * for a detailed description of the method for the divisions see *
! 1102: * donald e.knuth 'the art of computer programming' vol.2 p.257 *
! 1103: ******************************************************************************/
! 1104: void
! 1105: mdiv (uu, v, typ) /* divide string arithmetic */
! 1106: char *uu, /* dividend and result */
! 1107: *v; /* divisor */
! 1108: short typ; /* type: '/' or '\' or '#' */
! 1109:
! 1110: {
! 1111: char q[STRLEN + 2 /*was 257*/]; /* quotient */
! 1112: char u[2*(STRLEN + 1)/*was 512*/];/* intermediate result */
! 1113: char vv[STRLEN +1 /*was 256*/];
! 1114: short d,
! 1115: d1,
! 1116: k1,
! 1117: m,
! 1118: ulen,
! 1119: vlen,
! 1120: dpu,
! 1121: dpv,
! 1122: guess,
! 1123: mi,
! 1124: plus,
! 1125: v1;
! 1126: register long int i;
! 1127: register int j;
! 1128: register int k;
! 1129: register int carry = 0;
! 1130:
! 1131: #if !defined(_AIX)
! 1132: if (fp_mode) {
! 1133: double fp_uu;
! 1134: double fp_v;
! 1135:
! 1136: stcnv_m2c (uu);
! 1137: stcnv_m2c (v);
! 1138:
! 1139: fp_uu = atof (uu);
! 1140: fp_v = atof (v);
! 1141:
! 1142: switch (typ) {
! 1143:
! 1144: case '/':
! 1145: snprintf (uu, STRLEN - 1, fp_conversion, fp_uu / fp_v);
! 1146: break;
! 1147:
! 1148: case '\\':
! 1149: snprintf (uu, STRLEN - 1, "%ld\201", (long) (fp_uu / fp_v));
! 1150: break;
! 1151:
! 1152: case '#':
! 1153: snprintf (uu, STRLEN - 1, "%ld\201", (((long) fp_uu) % ((long) fp_v)));
! 1154: //snprintf (uu, STRLEN - 1, "%ld\201", ((long) fp_uu < 0 ? (((long) fp_uu % (long) fp_v) + (long) fp_v) % (long) fp_v : (long) fp_uu % (long) fp_v));
! 1155: break;
! 1156: }
! 1157:
! 1158: trim_decimal (uu);
! 1159:
! 1160: return;
! 1161: }
! 1162: #endif
! 1163:
! 1164:
! 1165: if (merr () > OK)
! 1166: return; /* avoid nonsense in recursion */
! 1167: if (uu[0] == ZERO)
! 1168: return;
! 1169: #ifdef NEVER
! 1170: /* should be faster on DIV 2, but causes some error
! 1171: * in connection with SQRT */
! 1172: if (v[1] == EOL && typ == '/') {
! 1173: if (v[0] == ONE)
! 1174: return;
! 1175: if (v[0] == TWO) {
! 1176: carry = 0;
! 1177: k = (-1);
! 1178: k1 = (-1);
! 1179: while ((i = uu[++k]) != EOL) {
! 1180: if (i < ZERO) {
! 1181: if (i == POINT) {
! 1182: k1 = k;
! 1183: if (k + zprecise < STRLEN)
! 1184: uu[k + zprecise] = EOL;
! 1185: }
! 1186: continue;
! 1187: }
! 1188: if (i == EOL)
! 1189: break;
! 1190: if (i & 01)
! 1191: j = NUMBASE;
! 1192: i = (i + ZERO + carry) / 2;
! 1193: carry = j;
! 1194: j = 0;
! 1195: uu[k] = i;
! 1196: }
! 1197: j = (uu[0] == MINUS);
! 1198: if (uu[j] == ZERO) {
! 1199: while (j < k) {
! 1200: uu[j] = uu[j + 1];
! 1201: j++;
! 1202: }
! 1203: k--;
! 1204: }
! 1205: if (carry && k < (STRLEN - 2) /*was 253*/) {
! 1206: if (k1 < 0) {
! 1207: k1 = k;
! 1208: uu[k++] = POINT;
! 1209: }
! 1210: uu[k++] = FIVE;
! 1211: uu[k] = EOL;
! 1212: }
! 1213: return;
! 1214: }
! 1215: }
! 1216: #endif /* NEVER */
! 1217: /* look at the signs */
! 1218: stcpy (u, uu);
! 1219: mi = 0;
! 1220: plus = 0;
! 1221: if (typ != '#') {
! 1222: if (u[0] == MINUS) {
! 1223: u[0] = ZERO;
! 1224: mi = 1;
! 1225: }
! 1226: if (v[0] == MINUS) {
! 1227: v[0] = ZERO;
! 1228: toggle (mi);
! 1229: }
! 1230: } else {
! 1231: stcpy (vv, v);
! 1232: if (u[0] == MINUS) {
! 1233: u[0] = ZERO;
! 1234: plus = 1;
! 1235: }
! 1236: if (v[0] == MINUS) {
! 1237: v[0] = ZERO;
! 1238: mi = 1;
! 1239: toggle (plus);
! 1240: }
! 1241: }
! 1242: /* convert from ASCII to 'number' */
! 1243: i = 0;
! 1244: dpv = (-1);
! 1245: k = 0;
! 1246: while ((j = v[i]) != EOL) {
! 1247: j -= ZERO;
! 1248: if (j == point)
! 1249: dpv = i;
! 1250: if (j == 0)
! 1251: k++;
! 1252: v[i++] = j;
! 1253: }
! 1254:
! 1255: v[vlen = i] = 0;
! 1256: v[i + 1] = 0;
! 1257: v[i + 2] = 0;
! 1258: if (v[0] != 0) {
! 1259: while (i >= 0) {
! 1260: v[i + 1] = v[i];
! 1261: i--;
! 1262: }
! 1263: v[0] = 0;
! 1264: dpv++;
! 1265: } else {
! 1266: vlen--;
! 1267: }
! 1268: d1 = 0;
! 1269:
! 1270: i = 0;
! 1271: dpu = (-1);
! 1272: while (u[i] != EOL) {
! 1273: u[i] -= ZERO;
! 1274: if (u[i] == point)
! 1275: dpu = i;
! 1276: i++;
! 1277: }
! 1278: if (dpu < 0) {
! 1279: u[dpu = i++] = point;
! 1280: }
! 1281: /* u[ulen=i]=0; u[i+1]=0; u[i+2]=0; */
! 1282: ulen = i;
! 1283: while (i < 512)
! 1284: u[i++] = 0;
! 1285: i = ulen; /* somehow that's necessary - sometimes I check why */
! 1286: if (u[0] != 0) {
! 1287: while (i >= 0) {
! 1288: u[i + 1] = u[i];
! 1289: i--;
! 1290: }
! 1291: u[0] = 0;
! 1292: dpu++;
! 1293: } else {
! 1294: ulen--;
! 1295: }
! 1296: if ((vlen + zprecise) > STRLEN && (dpv + zprecise) < vlen)
! 1297: vlen -= zprecise;
! 1298:
! 1299: if (dpv > 0) { /* make v an integer *//* shift v */
! 1300: d1 = vlen - dpv;
! 1301: for (i = dpv; i < vlen; i++)
! 1302: v[i] = v[i + 1];
! 1303: vlen--;
! 1304: /* remove leading zeroes */
! 1305: while (v[1] == 0) {
! 1306: for (i = 1; i <= vlen; i++)
! 1307: v[i] = v[i + 1];
! 1308: vlen--;
! 1309: }
! 1310: v[vlen + 1] = 0;
! 1311: v[vlen + 2] = 0;
! 1312: /* shift u */
! 1313: i = dpu;
! 1314: for (j = 0; j < d1; j++) {
! 1315: if (i >= ulen) {
! 1316: u[i + 1] = 0;
! 1317: ulen++;
! 1318: }
! 1319: u[i] = u[i + 1];
! 1320: i++;
! 1321: }
! 1322: u[i] = point;
! 1323: dpu = i;
! 1324: }
! 1325: d = dpu + 1 - ulen;
! 1326: if (dpv > dpu)
! 1327: d += dpv - dpu;
! 1328: if (typ == '/')
! 1329: d += zprecise;
! 1330: if ((d + ulen) > STRLEN) {
! 1331: u[0] = EOL;
! 1332: merr_raise (M75);
! 1333: return;
! 1334: }
! 1335: while (d > 0) {
! 1336: u[++ulen] = 0;
! 1337: d--;
! 1338: }
! 1339: /* normalize */
! 1340: if ((d = NUMBASE / (v[1] + 1)) > 1) {
! 1341: i = ulen;
! 1342: carry = 0;
! 1343: while (i > 0) {
! 1344: if (u[i] != point) {
! 1345: carry += u[i] * d;
! 1346: u[i] = carry % NUMBASE;
! 1347: carry = carry / NUMBASE;
! 1348: }
! 1349: i--;
! 1350: }
! 1351: u[0] = carry;
! 1352: i = vlen;
! 1353: carry = 0;
! 1354: while (i > 0) {
! 1355: carry += v[i] * d;
! 1356: v[i] = carry % NUMBASE;
! 1357: carry = carry / NUMBASE;
! 1358: i--;
! 1359: }
! 1360: v[0] = carry;
! 1361: }
! 1362: /* initialize */
! 1363: j = 0;
! 1364: m = ulen - vlen + 1;
! 1365: if (m <= dpu)
! 1366: m = dpu + 1;
! 1367: for (i = 0; i <= m; q[i++] = ZERO) ;
! 1368: if (typ == '#') {
! 1369: m = dpu - vlen;
! 1370: }
! 1371: v1 = v[1];
! 1372:
! 1373: while (j < m) {
! 1374: if (u[j] != point) { /* calculate guess */
! 1375: if ((k = u[j] * NUMBASE + (u[j + 1] == point ? u[j + 2] : u[j + 1])) == 0) {
! 1376: j++;
! 1377: continue;
! 1378: }
! 1379: k1 = (u[j + 1] == point || u[j + 2] == point ? u[j + 3] : u[j + 2]);
! 1380: guess = (u[j] == v1 ? (NUMBASE - 1) : k / v1);
! 1381: if (v[2] * guess > (k - guess * v1) * NUMBASE + k1) {
! 1382: guess--;
! 1383: if (v[2] * guess > (k - guess * v1) * NUMBASE + k1)
! 1384: guess--;
! 1385: }
! 1386: /* multiply and subtract */
! 1387: i = vlen;
! 1388: carry = 0;
! 1389: k = j + i;
! 1390: if (j < dpu && k >= dpu)
! 1391: k++;
! 1392: while (k >= 0) {
! 1393: if (u[k] == point)
! 1394: k--;
! 1395: if (i >= 0) {
! 1396: u[k] -= v[i--] * guess + carry;
! 1397: } else {
! 1398: if (carry == 0)
! 1399: break;
! 1400: u[k] -= carry;
! 1401: }
! 1402: carry = 0;
! 1403: while (u[k] < 0) {
! 1404: u[k] += NUMBASE;
! 1405: carry++;
! 1406: }
! 1407: k--;
! 1408: }
! 1409: /* test remainder / add back */
! 1410: if (carry) {
! 1411: guess--;
! 1412: i = vlen;
! 1413: carry = 0;
! 1414: k = j + i;
! 1415: if (j < dpu && k >= dpu)
! 1416: k++;
! 1417: while (k >= 0) {
! 1418: if (u[k] == point)
! 1419: k--;
! 1420: if (i >= 0) {
! 1421: u[k] += v[i--] + carry;
! 1422: } else {
! 1423: if (carry == 0)
! 1424: break;
! 1425: u[k] += carry;
! 1426: }
! 1427: carry = u[k] / NUMBASE;
! 1428: u[k] = u[k] % NUMBASE;
! 1429: k--;
! 1430: }
! 1431: }
! 1432: q[j++] = guess + ZERO;
! 1433: u[0] = 0;
! 1434: } else {
! 1435: q[j++] = POINT;
! 1436: }
! 1437: }
! 1438: /* unnormalize */
! 1439: if (typ != '#') {
! 1440: i = 0;
! 1441: while (i <= m) {
! 1442: if ((u[i] = q[i]) == POINT)
! 1443: dpv = i;
! 1444: i++;
! 1445: }
! 1446: k = vlen;
! 1447: k1 = dpv;
! 1448: while (k-- > 0) {
! 1449: while (k1 <= 0) {
! 1450: for (i = (++m); i > 0; i--)
! 1451: u[i] = u[i - 1];
! 1452: k1++;
! 1453: u[0] = ZERO;
! 1454: }
! 1455: u[k1] = u[k1 - 1];
! 1456: u[--k1] = POINT;
! 1457: dpv = k1;
! 1458: }
! 1459: u[m] = EOL;
! 1460: /* rounding */
! 1461:
! 1462: if (typ != '/')
! 1463: u[dpv + 1] = EOL;
! 1464: else {
! 1465: k = dpv + zprecise;
! 1466: k1 = u[k + 1] >= FIVE;
! 1467: u[k + 1] = EOL;
! 1468: if (k1) {
! 1469: do {
! 1470: if (u[k] != POINT) {
! 1471: if ((carry = (u[k] == NINE)))
! 1472: u[k] = ZERO;
! 1473: else
! 1474: u[k]++;
! 1475: }
! 1476: k--;
! 1477: } while (carry);
! 1478: }
! 1479: }
! 1480: } else { /* return the remainder */
! 1481: carry = 0;
! 1482: if (d > 1) {
! 1483: for (i = 0; i <= ulen; i++) {
! 1484: if (u[i] == point) {
! 1485: u[i] = POINT;
! 1486: dpu = i;
! 1487: } else {
! 1488: u[i] = (j = carry + u[i]) / d + ZERO;
! 1489: carry = j % d * NUMBASE;
! 1490: }
! 1491: }
! 1492: } else {
! 1493: for (i = 0; i <= ulen; i++)
! 1494: if (u[i] == point)
! 1495: u[dpu = i] = POINT;
! 1496: else
! 1497: u[i] += ZERO;
! 1498: }
! 1499: u[i] = EOL;
! 1500: if (d1 > 0) {
! 1501: u[i + 1] = EOL;
! 1502: u[i + 2] = EOL;
! 1503: i = dpu;
! 1504: for (j = 0; j < d1; j++) {
! 1505: u[i] = u[i - 1];
! 1506: i--;
! 1507: }
! 1508: u[i] = POINT;
! 1509: }
! 1510: }
! 1511: /* remove trailing zeroes */
! 1512: i = 0;
! 1513: k = (-1);
! 1514: while (u[i] != EOL) {
! 1515: if (u[i] == POINT)
! 1516: k = i;
! 1517: i++;
! 1518: }
! 1519: i--;
! 1520: if (k >= 0) {
! 1521: while (u[i] == ZERO && i > k)
! 1522: u[i--] = EOL;
! 1523: }
! 1524: /* remove trailing point */
! 1525: if (u[i] == POINT)
! 1526: u[i] = EOL;
! 1527: /* remove leading zeroes */
! 1528: while (u[0] == ZERO) {
! 1529: i = 0;
! 1530: while ((u[i] = u[i + 1]) != EOL)
! 1531: i++;
! 1532: }
! 1533: if (u[0] == EOL) {
! 1534: u[0] = ZERO;
! 1535: u[1] = EOL;
! 1536: mi = 0;
! 1537: }
! 1538: if ((mi || plus) && (u[0] != ZERO)) {
! 1539: if (mi != plus) {
! 1540: i = stlen (u) + 1;
! 1541: do {
! 1542: u[i] = u[i - 1];
! 1543: i--;
! 1544: } while (i > 0);
! 1545: u[0] = MINUS;
! 1546: }
! 1547: if (plus)
! 1548: add (u, vv);
! 1549: }
! 1550: stcpy (uu, u);
! 1551: return;
! 1552: } /* end div() */
! 1553: /******************************************************************************/
! 1554: void
! 1555: power (a, b) /* raise a to the b-th power */
! 1556: char *a,
! 1557: *b;
! 1558: {
! 1559: char c[STRLEN + 2/*was 257*/];
! 1560: char d[4*(STRLEN + 1)/*was 1024*/];/* 257 should be sufficient, but somewhere there */
! 1561:
! 1562: /* is a memory leak resulting in wrong results */
! 1563: /* with fractional powers, e.g. 2**(3/7) */
! 1564: /* even a value of 513 is too small */
! 1565: char e[STRLEN + 2/*was 257*/];
! 1566: register long i;
! 1567: register long j;
! 1568:
! 1569: #if !defined(_AIX)
! 1570: if (fp_mode) {
! 1571: double fp_a;
! 1572: double fp_b;
! 1573:
! 1574: stcnv_m2c (a);
! 1575: stcnv_m2c (b);
! 1576:
! 1577: fp_a = atof (a);
! 1578: fp_b = atof (b);
! 1579:
! 1580: snprintf (a, STRLEN - 1, fp_conversion, pow (fp_a, fp_b));
! 1581:
! 1582: trim_decimal (a);
! 1583:
! 1584: return;
! 1585: }
! 1586: #endif
! 1587:
! 1588:
! 1589: if (merr () > OK)
! 1590: return; /* avoid nonsense in recursion */
! 1591: /* if zero or one there's not much to do */
! 1592: if (a[1] == EOL) {
! 1593: if (a[0] == ZERO) {
! 1594: if (b[1] == EOL && b[0] == ZERO)
! 1595: merr_raise (M9);
! 1596: return;
! 1597: } /* undef */
! 1598: if (a[0] == ONE)
! 1599: return;
! 1600: }
! 1601: if (b[0] == MINUS) {
! 1602: power (a, &b[1]);
! 1603: if (merr () == M75) {
! 1604: a[0] = ZERO;
! 1605: a[1] = EOL;
! 1606: merr_raise (OK);
! 1607: return;
! 1608: }
! 1609: stcpy (c, a);
! 1610: a[0] = ONE;
! 1611: a[1] = EOL;
! 1612: mdiv (a, c, '/');
! 1613: return;
! 1614: }
! 1615: if (b[1] == EOL) {
! 1616: switch (b[0]) {
! 1617: case ZERO:
! 1618: a[0] = ONE;
! 1619: a[1] = EOL;
! 1620: case ONE:
! 1621: return;
! 1622: case TWO:
! 1623: stcpy (c, a);
! 1624: mul (a, c);
! 1625: return;
! 1626: }
! 1627: }
! 1628: /* look for decimal point */
! 1629: e[0] = EOL;
! 1630: i = 0;
! 1631: while (b[i] != EOL) {
! 1632: if (b[i] == POINT) {
! 1633: if (a[0] == MINUS) {
! 1634: merr_raise (M9);
! 1635: return;
! 1636: } /* undefined case */
! 1637: if (b[i + 1] == FIVE && b[i + 2] == EOL) { /* half-integer: extra solution */
! 1638: if (i) {
! 1639: stcpy (c, b);
! 1640: add (b, c);
! 1641: power (a, b);
! 1642: if (merr () > OK) {
! 1643: a[0] = ONE;
! 1644: a[1] = EOL;
! 1645: return;
! 1646: }
! 1647: }
! 1648: g_sqrt (a);
! 1649: return;
! 1650: }
! 1651: stcpy (e, &b[i]);
! 1652: b[i] = EOL;
! 1653: break;
! 1654: }
! 1655: i++;
! 1656: }
! 1657: stcpy (d, a);
! 1658: i = intexpr (b);
! 1659: if (merr () == MXNUM)
! 1660: return;
! 1661: /* do it with a small number of multiplications */
! 1662: /* the number of multiplications is not optimum, but reasonably small */
! 1663: /* see donald e. knuth "The Art of Computer Programming" Vol.II p.441 */
! 1664: if (i == 0) {
! 1665: a[0] = ONE;
! 1666: a[1] = EOL;
! 1667: } else {
! 1668: j = 1;
! 1669: while (j < i) {
! 1670: j = j * 2;
! 1671: if (j < 0) {
! 1672: merr_raise (MXNUM);
! 1673: return;
! 1674: }
! 1675: }
! 1676: if (i != j)
! 1677: j = j / 2;
! 1678: j = j / 2;
! 1679: while (j) {
! 1680: stcpy (c, a);
! 1681: mul (a, c);
! 1682: if (i & j) {
! 1683: stcpy (c, d);
! 1684: mul (a, c);
! 1685: }
! 1686: j = j / 2;
! 1687: if (merr () == MXNUM)
! 1688: return;
! 1689: }
! 1690: if (e[0] == EOL)
! 1691: return;
! 1692: }
! 1693: /* non integer exponent */
! 1694: /* state of computation at this point: */
! 1695: /* d == saved value of a */
! 1696: /* a == d^^int(b); */
! 1697: /* e == frac(b); */
! 1698: {
! 1699: char Z[STRLEN + 2/*was 257*/];
! 1700:
! 1701: /* is fraction the inverse of an integer? */
! 1702: Z[0] = ONE;
! 1703: Z[1] = EOL;
! 1704: stcpy (c, e);
! 1705: mdiv (Z, c, '/');
! 1706: i = 0;
! 1707: for (;;)
! 1708: {
! 1709: if ((j = Z[i++]) == EOL) {
! 1710: j = intexpr (Z);
! 1711: break;
! 1712: }
! 1713: if (j != POINT)
! 1714: continue;
! 1715: j = intexpr (Z);
! 1716: if (Z[i] == NINE)
! 1717: j++;
! 1718: break;
! 1719: }
! 1720: Z[0] = ONE;
! 1721: Z[1] = EOL;
! 1722: lintstr (c, j);
! 1723: mdiv (Z, c, '/');
! 1724: /* if integer */
! 1725:
! 1726: if (stcmp (Z, e) == 0) {
! 1727: stcpy (Z, d);
! 1728: root (Z, j);
! 1729: if (merr () <= OK) {
! 1730: mul (a, Z);
! 1731: return;
! 1732: } /* on error try other method */
! 1733: merr_raise (OK);
! 1734: }
! 1735: Z[0] = ONE;
! 1736: Z[1] = EOL;
! 1737: zprecise += 2;
! 1738: for (;;)
! 1739: {
! 1740: c[0] = TWO;
! 1741: c[1] = EOL;
! 1742: mul (e, c);
! 1743: g_sqrt (d);
! 1744: if (e[0] == ONE) {
! 1745: e[0] = ZERO;
! 1746: numlit (e);
! 1747: stcpy (c, d);
! 1748: mul (Z, c);
! 1749: mround (Z, zprecise);
! 1750: }
! 1751: if (e[0] == ZERO)
! 1752: break;
! 1753: i = 0;
! 1754: j = (d[0] == ONE ? ZERO : NINE);
! 1755: for (;;) {
! 1756: ++i;
! 1757: if (d[i] != j && d[i] != '.')
! 1758: break;
! 1759: }
! 1760: if (d[i] == EOL || (i > zprecise))
! 1761: break;
! 1762: }
! 1763: zprecise -= 2;
! 1764: mul (a, Z);
! 1765: mround (a, zprecise + 1);
! 1766: }
! 1767: return;
! 1768: } /* end power() */
! 1769: /******************************************************************************/
! 1770: void
! 1771: g_sqrt (a) /* square root */
! 1772: char *a;
! 1773:
! 1774: {
! 1775: register int i,
! 1776: ch;
! 1777:
! 1778: if (a[0] == ZERO)
! 1779: return;
! 1780: if (a[0] == MINUS) {
! 1781: merr_raise (M9);
! 1782: return;
! 1783: }
! 1784: if (merr () > OK)
! 1785: return; /* avoid nonsense in recursion */
! 1786: {
! 1787: char tmp1[STRLEN +2 /*was 257*/],
! 1788: tmp2[STRLEN +2 /*was 257*/],
! 1789: XX[STRLEN +2 /*was 257*/],
! 1790: XXX[STRLEN +2 /*was 257*/];
! 1791:
! 1792: stcpy (XX, a);
! 1793: /* look for good initial value */
! 1794: if (a[0] > ONE || (a[0] == ONE && a[1] != POINT)) {
! 1795: i = 0;
! 1796: while ((ch = a[i++]) != EOL) {
! 1797: if (ch == POINT)
! 1798: break;
! 1799: }
! 1800: if ((i = (i + 1) / 2))
! 1801: a[i] = EOL;
! 1802: } else if (a[0] != ONE) {
! 1803: a[0] = ONE;
! 1804: a[1] = EOL;
! 1805: }
! 1806: /* "Newton's" algorithm with quadratic convergence */
! 1807: zprecise++;
! 1808: do {
! 1809: stcpy (XXX, a);
! 1810: stcpy (tmp1, XX);
! 1811: stcpy (tmp2, a);
! 1812: mdiv (tmp1, tmp2, '/');
! 1813: if (merr () > OK)
! 1814: break;
! 1815: add (a, tmp1);
! 1816: tmp2[0] = TWO;
! 1817: tmp2[1] = EOL;
! 1818: mdiv (a, tmp2, '/');
! 1819: } while (comp (a, XXX));
! 1820: zprecise--;
! 1821: return;
! 1822: }
! 1823: } /* end g_sqrt() */
! 1824: /******************************************************************************/
! 1825: void
! 1826: root (a, n) /* n.th root */
! 1827: char *a;
! 1828: long n;
! 1829:
! 1830: {
! 1831: register int i,
! 1832: ch;
! 1833:
! 1834: if (a[0] == ZERO)
! 1835: return;
! 1836: if (a[0] == MINUS || n == 0) {
! 1837: merr_raise (M9);
! 1838: return;
! 1839: }
! 1840: if (merr () > OK)
! 1841: return; /* avoid nonsense in recursion */
! 1842: {
! 1843: char tmp1[STRLEN +2/*was 257*/],
! 1844: tmp2[STRLEN +2/*was 257*/],
! 1845: XX[STRLEN +2/*was 257*/],
! 1846: XXX[STRLEN +2/*was 257*/];
! 1847: short again;
! 1848:
! 1849: stcpy (XX, a);
! 1850: /* look for good initial value */
! 1851: if (a[0] > ONE || (a[0] == ONE && a[1] != POINT)) {
! 1852: i = 0;
! 1853: while ((ch = a[i++]) != EOL && ch != POINT) ;
! 1854: if ((i = (i + n - 2) / n) > 0) {
! 1855: a[0] = THREE;
! 1856: a[i] = EOL;
! 1857: }
! 1858: } else if (a[0] != ONE) {
! 1859: a[0] = ONE;
! 1860: a[1] = EOL;
! 1861: }
! 1862: /* "Newton's" algorithm with quadratic convergence */
! 1863:
! 1864: if (zprecise <= 3)
! 1865: again = 0; /* speedup div with small zprec. */
! 1866: else {
! 1867: again = zprecise;
! 1868: zprecise = 2;
! 1869: }
! 1870: second:;
! 1871: zprecise++;
! 1872: for (;;) {
! 1873: stcpy (XXX, a);
! 1874: lintstr (tmp1, n - 1);
! 1875: stcpy (tmp2, a);
! 1876: power (tmp2, tmp1);
! 1877: stcpy (tmp1, XX);
! 1878: mdiv (tmp1, tmp2, '/');
! 1879: if (merr () > OK)
! 1880: break;
! 1881: lintstr (tmp2, n - 1);
! 1882: mul (a, tmp2);
! 1883: add (a, tmp1);
! 1884: lintstr (tmp2, n);
! 1885: mdiv (a, tmp2, '/');
! 1886: stcpy (tmp2, a);
! 1887: mdiv (XXX, tmp2, '/');
! 1888: tmp2[0] = ONE;
! 1889: if (zprecise <= 0)
! 1890: tmp2[1] = EOL;
! 1891: else {
! 1892: tmp2[1] = POINT;
! 1893: for (i = 2; i < zprecise; i++)
! 1894: tmp2[i] = ZERO;
! 1895: tmp2[i++] = FIVE;
! 1896: tmp2[i] = EOL;
! 1897: }
! 1898: if (!comp (XXX, tmp2))
! 1899: continue;
! 1900: if (zprecise <= 0)
! 1901: break;
! 1902: tmp2[0] = POINT;
! 1903: for (i = 1; i < zprecise; i++)
! 1904: tmp2[i] = NINE;
! 1905: tmp2[i - 1] = FIVE;
! 1906: tmp2[i] = EOL;
! 1907: if (comp (tmp2, XXX))
! 1908: break;
! 1909: }
! 1910: zprecise--;
! 1911: if (again) {
! 1912: zprecise = again;
! 1913: again = 0;
! 1914: goto second;
! 1915: }
! 1916: return;
! 1917: }
! 1918: } /* end root() */
! 1919: /******************************************************************************/
! 1920: int
! 1921: numlit (str)
! 1922: /** str is interpreted as a MUMPS number
! 1923: * and is converted to canonical form
! 1924: * return value: 1=EUR, 4=DM, 0=other */
! 1925: char *str;
! 1926: {
! 1927: long j,
! 1928: mi = 0,
! 1929: pointx = (-1),
! 1930: expflg = 0;
! 1931: long val,
! 1932: exp = 0L;
! 1933: int result;
! 1934: register int i = 0;
! 1935: register int ch;
! 1936:
! 1937: result = 0;
! 1938: if (str[0] < ONE) {
! 1939: if (str[0] == EOL) {
! 1940: str[0] = ZERO;
! 1941: str[1] = EOL;
! 1942: return (0);
! 1943: }
! 1944: /* compact signs */
! 1945: while (str[i] == PLUS || str[i] == MINUS)
! 1946: if (str[i++] == MINUS)
! 1947: mi++;
! 1948: if ((j = (mi &= 01)))
! 1949: str[0] = MINUS;
! 1950: /* compact leading zeroes */
! 1951: while (str[i] == ZERO)
! 1952: i++;
! 1953: if (i > j)
! 1954: stcpy (&str[j], &str[i]);
! 1955: if (str[mi] == EOL) {
! 1956: str[0] = ZERO;
! 1957: str[1] = EOL;
! 1958: return (0);
! 1959: }
! 1960: i = mi;
! 1961: }
! 1962: while ((ch = str[i]) <= NINE && ch >= ZERO)
! 1963: i++;
! 1964: if ((result = unit (&str[i]))) {
! 1965: if (i == mi) {
! 1966: str[0] = '0';
! 1967: i = 1;
! 1968: }
! 1969: ch = str[i] = EOL;
! 1970: }
! 1971: if (ch == EOL)
! 1972: return (result);
! 1973: if (ch == POINT) {
! 1974: pointx = i++;
! 1975: while ((ch = str[i]) <= NINE && ch >= ZERO)
! 1976: i++;
! 1977: if ((result = unit (&str[i])))
! 1978: ch = str[i] = EOL;
! 1979: if (ch == EOL) {
! 1980: i = pointx;
! 1981: goto point0;
! 1982: }
! 1983: }
! 1984: /* check for zero mantissa */
! 1985: j = 0;
! 1986: while (j < i) {
! 1987: if (str[j] > ZERO) {
! 1988: j = (-1);
! 1989: break;
! 1990: }
! 1991: j++;
! 1992: }
! 1993: if (j >= 0) {
! 1994: str[0] = ZERO;
! 1995: str[1] = EOL;
! 1996: return (result);
! 1997: }
! 1998: /* search for exponent */
! 1999: for (; ch != EOL; ch = str[++i]) {
! 2000: if (ch <= NINE && ch >= ZERO) {
! 2001: if (expflg) {
! 2002: ch -= ZERO;
! 2003: val = exp * NUMBASE + ch;
! 2004: /* check for numeric overflow */
! 2005: if (((val - ch) / NUMBASE) != exp) {
! 2006: merr_raise (MXNUM);
! 2007: return (0);
! 2008: }
! 2009: exp = val;
! 2010: }
! 2011: continue;
! 2012: }
! 2013: if (expflg) {
! 2014: if (ch == MINUS) {
! 2015: expflg = (-expflg);
! 2016: continue;
! 2017: }
! 2018: if (ch == PLUS)
! 2019: continue;
! 2020: }
! 2021: if ((result = unit (&str[i])))
! 2022: ch = str[i] = EOL;
! 2023: str[i] = EOL;
! 2024: if (ch == 'E' || (lowerflag && ch == 'e')) {
! 2025: expflg++;
! 2026: continue;
! 2027: }
! 2028: break;
! 2029: }
! 2030: /* append a point at the right end */
! 2031: if (expflg) {
! 2032: if (pointx < 0) {
! 2033: i = mi;
! 2034: while (str[i] != EOL)
! 2035: i++;
! 2036: if (i >= (STRLEN - 1)/*was 254*/) {
! 2037: str[STRLEN] = EOL;
! 2038: merr_raise (M75);
! 2039: return (0);
! 2040: }
! 2041: str[pointx = i] = POINT;
! 2042: str[++i] = EOL;
! 2043: }
! 2044: /* if exp shift decimal point */
! 2045: if (expflg > 0) {
! 2046: while (exp-- > 0) {
! 2047: if ((str[pointx] = str[pointx + 1]) == EOL) {
! 2048: if (pointx >= (STRLEN - 1)/*was 254*/)
! 2049: break;
! 2050: str[pointx] = ZERO;
! 2051: str[pointx + 1] = str[pointx + 2] = EOL;
! 2052: }
! 2053: pointx++;
! 2054: }
! 2055: if (pointx >= (STRLEN - 1)/*was 254*/) {
! 2056: str[STRLEN] = EOL;
! 2057: merr_raise (M75);
! 2058: return (0);
! 2059: }
! 2060: str[pointx] = POINT;
! 2061: } else { /* (expflg<0) */
! 2062: while (exp-- > 0) {
! 2063: if (--pointx < 0) {
! 2064: i = pointx = 0;
! 2065: while (str[i++] != EOL) ;
! 2066: if (i >= (STRLEN - 1)/*was 254*/) {
! 2067: merr_raise (M75);
! 2068: return (0);
! 2069: }
! 2070: while (i-- > 0)
! 2071: str[i + 1] = str[i];
! 2072: str[0] = ZERO;
! 2073: }
! 2074: str[pointx + 1] = str[pointx];
! 2075: }
! 2076: str[pointx] = POINT;
! 2077: }
! 2078: }
! 2079: if ((i = pointx) >= 0) {
! 2080:
! 2081: point0:
! 2082:
! 2083: while (str[++i] != EOL) ;
! 2084:
! 2085: i--;
! 2086: while (str[i] == ZERO)
! 2087: str[i--] = EOL; /* remove trailing zeroes */
! 2088: if (str[i] == POINT)
! 2089: str[i] = EOL; /* remove trailing point */
! 2090: }
! 2091: if (str[mi] == EOL) {
! 2092: str[0] = ZERO;
! 2093: str[1] = EOL;
! 2094: }
! 2095: return (result);
! 2096: }
! 2097: /******************************************************************************/
! 2098: int unit (char *str)
! 2099: /**
! 2100: * str is interpreted as a currency
! 2101: * symbol
! 2102: * return value: 1=EUR, 4=DM, ...
! 2103: * 0=other */
! 2104:
! 2105: {
! 2106: char ch;
! 2107:
! 2108: ch = str[0];
! 2109: if ((ch < 'A') || (ch > 'P'))
! 2110: return 0;
! 2111: switch (ch) {
! 2112: case 'E':
! 2113: if (str[1] == 'U' && str[2] == 'R')
! 2114: return 1;
! 2115: if (str[1] == 'S' && str[2] == 'P')
! 2116: return 5;
! 2117: return 0;
! 2118: case 'D':
! 2119: if (str[1] == 'M')
! 2120: return 4;
! 2121: if (str[1] == 'E' && str[2] == 'M')
! 2122: return 4;
! 2123: return 0;
! 2124: case 'A':
! 2125: if (str[1] == 'T' && str[2] == 'S')
! 2126: return 2;
! 2127: return 0;
! 2128: case 'B':
! 2129: if (str[1] == 'F' && str[2] == 'R')
! 2130: return 3;
! 2131: return 0;
! 2132: case 'F':
! 2133: if (str[1] == 'F')
! 2134: return 7;
! 2135: if (str[1] == 'M' && str[2] == 'K')
! 2136: return 6;
! 2137: if (str[1] == 'R' && str[2] == 'F')
! 2138: return 7;
! 2139: return 0;
! 2140: case 'I':
! 2141: if (str[1] == 'E' && str[2] == 'P')
! 2142: return 8;
! 2143: if (str[1] == 'T' && str[2] == 'L')
! 2144: return 9;
! 2145: return 0;
! 2146: case 'N':
! 2147: if (str[1] == 'L' && str[2] == 'G')
! 2148: return 10;
! 2149: return 0;
! 2150: case 'P':
! 2151: if (str[1] == 'T' && str[2] == 'E')
! 2152: return 11;
! 2153: }
! 2154: return 0;
! 2155: }
! 2156: /******************************************************************************/
! 2157: long
! 2158: intexpr (str)
! 2159: char *str;
! 2160:
! 2161: /* 'str' is interpreted as integer and converted to int */
! 2162: {
! 2163: {
! 2164: register int ch;
! 2165: register int i = 0;
! 2166: register long value;
! 2167: register long newval;
! 2168: short minus = FALSE;
! 2169:
! 2170: if ((ch = str[0]) == MINUS) {
! 2171: ch = str[1];
! 2172: minus = TRUE;
! 2173: i = 1;
! 2174: }
! 2175: if (ch >= ZERO && ch <= NINE) {
! 2176: value = ch - ZERO;
! 2177: while ((ch = str[++i]) >= ZERO && ch <= NINE) {
! 2178: newval = value * NUMBASE;
! 2179: if (newval < 0 || ((newval / NUMBASE) != value)) {
! 2180: merr_raise (MXNUM);
! 2181: return (minus ? -1 : 1);
! 2182: };
! 2183: newval += ((long) ch - ZERO);
! 2184: if (newval < 0) {
! 2185: merr_raise (MXNUM);
! 2186: return (minus ? -1 : 1);
! 2187: };
! 2188: value = newval;
! 2189: }
! 2190: if (minus)
! 2191: value = (-value);
! 2192: if (ch == EOL)
! 2193: return value;
! 2194: } else if (ch != ZERO && ch != PLUS && ch != MINUS && ch != POINT)
! 2195: return 0L;
! 2196: }
! 2197: {
! 2198: register int ch;
! 2199: register int i = 0;
! 2200: register long value;
! 2201: register long newval;
! 2202: char tmp[STRLEN +2/*was 257*/];
! 2203:
! 2204: stcpy (tmp, str);
! 2205: numlit (tmp);
! 2206: i = (tmp[0] == MINUS);
! 2207: if (merr () == MXNUM)
! 2208: return (i ? -1 : 1);
! 2209: value = 0L;
! 2210: while ((ch = tmp[i++]) >= ZERO && ch <= NINE) {
! 2211: newval = value * NUMBASE;
! 2212: if (newval < 0 || ((newval / NUMBASE) != value)) {
! 2213: merr_raise (MXNUM);
! 2214: value = 1;
! 2215: break;
! 2216: }
! 2217: newval += ((long) ch - ZERO);
! 2218: if (newval < 0) {
! 2219: merr_raise (MXNUM);
! 2220: value = 1;
! 2221: break;
! 2222: }
! 2223: value = newval;
! 2224: }
! 2225: if (tmp[0] == MINUS)
! 2226: value = (-value);
! 2227: return value;
! 2228: }
! 2229: } /* end of intexpr */
! 2230: /******************************************************************************/
! 2231: short int
! 2232: tvexpr (str) /* str is interpreted as truth valued expression */
! 2233: char *str;
! 2234: {
! 2235: if (str[0] > ZERO && str[0] <= NINE) {
! 2236: str[0] = ONE;
! 2237: str[1] = EOL;
! 2238: return TRUE;
! 2239: }
! 2240: if (str[1] == EOL) {
! 2241: str[0] = ZERO;
! 2242: return FALSE;
! 2243: } {
! 2244: register int ch;
! 2245: register int i = 0;
! 2246: register int pointx = FALSE;
! 2247: register int sign = FALSE;
! 2248:
! 2249: for (;;)
! 2250: {
! 2251: if ((ch = str[i]) > ZERO && ch <= NINE) {
! 2252: str[0] = ONE;
! 2253: str[1] = EOL;
! 2254: return TRUE;
! 2255: }
! 2256: i++;
! 2257: if ((ch == PLUS || ch == MINUS) && sign == FALSE)
! 2258: continue;
! 2259: sign = TRUE;
! 2260: if (ch == ZERO)
! 2261: continue;
! 2262: if (ch == POINT && pointx == FALSE) {
! 2263: sign = TRUE;
! 2264: pointx = TRUE;
! 2265: continue;
! 2266: }
! 2267: str[0] = ZERO;
! 2268: str[1] = EOL;
! 2269: return FALSE;
! 2270: }
! 2271: }
! 2272: }
! 2273: /******************************************************************************/
! 2274: void
! 2275: m_op (a, b, op)
! 2276: char *a,
! 2277: *b;
! 2278: short op;
! 2279: {
! 2280: int atyp,
! 2281: btyp; /* DM/EUR currency types */
! 2282: char tmp[(STRLEN + 1)/*was 256*/];
! 2283:
! 2284: switch (op & 0177) { /* binary operators */
! 2285: case '_':
! 2286: if (op & 0200)
! 2287: break; /* NOT_OPERAND */
! 2288: if (stcat (a, b) == 0) {
! 2289: merr_raise (M75);
! 2290: }
! 2291: return;
! 2292: case '=':
! 2293: if (stcmp (a, b))
! 2294: *a = ZERO;
! 2295: else
! 2296: *a = ONE;
! 2297: if (op & 0200)
! 2298: toggle (*a); /* NOT_OPERAND */
! 2299: a[1] = EOL;
! 2300: return;
! 2301: case '[':
! 2302: if (*b == EOL || find (a, b))
! 2303: *a = ONE;
! 2304: else
! 2305: *a = ZERO;
! 2306: if (op & 0200)
! 2307: toggle (*a); /* NOT_OPERAND */
! 2308: a[1] = EOL;
! 2309: return;
! 2310: case ']':
! 2311: if (*b == EOL) {
! 2312: if (*a == EOL)
! 2313: *a = ZERO;
! 2314: else
! 2315: *a = ONE;
! 2316: }
! 2317: /* frequent special case */
! 2318: else if (stcmp (a, b) <= 0)
! 2319: *a = ZERO;
! 2320: else
! 2321: *a = ONE;
! 2322: if (op & 0200)
! 2323: toggle (*a); /* NOT_OPERAND */
! 2324: a[1] = EOL;
! 2325: return;
! 2326: } /* end switch */
! 2327: atyp = numlit (a);
! 2328: if (op == '-') {
! 2329: stcpy (&tmp[1], b);
! 2330: tmp[0] = '-';
! 2331: op = '+';
! 2332: } else
! 2333: stcpy (tmp, b);
! 2334: btyp = numlit (tmp);
! 2335: switch (op & 0177) { /* binary operators, NOT OMITTED */
! 2336: case '+':
! 2337: if (op & 0200) {
! 2338: merr_raise (ASSIGNER);
! 2339: return;
! 2340: } /* NOT_OPERAND */
! 2341: #ifdef EUR2DEM
! 2342: if (atyp != btyp) {
! 2343: char tmp2[256];
! 2344:
! 2345: if ((atyp == 0) && (a[0] == '0'))
! 2346: atyp = btyp; /* zero is any currency */
! 2347: if ((btyp == 0) && (tmp[0] == '0'))
! 2348: btyp = atyp; /* zero is any currency */
! 2349: if (atyp && btyp) {
! 2350: if (atyp > 1) {
! 2351: stcpy (tmp2, EUR2WHR[atyp]);
! 2352: mul (tmp, tmp2);
! 2353: }
! 2354: if (btyp > 1) {
! 2355: zprecise += 4;
! 2356: stcpy (tmp2, EUR2WHR[btyp]);
! 2357: mdiv (tmp, tmp2, '/');
! 2358: zprecise -= 4;
! 2359: }
! 2360: } else if (atyp != btyp && typemmflag) {
! 2361: merr_raise (TYPEMISMATCH);
! 2362: return;
! 2363: }
! 2364: }
! 2365: #endif /* EUR2DEM */
! 2366: add (a, tmp);
! 2367: plus02:
! 2368: #ifdef EUR2DEM
! 2369: if (atyp == 0)
! 2370: break;
! 2371: if (atyp != btyp)
! 2372: cond_round (a, zprecise);
! 2373: if (atyp)
! 2374: stcat (a, WHR[atyp]);
! 2375: #endif /* EUR2DEM */
! 2376: break;
! 2377:
! 2378: case '*':
! 2379: if (op & 0200) {
! 2380: merr_raise (ASSIGNER);
! 2381: return;
! 2382: } /* NOT_OPERAND */
! 2383: #ifdef EUR2DEM
! 2384: if (btyp && atyp == 0) {
! 2385: atyp = btyp;
! 2386: btyp = 0;
! 2387: }
! 2388: if (atyp && btyp) {
! 2389: if (typemmflag) {
! 2390: merr_raise (TYPEMISMATCH);
! 2391: return;
! 2392: }
! 2393: atyp = btyp = 0;
! 2394: }
! 2395: #endif /* EUR2DEM */
! 2396: mul (a, tmp);
! 2397:
! 2398: #ifdef EUR2DEM
! 2399: if (atyp) {
! 2400: cond_round (a, zprecise);
! 2401: stcat (a, WHR[atyp]);
! 2402: }
! 2403: #endif /* EUR2DEM */
! 2404: break;
! 2405:
! 2406: case '/':
! 2407: case '\\':
! 2408: case '#':
! 2409:
! 2410: if (op & 0200) {
! 2411: merr_raise (ASSIGNER);
! 2412: return;
! 2413: } /* NOT_OPERAND */
! 2414: #ifdef EUR2DEM
! 2415: if (atyp != btyp) {
! 2416: char tmp2[(STRLEN + 1)/*was 256*/];
! 2417:
! 2418: if (atyp && btyp) {
! 2419: if (op == '#') {
! 2420: if (atyp > 1) {
! 2421: stcpy (tmp2, EUR2WHR[atyp]);
! 2422: mul (tmp, tmp2);
! 2423: }
! 2424: if (btyp > 1) {
! 2425: stcpy (tmp2, EUR2WHR[btyp]);
! 2426: mdiv (tmp, tmp2, '/');
! 2427: }
! 2428: } else {
! 2429: if (atyp > 1) {
! 2430: stcpy (tmp2, EUR2WHR[atyp]);
! 2431: mul (tmp, tmp2);
! 2432: }
! 2433: if (btyp > 1) {
! 2434: stcpy (tmp2, EUR2WHR[btyp]);
! 2435: mul (a, tmp2);
! 2436: }
! 2437: atyp = btyp = 0;
! 2438: }
! 2439: } else if (btyp && typemmflag && (*a != '0' || op == '#')) {
! 2440: merr_raise (TYPEMISMATCH);
! 2441: return;
! 2442: }
! 2443: } else if (op != '#')
! 2444: atyp = 0;
! 2445: #endif /* EUR2DEM */
! 2446: if (tmp[0] == ZERO) {
! 2447: merr_raise (M9);
! 2448: break;
! 2449: }
! 2450: mdiv (a, tmp, op);
! 2451: goto plus02;
! 2452:
! 2453: case ' ':
! 2454: power (a, tmp);
! 2455: break; /* ' ' stands for power */
! 2456:
! 2457: case '>':
! 2458:
! 2459: #ifdef EUR2DEM
! 2460: if (atyp != btyp) {
! 2461: char tmp2[(STRLEN + 1)/*was 256*/];
! 2462:
! 2463: if ((atyp == 0) && (a[0] == '0'))
! 2464: atyp = btyp; /* zero is any currency */
! 2465: if ((btyp == 0) && (tmp[0] == '0'))
! 2466: btyp = atyp; /* zero is any currency */
! 2467: if (atyp && btyp) {
! 2468: if (atyp > 1) {
! 2469: stcpy (tmp2, EUR2WHR[atyp]);
! 2470: mul (tmp, tmp2);
! 2471: }
! 2472: if (btyp > 1) {
! 2473: stcpy (tmp2, EUR2WHR[btyp]);
! 2474: mul (a, tmp2);
! 2475: }
! 2476: cond_round (a, zprecise + 2);
! 2477: cond_round (tmp, zprecise + 2);
! 2478: } else if (atyp != btyp && typemmflag) {
! 2479: merr_raise (TYPEMISMATCH);
! 2480: return;
! 2481: }
! 2482: }
! 2483: #endif /* EUR2DEM */
! 2484: if (comp (tmp, a))
! 2485: *a = ONE;
! 2486: else
! 2487: *a = ZERO;
! 2488: if (op & 0200)
! 2489: toggle (*a); /* NOT_OPERAND */
! 2490: a[1] = EOL;
! 2491: break;
! 2492:
! 2493: case '<':
! 2494:
! 2495: #ifdef EUR2DEM
! 2496: if (atyp != btyp) {
! 2497: char tmp2[(STRLEN + 1)/*was 256*/];
! 2498:
! 2499: if ((atyp == 0) && (a[0] == '0'))
! 2500: atyp = btyp; /* zero is any currency */
! 2501: if ((btyp == 0) && (tmp[0] == '0'))
! 2502: btyp = atyp; /* zero is any currency */
! 2503: if (atyp && btyp) {
! 2504: if (atyp > 1) {
! 2505: stcpy (tmp2, EUR2WHR[atyp]);
! 2506: mul (tmp, tmp2);
! 2507: }
! 2508: if (btyp > 1) {
! 2509: stcpy (tmp2, EUR2WHR[btyp]);
! 2510: mul (a, tmp2);
! 2511: }
! 2512: cond_round (a, zprecise + 2);
! 2513: cond_round (tmp, zprecise + 2);
! 2514: } else if (atyp != btyp && typemmflag) {
! 2515: merr_raise (TYPEMISMATCH);
! 2516: return;
! 2517: }
! 2518: }
! 2519: #endif /* EUR2DEM */
! 2520: if (comp (a, tmp))
! 2521: *a = ONE;
! 2522: else
! 2523: *a = ZERO;
! 2524: if (op & 0200)
! 2525: toggle (*a); /* NOT_OPERAND */
! 2526: a[1] = EOL;
! 2527: break;
! 2528:
! 2529: case '&':
! 2530:
! 2531: if (tvexpr (a)) {
! 2532: tvexpr (tmp);
! 2533: *a = *tmp;
! 2534: }
! 2535: if (op & 0200)
! 2536: toggle (*a); /* NOT_OPERAND */
! 2537: a[1] = EOL;
! 2538: break;
! 2539:
! 2540: case '!':
! 2541:
! 2542: if (tvexpr (a) == FALSE && tvexpr (tmp))
! 2543: *a = ONE;
! 2544: if (op & 0200)
! 2545: toggle (*a); /* NOT_OPERAND */
! 2546: a[1] = EOL;
! 2547: break;
! 2548:
! 2549: default:
! 2550: merr_raise (ASSIGNER);
! 2551:
! 2552: }
! 2553: return;
! 2554: } /* end m_op */
! 2555: /******************************************************************************/
! 2556: /* rounding */
! 2557: /* 'a' is assumed to be a 'canonic' numeric string */
! 2558: /* it is rounded to 'digits' fractional digits */
! 2559: void
! 2560: mround (a, digits)
! 2561: char *a;
! 2562: int digits;
! 2563: {
! 2564: int ch,
! 2565: i,
! 2566: pointpos,
! 2567: lena;
! 2568:
! 2569: pointpos = -1;
! 2570: i = 0;
! 2571: i = 0;
! 2572: while (a[i] != EOL) {
! 2573: if (a[i] == POINT)
! 2574: pointpos = i;
! 2575: i++;
! 2576: }
! 2577: lena = i;
! 2578: if (pointpos < 0)
! 2579: pointpos = i;
! 2580: if ((pointpos + digits + 1) >= i)
! 2581: return; /* nothing to round */
! 2582: i = pointpos + digits + 1;
! 2583: if (a[i] < FIVE) {
! 2584: a[i] = EOL;
! 2585: while (a[--i] == ZERO)
! 2586: a[i] = EOL;
! 2587: if (a[i] == POINT) {
! 2588: a[i] = EOL;
! 2589: if (i == 0 || (i == 1 && a[0] == MINUS))
! 2590: a[0] = ZERO;
! 2591: }
! 2592: return;
! 2593: }
! 2594: for (;;)
! 2595: {
! 2596: if (i >= pointpos)
! 2597: a[i] = EOL;
! 2598: else
! 2599: a[i] = ZERO;
! 2600: if (--i < (a[0] == MINUS)) {
! 2601: for (i = lena; i >= 0; i--)
! 2602: a[i + 1] = a[i];
! 2603: a[a[0] == '-'] = ONE;
! 2604: break;
! 2605: }
! 2606: if ((ch = a[i]) == POINT)
! 2607: continue;
! 2608: if (a[i] < NINE && ch >= ZERO) {
! 2609: a[i] = ++ch;
! 2610: break;
! 2611: }
! 2612: }
! 2613: return;
! 2614: } /* end mround */
! 2615:
! 2616: /* End of $Source: /cvsroot-fuse/gump/FreeM/src/operator.c,v $ */
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>