/* FRACTALS.C, FRACTALP.C and CALCFRAC.C actually calculate the fractal images (well, SOMEBODY had to do it!). The modules are set up so that all logic that is independent of any fractal-specific code is in CALCFRAC.C, the code that IS fractal-specific is in FRACTALS.C, and the structure that ties (we hope!) everything together is in FRACTALP.C. Original author Tim Wegner, but just about ALL the authors have contributed SOME code to this routine at one time or another, or contributed to one of the many massive restructurings. The Fractal-specific routines are divided into three categories: 1. Routines that are called once-per-orbit to calculate the orbit value. These have names like "XxxxFractal", and their function pointers are stored in fractalspecific[fractype].orbitcalc. EVERY new fractal type needs one of these. Return 0 to continue iterations, 1 if we're done. Results for integer fractals are left in 'lnew.x' and 'lnew.y', for floating point fractals in 'new.x' and 'new.y'. 2. Routines that are called once per pixel to set various variables prior to the orbit calculation. These have names like xxx_per_pixel and are fairly generic - chances are one is right for your new type. They are stored in fractalspecific[fractype].per_pixel. 3. Routines that are called once per screen to set various variables. These have names like XxxxSetup, and are stored in fractalspecific[fractype].per_image. 4. The main fractal routine. Usually this will be StandardFractal(), but if you have written a stand-alone fractal routine independent of the StandardFractal mechanisms, your routine name goes here, stored in fractalspecific[fractype].calctype.per_image. Adding a new fractal type should be simply a matter of adding an item to the 'fractalspecific' structure, writing (or re-using one of the existing) an appropriate setup, per_image, per_pixel, and orbit routines. -------------------------------------------------------------------- */ #include #include #ifdef __TURBOC__ #include #elif !defined(__386BSD__) #include #endif /* see Fractint.c for a description of the "include" hierarchy */ #include "port.h" #include "prototyp.h" #include "helpdefs.h" #include "fractype.h" #include "externs.h" #define NEWTONDEGREELIMIT 100 _LCMPLX lcoefficient,lold,lnew,lparm, linit,ltmp,ltmp2,lparm2; long ltempsqrx,ltempsqry; int maxcolor; int root, degree,basin; double floatmin,floatmax; double roverd, d1overd, threshold; _CMPLX tmp2; _CMPLX coefficient; _CMPLX staticroots[16]; /* roots array for degree 16 or less */ _CMPLX *roots = staticroots; struct MPC *MPCroots; long FgHalf; _CMPLX pwr; int bitshiftless1; /* bit shift less 1 */ #ifndef sqr #define sqr(x) ((x)*(x)) #endif #ifndef lsqr #define lsqr(x) (multiply((x),(x),bitshift)) #endif #define modulus(z) (sqr((z).x)+sqr((z).y)) #define conjugate(pz) ((pz)->y = 0.0 - (pz)->y) #define distance(z1,z2) (sqr((z1).x-(z2).x)+sqr((z1).y-(z2).y)) #define pMPsqr(z) (*pMPmul((z),(z))) #define MPdistance(z1,z2) (*pMPadd(pMPsqr(*pMPsub((z1).x,(z2).x)),pMPsqr(*pMPsub((z1).y,(z2).y)))) double twopi = PI*2.0; int c_exp; /* These are local but I don't want to pass them as parameters */ _CMPLX parm,parm2; _CMPLX *floatparm; _LCMPLX *longparm; /* used here and in jb.c */ /* -------------------------------------------------------------------- */ /* These variables are external for speed's sake only */ /* -------------------------------------------------------------------- */ double sinx,cosx; double siny,cosy; double tmpexp; double tempsqrx,tempsqry; double foldxinitx,foldyinity,foldxinity,foldyinitx; long oldxinitx,oldyinity,oldxinity,oldyinitx; long longtmp; /* These are for quaternions */ double qc,qci,qcj,qck; /* temporary variables for trig use */ long lcosx, lsinx; long lcosy, lsiny; /* ** details of finite attractors (required for Magnet Fractals) ** (can also be used in "coloring in" the lakes of Julia types) */ /* ** pre-calculated values for fractal types Magnet2M & Magnet2J */ _CMPLX T_Cm1; /* 3 * (floatparm - 1) */ _CMPLX T_Cm2; /* 3 * (floatparm - 2) */ _CMPLX T_Cm1Cm2; /* (floatparm - 1) * (floatparm - 2) */ void FloatPreCalcMagnet2(void) /* precalculation for Magnet2 (M & J) for speed */ { T_Cm1.x = floatparm->x - 1.0; T_Cm1.y = floatparm->y; T_Cm2.x = floatparm->x - 2.0; T_Cm2.y = floatparm->y; T_Cm1Cm2.x = (T_Cm1.x * T_Cm2.x) - (T_Cm1.y * T_Cm2.y); T_Cm1Cm2.y = (T_Cm1.x * T_Cm2.y) + (T_Cm1.y * T_Cm2.x); T_Cm1.x += T_Cm1.x + T_Cm1.x; T_Cm1.y += T_Cm1.y + T_Cm1.y; T_Cm2.x += T_Cm2.x + T_Cm2.x; T_Cm2.y += T_Cm2.y + T_Cm2.y; } /* -------------------------------------------------------------------- */ /* Bailout Routines Macros */ /* -------------------------------------------------------------------- */ int (near *floatbailout)(void); int (near *longbailout)(void); int (near *bignumbailout)(void); int (near *bigfltbailout)(void); #if 0 int near fpMODbailout(void) { if ( ( magnitude = ( tempsqrx=sqr(new.x) ) + ( tempsqry=sqr(new.y) ) ) >= rqlim ) return(1); old = new; return(0); } #endif int near fpMODbailout(void) { tempsqrx=sqr(new.x); tempsqry=sqr(new.y); magnitude = tempsqrx + tempsqry; if(magnitude >= rqlim) return(1); old = new; return(0); } int near fpREALbailout(void) { tempsqrx=sqr(new.x); tempsqry=sqr(new.y); magnitude = tempsqrx + tempsqry; if(tempsqrx >= rqlim) return(1); old = new; return(0); } int near fpIMAGbailout(void) { tempsqrx=sqr(new.x); tempsqry=sqr(new.y); magnitude = tempsqrx + tempsqry; if(tempsqry >= rqlim) return(1); old = new; return(0); } int near fpORbailout(void) { tempsqrx=sqr(new.x); tempsqry=sqr(new.y); magnitude = tempsqrx + tempsqry; if(tempsqrx >= rqlim || tempsqry >= rqlim) return(1); old = new; return(0); } int near fpANDbailout(void) { tempsqrx=sqr(new.x); tempsqry=sqr(new.y); magnitude = tempsqrx + tempsqry; if(tempsqrx >= rqlim && tempsqry >= rqlim) return(1); old = new; return(0); } int near fpMANHbailout(void) { double manhmag; tempsqrx=sqr(new.x); tempsqry=sqr(new.y); magnitude = tempsqrx + tempsqry; manhmag = fabs(new.x) + fabs(new.y); if((manhmag * manhmag) >= rqlim) return(1); old = new; return(0); } int near fpMANRbailout(void) { double manrmag; tempsqrx=sqr(new.x); tempsqry=sqr(new.y); magnitude = tempsqrx + tempsqry; manrmag = new.x + new.y; /* don't need abs() since we square it next */ if((manrmag * manrmag) >= rqlim) return(1); old = new; return(0); } #define FLOATTRIGBAILOUT() \ if (fabs(old.y) >= rqlim2) return(1); #define LONGTRIGBAILOUT() \ if(labs(lold.y) >= llimit2) { return(1);} #define LONGXYTRIGBAILOUT() \ if(labs(lold.x) >= llimit2 || labs(lold.y) >= llimit2)\ { return(1);} #define FLOATXYTRIGBAILOUT() \ if (fabs(old.x) >= rqlim2 || fabs(old.y) >= rqlim2) return(1); #define FLOATHTRIGBAILOUT() \ if (fabs(old.x) >= rqlim2) return(1); #define LONGHTRIGBAILOUT() \ if(labs(lold.x) >= llimit2) { return(1);} #define TRIG16CHECK(X) \ if(labs((X)) > l16triglim) { return(1);} #define OLD_FLOATEXPBAILOUT() \ if (fabs(old.y) >= 1.0e8) return(1);\ if (fabs(old.x) >= 6.4e2) return(1); #define FLOATEXPBAILOUT() \ if (fabs(old.y) >= 1.0e3) return(1);\ if (fabs(old.x) >= 8) return(1); #define LONGEXPBAILOUT() \ if (labs(lold.y) >= (1000L<= (8L< l16triglim)\ {\ double tmp;\ tmp = (X);\ tmp /= fudge;\ tmp = fmod(tmp,twopi);\ tmp *= fudge;\ (X) = (long)tmp;\ }\ static int near Halleybailout(void) { if ( fabs(modulus(new)-modulus(old)) < parm2.x) return(1); old = new; return(0); } #ifndef XFRACT #define MPCmod(m) (*pMPadd(*pMPmul((m).x, (m).x), *pMPmul((m).y, (m).y))) struct MPC mpcold, mpcnew, mpctmp, mpctmp1; struct MP mptmpparm2x; #if (_MSC_VER >= 700) #pragma code_seg ("mpmath1_text") /* place following in an overlay */ #endif static int near MPCHalleybailout(void) { static struct MP mptmpbailout; mptmpbailout = *MPabs(*pMPsub(MPCmod(mpcnew), MPCmod(mpcold))); if (pMPcmp(mptmpbailout, mptmpparm2x) < 0) return(1); mpcold = mpcnew; return(0); } #if (_MSC_VER >= 700) #pragma code_seg () /* back to normal segment */ #endif #endif #ifdef XFRACT int asmlMODbailout(void) { return 0;} int asmlREALbailout(void) { return 0;} int asmlIMAGbailout(void) { return 0;} int asmlORbailout(void) { return 0;} int asmlANDbailout(void) { return 0;} int asmlMANHbailout(void) { return 0;} int asmlMANRbailout(void) { return 0;} int asm386lMODbailout(void) { return 0;} int asm386lREALbailout(void) { return 0;} int asm386lIMAGbailout(void) { return 0;} int asm386lORbailout(void) { return 0;} int asm386lANDbailout(void) { return 0;} int asm386lMANHbailout(void) { return 0;} int asm386lMANRbailout(void) { return 0;} int asmfpMODbailout(void) { return 0;} int asmfpREALbailout(void) { return 0;} int asmfpIMAGbailout(void) { return 0;} int asmfpORbailout(void) { return 0;} int asmfpANDbailout(void) { return 0;} int asmfpMANHbailout(void) { return 0;} int asmfpMANRbailout(void) { return 0;} #endif /* -------------------------------------------------------------------- */ /* Fractal (once per iteration) routines */ /* -------------------------------------------------------------------- */ static double xt, yt, t2; /* Raise complex number (base) to the (exp) power, storing the result ** in complex (result). */ void cpower(_CMPLX *base, int exp, _CMPLX *result) { if (exp<0) { cpower(base,-exp,result); CMPLXrecip(*result,*result); return; } xt = base->x; yt = base->y; if (exp & 1) { result->x = xt; result->y = yt; } else { result->x = 1.0; result->y = 0.0; } exp >>= 1; while (exp) { t2 = xt * xt - yt * yt; yt = 2 * xt * yt; xt = t2; if (exp & 1) { t2 = xt * result->x - yt * result->y; result->y = result->y * xt + yt * result->x; result->x = t2; } exp >>= 1; } } #ifndef XFRACT /* long version */ static long lxt, lyt, lt2; int lcpower(_LCMPLX *base, int exp, _LCMPLX *result, int bitshift) { static long maxarg; maxarg = 64L<x; lyt = base->y; if (exp & 1) { result->x = lxt; result->y = lyt; } else { result->x = 1L<y = 0L; } exp >>= 1; while (exp) { /* if(labs(lxt) >= maxarg || labs(lyt) >= maxarg) return(-1); */ lt2 = multiply(lxt, lxt, bitshift) - multiply(lyt,lyt,bitshift); lyt = multiply(lxt,lyt,bitshiftless1); if(overflow) return(overflow); lxt = lt2; if (exp & 1) { lt2 = multiply(lxt,result->x, bitshift) - multiply(lyt,result->y,bitshift); result->y = multiply(result->y,lxt,bitshift) + multiply(lyt,result->x,bitshift); result->x = lt2; } exp >>= 1; } if(result->x == 0 && result->y == 0) overflow = 1; return(overflow); } #if 0 int z_to_the_z(_CMPLX *z, _CMPLX *out) { static _CMPLX tmp1,tmp2; /* raises complex z to the z power */ int errno_xxx; errno_xxx = 0; if(fabs(z->x) < DBL_EPSILON) return(-1); /* log(x + iy) = 1/2(log(x*x + y*y) + i(arc_tan(y/x)) */ tmp1.x = .5*log(sqr(z->x)+sqr(z->y)); /* the fabs in next line added to prevent discontinuity in image */ tmp1.y = atan(fabs(z->y/z->x)); /* log(z)*z */ tmp2.x = tmp1.x * z->x - tmp1.y * z->y; tmp2.y = tmp1.x * z->y + tmp1.y * z->x; /* z*z = e**(log(z)*z) */ /* e**(x + iy) = e**x * (cos(y) + isin(y)) */ tmpexp = exp(tmp2.x); FPUsincos(&tmp2.y,&siny,&cosy); out->x = tmpexp*cosy; out->y = tmpexp*siny; return(errno_xxx); } #endif #endif #ifdef XFRACT /* fractint uses the NewtonFractal2 code in newton.asm */ int complex_div(_CMPLX arg1,_CMPLX arg2,_CMPLX *pz); int complex_mult(_CMPLX arg1,_CMPLX arg2,_CMPLX *pz); /* Distance of complex z from unit circle */ #define DIST1(z) (((z).x-1.0)*((z).x-1.0)+((z).y)*((z).y)) #define LDIST1(z) (lsqr((((z).x)-fudge)) + lsqr(((z).y))) int NewtonFractal2(void) { static char start=1; if(start) { start = 0; } cpower(&old, degree-1, &tmp); complex_mult(tmp, old, &new); if (DIST1(new) < threshold) { if(fractype==NEWTBASIN || fractype==MPNEWTBASIN) { long tmpcolor; int i; tmpcolor = -1; /* this code determines which degree-th root of root the Newton formula converges to. The roots of a 1 are distributed on a circle of radius 1 about the origin. */ for(i=0;ix = arg1.x*arg2.x - arg1.y*arg2.y; pz->y = arg1.x*arg2.y+arg1.y*arg2.x; return(0); } int complex_div(_CMPLX numerator,_CMPLX denominator,_CMPLX *pout) { double mod; if((mod = modulus(denominator)) < FLT_MIN) return(1); conjugate(&denominator); complex_mult(numerator,denominator,pout); pout->x = pout->x/mod; pout->y = pout->y/mod; return(0); } #endif /* newton code only used by xfractint */ #ifndef XFRACT struct MP mproverd, mpd1overd, mpthreshold; struct MP mpt2; struct MP mpone; #endif #if (_MSC_VER >= 700) #pragma code_seg ("mpmath1_text") /* place following in an overlay */ #endif int MPCNewtonFractal(void) { #ifndef XFRACT MPOverflow = 0; mpctmp = MPCpow(mpcold,degree-1); mpcnew.x = *pMPsub(*pMPmul(mpctmp.x,mpcold.x),*pMPmul(mpctmp.y,mpcold.y)); mpcnew.y = *pMPadd(*pMPmul(mpctmp.x,mpcold.y),*pMPmul(mpctmp.y,mpcold.x)); mpctmp1.x = *pMPsub(mpcnew.x, MPCone.x); mpctmp1.y = *pMPsub(mpcnew.y, MPCone.y); if(pMPcmp(MPCmod(mpctmp1),mpthreshold)< 0) { if(fractype==MPNEWTBASIN) { long tmpcolor; int i; tmpcolor = -1; for(i=0;i= 700) #pragma code_seg () /* back to normal segment */ #endif int Barnsley1Fractal(void) { #ifndef XFRACT /* Barnsley's Mandelbrot type M1 from "Fractals Everywhere" by Michael Barnsley, p. 322 */ /* calculate intermediate products */ oldxinitx = multiply(lold.x, longparm->x, bitshift); oldyinity = multiply(lold.y, longparm->y, bitshift); oldxinity = multiply(lold.x, longparm->y, bitshift); oldyinitx = multiply(lold.y, longparm->x, bitshift); /* orbit calculation */ if(lold.x >= 0) { lnew.x = (oldxinitx - longparm->x - oldyinity); lnew.y = (oldyinitx - longparm->y + oldxinity); } else { lnew.x = (oldxinitx + longparm->x - oldyinity); lnew.y = (oldyinitx + longparm->y + oldxinity); } return(longbailout()); #else return(0); #endif } int Barnsley1FPFractal(void) { /* Barnsley's Mandelbrot type M1 from "Fractals Everywhere" by Michael Barnsley, p. 322 */ /* note that fast >= 287 equiv in fracsuba.asm must be kept in step */ /* calculate intermediate products */ foldxinitx = old.x * floatparm->x; foldyinity = old.y * floatparm->y; foldxinity = old.x * floatparm->y; foldyinitx = old.y * floatparm->x; /* orbit calculation */ if(old.x >= 0) { new.x = (foldxinitx - floatparm->x - foldyinity); new.y = (foldyinitx - floatparm->y + foldxinity); } else { new.x = (foldxinitx + floatparm->x - foldyinity); new.y = (foldyinitx + floatparm->y + foldxinity); } return(floatbailout()); } int Barnsley2Fractal(void) { #ifndef XFRACT /* An unnamed Mandelbrot/Julia function from "Fractals Everywhere" by Michael Barnsley, p. 331, example 4.2 */ /* note that fast >= 287 equiv in fracsuba.asm must be kept in step */ /* calculate intermediate products */ oldxinitx = multiply(lold.x, longparm->x, bitshift); oldyinity = multiply(lold.y, longparm->y, bitshift); oldxinity = multiply(lold.x, longparm->y, bitshift); oldyinitx = multiply(lold.y, longparm->x, bitshift); /* orbit calculation */ if(oldxinity + oldyinitx >= 0) { lnew.x = oldxinitx - longparm->x - oldyinity; lnew.y = oldyinitx - longparm->y + oldxinity; } else { lnew.x = oldxinitx + longparm->x - oldyinity; lnew.y = oldyinitx + longparm->y + oldxinity; } return(longbailout()); #else return(0); #endif } int Barnsley2FPFractal(void) { /* An unnamed Mandelbrot/Julia function from "Fractals Everywhere" by Michael Barnsley, p. 331, example 4.2 */ /* calculate intermediate products */ foldxinitx = old.x * floatparm->x; foldyinity = old.y * floatparm->y; foldxinity = old.x * floatparm->y; foldyinitx = old.y * floatparm->x; /* orbit calculation */ if(foldxinity + foldyinitx >= 0) { new.x = foldxinitx - floatparm->x - foldyinity; new.y = foldyinitx - floatparm->y + foldxinity; } else { new.x = foldxinitx + floatparm->x - foldyinity; new.y = foldyinitx + floatparm->y + foldxinity; } return(floatbailout()); } int JuliaFractal(void) { #ifndef XFRACT /* used for C prototype of fast integer math routines for classic Mandelbrot and Julia */ lnew.x = ltempsqrx - ltempsqry + longparm->x; lnew.y = multiply(lold.x, lold.y, bitshiftless1) + longparm->y; return(longbailout()); #elif !defined(__386BSD__) fprintf(stderr,"JuliaFractal called\n"); exit(-1); #endif } int JuliafpFractal(void) { /* floating point version of classical Mandelbrot/Julia */ /* note that fast >= 287 equiv in fracsuba.asm must be kept in step */ new.x = tempsqrx - tempsqry + floatparm->x; new.y = 2.0 * old.x * old.y + floatparm->y; return(floatbailout()); } int LambdaFPFractal(void) { /* variation of classical Mandelbrot/Julia */ /* note that fast >= 287 equiv in fracsuba.asm must be kept in step */ tempsqrx = old.x - tempsqrx + tempsqry; tempsqry = -(old.y * old.x); tempsqry += tempsqry + old.y; new.x = floatparm->x * tempsqrx - floatparm->y * tempsqry; new.y = floatparm->x * tempsqry + floatparm->y * tempsqrx; return(floatbailout()); } int LambdaFractal(void) { #ifndef XFRACT /* variation of classical Mandelbrot/Julia */ /* in complex math) temp = Z * (1-Z) */ ltempsqrx = lold.x - ltempsqrx + ltempsqry; ltempsqry = lold.y - multiply(lold.y, lold.x, bitshiftless1); /* (in complex math) Z = Lambda * Z */ lnew.x = multiply(longparm->x, ltempsqrx, bitshift) - multiply(longparm->y, ltempsqry, bitshift); lnew.y = multiply(longparm->x, ltempsqry, bitshift) + multiply(longparm->y, ltempsqrx, bitshift); return(longbailout()); #else return(0); #endif } int SierpinskiFractal(void) { #ifndef XFRACT /* following code translated from basic - see "Fractals Everywhere" by Michael Barnsley, p. 251, Program 7.1.1 */ lnew.x = (lold.x << 1); /* new.x = 2 * old.x */ lnew.y = (lold.y << 1); /* new.y = 2 * old.y */ if(lold.y > ltmp.y) /* if old.y > .5 */ lnew.y = lnew.y - ltmp.x; /* new.y = 2 * old.y - 1 */ else if(lold.x > ltmp.y) /* if old.x > .5 */ lnew.x = lnew.x - ltmp.x; /* new.x = 2 * old.x - 1 */ /* end barnsley code */ return(longbailout()); #else return(0); #endif } int SierpinskiFPFractal(void) { /* following code translated from basic - see "Fractals Everywhere" by Michael Barnsley, p. 251, Program 7.1.1 */ new.x = old.x + old.x; new.y = old.y + old.y; if(old.y > .5) new.y = new.y - 1; else if (old.x > .5) new.x = new.x - 1; /* end barnsley code */ return(floatbailout()); } int LambdaexponentFractal(void) { /* found this in "Science of Fractal Images" */ if (save_release > 2002) { /* need braces since these are macros */ FLOATEXPBAILOUT(); } else { OLD_FLOATEXPBAILOUT(); } FPUsincos (&old.y,&siny,&cosy); if (old.x >= rqlim && cosy >= 0.0) return(1); tmpexp = exp(old.x); tmp.x = tmpexp*cosy; tmp.y = tmpexp*siny; /*multiply by lamda */ new.x = floatparm->x*tmp.x - floatparm->y*tmp.y; new.y = floatparm->y*tmp.x + floatparm->x*tmp.y; old = new; return(0); } int LongLambdaexponentFractal(void) { #ifndef XFRACT /* found this in "Science of Fractal Images" */ LONGEXPBAILOUT(); SinCos086 (lold.y, &lsiny, &lcosy); if (lold.x >= llimit && lcosy >= 0L) return(1); longtmp = Exp086(lold.x); ltmp.x = multiply(longtmp, lcosy, bitshift); ltmp.y = multiply(longtmp, lsiny, bitshift); lnew.x = multiply(longparm->x, ltmp.x, bitshift) - multiply(longparm->y, ltmp.y, bitshift); lnew.y = multiply(longparm->x, ltmp.y, bitshift) + multiply(longparm->y, ltmp.x, bitshift); lold = lnew; return(0); #else return(0); #endif } int FloatTrigPlusExponentFractal(void) { /* another Scientific American biomorph type */ /* z(n+1) = e**z(n) + trig(z(n)) + C */ if (fabs(old.x) >= 6.4e2) return(1); /* DOMAIN errors */ tmpexp = exp(old.x); FPUsincos (&old.y,&siny,&cosy); CMPLXtrig0(old,new); /*new = trig(old) + e**old + C */ new.x += tmpexp*cosy + floatparm->x; new.y += tmpexp*siny + floatparm->y; return(floatbailout()); } int LongTrigPlusExponentFractal(void) { #ifndef XFRACT /* calculate exp(z) */ /* domain check for fast transcendental functions */ TRIG16CHECK(lold.x); TRIG16CHECK(lold.y); longtmp = Exp086(lold.x); SinCos086 (lold.y, &lsiny, &lcosy); LCMPLXtrig0(lold,lnew); lnew.x += multiply(longtmp, lcosy, bitshift) + longparm->x; lnew.y += multiply(longtmp, lsiny, bitshift) + longparm->y; return(longbailout()); #else return(0); #endif } int MarksLambdaFractal(void) { /* Mark Peterson's variation of "lambda" function */ /* Z1 = (C^(exp-1) * Z**2) + C */ #ifndef XFRACT ltmp.x = ltempsqrx - ltempsqry; ltmp.y = multiply(lold.x ,lold.y ,bitshiftless1); lnew.x = multiply(lcoefficient.x, ltmp.x, bitshift) - multiply(lcoefficient.y, ltmp.y, bitshift) + longparm->x; lnew.y = multiply(lcoefficient.x, ltmp.y, bitshift) + multiply(lcoefficient.y, ltmp.x, bitshift) + longparm->y; return(longbailout()); #else return(0); #endif } int MarksLambdafpFractal(void) { /* Mark Peterson's variation of "lambda" function */ /* Z1 = (C^(exp-1) * Z**2) + C */ tmp.x = tempsqrx - tempsqry; tmp.y = old.x * old.y *2; new.x = coefficient.x * tmp.x - coefficient.y * tmp.y + floatparm->x; new.y = coefficient.x * tmp.y + coefficient.y * tmp.x + floatparm->y; return(floatbailout()); } long XXOne, FgOne, FgTwo; int UnityFractal(void) { #ifndef XFRACT /* brought to you by Mark Peterson - you won't find this in any fractal books unless they saw it here first - Mark invented it! */ XXOne = multiply(lold.x, lold.x, bitshift) + multiply(lold.y, lold.y, bitshift); if((XXOne > FgTwo) || (labs(XXOne - FgOne) < delmin)) return(1); lold.y = multiply(FgTwo - XXOne, lold.x, bitshift); lold.x = multiply(FgTwo - XXOne, lold.y, bitshift); lnew=lold; /* TW added this line */ return(0); #else return(0); #endif } int UnityfpFractal(void) { double XXOne; /* brought to you by Mark Peterson - you won't find this in any fractal books unless they saw it here first - Mark invented it! */ XXOne = sqr(old.x) + sqr(old.y); if((XXOne > 2.0) || (fabs(XXOne - 1.0) < ddelmin)) return(1); old.y = (2.0 - XXOne)* old.x; old.x = (2.0 - XXOne)* old.y; new=old; /* TW added this line */ return(0); } int Mandel4Fractal(void) { /* By writing this code, Bert has left behind the excuse "don't know what a fractal is, just know how to make'em go fast". Bert is hereby declared a bonafide fractal expert! Supposedly this routine calculates the Mandelbrot/Julia set based on the polynomial z**4 + lambda, but I wouldn't know -- can't follow all that integer math speedup stuff - Tim */ /* first, compute (x + iy)**2 */ #ifndef XFRACT lnew.x = ltempsqrx - ltempsqry; lnew.y = multiply(lold.x, lold.y, bitshiftless1); if (longbailout()) return(1); /* then, compute ((x + iy)**2)**2 + lambda */ lnew.x = ltempsqrx - ltempsqry + longparm->x; lnew.y = multiply(lold.x, lold.y, bitshiftless1) + longparm->y; return(longbailout()); #else return(0); #endif } int Mandel4fpFractal(void) { /* first, compute (x + iy)**2 */ new.x = tempsqrx - tempsqry; new.y = old.x*old.y*2; if (floatbailout()) return(1); /* then, compute ((x + iy)**2)**2 + lambda */ new.x = tempsqrx - tempsqry + floatparm->x; new.y = old.x*old.y*2 + floatparm->y; return(floatbailout()); } int floatZtozPluszpwrFractal(void) { cpower(&old,(int)param[2],&new); old = ComplexPower(old,old); new.x = new.x + old.x +floatparm->x; new.y = new.y + old.y +floatparm->y; return(floatbailout()); } int longZpowerFractal(void) { #ifndef XFRACT if(lcpower(&lold,c_exp,&lnew,bitshift)) lnew.x = lnew.y = 8L<x; lnew.y += longparm->y; return(longbailout()); #else return(0); #endif } int longCmplxZpowerFractal(void) { #ifndef XFRACT _CMPLX x, y; x.x = (double)lold.x / fudge; x.y = (double)lold.y / fudge; y.x = (double)lparm2.x / fudge; y.y = (double)lparm2.y / fudge; x = ComplexPower(x, y); if(fabs(x.x) < fgLimit && fabs(x.y) < fgLimit) { lnew.x = (long)(x.x * fudge); lnew.y = (long)(x.y * fudge); } else overflow = 1; lnew.x += longparm->x; lnew.y += longparm->y; return(longbailout()); #else return(0); #endif } int floatZpowerFractal(void) { cpower(&old,c_exp,&new); new.x += floatparm->x; new.y += floatparm->y; return(floatbailout()); } int floatCmplxZpowerFractal(void) { new = ComplexPower(old, parm2); new.x += floatparm->x; new.y += floatparm->y; return(floatbailout()); } int Barnsley3Fractal(void) { /* An unnamed Mandelbrot/Julia function from "Fractals Everywhere" by Michael Barnsley, p. 292, example 4.1 */ /* calculate intermediate products */ #ifndef XFRACT oldxinitx = multiply(lold.x, lold.x, bitshift); oldyinity = multiply(lold.y, lold.y, bitshift); oldxinity = multiply(lold.x, lold.y, bitshift); /* orbit calculation */ if(lold.x > 0) { lnew.x = oldxinitx - oldyinity - fudge; lnew.y = oldxinity << 1; } else { lnew.x = oldxinitx - oldyinity - fudge + multiply(longparm->x,lold.x,bitshift); lnew.y = oldxinity <<1; /* This term added by Tim Wegner to make dependent on the imaginary part of the parameter. (Otherwise Mandelbrot is uninteresting. */ lnew.y += multiply(longparm->y,lold.x,bitshift); } return(longbailout()); #else return(0); #endif } int Barnsley3FPFractal(void) { /* An unnamed Mandelbrot/Julia function from "Fractals Everywhere" by Michael Barnsley, p. 292, example 4.1 */ /* calculate intermediate products */ foldxinitx = old.x * old.x; foldyinity = old.y * old.y; foldxinity = old.x * old.y; /* orbit calculation */ if(old.x > 0) { new.x = foldxinitx - foldyinity - 1.0; new.y = foldxinity * 2; } else { new.x = foldxinitx - foldyinity -1.0 + floatparm->x * old.x; new.y = foldxinity * 2; /* This term added by Tim Wegner to make dependent on the imaginary part of the parameter. (Otherwise Mandelbrot is uninteresting. */ new.y += floatparm->y * old.x; } return(floatbailout()); } int TrigPlusZsquaredFractal(void) { #ifndef XFRACT /* From Scientific American, July 1989 */ /* A Biomorph */ /* z(n+1) = trig(z(n))+z(n)**2+C */ LCMPLXtrig0(lold,lnew); lnew.x += ltempsqrx - ltempsqry + longparm->x; lnew.y += multiply(lold.x, lold.y, bitshiftless1) + longparm->y; return(longbailout()); #else return(0); #endif } int TrigPlusZsquaredfpFractal(void) { /* From Scientific American, July 1989 */ /* A Biomorph */ /* z(n+1) = trig(z(n))+z(n)**2+C */ CMPLXtrig0(old,new); new.x += tempsqrx - tempsqry + floatparm->x; new.y += 2.0 * old.x * old.y + floatparm->y; return(floatbailout()); } int Richard8fpFractal(void) { /* Richard8 {c = z = pixel: z=sin(z)+sin(pixel),|z|<=50} */ CMPLXtrig0(old,new); /* CMPLXtrig1(*floatparm,tmp); */ new.x += tmp.x; new.y += tmp.y; return(floatbailout()); } int Richard8Fractal(void) { #ifndef XFRACT /* Richard8 {c = z = pixel: z=sin(z)+sin(pixel),|z|<=50} */ LCMPLXtrig0(lold,lnew); /* LCMPLXtrig1(*longparm,ltmp); */ lnew.x += ltmp.x; lnew.y += ltmp.y; return(longbailout()); #else return(0); #endif } int PopcornFractal_Old(void) { tmp = old; tmp.x *= 3.0; tmp.y *= 3.0; FPUsincos(&tmp.x,&sinx,&cosx); FPUsincos(&tmp.y,&siny,&cosy); tmp.x = sinx/cosx + old.x; tmp.y = siny/cosy + old.y; FPUsincos(&tmp.x,&sinx,&cosx); FPUsincos(&tmp.y,&siny,&cosy); new.x = old.x - parm.x*siny; new.y = old.y - parm.x*sinx; /* new.x = old.x - parm.x*sin(old.y+tan(3*old.y)); new.y = old.y - parm.x*sin(old.x+tan(3*old.x)); */ if(plot == noplot) { plot_orbit(new.x,new.y,1+row%colors); old = new; } else /* FLOATBAILOUT(); */ /* PB The above line was weird, not what it seems to be! But, bracketing it or always doing it (either of which seem more likely to be what was intended) changes the image for the worse, so I'm not touching it. Same applies to int form in next routine. */ /* PB later: recoded inline, still leaving it weird */ tempsqrx = sqr(new.x); tempsqry = sqr(new.y); if((magnitude = tempsqrx + tempsqry) >= rqlim) return(1); old = new; return(0); } int PopcornFractal(void) { tmp = old; tmp.x *= 3.0; tmp.y *= 3.0; FPUsincos(&tmp.x,&sinx,&cosx); FPUsincos(&tmp.y,&siny,&cosy); tmp.x = sinx/cosx + old.x; tmp.y = siny/cosy + old.y; FPUsincos(&tmp.x,&sinx,&cosx); FPUsincos(&tmp.y,&siny,&cosy); new.x = old.x - parm.x*siny; new.y = old.y - parm.x*sinx; /* new.x = old.x - parm.x*sin(old.y+tan(3*old.y)); new.y = old.y - parm.x*sin(old.x+tan(3*old.x)); */ if(plot == noplot) { plot_orbit(new.x,new.y,1+row%colors); old = new; } /* else */ /* FLOATBAILOUT(); */ /* PB The above line was weird, not what it seems to be! But, bracketing it or always doing it (either of which seem more likely to be what was intended) changes the image for the worse, so I'm not touching it. Same applies to int form in next routine. */ /* PB later: recoded inline, still leaving it weird */ /* JCO: sqr's should always be done, else magnitude could be wrong */ tempsqrx = sqr(new.x); tempsqry = sqr(new.y); if((magnitude = tempsqrx + tempsqry) >= rqlim || fabs(new.x) > rqlim2 || fabs(new.y) > rqlim2 ) return(1); old = new; return(0); } int LPopcornFractal_Old(void) { #ifndef XFRACT ltmp = lold; ltmp.x *= 3L; ltmp.y *= 3L; LTRIGARG(ltmp.x); LTRIGARG(ltmp.y); SinCos086(ltmp.x,&lsinx,&lcosx); SinCos086(ltmp.y,&lsiny,&lcosy); ltmp.x = divide(lsinx,lcosx,bitshift) + lold.x; ltmp.y = divide(lsiny,lcosy,bitshift) + lold.y; LTRIGARG(ltmp.x); LTRIGARG(ltmp.y); SinCos086(ltmp.x,&lsinx,&lcosx); SinCos086(ltmp.y,&lsiny,&lcosy); lnew.x = lold.x - multiply(lparm.x,lsiny,bitshift); lnew.y = lold.y - multiply(lparm.x,lsinx,bitshift); if(plot == noplot) { iplot_orbit(lnew.x,lnew.y,1+row%colors); lold = lnew; } else /* LONGBAILOUT(); */ /* PB above still the old way, is weird, see notes in FP popcorn case */ { ltempsqrx = lsqr(lnew.x); ltempsqry = lsqr(lnew.y); } lmagnitud = ltempsqrx + ltempsqry; if (lmagnitud >= llimit || lmagnitud < 0 || labs(lnew.x) > llimit2 || labs(lnew.y) > llimit2) return(1); lold = lnew; return(0); #else return(0); #endif } int LPopcornFractal(void) { #ifndef XFRACT ltmp = lold; ltmp.x *= 3L; ltmp.y *= 3L; LTRIGARG(ltmp.x); LTRIGARG(ltmp.y); SinCos086(ltmp.x,&lsinx,&lcosx); SinCos086(ltmp.y,&lsiny,&lcosy); ltmp.x = divide(lsinx,lcosx,bitshift) + lold.x; ltmp.y = divide(lsiny,lcosy,bitshift) + lold.y; LTRIGARG(ltmp.x); LTRIGARG(ltmp.y); SinCos086(ltmp.x,&lsinx,&lcosx); SinCos086(ltmp.y,&lsiny,&lcosy); lnew.x = lold.x - multiply(lparm.x,lsiny,bitshift); lnew.y = lold.y - multiply(lparm.x,lsinx,bitshift); if(plot == noplot) { iplot_orbit(lnew.x,lnew.y,1+row%colors); lold = lnew; } /* else */ /* JCO: sqr's should always be done, else magnitude could be wrong */ ltempsqrx = lsqr(lnew.x); ltempsqry = lsqr(lnew.y); lmagnitud = ltempsqrx + ltempsqry; if (lmagnitud >= llimit || lmagnitud < 0 || labs(lnew.x) > llimit2 || labs(lnew.y) > llimit2) return(1); lold = lnew; return(0); #else return(0); #endif } /* Popcorn generalization proposed by HB */ int PopcornFractalFn(void) { _CMPLX tmpx; _CMPLX tmpy; /* tmpx contains the generalized value of the old real "x" equation */ CMPLXtimesreal(parm2,old.y,tmp); /* tmp = (C * old.y) */ CMPLXtrig1(tmp,tmpx); /* tmpx = trig1(tmp) */ tmpx.x += old.y; /* tmpx = old.y + trig1(tmp) */ CMPLXtrig0(tmpx,tmp); /* tmp = trig0(tmpx) */ CMPLXmult(tmp,parm,tmpx); /* tmpx = tmp * h */ /* tmpy contains the generalized value of the old real "y" equation */ CMPLXtimesreal(parm2,old.x,tmp); /* tmp = (C * old.x) */ CMPLXtrig3(tmp,tmpy); /* tmpy = trig3(tmp) */ tmpy.x += old.x; /* tmpy = old.x + trig1(tmp) */ CMPLXtrig2(tmpy,tmp); /* tmp = trig2(tmpy) */ CMPLXmult(tmp,parm,tmpy); /* tmpy = tmp * h */ new.x = old.x - tmpx.x - tmpy.y; new.y = old.y - tmpy.x - tmpx.y; if(plot == noplot) { plot_orbit(new.x,new.y,1+row%colors); old = new; } tempsqrx = sqr(new.x); tempsqry = sqr(new.y); if((magnitude = tempsqrx + tempsqry) >= rqlim || fabs(new.x) > rqlim2 || fabs(new.y) > rqlim2 ) return(1); old = new; return(0); } #define FIX_OVERFLOW(arg) if(overflow) \ { \ (arg).x = fudge;\ (arg).y = 0;\ overflow = 0;\ } int LPopcornFractalFn(void) { #ifndef XFRACT _LCMPLX ltmpx, ltmpy; overflow = 0; /* ltmpx contains the generalized value of the old real "x" equation */ LCMPLXtimesreal(lparm2,lold.y,ltmp); /* tmp = (C * old.y) */ LCMPLXtrig1(ltmp,ltmpx); /* tmpx = trig1(tmp) */ FIX_OVERFLOW(ltmpx); ltmpx.x += lold.y; /* tmpx = old.y + trig1(tmp) */ LCMPLXtrig0(ltmpx,ltmp); /* tmp = trig0(tmpx) */ FIX_OVERFLOW(ltmp); LCMPLXmult(ltmp,lparm,ltmpx); /* tmpx = tmp * h */ /* ltmpy contains the generalized value of the old real "y" equation */ LCMPLXtimesreal(lparm2,lold.x,ltmp); /* tmp = (C * old.x) */ LCMPLXtrig3(ltmp,ltmpy); /* tmpy = trig3(tmp) */ FIX_OVERFLOW(ltmpy); ltmpy.x += lold.x; /* tmpy = old.x + trig1(tmp) */ LCMPLXtrig2(ltmpy,ltmp); /* tmp = trig2(tmpy) */ FIX_OVERFLOW(ltmp); LCMPLXmult(ltmp,lparm,ltmpy); /* tmpy = tmp * h */ lnew.x = lold.x - ltmpx.x - ltmpy.y; lnew.y = lold.y - ltmpy.x - ltmpx.y; if(plot == noplot) { iplot_orbit(lnew.x,lnew.y,1+row%colors); lold = lnew; } ltempsqrx = lsqr(lnew.x); ltempsqry = lsqr(lnew.y); lmagnitud = ltempsqrx + ltempsqry; if (lmagnitud >= llimit || lmagnitud < 0 || labs(lnew.x) > llimit2 || labs(lnew.y) > llimit2) return(1); lold = lnew; return(0); #else return(0); #endif } int MarksCplxMand(void) { tmp.x = tempsqrx - tempsqry; tmp.y = 2*old.x*old.y; FPUcplxmul(&tmp, &coefficient, &new); new.x += floatparm->x; new.y += floatparm->y; return(floatbailout()); } int SpiderfpFractal(void) { /* Spider(XAXIS) { c=z=pixel: z=z*z+c; c=c/2+z, |z|<=4 } */ new.x = tempsqrx - tempsqry + tmp.x; new.y = 2 * old.x * old.y + tmp.y; tmp.x = tmp.x/2 + new.x; tmp.y = tmp.y/2 + new.y; return(floatbailout()); } int SpiderFractal(void) { #ifndef XFRACT /* Spider(XAXIS) { c=z=pixel: z=z*z+c; c=c/2+z, |z|<=4 } */ lnew.x = ltempsqrx - ltempsqry + ltmp.x; lnew.y = multiply(lold.x, lold.y, bitshiftless1) + ltmp.y; ltmp.x = (ltmp.x >> 1) + lnew.x; ltmp.y = (ltmp.y >> 1) + lnew.y; return(longbailout()); #else return(0); #endif } int TetratefpFractal(void) { /* Tetrate(XAXIS) { c=z=pixel: z=c^z, |z|<=(P1+3) } */ new = ComplexPower(*floatparm,old); return(floatbailout()); } int ZXTrigPlusZFractal(void) { #ifndef XFRACT /* z = (p1*z*trig(z))+p2*z */ LCMPLXtrig0(lold,ltmp); /* ltmp = trig(old) */ LCMPLXmult(lparm,ltmp,ltmp); /* ltmp = p1*trig(old) */ LCMPLXmult(lold,ltmp,ltmp2); /* ltmp2 = p1*old*trig(old) */ LCMPLXmult(lparm2,lold,ltmp); /* ltmp = p2*old */ LCMPLXadd(ltmp2,ltmp,lnew); /* lnew = p1*trig(old) + p2*old */ return(longbailout()); #else return(0); #endif } int ScottZXTrigPlusZFractal(void) { #ifndef XFRACT /* z = (z*trig(z))+z */ LCMPLXtrig0(lold,ltmp); /* ltmp = trig(old) */ LCMPLXmult(lold,ltmp,lnew); /* lnew = old*trig(old) */ LCMPLXadd(lnew,lold,lnew); /* lnew = trig(old) + old */ return(longbailout()); #else return(0); #endif } int SkinnerZXTrigSubZFractal(void) { #ifndef XFRACT /* z = (z*trig(z))-z */ LCMPLXtrig0(lold,ltmp); /* ltmp = trig(old) */ LCMPLXmult(lold,ltmp,lnew); /* lnew = old*trig(old) */ LCMPLXsub(lnew,lold,lnew); /* lnew = trig(old) - old */ return(longbailout()); #else return(0); #endif } int ZXTrigPlusZfpFractal(void) { /* z = (p1*z*trig(z))+p2*z */ CMPLXtrig0(old,tmp); /* tmp = trig(old) */ CMPLXmult(parm,tmp,tmp); /* tmp = p1*trig(old) */ CMPLXmult(old,tmp,tmp2); /* tmp2 = p1*old*trig(old) */ CMPLXmult(parm2,old,tmp); /* tmp = p2*old */ CMPLXadd(tmp2,tmp,new); /* new = p1*trig(old) + p2*old */ return(floatbailout()); } int ScottZXTrigPlusZfpFractal(void) { /* z = (z*trig(z))+z */ CMPLXtrig0(old,tmp); /* tmp = trig(old) */ CMPLXmult(old,tmp,new); /* new = old*trig(old) */ CMPLXadd(new,old,new); /* new = trig(old) + old */ return(floatbailout()); } int SkinnerZXTrigSubZfpFractal(void) { /* z = (z*trig(z))-z */ CMPLXtrig0(old,tmp); /* tmp = trig(old) */ CMPLXmult(old,tmp,new); /* new = old*trig(old) */ CMPLXsub(new,old,new); /* new = trig(old) - old */ return(floatbailout()); } int Sqr1overTrigFractal(void) { #ifndef XFRACT /* z = sqr(1/trig(z)) */ LCMPLXtrig0(lold,lold); LCMPLXrecip(lold,lold); LCMPLXsqr(lold,lnew); return(longbailout()); #else return(0); #endif } int Sqr1overTrigfpFractal(void) { /* z = sqr(1/trig(z)) */ CMPLXtrig0(old,old); CMPLXrecip(old,old); CMPLXsqr(old,new); return(floatbailout()); } int TrigPlusTrigFractal(void) { #ifndef XFRACT /* z = trig(0,z)*p1+trig1(z)*p2 */ LCMPLXtrig0(lold,ltmp); LCMPLXmult(lparm,ltmp,ltmp); LCMPLXtrig1(lold,ltmp2); LCMPLXmult(lparm2,ltmp2,lold); LCMPLXadd(ltmp,lold,lnew); return(longbailout()); #else return(0); #endif } int TrigPlusTrigfpFractal(void) { /* z = trig0(z)*p1+trig1(z)*p2 */ CMPLXtrig0(old,tmp); CMPLXmult(parm,tmp,tmp); CMPLXtrig1(old,old); CMPLXmult(parm2,old,old); CMPLXadd(tmp,old,new); return(floatbailout()); } /* The following four fractals are based on the idea of parallel or alternate calculations. The shift is made when the mod reaches a given value. JCO 5/6/92 */ int LambdaTrigOrTrigFractal(void) { #ifndef XFRACT /* z = trig0(z)*p1 if mod(old) < p2.x and trig1(z)*p1 if mod(old) >= p2.x */ if ((LCMPLXmod(lold)) < lparm2.x){ LCMPLXtrig0(lold,ltmp); LCMPLXmult(*longparm,ltmp,lnew);} else{ LCMPLXtrig1(lold,ltmp); LCMPLXmult(*longparm,ltmp,lnew);} return(longbailout()); #else return(0); #endif } int LambdaTrigOrTrigfpFractal(void) { /* z = trig0(z)*p1 if mod(old) < p2.x and trig1(z)*p1 if mod(old) >= p2.x */ if (CMPLXmod(old) < parm2.x){ CMPLXtrig0(old,old); FPUcplxmul(floatparm,&old,&new);} else{ CMPLXtrig1(old,old); FPUcplxmul(floatparm,&old,&new);} return(floatbailout()); } int JuliaTrigOrTrigFractal(void) { #ifndef XFRACT /* z = trig0(z)+p1 if mod(old) < p2.x and trig1(z)+p1 if mod(old) >= p2.x */ if (LCMPLXmod(lold) < lparm2.x){ LCMPLXtrig0(lold,ltmp); LCMPLXadd(*longparm,ltmp,lnew);} else{ LCMPLXtrig1(lold,ltmp); LCMPLXadd(*longparm,ltmp,lnew);} return(longbailout()); #else return(0); #endif } int JuliaTrigOrTrigfpFractal(void) { /* z = trig0(z)+p1 if mod(old) < p2.x and trig1(z)+p1 if mod(old) >= p2.x */ if (CMPLXmod(old) < parm2.x){ CMPLXtrig0(old,old); CMPLXadd(*floatparm,old,new);} else{ CMPLXtrig1(old,old); CMPLXadd(*floatparm,old,new);} return(floatbailout()); } int AplusOne, Ap1deg; struct MP mpAplusOne, mpAp1deg; struct MPC mpctmpparm; #if (_MSC_VER >= 700) #pragma code_seg ("mpmath1_text") /* place following in an overlay */ #endif int MPCHalleyFractal(void) { #ifndef XFRACT /* X(X^a - 1) = 0, Halley Map */ /* a = parm.x, relaxation coeff. = parm.y, epsilon = parm2.x */ int ihal; struct MPC mpcXtoAlessOne, mpcXtoA; struct MPC mpcXtoAplusOne; /* a-1, a, a+1 */ struct MPC mpcFX, mpcF1prime, mpcF2prime, mpcHalnumer1; struct MPC mpcHalnumer2, mpcHaldenom, mpctmp; MPOverflow = 0; mpcXtoAlessOne.x = mpcold.x; mpcXtoAlessOne.y = mpcold.y; for(ihal=2; ihal= 700) #pragma code_seg () /* back to normal segment */ #endif int HalleyFractal(void) { /* X(X^a - 1) = 0, Halley Map */ /* a = parm.x = degree, relaxation coeff. = parm.y, epsilon = parm2.x */ int ihal; _CMPLX XtoAlessOne, XtoA, XtoAplusOne; /* a-1, a, a+1 */ _CMPLX FX, F1prime, F2prime, Halnumer1, Halnumer2, Haldenom; _CMPLX relax; XtoAlessOne = old; for(ihal=2; ihalx+multiply(longparm->y,ltmp2.x,bitshift); lnew.y = (ltmp.x + ltmp.x) + multiply(longparm->y,ltmp2.y,bitshift); ltmp2 = lold; /* set ltmp2 to Y value */ return(longbailout()); #else return(0); #endif } int PhoenixFractal(void) { /* z(n+1) = z(n)^2 + p + qy(n), y(n+1) = z(n) */ tmp.x = old.x * old.y; new.x = tempsqrx - tempsqry + floatparm->x + (floatparm->y * tmp2.x); new.y = (tmp.x + tmp.x) + (floatparm->y * tmp2.y); tmp2 = old; /* set tmp2 to Y value */ return(floatbailout()); } int LongPhoenixFractalcplx(void) { #ifndef XFRACT /* z(n+1) = z(n)^2 + p + qy(n), y(n+1) = z(n) */ ltmp.x = multiply(lold.x, lold.y, bitshift); lnew.x = ltempsqrx-ltempsqry+longparm->x+multiply(lparm2.x,ltmp2.x,bitshift)-multiply(lparm2.y,ltmp2.y,bitshift); lnew.y = (ltmp.x + ltmp.x)+longparm->y+multiply(lparm2.x,ltmp2.y,bitshift)+multiply(lparm2.y,ltmp2.x,bitshift); ltmp2 = lold; /* set ltmp2 to Y value */ return(longbailout()); #else return(0); #endif } int PhoenixFractalcplx(void) { /* z(n+1) = z(n)^2 + p1 + p2*y(n), y(n+1) = z(n) */ tmp.x = old.x * old.y; new.x = tempsqrx - tempsqry + floatparm->x + (parm2.x * tmp2.x) - (parm2.y * tmp2.y); new.y = (tmp.x + tmp.x) + floatparm->y + (parm2.x * tmp2.y) + (parm2.y * tmp2.x); tmp2 = old; /* set tmp2 to Y value */ return(floatbailout()); } int LongPhoenixPlusFractal(void) { #ifndef XFRACT /* z(n+1) = z(n)^(degree-1) * (z(n) + p) + qy(n), y(n+1) = z(n) */ int i; _LCMPLX loldplus, lnewminus; loldplus = lold; ltmp = lold; for(i=1; i= 2, degree=degree-1 in setup */ LCMPLXmult(lold,ltmp,ltmp); /* = old^(degree-1) */ } loldplus.x += longparm->x; LCMPLXmult(ltmp, loldplus, lnewminus); lnew.x = lnewminus.x + multiply(longparm->y,ltmp2.x,bitshift); lnew.y = lnewminus.y + multiply(longparm->y,ltmp2.y,bitshift); ltmp2 = lold; /* set ltmp2 to Y value */ return(longbailout()); #else return(0); #endif } int PhoenixPlusFractal(void) { /* z(n+1) = z(n)^(degree-1) * (z(n) + p) + qy(n), y(n+1) = z(n) */ int i; _CMPLX oldplus, newminus; oldplus = old; tmp = old; for(i=1; i= 2, degree=degree-1 in setup */ FPUcplxmul(&old, &tmp, &tmp); /* = old^(degree-1) */ } oldplus.x += floatparm->x; FPUcplxmul(&tmp, &oldplus, &newminus); new.x = newminus.x + (floatparm->y * tmp2.x); new.y = newminus.y + (floatparm->y * tmp2.y); tmp2 = old; /* set tmp2 to Y value */ return(floatbailout()); } int LongPhoenixMinusFractal(void) { #ifndef XFRACT /* z(n+1) = z(n)^(degree-2) * (z(n)^2 + p) + qy(n), y(n+1) = z(n) */ int i; _LCMPLX loldsqr, lnewminus; LCMPLXmult(lold,lold,loldsqr); ltmp = lold; for(i=1; i= 3, degree=degree-2 in setup */ LCMPLXmult(lold,ltmp,ltmp); /* = old^(degree-2) */ } loldsqr.x += longparm->x; LCMPLXmult(ltmp, loldsqr, lnewminus); lnew.x = lnewminus.x + multiply(longparm->y,ltmp2.x,bitshift); lnew.y = lnewminus.y + multiply(longparm->y,ltmp2.y,bitshift); ltmp2 = lold; /* set ltmp2 to Y value */ return(longbailout()); #else return(0); #endif } int PhoenixMinusFractal(void) { /* z(n+1) = z(n)^(degree-2) * (z(n)^2 + p) + qy(n), y(n+1) = z(n) */ int i; _CMPLX oldsqr, newminus; FPUcplxmul(&old, &old, &oldsqr); tmp = old; for(i=1; i= 3, degree=degree-2 in setup */ FPUcplxmul(&old, &tmp, &tmp); /* = old^(degree-2) */ } oldsqr.x += floatparm->x; FPUcplxmul(&tmp, &oldsqr, &newminus); new.x = newminus.x + (floatparm->y * tmp2.x); new.y = newminus.y + (floatparm->y * tmp2.y); tmp2 = old; /* set tmp2 to Y value */ return(floatbailout()); } int LongPhoenixCplxPlusFractal(void) { #ifndef XFRACT /* z(n+1) = z(n)^(degree-1) * (z(n) + p) + qy(n), y(n+1) = z(n) */ int i; _LCMPLX loldplus, lnewminus; loldplus = lold; ltmp = lold; for(i=1; i= 2, degree=degree-1 in setup */ LCMPLXmult(lold,ltmp,ltmp); /* = old^(degree-1) */ } loldplus.x += longparm->x; loldplus.y += longparm->y; LCMPLXmult(ltmp, loldplus, lnewminus); LCMPLXmult(lparm2, ltmp2, ltmp); lnew.x = lnewminus.x + ltmp.x; lnew.y = lnewminus.y + ltmp.y; ltmp2 = lold; /* set ltmp2 to Y value */ return(longbailout()); #else return(0); #endif } int PhoenixCplxPlusFractal(void) { /* z(n+1) = z(n)^(degree-1) * (z(n) + p) + qy(n), y(n+1) = z(n) */ int i; _CMPLX oldplus, newminus; oldplus = old; tmp = old; for(i=1; i= 2, degree=degree-1 in setup */ FPUcplxmul(&old, &tmp, &tmp); /* = old^(degree-1) */ } oldplus.x += floatparm->x; oldplus.y += floatparm->y; FPUcplxmul(&tmp, &oldplus, &newminus); FPUcplxmul(&parm2, &tmp2, &tmp); new.x = newminus.x + tmp.x; new.y = newminus.y + tmp.y; tmp2 = old; /* set tmp2 to Y value */ return(floatbailout()); } int LongPhoenixCplxMinusFractal(void) { #ifndef XFRACT /* z(n+1) = z(n)^(degree-2) * (z(n)^2 + p) + qy(n), y(n+1) = z(n) */ int i; _LCMPLX loldsqr, lnewminus; LCMPLXmult(lold,lold,loldsqr); ltmp = lold; for(i=1; i= 3, degree=degree-2 in setup */ LCMPLXmult(lold,ltmp,ltmp); /* = old^(degree-2) */ } loldsqr.x += longparm->x; loldsqr.y += longparm->y; LCMPLXmult(ltmp, loldsqr, lnewminus); LCMPLXmult(lparm2, ltmp2, ltmp); lnew.x = lnewminus.x + ltmp.x; lnew.y = lnewminus.y + ltmp.y; ltmp2 = lold; /* set ltmp2 to Y value */ return(longbailout()); #else return(0); #endif } int PhoenixCplxMinusFractal(void) { /* z(n+1) = z(n)^(degree-2) * (z(n)^2 + p) + qy(n), y(n+1) = z(n) */ int i; _CMPLX oldsqr, newminus; FPUcplxmul(&old, &old, &oldsqr); tmp = old; for(i=1; i= 3, degree=degree-2 in setup */ FPUcplxmul(&old, &tmp, &tmp); /* = old^(degree-2) */ } oldsqr.x += floatparm->x; oldsqr.y += floatparm->y; FPUcplxmul(&tmp, &oldsqr, &newminus); FPUcplxmul(&parm2, &tmp2, &tmp); new.x = newminus.x + tmp.x; new.y = newminus.y + tmp.y; tmp2 = old; /* set tmp2 to Y value */ return(floatbailout()); } int ScottTrigPlusTrigFractal(void) { #ifndef XFRACT /* z = trig0(z)+trig1(z) */ LCMPLXtrig0(lold,ltmp); LCMPLXtrig1(lold,lold); LCMPLXadd(ltmp,lold,lnew); return(longbailout()); #else return(0); #endif } int ScottTrigPlusTrigfpFractal(void) { /* z = trig0(z)+trig1(z) */ CMPLXtrig0(old,tmp); CMPLXtrig1(old,tmp2); CMPLXadd(tmp,tmp2,new); return(floatbailout()); } int SkinnerTrigSubTrigFractal(void) { #ifndef XFRACT /* z = trig(0,z)-trig1(z) */ LCMPLXtrig0(lold,ltmp); LCMPLXtrig1(lold,ltmp2); LCMPLXsub(ltmp,ltmp2,lnew); return(longbailout()); #else return(0); #endif } int SkinnerTrigSubTrigfpFractal(void) { /* z = trig0(z)-trig1(z) */ CMPLXtrig0(old,tmp); CMPLXtrig1(old,tmp2); CMPLXsub(tmp,tmp2,new); return(floatbailout()); } int TrigXTrigfpFractal(void) { /* z = trig0(z)*trig1(z) */ CMPLXtrig0(old,tmp); CMPLXtrig1(old,old); CMPLXmult(tmp,old,new); return(floatbailout()); } #ifndef XFRACT /* call float version of fractal if integer math overflow */ static int TryFloatFractal(int (*fpFractal)(void)) { overflow=0; /* lold had better not be changed! */ old.x = lold.x; old.x /= fudge; old.y = lold.y; old.y /= fudge; tempsqrx = sqr(old.x); tempsqry = sqr(old.y); fpFractal(); if (save_release < 1900) { /* for backwards compatibility */ lnew.x = (long)(new.x/fudge); /* this error has been here a long time */ lnew.y = (long)(new.y/fudge); } else { lnew.x = (long)(new.x*fudge); lnew.y = (long)(new.y*fudge); } return(0); } #endif int TrigXTrigFractal(void) { #ifndef XFRACT _LCMPLX ltmp2; /* z = trig0(z)*trig1(z) */ LCMPLXtrig0(lold,ltmp); LCMPLXtrig1(lold,ltmp2); LCMPLXmult(ltmp,ltmp2,lnew); if(overflow) TryFloatFractal(TrigXTrigfpFractal); return(longbailout()); #else return(0); #endif } /********************************************************************/ /* Next six orbit functions are one type - extra functions are */ /* special cases written for speed. */ /********************************************************************/ int TrigPlusSqrFractal(void) /* generalization of Scott and Skinner types */ { #ifndef XFRACT /* { z=pixel: z=(p1,p2)*trig(z)+(p3,p4)*sqr(z), |z| l16triglim_2 || labs(ltmp.y) > l16triglim_2) && save_release > 1900) overflow = 1; else { LCMPLXtrig0(ltmp,lnew); } if(overflow) TryFloatFractal(TrigZsqrdfpFractal); return(longbailout()); #else return(0); #endif } int SqrTrigFractal(void) { #ifndef XFRACT /* { z=pixel: z=sqr(trig(z)), |z|x - 1; /* top = Z**2+C-1 */ top.y = old.x * old.y; top.y = top.y + top.y + floatparm->y; bot.x = old.x + old.x + floatparm->x - 2; /* bot = 2*Z+C-2 */ bot.y = old.y + old.y + floatparm->y; div = bot.x*bot.x + bot.y*bot.y; /* tmp = top/bot */ if (div < FLT_MIN) return(1); tmp.x = (top.x*bot.x + top.y*bot.y)/div; tmp.y = (top.y*bot.x - top.x*bot.y)/div; new.x = (tmp.x + tmp.y) * (tmp.x - tmp.y); /* Z = tmp**2 */ new.y = tmp.x * tmp.y; new.y += new.y; return(floatbailout()); } int Magnet2Fractal(void) /* Z = ((Z**3 + 3(C-1)Z + (C-1)(C-2) ) / */ /* (3Z**2 + 3(C-2)Z + (C-1)(C-2)+1) )**2 */ { /* In "Beauty of Fractals", code by Kev Allen. */ _CMPLX top, bot, tmp; double div; top.x = old.x * (tempsqrx-tempsqry-tempsqry-tempsqry + T_Cm1.x) - old.y * T_Cm1.y + T_Cm1Cm2.x; top.y = old.y * (tempsqrx+tempsqrx+tempsqrx-tempsqry + T_Cm1.x) + old.x * T_Cm1.y + T_Cm1Cm2.y; bot.x = tempsqrx - tempsqry; bot.x = bot.x + bot.x + bot.x + old.x * T_Cm2.x - old.y * T_Cm2.y + T_Cm1Cm2.x + 1.0; bot.y = old.x * old.y; bot.y += bot.y; bot.y = bot.y + bot.y + bot.y + old.x * T_Cm2.y + old.y * T_Cm2.x + T_Cm1Cm2.y; div = bot.x*bot.x + bot.y*bot.y; /* tmp = top/bot */ if (div < FLT_MIN) return(1); tmp.x = (top.x*bot.x + top.y*bot.y)/div; tmp.y = (top.y*bot.x - top.x*bot.y)/div; new.x = (tmp.x + tmp.y) * (tmp.x - tmp.y); /* Z = tmp**2 */ new.y = tmp.x * tmp.y; new.y += new.y; return(floatbailout()); } int LambdaTrigFractal(void) { #ifndef XFRACT LONGXYTRIGBAILOUT(); LCMPLXtrig0(lold,ltmp); /* ltmp = trig(lold) */ LCMPLXmult(*longparm,ltmp,lnew); /* lnew = longparm*trig(lold) */ lold = lnew; return(0); #else return(0); #endif } int LambdaTrigfpFractal(void) { FLOATXYTRIGBAILOUT(); CMPLXtrig0(old,tmp); /* tmp = trig(old) */ CMPLXmult(*floatparm,tmp,new); /* new = longparm*trig(old) */ old = new; return(0); } /* bailouts are different for different trig functions */ int LambdaTrigFractal1(void) { #ifndef XFRACT LONGTRIGBAILOUT(); /* sin,cos */ LCMPLXtrig0(lold,ltmp); /* ltmp = trig(lold) */ LCMPLXmult(*longparm,ltmp,lnew); /* lnew = longparm*trig(lold) */ lold = lnew; return(0); #else return(0); #endif } int LambdaTrigfpFractal1(void) { FLOATTRIGBAILOUT(); /* sin,cos */ CMPLXtrig0(old,tmp); /* tmp = trig(old) */ CMPLXmult(*floatparm,tmp,new); /* new = longparm*trig(old) */ old = new; return(0); } int LambdaTrigFractal2(void) { #ifndef XFRACT LONGHTRIGBAILOUT(); /* sinh,cosh */ LCMPLXtrig0(lold,ltmp); /* ltmp = trig(lold) */ LCMPLXmult(*longparm,ltmp,lnew); /* lnew = longparm*trig(lold) */ lold = lnew; return(0); #else return(0); #endif } int LambdaTrigfpFractal2(void) { #ifndef XFRACT FLOATHTRIGBAILOUT(); /* sinh,cosh */ CMPLXtrig0(old,tmp); /* tmp = trig(old) */ CMPLXmult(*floatparm,tmp,new); /* new = longparm*trig(old) */ old = new; return(0); #else return(0); #endif } int ManOWarFractal(void) { #ifndef XFRACT /* From Art Matrix via Lee Skinner */ lnew.x = ltempsqrx - ltempsqry + ltmp.x + longparm->x; lnew.y = multiply(lold.x, lold.y, bitshiftless1) + ltmp.y + longparm->y; ltmp = lold; return(longbailout()); #else return(0); #endif } int ManOWarfpFractal(void) { /* From Art Matrix via Lee Skinner */ /* note that fast >= 287 equiv in fracsuba.asm must be kept in step */ new.x = tempsqrx - tempsqry + tmp.x + floatparm->x; new.y = 2.0 * old.x * old.y + tmp.y + floatparm->y; tmp = old; return(floatbailout()); } /* MarksMandelPwr (XAXIS) { z = pixel, c = z ^ (z - 1): z = c * sqr(z) + pixel, |z| <= 4 } */ int MarksMandelPwrfpFractal(void) { CMPLXtrig0(old,new); CMPLXmult(tmp,new,new); new.x += floatparm->x; new.y += floatparm->y; return(floatbailout()); } int MarksMandelPwrFractal(void) { #ifndef XFRACT LCMPLXtrig0(lold,lnew); LCMPLXmult(ltmp,lnew,lnew); lnew.x += longparm->x; lnew.y += longparm->y; return(longbailout()); #else return(0); #endif } /* I was coding Marksmandelpower and failed to use some temporary variables. The result was nice, and since my name is not on any fractal, I thought I would immortalize myself with this error! Tim Wegner */ int TimsErrorfpFractal(void) { CMPLXtrig0(old,new); new.x = new.x * tmp.x - new.y * tmp.y; new.y = new.x * tmp.y - new.y * tmp.x; new.x += floatparm->x; new.y += floatparm->y; return(floatbailout()); } int TimsErrorFractal(void) { #ifndef XFRACT LCMPLXtrig0(lold,lnew); lnew.x = multiply(lnew.x,ltmp.x,bitshift)-multiply(lnew.y,ltmp.y,bitshift); lnew.y = multiply(lnew.x,ltmp.y,bitshift)-multiply(lnew.y,ltmp.x,bitshift); lnew.x += longparm->x; lnew.y += longparm->y; return(longbailout()); #else return(0); #endif } int CirclefpFractal(void) { long i; i = (long)(param[0]*(tempsqrx+tempsqry)); coloriter = i%colors; return(1); } /* CirclelongFractal() { long i; i = multiply(lparm.x,(ltempsqrx+ltempsqry),bitshift); i = i >> bitshift; coloriter = i%colors); return(1); } */ /* -------------------------------------------------------------------- */ /* Initialization (once per pixel) routines */ /* -------------------------------------------------------------------- */ #ifdef XFRACT /* this code translated to asm - lives in newton.asm */ /* transform points with reciprocal function */ void invertz2(_CMPLX *z) { z->x = dxpixel(); z->y = dypixel(); z->x -= f_xcenter; z->y -= f_ycenter; /* Normalize values to center of circle */ tempsqrx = sqr(z->x) + sqr(z->y); /* Get old radius */ if(fabs(tempsqrx) > FLT_MIN) tempsqrx = f_radius / tempsqrx; else tempsqrx = FLT_MAX; /* a big number, but not TOO big */ z->x *= tempsqrx; z->y *= tempsqrx; /* Perform inversion */ z->x += f_xcenter; z->y += f_ycenter; /* Renormalize */ } #endif int long_julia_per_pixel(void) { #ifndef XFRACT /* integer julia types */ /* lambda */ /* barnsleyj1 */ /* barnsleyj2 */ /* sierpinski */ if(invert) { /* invert */ invertz2(&old); /* watch out for overflow */ if(sqr(old.x)+sqr(old.y) >= 127) { old.x = 8; /* value to bail out in one iteration */ old.y = 8; } /* convert to fudged longs */ lold.x = (long)(old.x*fudge); lold.y = (long)(old.y*fudge); } else { lold.x = lxpixel(); lold.y = lypixel(); } return(0); #else printf("Called long_julia_per_pixel\n"); return(0); #endif } int long_richard8_per_pixel(void) { #ifndef XFRACT long_mandel_per_pixel(); LCMPLXtrig1(*longparm,ltmp); LCMPLXmult(ltmp,lparm2,ltmp); return(1); #else return(0); #endif } int long_mandel_per_pixel(void) { #ifndef XFRACT /* integer mandel types */ /* barnsleym1 */ /* barnsleym2 */ linit.x = lxpixel(); if(save_release >= 2004) linit.y = lypixel(); if(invert) { /* invert */ invertz2(&init); /* watch out for overflow */ if(sqr(init.x)+sqr(init.y) >= 127) { init.x = 8; /* value to bail out in one iteration */ init.y = 8; } /* convert to fudged longs */ linit.x = (long)(init.x*fudge); linit.y = (long)(init.y*fudge); } if(useinitorbit == 1) lold = linitorbit; else lold = linit; lold.x += lparm.x; /* initial pertubation of parameters set */ lold.y += lparm.y; return(1); /* 1st iteration has been done */ #else printf("Called long_mandel_per_pixel\n"); return(0); #endif } int julia_per_pixel(void) { /* julia */ if(invert) { /* invert */ invertz2(&old); /* watch out for overflow */ if(bitshift <= 24) if (sqr(old.x)+sqr(old.y) >= 127) { old.x = 8; /* value to bail out in one iteration */ old.y = 8; } if(bitshift > 24) if (sqr(old.x)+sqr(old.y) >= 4.0) { old.x = 2; /* value to bail out in one iteration */ old.y = 2; } /* convert to fudged longs */ lold.x = (long)(old.x*fudge); lold.y = (long)(old.y*fudge); } else { lold.x = lxpixel(); lold.y = lypixel(); } ltempsqrx = multiply(lold.x, lold.x, bitshift); ltempsqry = multiply(lold.y, lold.y, bitshift); ltmp = lold; return(0); } int marks_mandelpwr_per_pixel(void) { #ifndef XFRACT mandel_per_pixel(); ltmp = lold; ltmp.x -= fudge; LCMPLXpwr(lold,ltmp,ltmp); return(1); #else return(0); #endif } int mandel_per_pixel(void) { /* mandel */ if(invert) { invertz2(&init); /* watch out for overflow */ if(bitshift <= 24) if (sqr(init.x)+sqr(init.y) >= 127) { init.x = 8; /* value to bail out in one iteration */ init.y = 8; } if(bitshift > 24) if (sqr(init.x)+sqr(init.y) >= 4) { init.x = 2; /* value to bail out in one iteration */ init.y = 2; } /* convert to fudged longs */ linit.x = (long)(init.x*fudge); linit.y = (long)(init.y*fudge); } else { linit.x = lxpixel(); if(save_release >= 2004) linit.y = lypixel(); } switch (fractype) { case MANDELLAMBDA: /* Critical Value 0.5 + 0.0i */ lold.x = FgHalf; lold.y = 0; break; default: lold = linit; break; } /* alter init value */ if(useinitorbit == 1) lold = linitorbit; else if(useinitorbit == 2) lold = linit; if((inside == BOF60 || inside == BOF61) && !nobof) { /* kludge to match "Beauty of Fractals" picture since we start Mandelbrot iteration with init rather than 0 */ lold.x = lparm.x; /* initial pertubation of parameters set */ lold.y = lparm.y; coloriter = -1; } else { lold.x += lparm.x; /* initial pertubation of parameters set */ lold.y += lparm.y; } ltmp = linit; /* for spider */ ltempsqrx = multiply(lold.x, lold.x, bitshift); ltempsqry = multiply(lold.y, lold.y, bitshift); return(1); /* 1st iteration has been done */ } int marksmandel_per_pixel() { #ifndef XFRACT /* marksmandel */ if(invert) { invertz2(&init); /* watch out for overflow */ if(sqr(init.x)+sqr(init.y) >= 127) { init.x = 8; /* value to bail out in one iteration */ init.y = 8; } /* convert to fudged longs */ linit.x = (long)(init.x*fudge); linit.y = (long)(init.y*fudge); } else { linit.x = lxpixel(); if(save_release >= 2004) linit.y = lypixel(); } if(useinitorbit == 1) lold = linitorbit; else lold = linit; lold.x += lparm.x; /* initial pertubation of parameters set */ lold.y += lparm.y; if(c_exp > 3) lcpower(&lold,c_exp-1,&lcoefficient,bitshift); else if(c_exp == 3) { lcoefficient.x = multiply(lold.x, lold.x, bitshift) - multiply(lold.y, lold.y, bitshift); lcoefficient.y = multiply(lold.x, lold.y, bitshiftless1); } else if(c_exp == 2) lcoefficient = lold; else if(c_exp < 2) { lcoefficient.x = 1L << bitshift; lcoefficient.y = 0L; } ltempsqrx = multiply(lold.x, lold.x, bitshift); ltempsqry = multiply(lold.y, lold.y, bitshift); #endif return(1); /* 1st iteration has been done */ } int marksmandelfp_per_pixel() { /* marksmandel */ if(invert) invertz2(&init); else { init.x = dxpixel(); if(save_release >= 2004) init.y = dypixel(); } if(useinitorbit == 1) old = initorbit; else old = init; old.x += parm.x; /* initial pertubation of parameters set */ old.y += parm.y; tempsqrx = sqr(old.x); tempsqry = sqr(old.y); if(c_exp > 3) cpower(&old,c_exp-1,&coefficient); else if(c_exp == 3) { coefficient.x = tempsqrx - tempsqry; coefficient.y = old.x * old.y * 2; } else if(c_exp == 2) coefficient = old; else if(c_exp < 2) { coefficient.x = 1.0; coefficient.y = 0.0; } return(1); /* 1st iteration has been done */ } int marks_mandelpwrfp_per_pixel(void) { mandelfp_per_pixel(); tmp = old; tmp.x -= 1; CMPLXpwr(old,tmp,tmp); return(1); } int mandelfp_per_pixel(void) { /* floating point mandelbrot */ /* mandelfp */ if(invert) invertz2(&init); else { init.x = dxpixel(); if(save_release >= 2004) init.y = dypixel(); } switch (fractype) { case MAGNET2M: FloatPreCalcMagnet2(); case MAGNET1M: /* Crit Val Zero both, but neither */ old.x = old.y = 0.0; /* is of the form f(Z,C) = Z*g(Z)+C */ break; case MANDELLAMBDAFP: /* Critical Value 0.5 + 0.0i */ old.x = 0.5; old.y = 0.0; break; default: old = init; break; } /* alter init value */ if(useinitorbit == 1) old = initorbit; else if(useinitorbit == 2) old = init; if((inside == BOF60 || inside == BOF61) && !nobof) { /* kludge to match "Beauty of Fractals" picture since we start Mandelbrot iteration with init rather than 0 */ old.x = parm.x; /* initial pertubation of parameters set */ old.y = parm.y; coloriter = -1; } else { old.x += parm.x; old.y += parm.y; } tmp = init; /* for spider */ tempsqrx = sqr(old.x); /* precalculated value for regular Mandelbrot */ tempsqry = sqr(old.y); return(1); /* 1st iteration has been done */ } int juliafp_per_pixel(void) { /* floating point julia */ /* juliafp */ if(invert) invertz2(&old); else { old.x = dxpixel(); old.y = dypixel(); } tempsqrx = sqr(old.x); /* precalculated value for regular Julia */ tempsqry = sqr(old.y); tmp = old; return(0); } #if (_MSC_VER >= 700) #pragma code_seg ("mpmath1_text") /* place following in an overlay */ #endif int MPCjulia_per_pixel(void) { #ifndef XFRACT /* floating point julia */ /* juliafp */ if(invert) invertz2(&old); else { old.x = dxpixel(); old.y = dypixel(); } mpcold.x = *pd2MP(old.x); mpcold.y = *pd2MP(old.y); return(0); #else return(0); #endif } #if (_MSC_VER >= 700) #pragma code_seg () /* back to normal segment */ #endif int otherrichard8fp_per_pixel(void) { othermandelfp_per_pixel(); CMPLXtrig1(*floatparm,tmp); CMPLXmult(tmp,parm2,tmp); return(1); } int othermandelfp_per_pixel(void) { if(invert) invertz2(&init); else { init.x = dxpixel(); if(save_release >= 2004) init.y = dypixel(); } if(useinitorbit == 1) old = initorbit; else old = init; old.x += parm.x; /* initial pertubation of parameters set */ old.y += parm.y; return(1); /* 1st iteration has been done */ } #if (_MSC_VER >= 700) #pragma code_seg ("mpmath1_text") /* place following in an overlay */ #endif int MPCHalley_per_pixel(void) { #ifndef XFRACT /* MPC halley */ if(invert) invertz2(&init); else { init.x = dxpixel(); if(save_release >= 2004) init.y = dypixel(); } mpcold.x = *pd2MP(init.x); mpcold.y = *pd2MP(init.y); return(0); #else return(0); #endif } #if (_MSC_VER >= 700) #pragma code_seg () /* back to normal segment */ #endif int Halley_per_pixel(void) { if(invert) invertz2(&init); else { init.x = dxpixel(); if(save_release >= 2004) init.y = dypixel(); } old = init; return(0); /* 1st iteration is not done */ } int otherjuliafp_per_pixel(void) { if(invert) invertz2(&old); else { old.x = dxpixel(); old.y = dypixel(); } return(0); } #if 0 #define Q0 .113 #define Q1 .01 #else #define Q0 0 #define Q1 0 #endif int quaternionjulfp_per_pixel(void) { old.x = dxpixel(); old.y = dypixel(); floatparm->x = param[4]; floatparm->y = param[5]; qc = param[0]; qci = param[1]; qcj = param[2]; qck = param[3]; return(0); } int quaternionfp_per_pixel(void) { old.x = 0; old.y = 0; floatparm->x = 0; floatparm->y = 0; qc = dxpixel(); qci = dypixel(); qcj = param[2]; qck = param[3]; return(0); } int MarksCplxMandperp(void) { if(invert) invertz2(&init); else { init.x = dxpixel(); if(save_release >= 2004) init.y = dypixel(); } old.x = init.x + parm.x; /* initial pertubation of parameters set */ old.y = init.y + parm.y; tempsqrx = sqr(old.x); /* precalculated value */ tempsqry = sqr(old.y); coefficient = ComplexPower(init, pwr); return(1); } int long_phoenix_per_pixel(void) { #ifndef XFRACT if(invert) { /* invert */ invertz2(&old); /* watch out for overflow */ if(sqr(old.x)+sqr(old.y) >= 127) { old.x = 8; /* value to bail out in one iteration */ old.y = 8; } /* convert to fudged longs */ lold.x = (long)(old.x*fudge); lold.y = (long)(old.y*fudge); } else { lold.x = lxpixel(); lold.y = lypixel(); } ltempsqrx = multiply(lold.x, lold.x, bitshift); ltempsqry = multiply(lold.y, lold.y, bitshift); ltmp2.x = 0; /* use ltmp2 as the complex Y value */ ltmp2.y = 0; return(0); #else printf("Called long_phoenix_per_pixel\n"); return(0); #endif } int phoenix_per_pixel(void) { if(invert) invertz2(&old); else { old.x = dxpixel(); old.y = dypixel(); } tempsqrx = sqr(old.x); /* precalculated value */ tempsqry = sqr(old.y); tmp2.x = 0; /* use tmp2 as the complex Y value */ tmp2.y = 0; return(0); } int long_mandphoenix_per_pixel(void) { #ifndef XFRACT linit.x = lxpixel(); if(save_release >= 2004) linit.y = lypixel(); if(invert) { /* invert */ invertz2(&init); /* watch out for overflow */ if(sqr(init.x)+sqr(init.y) >= 127) { init.x = 8; /* value to bail out in one iteration */ init.y = 8; } /* convert to fudged longs */ linit.x = (long)(init.x*fudge); linit.y = (long)(init.y*fudge); } if(useinitorbit == 1) lold = linitorbit; else lold = linit; lold.x += lparm.x; /* initial pertubation of parameters set */ lold.y += lparm.y; ltempsqrx = multiply(lold.x, lold.x, bitshift); ltempsqry = multiply(lold.y, lold.y, bitshift); ltmp2.x = 0; ltmp2.y = 0; return(1); /* 1st iteration has been done */ #else printf("Called long_mandphoenix_per_pixel\n"); return(0); #endif } int mandphoenix_per_pixel(void) { if(invert) invertz2(&init); else { init.x = dxpixel(); if(save_release >= 2004) init.y = dypixel(); } if(useinitorbit == 1) old = initorbit; else old = init; old.x += parm.x; /* initial pertubation of parameters set */ old.y += parm.y; tempsqrx = sqr(old.x); /* precalculated value */ tempsqry = sqr(old.y); tmp2.x = 0; tmp2.y = 0; return(1); /* 1st iteration has been done */ } int QuaternionFPFractal(void) { double a0,a1,a2,a3,n0,n1,n2,n3; a0 = old.x; a1 = old.y; a2 = floatparm->x; a3 = floatparm->y; n0 = a0*a0-a1*a1-a2*a2-a3*a3 + qc; n1 = 2*a0*a1 + qci; n2 = 2*a0*a2 + qcj; n3 = 2*a0*a3 + qck; /* Check bailout */ magnitude = a0*a0+a1*a1+a2*a2+a3*a3; if (magnitude>rqlim) { return 1; } old.x = new.x = n0; old.y = new.y = n1; floatparm->x = n2; floatparm->y = n3; return(0); } int HyperComplexFPFractal(void) { _HCMPLX hold, hnew; hold.x = old.x; hold.y = old.y; hold.z = floatparm->x; hold.t = floatparm->y; /* HComplexSqr(&hold,&hnew); */ HComplexTrig0(&hold,&hnew); hnew.x += qc; hnew.y += qci; hnew.z += qcj; hnew.t += qck; old.x = new.x = hnew.x; old.y = new.y = hnew.y; floatparm->x = hnew.z; floatparm->y = hnew.t; /* Check bailout */ magnitude = sqr(old.x)+sqr(old.y)+sqr(floatparm->x)+sqr(floatparm->y); if (magnitude>rqlim) { return 1; } return(0); } int VLfpFractal(void) /* Beauty of Fractals pp. 125 - 127 */ { double a, b, ab, half, u, w, xy; half = param[0] / 2.0; xy = old.x * old.y; u = old.x - xy; w = -old.y + xy; a = old.x + param[1] * u; b = old.y + param[1] * w; ab = a * b; new.x = old.x + half * (u + (a - ab)); new.y = old.y + half * (w + (-b + ab)); return(floatbailout()); } int EscherfpFractal(void) /* Science of Fractal Images pp. 185, 187 */ { _CMPLX oldtest, newtest, testsqr; double testsize = 0.0; long testiter = 0; new.x = tempsqrx - tempsqry; /* standard Julia with C == (0.0, 0.0i) */ new.y = 2.0 * old.x * old.y; oldtest.x = new.x * 15.0; /* scale it */ oldtest.y = new.y * 15.0; testsqr.x = sqr(oldtest.x); /* set up to test with user-specified ... */ testsqr.y = sqr(oldtest.y); /* ... Julia as the target set */ while (testsize <= rqlim && testiter < maxit) /* nested Julia loop */ { newtest.x = testsqr.x - testsqr.y + param[0]; newtest.y = 2.0 * oldtest.x * oldtest.y + param[1]; testsize = (testsqr.x = sqr(newtest.x)) + (testsqr.y = sqr(newtest.y)); oldtest = newtest; testiter++; } if (testsize > rqlim) return(floatbailout()); /* point not in target set */ else /* make distinct level sets if point stayed in target set */ { coloriter = ((3L * coloriter) % 255L) + 1L; return 1; } } /* re-use static roots variable memory for mandelmix4 */ #define A staticroots[ 0] #define B staticroots[ 1] #define C staticroots[ 2] #define D staticroots[ 3] #define F staticroots[ 4] #define G staticroots[ 5] #define H staticroots[ 6] #define J staticroots[ 7] #define K staticroots[ 8] #define L staticroots[ 9] #define Z staticroots[10] int MandelbrotMix4Setup(void) { int sign_array = 0; A.x=param[0]; A.y=0.0; /* a=real(p1), */ B.x=param[1]; B.y=0.0; /* b=imag(p1), */ D.x=param[2]; D.y=0.0; /* d=real(p2), */ F.x=param[3]; F.y=0.0; /* f=imag(p2), */ K.x=param[4]+1.0; K.y=0.0; /* k=real(p3)+1, */ L.x=param[5]+100.0; L.y=0.0; /* l=imag(p3)+100, */ CMPLXrecip(F,G); /* g=1/f, */ CMPLXrecip(D,H); /* h=1/d, */ CMPLXsub(F,B,tmp); /* tmp = f-b */ CMPLXrecip(tmp,J); /* j = 1/(f-b) */ CMPLXneg(A,tmp); CMPLXmult(tmp,B,tmp); /* z=(-a*b*g*h)^j, */ CMPLXmult(tmp,G,tmp); CMPLXmult(tmp,H,tmp); /* This code kludge attempts to duplicate the behavior of the parser in determining the sign of zero of the imaginary part of the argument of the power function. The reason this is important is that the complex arctangent returns PI in one case and -PI in the other, depending on the sign bit of zero, and we wish the results to be compatible with Jim Muth's mix4 formula using the parser. First create a number encoding the signs of a, b, g , h. Our kludge assumes that those signs determine the behavior. */ if(A.x < 0.0) sign_array += 8; if(B.x < 0.0) sign_array += 4; if(G.x < 0.0) sign_array += 2; if(H.x < 0.0) sign_array += 1; if(tmp.y == 0.0) /* we know tmp.y IS zero but ... */ { switch(sign_array) { /* Add to this list the magic numbers of any cases in which the fractal does not match the formula version */ case 15: /* 1111 */ case 10: /* 1010 */ case 6: /* 0110 */ case 5: /* 0101 */ case 3: /* 0011 */ case 0: /* 0000 */ tmp.y = -tmp.y; /* swap sign bit */ default: /* do nothing - remaining cases already OK */ ; } /* in case our kludge failed, let the user fix it */ if(debugflag == 1012) tmp.y = -tmp.y; } CMPLXpwr(tmp,J,tmp); /* note: z is old */ /* in case our kludge failed, let the user fix it */ if(param[6] < 0.0) tmp.y = -tmp.y; if(bailout == 0) { rqlim = L.x; rqlim2 = rqlim*rqlim; } return(1); } int MandelbrotMix4fp_per_pixel(void) { if(invert) invertz2(&init); else { init.x = dxpixel(); init.y = dypixel(); } old = tmp; CMPLXtrig0(init,C); /* c=fn1(pixel): */ return(0); /* 1st iteration has been NOT been done */ } int MandelbrotMix4fpFractal(void) /* from formula by Jim Muth */ { /* z=k*((a*(z^b))+(d*(z^f)))+c, */ _CMPLX z_b, z_f; CMPLXpwr(old,B,z_b); /* (z^b) */ CMPLXpwr(old,F,z_f); /* (z^f) */ new.x = K.x*A.x*z_b.x + K.x*D.x*z_f.x + C.x; new.y = K.x*A.x*z_b.y + K.x*D.x*z_f.y + C.y; return(floatbailout()); } #undef A #undef B #undef C #undef D #undef F #undef G #undef H #undef J #undef K #undef L /* * The following functions calculate the real and imaginary complex * coordinates of the point in the complex plane corresponding to * the screen coordinates (col,row) at the current zoom corners * settings. The functions come in two flavors. One looks up the pixel * values using the precalculated grid arrays dx0, dx1, dy0, and dy1, * which has a speed advantage but is limited to MAXPIXELS image * dimensions. The other calculates the complex coordinates at a * cost of two additions and two multiplications for each component, * but works at any resolution. * * With Microsoft C's _fastcall keyword, the function call overhead * appears to be negligible. It also appears that the speed advantage * of the lookup vs the calculation is negligible on machines with * coprocessors. Bert Tyler's original implementation was designed for * machines with no coprocessor; on those machines the saving was * significant. For the time being, the table lookup capability will * be maintained. */ /* Real component, grid lookup version - requires dx0/dx1 arrays */ static double _fastcall dxpixel_grid(void) { return(dx0[col]+dx1[row]); } /* Real component, calculation version - does not require arrays */ static double _fastcall dxpixel_calc(void) { return((double)(xxmin + col*delxx + row*delxx2)); } /* Imaginary component, grid lookup version - requires dy0/dy1 arrays */ static double _fastcall dypixel_grid(void) { return(dy0[row]+dy1[col]); } /* Imaginary component, calculation version - does not require arrays */ static double _fastcall dypixel_calc(void) { return((double)(yymax - row*delyy - col*delyy2)); } /* Real component, grid lookup version - requires lx0/lx1 arrays */ static long _fastcall lxpixel_grid(void) { return(lx0[col]+lx1[row]); } /* Real component, calculation version - does not require arrays */ static long _fastcall lxpixel_calc(void) { return(xmin + col*delx + row*delx2); } /* Imaginary component, grid lookup version - requires ly0/ly1 arrays */ static long _fastcall lypixel_grid(void) { return(ly0[row]+ly1[col]); } /* Imaginary component, calculation version - does not require arrays */ static long _fastcall lypixel_calc(void) { return(ymax - row*dely - col*dely2); } double (_fastcall *dxpixel)(void) = dxpixel_calc; double (_fastcall *dypixel)(void) = dypixel_calc; long (_fastcall *lxpixel)(void) = lxpixel_calc; long (_fastcall *lypixel)(void) = lypixel_calc; void set_pixel_calc_functions(void) { if(use_grid) { dxpixel = dxpixel_grid; dypixel = dypixel_grid; lxpixel = lxpixel_grid; lypixel = lypixel_grid; } else { dxpixel = dxpixel_calc; dypixel = dypixel_calc; lxpixel = lxpixel_calc; lypixel = lypixel_calc; } }