/******************************************************************* Copyright (C) 2018 AMPL Optimization, Inc.; written by David M. Gay. Permission to use, copy, modify, and distribute this software and its documentation for any purpose and without fee is hereby granted, provided that the above copyright notice appear in all copies and that both that the copyright notice and this permission notice and warranty disclaimer appear in supporting documentation. The author and AMPL Optimization, Inc. disclaim all warranties with regard to this software, including all implied warranties of merchantability and fitness. In no event shall the author be liable for any special, indirect or consequential damages or any damages whatsoever resulting from loss of use, data or profits, whether in an action of contract, negligence or other tortious action, arising out of or in connection with the use or performance of this software. *******************************************************************/ #include "nlp.h" #include "errchk.h" #ifdef MULTIPLE_THREADS #define MTNot_Used(x) Not_Used(x) #else #define MTNot_Used(x) /*nothing*/ #define asl cur_ASL #endif #define introuble(who, a, jv) introuble_ASL(asl, who, a, jv) #define introuble2(who, a, b, jv) introuble2_ASL(asl, who, a, b, jv) #define zero_div(L,s) zero_div_ASL(asl, L, s) static real f_OPPLUS(expr *e A_ASL) { expr *L; /* e->dL = e->dR = 1.; */ L = e->L.e; e = e->R.e; return (*L->op)(L K_ASL) + (*e->op)(e K_ASL); } static real f_OPMINUS(expr *e A_ASL) { expr *L; /* e->dL = 1.; */ /* e->dR = -1.; */ L = e->L.e; e = e->R.e; return (*L->op)(L K_ASL) - (*e->op)(e K_ASL); } static real f_OPMULT(expr *e A_ASL) { expr *e1, *e2; e1 = e->L.e; e2 = e->R.e; return (e->dR = (*e1->op)(e1 K_ASL)) * (e->dL = (*e2->op)(e2 K_ASL)); } static real f_OPDIV(expr *e A_ASL) { expr *e1; real L, R, rv; e1 = e->L.e; L = (*e1->op)(e1 K_ASL); e1 = e->R.e; if (!(R = (*e1->op)(e1 K_ASL)) #ifdef WANT_INFNAN && !L #endif ) zero_div(L, "/"); rv = L / R; if (want_deriv) e->dR = -rv * (e->dL = 1. / R); return rv; } static real f_OPREM(expr *e A_ASL) { U rv; expr *e1; real L, R, t; /* e->dL = 1.; */ e1 = e->L.e; L = (*e1->op)(e1 K_ASL); e1 = e->R.e; R = (*e1->op)(e1 K_ASL); rv.d = fmod(L,R); if (errchk(rv)) introuble2("fmod",L,R,1); else if (want_deriv) { t = L / R; e->dR = t > 0. ? -floor(t) : -ceil(t); } return rv.d; } static real f_OPPOW(expr *e A_ASL) { U rv; expr *e1; real L, R; e1 = e->L.e; L = (*e1->op)(e1 K_ASL); e1 = e->R.e; R = (*e1->op)(e1 K_ASL); rv.d = mypow(L,R); if (errchk(rv)) { #ifdef WANT_INFNAN if (!L && R < 0.) { errno_set(0); if (want_deriv) e->dL = e->dR = Infinity; return Infinity; } #endif introuble2("pow",L,R,1); } if (want_deriv) { if (L > 0.) { e->dL = R * (rv.d/L); e->dR = log(L) * rv.d; } else if (L != 0.) { #ifndef WANT_INFNAN bad: #endif introuble2("pow'",L,R,2); } else { if (R > 1.) e->dL = e->dR = 0.; else if (R == 1.) { e->dL = 1.; e->dR = 0.; } else #ifdef WANT_INFNAN { e->dL = Infinity; e->dR = negInfinity; } #else goto bad; #endif } } return rv.d; } static real f_OP1POW(expr *e A_ASL) /* f_OPPOW for R = numeric constant */ { U rv; expr *e1; real L, R; e1 = e->L.e; L = (*e1->op)(e1 K_ASL); R = ((expr_n *)e->R.e)->v; rv.d = mypow(L,R); if (errchk(rv)) { #ifdef WANT_INFNAN if (!L && R < 0.) { errno_set(0); e->dL = negInfinity; return Infinity; } #endif introuble2("pow",L,R,1); } if (want_deriv) { if (L) e->dL = R * (rv.d/L); else if (R > 1.) e->dL = 0.; /* R == 1. does not arise here */ else { #ifdef WANT_INFNAN e->dL = negInfinity; #else introuble2("pow'",L,R,2); #endif } } return rv.d; } static real f_OP2POW(expr *e A_ASL) /* f_OPPOW for R = 2 */ { expr *e1; real L; e1 = e->L.e; L = (*e1->op)(e1 K_ASL); e->dL = L + L; return L*L; } static real f_OPCPOW(expr *e A_ASL) /* f_OPPOW for L = numeric constant */ { U rv; expr *e1; real L, R; e1 = e->R.e; rv.d = mypow(L = e->L.en->v, R = (*e1->op)(e1 K_ASL)); if (errchk(rv)) { #ifdef WANT_INFNAN if (!L && R < 0.) { errno_set(0); e->dR = 0.; return Infinity; } #endif introuble2("pow",L,R,1); } if (want_deriv) { if (L > 0.) { if (e->dL == 1) e->dL = log(L); /* cache value */ e->dR = e->dL * rv.d; } else if (L == 0.) e->dR = 0.; else { #if defined(WANT_INFNAN) && defined(QNaN0) union { unsigned int x[2]; real r; } u; u.x[0] = QNaN0; u.x[1] = QNaN1; e->dR = u.r; #else introuble2("pow'",L,R,2); #endif } } return rv.d; } static real f_OPLESS(expr *e A_ASL) { expr *e1; real L; e1 = e->L.e; L = (*e1->op)(e1 K_ASL); e1 = e->R.e; L -= (*e1->op)(e1 K_ASL); if (L < 0.) return e->dL = e->dR = 0; e->dL = 1.; e->dR = -1.; return L; } static real f_MINLIST(expr *e0 A_ASL) { de *d, *d1; real t, rv; expr *e1; derp *D; expr_va *e = (expr_va *)e0; d = e->L.d; e1 = d->e; rv = (*e1->op)(e1 K_ASL); for(d1 = d++; (e1 = d->e); d++) { t = (*e1->op)(e1 K_ASL); if (rv > t) { rv = t; d1 = d; } } if ((D = e->R.D)) { D->a.rp = d1->dv.rp; D->next = d1->d; } return rv; } static real f_MAXLIST(expr *e0 A_ASL) { de *d, *d1; derp *D; expr *e1; real t, rv; expr_va *e = (expr_va *)e0; d = e->L.d; e1 = d->e; rv = (*e1->op)(e1 K_ASL); for(d1 = d++; (e1 = d->e); d++) { t = (*e1->op)(e1 K_ASL); if (rv < t) { rv = t; d1 = d; } } if ((D = e->R.D)) { D->a.rp = d1->dv.rp; D->next = d1->d; } return rv; } static real f_FLOOR(expr *e A_ASL) { /* e->dL = 0.; */ e = e->L.e; return floor((*e->op)(e K_ASL)); } static real f_CEIL(expr *e A_ASL) { /* e->dL = 0.; */ e = e->L.e; return ceil((*e->op)(e K_ASL)); } static real f_ABS(expr *e A_ASL) { expr *e1; real rv; e1 = e->L.e; rv = (*e1->op)(e1 K_ASL); if (rv < 0.) { e->dL = -1.; return -rv; } e->dL = 1.; return rv; } static real f_OPUMINUS(expr *e A_ASL) { /* e->dL = -1.; */ e = e->L.e; return -(*e->op)(e K_ASL); } static real f_OP_tanh(expr *e A_ASL) { U rv, t1; expr *e1; real t, t2; e1 = e->L.e; t = (*e1->op)(e1 K_ASL); if (t >= 175.) { rv.d = 1.; e->dL = 0.; } else if (t <= -175.) { rv.d = -1.; e->dL = 0.; } else { rv.d = tanh(t); if (errchk(rv)) introuble("tanh",t,1); if (want_deriv) { t1.d = cosh(t); if (errchk(t1)) introuble("tanh'", t, 2); else { t2 = 1. / t1.d; e->dL = t2*t2; } } } return rv.d; } static real f_OP_tan(expr *e A_ASL) { U rv, t1; expr *e1; real t, t2; e1 = e->L.e; rv.d = tan(t = (*e1->op)(e1 K_ASL)); if (errchk(rv)) introuble("tan",t,1); if (want_deriv) { t1.d = cos(t); if (errchk(t1) || !t1.d) introuble("tan'",t,2); else { t2 = 1. / t1.d; e->dL = t2*t2; } } return rv.d; } static real f_OP_sqrt(expr *e A_ASL) { U rv; expr *e1; real t; e1 = e->L.e; t = (*e1->op)(e1 K_ASL); if (t < 0.) { badsqrt: introuble("sqrt",t,1); } rv.d = sqrt(t); if (errchk(rv)) goto badsqrt; if (want_deriv) { if (rv.d <= 0.) introuble("sqrt'",t,2); else e->dL = 0.5 / rv.d; } return rv.d; } static real f_OP_sinh(expr *e A_ASL) { U dL, rv; expr *e1; real t; e1 = e->L.e; rv.d = sinh(t = (*e1->op)(e1 K_ASL)); if (errchk(rv)) introuble("sinh",t,1); if (want_deriv) { dL.d = cosh(t); if (errchk(dL)) introuble("sinh'",t,2); e->dL = dL.d; } return rv.d; } static real f_OP_sin(expr *e A_ASL) { U dL, rv; expr *e1; real t; e1 = e->L.e; rv.d = sin(t = (*e1->op)(e1 K_ASL)); if (errchk(rv)) introuble("sin",t,1); if (want_deriv) { dL.d = cos(t); if (errchk(dL)) introuble("sin'",t,2); e->dL = dL.d; } return rv.d; } static real f_OP_log10(expr *e A_ASL) { U rv; expr *e1; real t; static real Le10; e1 = e->L.e; rv.d = log10(t = (*e1->op)(e1 K_ASL)); if (errchk(rv)) introuble("log10",t,1); if (want_deriv) { if (!Le10) Le10 = 1. / log(10.); e->dL = Le10 / t; } return rv.d; } static real f_OP_log(expr *e A_ASL) { U rv; expr *e1; real t; e1 = e->L.e; rv.d = log(t = (*e1->op)(e1 K_ASL)); if (errchk(rv)) introuble("log",t,1); if (want_deriv) e->dL = 1. / t; return rv.d; } static real f_OP_exp(expr *e A_ASL) { U rv; expr *e1; real t; e1 = e->L.e; rv.d = e->dL = exp(t = (*e1->op)(e1 K_ASL)); if (errchk(rv)) { if (t >= 0.) introuble("exp",t,1); else { errno_set(0); rv.d = 0.; } } return rv.d; } static real f_OP_cosh(expr *e A_ASL) { U dL, rv; expr *e1; real t; e1 = e->L.e; rv.d = cosh(t = (*e1->op)(e1 K_ASL)); if (errchk(rv)) introuble("cosh",t,1); if (want_deriv) { dL.d = sinh(t); if (errchk(dL)) introuble("cosh'",t,2); e->dL = dL.d; } return rv.d; } static real f_OP_cos(expr *e A_ASL) { U dL, rv; expr *e1; real t; e1 = e->L.e; rv.d = cos(t = (*e1->op)(e1 K_ASL)); if (errchk(rv)) introuble("cos",t,1); if (want_deriv) { dL.d = -sin(t); if (errchk(dL)) introuble("cos'",t,2); e->dL = dL.d; } return rv.d; } static real f_OP_atanh(expr *e A_ASL) { U rv; expr *e1; real t; e1 = e->L.e; t = (*e1->op)(e1 K_ASL); if (t <= -1. || t >= 1.) { errno_set(EDOM); bad_atan: rv.d = 0.; introuble("atanh",t,1); } else { rv.d = 0.5*log((1. + t) / (1. - t)); if (errchk(rv)) goto bad_atan; } if (want_deriv) e->dL = 1. / (1. - t*t); return rv.d; } static real f_OP_atan2(expr *e A_ASL) { U rv; expr *e1; real L, R, t, t1; e1 = e->L.e; L = (*e1->op)(e1 K_ASL); e1 = e->R.e; R = (*e1->op)(e1 K_ASL); rv.d = atan2(L,R); if (errchk(rv)) introuble2("atan2",L,R,1); if (want_deriv) { if ((t = L) < 0.) t = -t; if ((t1 = R) < 0.) t1 = -t1; if (t1 >= t) { t = L / R; t1 = 1. / (1. + t*t); e->dL = t1 /= R; e->dR = -t*t1; } else { t = R / L; t1 = -1. / (1. + t*t); e->dR = t1 /= L; e->dL = -t*t1; } } return rv.d; } static real f_OP_atan(expr *e A_ASL) { U rv; expr *e1; real t; e1 = e->L.e; rv.d = atan(t = (*e1->op)(e1 K_ASL)); if (errchk(rv)) introuble("atan",t,1); if (want_deriv) e->dL = 1. / (1. + t*t); return rv.d; } static real f_OP_asinh(expr *e A_ASL) { U rv; expr *e1; int sign; real t, t0, t1; e1 = e->L.e; t = t0 = (*e1->op)(e1 K_ASL); if ((sign = t < 0.)) t = -t; rv.d = log(t + (t1 = sqrt(t*t + 1.))); if (errchk(rv)) introuble("asinh",t0,1); if (sign) rv.d = -rv.d; if (want_deriv) e->dL = 1. / t1; return rv.d; } static real f_OP_asin(expr *e A_ASL) { U rv; expr *e1; real t, t1; e1 = e->L.e; rv.d = asin(t = (*e1->op)(e1 K_ASL)); if (errchk(rv)) introuble("asin",t,1); if (want_deriv) { if ((t1 = 1. - t*t) <= 0.) introuble("asin'",t,2); else e->dL = 1. / sqrt(t1); } return rv.d; } static real f_OP_acosh(expr *e A_ASL) { U rv; expr *e1; real t, t1; e1 = e->L.e; t = (*e1->op)(e1 K_ASL); if (t < 1.) { errno_set(EDOM); bad_acosh: rv.d = t1 = 0.; introuble("acosh",t,1); } else { rv.d = log(t + (t1 = sqrt(t*t - 1.))); if (errchk(rv)) goto bad_acosh; } if (want_deriv) { if (t1 <= 0.) introuble("acosh'",t,2); else e->dL = 1. / t1; } return rv.d; } static real f_OP_acos(expr *e A_ASL) { U rv; expr *e1; real t, t1; e1 = e->L.e; rv.d = acos(t = (*e1->op)(e1 K_ASL)); if (errchk(rv)) introuble("acos",t,1); if (want_deriv) { if ((t1 = 1. - t*t) <= 0.) introuble("acos'",t,2); else e->dL = -1. / sqrt(t1); } return rv.d; } static real f_OPIFnl(expr *e A_ASL) { derp *D; expr_if *eif = (expr_if *)e; e = eif->e; if ((*e->op)(e K_ASL)) { e = eif->T; if ((D = eif->D)) { D->a.rp = eif->Tv.rp; D->next = eif->dT; } } else { e = eif->F; if ((D = eif->D)) { D->a.rp = eif->Fv.rp; D->next = eif->dF; } } return (*e->op)(e K_ASL); } static real f_OPOR(expr *e A_ASL) { expr *e2 = e->R.e; e = e->L.e; return (*e->op)(e K_ASL) || (*e2->op)(e2 K_ASL) ? 1. : 0.; } static real f_OPAND(expr *e A_ASL) { expr *e2 = e->R.e; e = e->L.e; return (*e->op)(e K_ASL) && (*e2->op)(e2 K_ASL) ? 1. : 0.; } static real f_LT(expr *e A_ASL) { expr *e2 = e->R.e; e = e->L.e; return (*e->op)(e K_ASL) < (*e2->op)(e2 K_ASL) ? 1. : 0; } static real f_LE(expr *e A_ASL) { expr *e2 = e->R.e; e = e->L.e; return (*e->op)(e K_ASL) <= (*e2->op)(e2 K_ASL) ? 1. : 0; } static real f_EQ(expr *e A_ASL) { expr *e2 = e->R.e; e = e->L.e; return (*e->op)(e K_ASL) == (*e2->op)(e2 K_ASL) ? 1. : 0; } static real f_GE(expr *e A_ASL) { expr *e2 = e->R.e; e = e->L.e; return (*e->op)(e K_ASL) >= (*e2->op)(e2 K_ASL) ? 1. : 0; } static real f_GT(expr *e A_ASL) { expr *e2 = e->R.e; e = e->L.e; return (*e->op)(e K_ASL) > (*e2->op)(e2 K_ASL) ? 1. : 0; } static real f_NE(expr *e A_ASL) { expr *e2 = e->R.e; e = e->L.e; return (*e->op)(e K_ASL) != (*e2->op)(e2 K_ASL) ? 1. : 0; } static real f_OPNOT(expr *e A_ASL) { e = e->L.e; return (*e->op)(e K_ASL) ? 0. : 1.; } static real f_ANDLIST(expr *e A_ASL) { expr **ep, **epe; ep = e->L.ep; epe = e->R.ep; do { e = *ep++; if ((*e->op)(e K_ASL) == 0.) return 0.; } while(ep < epe); return 1.; } static real f_ORLIST(expr *e A_ASL) { expr **ep, **epe; ep = e->L.ep; epe = e->R.ep; do { e = *ep++; if ((*e->op)(e K_ASL) != 0.) return 1.; } while(ep < epe); return 0.; } static real f_OPIMPELSE(expr *e A_ASL) { expr_if *eif = (expr_if *)e; e = eif->e; e = ((*e->op)(e K_ASL) != 0.) ? eif->T : eif->F; return (*e->op)(e K_ASL); } static real f_OP_IFF(expr *e A_ASL) { expr *e1; int a, b; e1 = e->L.e; a = (*e1->op)(e1 K_ASL) != 0.; e1 = e->R.e; b = (*e1->op)(e1 K_ASL) != 0.; return a == b ? 1. : 0.; } static real f_OPSUMLIST(expr *e A_ASL) { expr **ep, **epe; real x; ep = e->L.ep; epe = e->R.ep; e = *ep++; x = (*e->op)(e K_ASL); do { e = *ep++; x += (*e->op)(e K_ASL); } while(ep < epe); return x; } static real f_OPintDIV(expr *e A_ASL) { expr *e1; real L, R; e1 = e->L.e; L = (*e1->op)(e1 K_ASL); e1 = e->R.e; if (!(R = (*e1->op)(e1 K_ASL))) zero_div(L, " div "); return (L /= R) >= 0 ? floor(L) : ceil(L); } static real f_OPprecision(expr *e A_ASL) { char buf[32]; expr *e1; real L, R; e1 = e->L.e; L = (*e1->op)(e1 K_ASL); e1 = e->R.e; R = (*e1->op)(e1 K_ASL); g_fmtp(buf, L, (int)R); return strtod(buf, (char **)0); } #ifdef No_dtoa static real Round(real x, int prec) { int flip; real scale; if (!x) return x; flip = 0; if (x < 0.) { x = -x; flip = 1; } if (!prec) x = floor(x + 0.5); else if (prec > 0) { scale = mypow(10., (real)prec); x = floor(x*scale + 0.5) / scale; } else { scale = mypow(10., -(real)prec); x = scale*floor(x/scale + 0.5); } return flip ? -x : x; } #else static real Round(real x, int prec) { char *b, *s, sbuf[400], *se; char buf[96]; int decpt, L, sign; if (!x) return x; s = dtoa_r(x, 3, prec, &decpt, &sign, &se, sbuf, sizeof(sbuf)); if (decpt == 9999) { zreturn: return x; } L = se - s; if (L <= 0) { x = 0.; goto zreturn; } if (L > 80) se = s + 80; b = buf; if (sign) *b++ = '-'; *b++ = '.'; while(s < se) *b++ = *s++; *b = 0; if (decpt) snprintf(b, buf + sizeof(buf) - b, "e%d", decpt); return strtod(buf, (char **)0); } #endif static real f_OPround(expr *e A_ASL) { expr *e1; real L, R; e1 = e->L.e; L = (*e1->op)(e1 K_ASL); e1 = e->R.e; R = (*e1->op)(e1 K_ASL); return Round(L, (int)R); } static real f_OPtrunc(expr *e A_ASL) { expr *e1; int prec; real L, R; e1 = e->L.e; L = (*e1->op)(e1 K_ASL); e1 = e->R.e; if (!(R = (*e1->op)(e1 K_ASL))) return L >= 0. ? floor(L) : ceil(L); R = Round(L, prec = (int)R); if (R != L) { R = 0.5*mypow(10., (real)-prec); R = Round(L > 0 ? L - R : L + R, prec); } return R; } static real f_OPPLTERM(expr *e A_ASL) { int n, z; plterm *p; real *bs; real r, t; MTNot_Used(asl); p = e->L.p; n = p->n; z = p->z; bs = p->bs + z; z >>= 1; r = ((expr_v *)e->R.e)->v; if (r >= 0) { n -= z; if (n <= 1 || r <= bs[1]) return r*(e->dL = *bs); for(t = bs[0]*bs[1]; --n > 1 && r > bs[3]; bs += 2) t += (bs[3]-bs[1])*bs[2]; return t + (r-bs[1])*(e->dL = bs[2]); } if (z <= 0) return r*(e->dL = bs[0]); for(t = bs[0]*bs[-1]; --z > 0 && r < bs[-3]; bs -= 2) { t += bs[-2]*(bs[-3] - bs[-1]); } return t + (r - bs[-1])*(e->dL = bs[-2]); } typedef char * sfunc ANSI((expr* A_ASL)); static char * f_OPHOL(expr *e A_ASL) { MTNot_Used(asl); return ((expr_h *)e)->sym; } static real f_OPVARVAL(expr *e A_ASL) { MTNot_Used(asl); return ((expr_v *)e)->v; } static char * f_OPIFSYM(expr *e A_ASL) { expr_if *eif = (expr_if *)e; e = eif->e; e = (*e->op)(e K_ASL) ? eif->T : eif->F; return (*(sfunc*)e->op)(e K_ASL); } static real f_OPFUNCALL(expr *e A_ASL) { TMInfo T, *T1, *T1prev; arglist *al; argpair *ap, *ape; const char *s; expr_f *f; func_info *fi; real rv; f = (expr_f *)e; fi = f->fi; for(ap = f->ap, ape = f->ape; ap < ape; ap++) { e = ap->e; *ap->u.v = (*e->op)(e K_ASL); } for(ap = f->sap, ape = f->sape; ap < ape; ap++) { e = ap->e; *ap->u.s = (*(sfunc*)e->op)(e K_ASL); } T.u.prev = 0; al = f->al; al->TMI = &T; al->Errmsg = 0; rv = (*fi->funcp)(al); errno_set(0); if ((s = al->Errmsg)) fintrouble_ASL(asl, fi, s, &T); for(T1 = T.u.prev; T1; T1 = T1prev) { T1prev = T1->u.prev; free(T1); } return rv; } static real f_OPCOUNT(expr *e A_ASL) { expr **ep, **epe; real x; ep = e->L.ep; epe = e->R.ep; e = *ep++; if ((x = (*e->op)(e K_ASL))) x = 1.; do { e = *ep++; if ((*e->op)(e K_ASL)) x++; } while(ep < epe); return x; } typedef struct jb_st { jmp_buf jb; } jb_st; static int rcompj(const void *a, const void *b, void *v) { jb_st *J; real t = *(real *)a - *(real *)b; if (!t) { J = (jb_st*)v; longjmp(J->jb, 1); } return t < 0 ? -1 : 1; } static real f_OPALLDIFF(expr *e A_ASL) { expr **ep, **epe; int n; jb_st J; real *r, *r1, rbuf[128], t; ep = e->L.ep; epe = e->R.ep; n = epe - ep; r = rbuf; if (n > sizeof(rbuf)/sizeof(real)) r = (real*)Malloc(n*sizeof(real)); r1 = r; while(ep < epe) { e = *ep++; *r1++ = (*e->op)(e K_ASL); } t = 1.; if (setjmp(J.jb)) { t = 0.; goto done; } qsortv(r, n, sizeof(real), rcompj, &J); done: if (r != rbuf) free(r); return t; } static real f_OPSOMESAME(expr *e A_ASL) { real t = f_OPALLDIFF(e K_ASL); return t == 0. ? 1. : 0.; } static real f_OPNUMBEROF(expr *e A_ASL) { expr **ep, **epe; real n, t; ep = e->L.ep; epe = e->R.ep; n = 0.; e = *ep++; t = (*e->op)(e K_ASL); while(ep < epe) { e = *ep++; if (t == (*e->op)(e K_ASL)) n++; } return n; } #define f_OPATLEAST f_LE #define f_OPATMOST f_GE #define f_OPEXACTLY f_EQ #define f_OPNOTATLEAST f_GT #define f_OPNOTATMOST f_LT #define f_OPNOTEXACTLY f_NE #define f_OPNUMBEROFs f_OPNUMBEROF /*TEMPORARY*/ efunc * r_ops[] = { #include "r_op.hd" }; /* It may be necessary to recompile this file after the change of 20150814 to errchk.h. */