home *** CD-ROM | disk | FTP | other *** search
/ Shareware Overload / ShartewareOverload.cdr / windows / winsrc.zip / CALCFRAC.C < prev    next >
C/C++ Source or Header  |  1990-11-19  |  81KB  |  2,834 lines

  1. /*
  2. CALCFRAC.C contains the high level ("engine") code for calculating the
  3. fractal images (well, SOMEBODY had to do it!).
  4. Original author Tim Wegner, but just about ALL the authors have contributed
  5. SOME code to this routine at one time or another, or contributed to one of
  6. the many massive restructurings.
  7. This module is linked as an overlay, use ENTER_OVLY and EXIT_OVLY.
  8. The following modules work very closely with CALCFRAC.C:
  9.   FRACTALS.C    the fractal-specific code for escape-time fractals.
  10.   FRACSUBR.C    assorted subroutines belonging mainly to calcfrac.
  11.   CALCMAND.ASM    fast Mandelbrot/Julia integer implementation
  12. Additional fractal-specific modules are also invoked from CALCFRAC:
  13.   LORENZ.C    engine level and fractal specific code for attractors.
  14.   JB.C        julibrot logic
  15.   PARSER.C    formula fractals
  16.   and more
  17.  -------------------------------------------------------------------- */
  18.  
  19. #include <stdio.h>
  20. #include <string.h>
  21. #include <stdlib.h>
  22. #include <float.h>
  23. #include <dos.h>
  24. #include <limits.h>
  25. #include "fractint.h"
  26. #include "fractype.h"
  27. #include "mpmath.h"
  28. #include "targa_lc.h"
  29.  
  30. /* routines in this module    */
  31.  
  32. void calcfrac_overlay(void);
  33. int  calcfract(void);
  34. /* the rest are called only within same overlay, thus don't need ENTER_OVLY */
  35. int  StandardFractal(void);
  36. int  calcmand(void);
  37. int  plasma(void);
  38. int  diffusion(void);
  39. int  test(void);
  40. int  Bifurcation(void);
  41. int  BifurcVerhulst(void),LongBifurcVerhulst(void),BifurcLambda(void);
  42. int  LongBifurcLambda(void),BifurcAddSinPi(void),BifurcSetSinPi(void);
  43. int  popcorn(void);
  44.  
  45. static void perform_worklist(void);
  46. static int  OneOrTwoPass(void);
  47. static int  StandardCalc(int);
  48. static int  potential(double,int);
  49. static void decomposition(void);
  50. static int  bound_trace_main(void);
  51. static int  boundary_trace(int,int);
  52. static int  calc_xy(int,int);
  53. static int  fillseg1(int,int,int,int);
  54. static int  fillseg(int,int,int,int);
  55. static void reverse_string(char *,char *,int);
  56. static int  solidguess(void);
  57. static int  guessrow(int,int,int);
  58. static void plotblock(int,int,int,int);
  59. static void setsymmetry(int,int);
  60. static int  xsym_split(int,int);
  61. static int  ysym_split(int,int);
  62. static void set_Plasma_palette();
  63. static void adjust();
  64. static void subDivide();
  65. static void verhulst(void);
  66. static void Bif_Period_Init(void);
  67. static int  Bif_Periodic(int);
  68.  
  69.  
  70. extern struct complex initorbit;
  71. extern char useinitorbit;
  72. struct lcomplex linitorbit;
  73.  
  74. extern unsigned int decoderline[];
  75. extern int overflow;
  76. long lmagnitud, llimit, llimit2, lclosenuff, l16triglim;
  77. struct complex init,tmp,old,new,saved;
  78. extern int biomorph;
  79. extern struct lcomplex linit;
  80. extern int basin;
  81. extern int cpu;
  82. extern char savename[80];   /* save files using this name */
  83. extern int resave_flag;
  84. extern int dotmode;
  85.  
  86. int color, oldcolor, row, col, passes;
  87. int realcolor;
  88. int iterations, invert;
  89. double f_radius,f_xcenter, f_ycenter; /* for inversion */
  90. extern double far *dx0, far *dy0;
  91. extern double far *dx1, far *dy1;
  92.  
  93. extern int LogFlag;
  94. extern unsigned char far *LogTable;
  95.  
  96. void (*plot)() = putcolor;
  97.  
  98. extern double inversion[];        /* inversion radius, f_xcenter, f_ycenter */
  99. extern int    xdots, ydots;        /* coordinates of dots on the screen  */
  100. extern int    sxdots,sydots;
  101. extern int    sxoffs,syoffs;
  102. extern int    colors;         /* maximum colors available */
  103. extern int    andcolor;        /* colors-1         */
  104. extern int    inside;         /* "inside" color to use    */
  105. extern int    outside;        /* "outside" color to use   */
  106. extern int    finattract;
  107. double        min_orbit;        /* orbit value closest to origin */
  108. int        min_index;        /* iteration of min_orbit */
  109. extern int    maxit;            /* try this many iterations */
  110. extern int    fractype;        /* fractal type */
  111. extern char    stdcalcmode;        /* '1', '2', 'g', 'b' */
  112. extern int    debugflag;        /* for debugging purposes */
  113. extern    int    diskvideo;        /* for disk-video klooges */
  114. extern int    calc_status;        /* status of calculations */
  115. extern long    calctime;        /* total calc time for image */
  116.  
  117. extern int rflag, rseed;
  118. extern int decomp[];
  119. extern int distest;
  120.  
  121. extern double    param[];        /* parameters */
  122. extern int    potflag;        /* potential enabled? */
  123. extern double    potparam[];        /* potential parameters */
  124. extern int    pot16bit;        /* store 16bit continuous potential */
  125. extern long    far *lx0, far *ly0; /* X, Y points */
  126. extern long    far *lx1, far *ly1; /* X, Y points */
  127. extern long    fudge;            /* fudge factor (2**n) */
  128. extern int    bitshift;        /* bit shift for fudge */
  129. extern long    delmin;         /* for periodicity checking */
  130.  
  131. extern double xxmin,xxmax,yymin,yymax,xx3rd,yy3rd; /* corners */
  132. extern long   xmin, xmax, ymin, ymax;           /* integer equivs */
  133. extern long   delx,dely;               /* X, Y increments */
  134. extern double delxx,delxx2,delyy,delyy2;
  135. double deltaX, deltaY;
  136. double magnitude, rqlim, rqlim2;
  137. extern struct complex parm,parm2;
  138. int (*calctype)();
  139. double closenuff;
  140. int pixelpi; /* value of pi in pixels */
  141. unsigned long lm;        /* magnitude limit (CALCMAND) */
  142. extern long linitx,linity;    /* in calcmand */
  143. extern unsigned long savedmask; /* in calcmand */
  144.  
  145. /* ORBIT variables */
  146. int    show_orbit;            /* flag to turn on and off */
  147. int    orbit_ptr;            /* pointer into save_orbit array */
  148. int far *save_orbit;            /* array to save orbit values */
  149. int    orbit_color=15;         /* XOR color */
  150.  
  151. int    ixstart, ixstop, iystart, iystop;    /* start, stop here */
  152. int    symmetry;       /* symmetry flag */
  153. int    reset_periodicity; /* nonzero if escape time pixel rtn to reset */
  154. int    kbdcount;       /* avoids checking keyboard too often */
  155.  
  156. extern    int    integerfractal;     /* TRUE if fractal uses integer math */
  157.  
  158. char far *resume_info = NULL;        /* pointer to resume info if allocated */
  159. int resuming;                /* nonzero if resuming after interrupt */
  160. int num_worklist;            /* resume worklist for standard engine */
  161. struct workliststuff worklist[MAXCALCWORK];
  162. int xxstart,xxstop;            /* these are same as worklist, */
  163. int yystart,yystop,yybegin;        /* declared as separate items  */
  164. int workpass,worksym;            /* for the sake of calcmand    */
  165.  
  166. extern long timer_interval;        /* timer(...) total */
  167.  
  168. struct complex far *dem_orbit = NULL;    /* temp used with distance estimator */
  169. static int dem_iter;            /* number of entries in dem_orbit */
  170. double dem_delta, dem_width;
  171. #define DEMOVERFLOW 100000000000000.0
  172.  
  173. /* variables which must be visible for tab_display */
  174. int got_status; /* -1 if not, 0 for 1or2pass, 1 for ssg, 2 for btm */
  175. int curpass,totpasses;
  176. int currow,curcol;
  177.  
  178. /* static vars for solidguess & its subroutines */
  179. static int maxblock,halfblock;
  180. static int guessplot;            /* paint 1st pass row at a time?   */
  181. static int right_guess,bottom_guess;
  182. #define maxyblk 7    /* maxxblk*maxyblk*2 <= 4096, the size of "prefix" */
  183. #define maxxblk 202  /* each maxnblk is oversize by 2 for a "border" */
  184.              /* maxxblk defn must match fracsubr.c */
  185. /* next has a skip bit for each maxblock unit;
  186.    1st pass sets bit  [1]... off only if block's contents guessed;
  187.    at end of 1st pass [0]... bits are set if any surrounding block not guessed;
  188.    bits are numbered [..][y/16+1][x+1]&(1<<(y&15)) */
  189. extern unsigned int prefix[2][maxyblk][maxxblk]; /* common temp */
  190. /* size of next puts a limit of 2048 pixels across on solid guessing logic */
  191. extern char dstack[4096];        /* common temp, two put_line calls */
  192.  
  193. int    attractors;            /* number of finite attractors  */
  194. struct complex    attr[N_ATTR];        /* finite attractor vals (f.p)  */
  195. struct lcomplex lattr[N_ATTR];        /* finite attractor vals (int)  */
  196.  
  197. extern void symPIplot(int,int,int);
  198. extern void symPIplot2J(int,int,int);
  199. extern void symPIplot4J(int,int,int);
  200. extern void symplot2(int,int,int);
  201. extern void symplot2Y(int,int,int);
  202. extern void symplot2J(int,int,int);
  203. extern void symplot4(int,int,int);
  204. extern void symplot2basin(int,int,int);
  205. extern void symplot4basin(int,int,int);
  206. extern void noplot(int,int,int);
  207.  
  208. #ifndef sqr
  209. #define sqr(x) ((x)*(x))
  210. #endif
  211.  
  212. #ifndef lsqr
  213. #define lsqr(x) (multiply((x),(x),bitshift))
  214. #endif
  215.  
  216. /* -------------------------------------------------------------------- */
  217. /*        These variables are external for speed's sake only      */
  218. /* -------------------------------------------------------------------- */
  219. extern struct lcomplex lold,lnew,lparm,lparm2;     /* added "lold" */
  220.  
  221. int periodicitycheck;
  222. extern long ltempsqrx,ltempsqry;
  223. extern double tempsqrx,tempsqry;
  224. extern LCMPLX ltmp;
  225. extern int display3d;
  226.  
  227.  
  228. void calcfrac_overlay() { }    /* for restore_active_ovly */
  229.  
  230.  
  231. /******* calcfract - the top level routine for generating an image *******/
  232.  
  233. int calcfract()
  234. {
  235.    int i;
  236.  
  237.    ENTER_OVLY(OVLY_CALCFRAC);
  238.  
  239.    attractors = 0;        /* default to no known finite attractors  */
  240.    display3d = 0;
  241.    basin = 0;
  242.  
  243.    init_misc();  /* set up some variables in parser.c */
  244.  
  245.    /* following delta values useful only for types with rotation disabled */
  246.    /* currently used only by bifurcation */
  247.    if (integerfractal)
  248.    {
  249.       distest = 0;
  250.       deltaX = (double)lx0[     1] / fudge - xxmin;
  251.       deltaY = yymax - (double)ly0[     1] / fudge;
  252.    }
  253.    else
  254.    {
  255.       deltaX = dx0[     1] - xxmin;
  256.       deltaY = yymax - dy0[     1];
  257.    }
  258.  
  259.    parm.x   = param[0];
  260.    parm.y   = param[1];
  261.    parm2.x  = param[2];
  262.    parm2.y  = param[3];
  263.  
  264.    if (LogFlag)
  265.       SetupLogTable();
  266.  
  267.    lm = 4L << bitshift;         /* CALCMAND magnitude limit */
  268.  
  269.    /* ORBIT stuff */
  270.    save_orbit = (int far *)((double huge *)dx0 + 4*MAXPIXELS);
  271.    show_orbit = 0;
  272.    orbit_ptr = 0;
  273.    orbit_color = 15;
  274.    if(colors < 16)
  275.       orbit_color = 1;
  276.  
  277.    if(inversion[0] != 0.0)
  278.    {
  279.       f_radius      = inversion[0];
  280.       f_xcenter   = inversion[1];
  281.       f_ycenter   = inversion[2];
  282.  
  283.       if (inversion[0] == AUTOINVERT)  /*  auto calc radius 1/6 screen */
  284.      inversion[0] = f_radius = min(fabs(xxmax - xxmin),
  285.          fabs(yymax - yymin)) / 6.0;
  286.  
  287.       if (invert < 2 || inversion[1] == AUTOINVERT)  /* xcenter not already set */
  288.       {
  289.      inversion[1] = f_xcenter = (xxmin + xxmax) / 2.0;
  290.      if (fabs(f_xcenter) < fabs(xxmax-xxmin) / 100)
  291.         inversion[1] = f_xcenter = 0.0;
  292.       }
  293.  
  294.       if (invert < 3 || inversion[2] == AUTOINVERT)  /* ycenter not already set */
  295.       {
  296.      inversion[2] = f_ycenter = (yymin + yymax) / 2.0;
  297.      if (fabs(f_ycenter) < fabs(yymax-yymin) / 100)
  298.         inversion[2] = f_ycenter = 0.0;
  299.       }
  300.  
  301.       invert = 3; /* so values will not be changed if we come back */
  302.    }
  303.  
  304.    closenuff = delmin >> abs(periodicitycheck); /* for periodicity checking */
  305.    closenuff /= fudge;
  306.    rqlim2 = sqrt(rqlim);
  307.    if (integerfractal)        /* for integer routines (lambda) */
  308.    {
  309.       lparm.x = parm.x * fudge;    /* real portion of Lambda */
  310.       lparm.y = parm.y * fudge;    /* imaginary portion of Lambda */
  311.       lparm2.x = parm2.x * fudge;  /* real portion of Lambda2 */
  312.       lparm2.y = parm2.y * fudge;  /* imaginary portion of Lambda2 */
  313.       llimit = rqlim * fudge;       /* stop if magnitude exceeds this */
  314.       if (llimit <= 0) llimit = 0x7fffffff; /* klooge for integer math */
  315.       llimit2 = rqlim2 * fudge;    /* stop if magnitude exceeds this */
  316.       lclosenuff = closenuff * fudge;    /* "close enough" value */
  317.       l16triglim = 8L<<16;       /* domain limit of fast trig functions */
  318.       linitorbit.x = initorbit.x * fudge;
  319.       linitorbit.y = initorbit.y * fudge;
  320.    }
  321.    resuming = (calc_status == 2);
  322.    if (!resuming) /* free resume_info memory if any is hanging around */
  323.    {
  324.       end_resume();
  325.       if (resave_flag) {
  326.      updatesavename(savename); /* do the pending increment */
  327.      resave_flag = 0;
  328.      }
  329.       calctime = 0;
  330.    }
  331.  
  332.    if (fractalspecific[fractype].calctype != StandardFractal
  333.        && fractalspecific[fractype].calctype != calcmand)
  334.    {
  335.       calctype = fractalspecific[fractype].calctype; /* per_image can override */
  336.       symmetry = fractalspecific[fractype].symmetry; /*   calctype & symmetry  */
  337.       plot = putcolor; /* defaults when setsymmetry not called or does nothing */
  338.       iystart = ixstart = yystart = xxstart = yybegin = 0;
  339.       iystop = yystop = ydots -1;
  340.       ixstop = xxstop = xdots -1;
  341.       calc_status = 1; /* mark as in-progress */
  342.       distest = 0; /* only standard escape time engine supports distest */
  343.       /* per_image routine is run here */
  344.       if (fractalspecific[fractype].per_image())
  345.       { /* not a stand-alone */
  346.      /* next two lines in case periodicity changed */
  347.      closenuff = delmin >> abs(periodicitycheck); /* for periodicity checking */
  348.      closenuff /= fudge;
  349.      lclosenuff = closenuff * fudge;    /* "close enough" value */
  350.      setsymmetry(symmetry,0);
  351.      timer(0,calctype); /* non-standard fractal engine */
  352.       }
  353.       if (check_key())
  354.       {
  355.      if (calc_status == 1) /* calctype didn't set this itself, */
  356.         calc_status = 3;   /* so mark it interrupted, non-resumable */
  357.       }
  358.       else
  359.      calc_status = 4; /* no key, so assume it completed */
  360.    }
  361.    else /* standard escape-time engine */
  362.       timer(0,(int (*)())perform_worklist);
  363.    calctime += timer_interval;
  364.  
  365.    if(LogTable)
  366.    {
  367.       farmemfree(LogTable);
  368.       LogTable = (char far *)0;
  369.    }
  370.  
  371.    EXIT_OVLY;
  372.    return((calc_status == 4) ? 0 : -1);
  373. }
  374.  
  375.  
  376. /**************** general escape-time engine routines *********************/
  377.  
  378. static void perform_worklist()
  379. {
  380.    int i;
  381.    long tmplong; /* this temp must be signed */
  382.  
  383.    if (potflag && pot16bit)
  384.    {
  385.       stdcalcmode = '1'; /* force 1 pass */
  386.       if (resuming == 0)
  387.      pot_startdisk();
  388.    }
  389.    if (stdcalcmode == 'b' && (fractalspecific[fractype].flags & NOTRACE))
  390.       stdcalcmode = '1';
  391.    if (stdcalcmode == 'g' && (fractalspecific[fractype].flags & NOGUESS))
  392.       stdcalcmode = '1';
  393.  
  394.    /* default setup a new worklist */
  395.    num_worklist = 1;
  396.    worklist[0].xxstart = 0;
  397.    worklist[0].yystart = worklist[0].yybegin = 0;
  398.    worklist[0].xxstop = xdots - 1;
  399.    worklist[0].yystop = ydots - 1;
  400.    worklist[0].pass = worklist[0].sym = 0;
  401.    if (resuming) /* restore worklist, if we can't the above will stay in place */
  402.    {
  403.       start_resume();
  404.       get_resume(sizeof(int),&num_worklist,sizeof(worklist),worklist,0);
  405.       end_resume();
  406.    }
  407.  
  408.    if (distest) /* setup stuff for distance estimator */
  409.    {
  410.       dem_delta = ( sqrt( sqr(delxx) + sqr(delxx2) )  /* half a pixel width */
  411.       + sqrt( sqr(delyy) + sqr(delyy2) ) ) / 4;
  412.       dem_width = ( sqrt( sqr(xxmax-xxmin) + sqr(xx3rd-xxmin) ) * ydots/xdots
  413.       + sqrt( sqr(yymax-yymin) + sqr(yy3rd-yymin) ) ) / distest;
  414.       dem_orbit = (struct complex far *)
  415.      farmemalloc(((long)(maxit+1)) * sizeof(*dem_orbit));
  416.       if (dem_orbit == NULL)
  417.       {
  418.      stopmsg(0,"\
  419. insufficient memory for distance estimator option.\n\
  420. Try reducing maximum iterations.");
  421.      calc_status = 0;
  422.      return;
  423.       }
  424.    }
  425.  
  426.    while (num_worklist > 0)
  427.    {
  428.       calctype = fractalspecific[fractype].calctype; /* per_image can override */
  429.       symmetry = fractalspecific[fractype].symmetry; /*   calctype & symmetry  */
  430.       plot = putcolor; /* defaults when setsymmetry not called or does nothing */
  431.  
  432.       /* pull top entry off worklist */
  433.       ixstart = xxstart = worklist[0].xxstart;
  434.       ixstop  = xxstop    = worklist[0].xxstop;
  435.       iystart = yystart = worklist[0].yystart;
  436.       iystop  = yystop    = worklist[0].yystop;
  437.       yybegin  = worklist[0].yybegin;
  438.       workpass = worklist[0].pass;
  439.       worksym  = worklist[0].sym;
  440.       --num_worklist;
  441.       for (i=0; i<num_worklist; ++i)
  442.      worklist[i] = worklist[i+1];
  443.  
  444.       calc_status = 1; /* mark as in-progress */
  445.  
  446.       fractalspecific[fractype].per_image();
  447.  
  448.       /* some common initialization for escape-time pixel level routines */
  449.       closenuff = delmin >> abs(periodicitycheck); /* for periodicity checking */
  450.       closenuff /= fudge;
  451.       lclosenuff = closenuff * fudge;    /* "close enough" value */
  452.       kbdcount=(cpu==386) ? 80 : 30;
  453.       /* savedmask is for calcmand's periodicity checking */
  454.       savedmask = 0xC0000000; /* top 2 bits on */
  455.       tmplong = (delmin >> abs(periodicitycheck)) | 1;
  456.       while (tmplong > 0) /* while top bit not on */
  457.       {
  458.      tmplong <<= 1;
  459.      savedmask = (savedmask >> 1) | 0x80000000;
  460.       }
  461.  
  462.       setsymmetry(symmetry,1);
  463.  
  464.       /* call the appropriate escape-time engine */
  465.       switch (stdcalcmode)
  466.       {
  467.      case 'b':
  468.         bound_trace_main();
  469.         break;
  470.      case 'g':
  471.         solidguess();
  472.         break;
  473.      default:
  474.         OneOrTwoPass();
  475.       }
  476.  
  477.       if (check_key()) /* interrupted? */
  478.      break;
  479.    }
  480.  
  481.    if (dem_orbit != NULL) /* release distance estimator work area */
  482.    {
  483.       farmemfree((unsigned char far *)dem_orbit);
  484.       dem_orbit = NULL;
  485.    }
  486.  
  487.    if (num_worklist > 0)
  488.    {  /* interrupted, resumable */
  489.       alloc_resume(sizeof(worklist)+10,1);
  490.       put_resume(sizeof(int),&num_worklist,sizeof(worklist),worklist,0);
  491.    }
  492.    else
  493.       calc_status = 4; /* completed */
  494. }
  495.  
  496. static int OneOrTwoPass()
  497. {
  498.    int i;
  499.    totpasses = 1;
  500.    if (stdcalcmode == '2') totpasses = 2;
  501.    if (stdcalcmode == '2' && workpass == 0) /* do 1st pass of two */
  502.    {
  503.       if (StandardCalc(1) == -1)
  504.       {
  505.      add_worklist(xxstart,xxstop,yystart,yystop,row,0,worksym);
  506.      return(-1);
  507.       }
  508.       if (num_worklist > 0) /* worklist not empty, defer 2nd pass */
  509.       {
  510.      add_worklist(xxstart,xxstop,yystart,yystop,yystart,1,worksym);
  511.      return(0);
  512.       }
  513.       workpass = 1;
  514.       yybegin = yystart;
  515.    }
  516.    /* second or only pass */
  517.    if (StandardCalc(2) == -1)
  518.    {
  519.       i = yystop;
  520.       if (iystop != yystop) /* must be due to symmetry */
  521.      i -= row - iystart;
  522.       add_worklist(xxstart,xxstop,row,i,row,workpass,worksym);
  523.       return(-1);
  524.    }
  525.    return(0);
  526. }
  527.  
  528. static int StandardCalc(int passnum)
  529. {
  530.    got_status = 0;
  531.    curpass = passnum;
  532.    row = yybegin;
  533.    while (row <= iystop)
  534.    {
  535.       currow = row;
  536.       reset_periodicity = 1;
  537.       col = ixstart;
  538.       while (col <= ixstop)
  539.       {
  540.      /* on 2nd pass of two, skip even pts */
  541.      if (passnum == 1 || stdcalcmode == '1' || (row&1) != 0 || (col&1) != 0)
  542.      {
  543.         if ((*calctype)() == -1) /* StandardFractal() or calcmand() */
  544.            return(-1); /* interrupted */
  545.         reset_periodicity = 0;
  546.         if (passnum == 1) /* first pass, copy pixel and bump col */
  547.         {
  548.            if ((row&1) == 0 && row < iystop)
  549.            {
  550.           (*plot)(col,row+1,color);
  551.           if ((col&1) == 0 && col < ixstop)
  552.              (*plot)(col+1,row+1,color);
  553.            }
  554.            if ((col&1) == 0 && col < ixstop)
  555.           (*plot)(++col,row,color);
  556.         }
  557.      }
  558.      ++col;
  559.       }
  560.       if (passnum == 1 && (row&1) == 0)
  561.      ++row;
  562.       ++row;
  563.    }
  564.    return(0);
  565. }
  566.  
  567. int calcmand()        /* fast per pixel 1/2/b/g, called with row & col set */
  568. {
  569.    /* setup values from far array to avoid using es reg in calcmand.asm */
  570.    linitx = lx0[col] + lx1[row];
  571.    linity = ly0[row] + ly1[col];
  572.    if (calcmandasm() >= 0)
  573.    {
  574.       if (LogFlag /* use logpal, but not if maxit & adjusted for inside,etc */
  575.     && (realcolor < maxit || (inside < 0 && color == maxit)))
  576.      color = LogTable[color];
  577.       if (color >= colors) /* don't use color 0 unless from inside/outside */
  578.      if (colors < 16)
  579.         color &= andcolor;
  580.      else
  581.         color = ((color - 1) % andcolor) + 1;  /* skip color zero */
  582.       (*plot) (col, row, color);
  583.    }
  584.    return (color);
  585. }
  586.  
  587. int StandardFractal()    /* per pixel 1/2/b/g, called with row & col set */
  588. {
  589.    int caught_a_cycle;
  590.    int savedand, savedincr;    /* for periodicity checking */
  591.    struct lcomplex lsaved;
  592.    int i, attracted;
  593.  
  594.    if (periodicitycheck == 0)
  595.       oldcolor = 32767;     /* don't check periodicity at all */
  596.    else if (reset_periodicity)
  597.       oldcolor = 250;        /* don't check periodicity 1st 250 iterations */
  598.  
  599.    /* really fractal specific, but we'll leave it here */
  600.    if (!integerfractal)
  601.    {
  602.  
  603.       if (useinitorbit == 1)
  604.      saved = initorbit;
  605.       else {
  606.      saved.x = 0;
  607.      saved.y = 0;
  608.      }
  609.       init.y = dy0[row] + dy1[col];
  610.    }
  611.    else
  612.    {
  613.       if (useinitorbit == 1)
  614.      lsaved = linitorbit;
  615.       else {
  616.      lsaved.x = 0;
  617.      lsaved.y = 0;
  618.      }
  619.       linit.y = ly0[row] + ly1[col];
  620.    }
  621.    orbit_ptr = 0;
  622.    color = 0;
  623.    caught_a_cycle = 0;
  624.    savedand = 1;        /* begin checking every other cycle */
  625.    savedincr = 1;        /* start checking the very first time */
  626.  
  627.    if (inside <= -60 && inside >= -61)
  628.    {
  629.       magnitude = lmagnitud = 0;
  630.       min_orbit = 100000.0;
  631.    }
  632.    overflow = 0;        /* reset integer math overflow flag */
  633.    if (distest)
  634.    {
  635.       dem_iter = 0;
  636.       if (fractalspecific[fractype].per_pixel()) /* mandels do the 1st iter */
  637.       {
  638.      dem_orbit[0].x = dem_orbit[0].y = 0;
  639.      ++dem_iter;
  640.       }
  641.    }
  642.    else
  643.       fractalspecific[fractype].per_pixel(); /* initialize the calculations */
  644.    attracted = FALSE;
  645.    while (++color < maxit)
  646.    {
  647.       if (distest)
  648.      dem_orbit[dem_iter++] = old;
  649.       /* calculation of one orbit goes here */
  650.       /* input in "old" -- output in "new" */
  651.  
  652.       if (fractalspecific[fractype].orbitcalc())
  653.      break;
  654.  
  655.       if (inside <= -60 && inside >= -61)
  656.       {
  657.      if (integerfractal)
  658.      {
  659.         if (lmagnitud == 0)
  660.            lmagnitud = lsqr(lnew.x) + lsqr(lnew.y);
  661.         magnitude = lmagnitud;
  662.         magnitude = magnitude / fudge;
  663.      }
  664.      else
  665.         if (magnitude == 0.0)
  666.            magnitude = sqr(new.x) + sqr(new.y);
  667.      if (magnitude < min_orbit)
  668.      {
  669.         min_orbit = magnitude;
  670.         min_index = color + 1;
  671.      }
  672.       }
  673.  
  674.       if (show_orbit)
  675.      if (!integerfractal)
  676.         plot_orbit(new.x, new.y, -1);
  677.      else
  678.         iplot_orbit(lnew.x, lnew.y, -1);
  679.  
  680.       if (attractors > 0)    /* finite attractor in the list   */
  681.       {             /* NOTE: Integer code is UNTESTED */
  682.      if (integerfractal)
  683.      {
  684.         for (i = 0; i < attractors; i++)
  685.         {
  686.            if (labs(lnew.x - lattr[i].x) < lclosenuff)
  687.           if (labs(lnew.y - lattr[i].y) < lclosenuff)
  688.           {
  689.              attracted = TRUE;
  690.              break;
  691.           }
  692.         }
  693.      }
  694.      else
  695.      {
  696.         for (i = 0; i < attractors; i++)
  697.         {
  698.            if (fabs(new.x - attr[i].x) < closenuff)
  699.           if (fabs(new.y - attr[i].y) < closenuff)
  700.           {
  701.              attracted = TRUE;
  702.              break;
  703.           }
  704.         }
  705.      }
  706.      if (attracted)
  707.         break;        /* AHA! Eaten by an attractor */
  708.       }
  709.  
  710.       if (color > oldcolor)    /* check periodicity */
  711.       {
  712.      if ((color & savedand) == 0)         /* time to save a new value */
  713.      {
  714.         if (!integerfractal)
  715.            saved = new;  /* floating pt fractals */
  716.         else
  717.            lsaved = lnew;/* integer fractals */
  718.         if (--savedincr == 0)    /* time to lengthen the periodicity? */
  719.         {
  720.            savedand = (savedand << 1) + 1;         /* longer periodicity */
  721.            savedincr = 4;/* restart counter */
  722.         }
  723.      }
  724.      else             /* check against an old save */
  725.      {
  726.         if (!integerfractal)     /* floating-pt periodicity chk */
  727.         {
  728.            if (fabs(saved.x - new.x) < closenuff)
  729.           if (fabs(saved.y - new.y) < closenuff)
  730.           {
  731.              caught_a_cycle = 1;
  732.              color = maxit - 1;
  733.           }
  734.         }
  735.         else         /* integer periodicity check */
  736.         {
  737.            if (labs(lsaved.x - lnew.x) < lclosenuff)
  738.           if (labs(lsaved.y - lnew.y) < lclosenuff)
  739.           {
  740.              caught_a_cycle = 1;
  741.              color = maxit - 1;
  742.           }
  743.         }
  744.      }
  745.       }
  746.    }
  747.  
  748.    realcolor = color;        /* save this before we start adjusting it */
  749.    if (color >= maxit)
  750.    {
  751.       oldcolor = 0;        /* check periodicity immediately next time */
  752.       if (periodicitycheck < 0 && caught_a_cycle)
  753.      color = caught_a_cycle = 7; /* show periodicity */
  754.    }
  755.    else
  756.       oldcolor = color + 10;    /* check when past this + 10 next time */
  757.    if (show_orbit)
  758.       scrub_orbit();
  759.    if (color == 0)
  760.       color = 1;        /* needed to make same as calcmand */
  761.  
  762.    if (distest)
  763.    {
  764.       double dist,temp;
  765.       struct complex deriv;
  766.       if (color < maxit && caught_a_cycle == 0) /* appears to be outside */
  767.       {
  768.      /* Distance estimator for points near Mandelbrot set */
  769.      /* Original code by Phil Wilson, hacked around by PB */
  770.      /* Algorithms from Peitgen & Saupe, Science of Fractal Images, p.198 */
  771.      deriv.x = 1; /* preset and skip 1st orbit */
  772.      deriv.y = 0;
  773.      i = 0;
  774.      while (++i < dem_iter)
  775.      {
  776.         temp = 2 * (dem_orbit[i].x * deriv.x - dem_orbit[i].y * deriv.y) + 1;
  777.         deriv.y = 2 * (dem_orbit[i].y * deriv.x + dem_orbit[i].x * deriv.y);
  778.         deriv.x = temp;
  779.         if (fabs(deriv.x) > DEMOVERFLOW || fabs(deriv.y) > DEMOVERFLOW)
  780.            break;
  781.      }
  782.      temp = sqr(new.x) + sqr(new.y);
  783.      dist = log(temp) * sqrt(temp) / sqrt(sqr(deriv.x) + sqr(deriv.y));
  784.      if (dist < dem_delta)
  785.         color = inside;
  786.      else if (colors == 2)
  787.         color = !inside;
  788.      else
  789.         color = sqrt(dist / dem_width + 1);
  790.       }
  791.    }
  792.    else if (potflag)
  793.    {
  794.       if (integerfractal)    /* adjust integer fractals */
  795.       {
  796.      new.x = ((double)lnew.x) / fudge;
  797.      new.y = ((double)lnew.y) / fudge;
  798.       }
  799.       magnitude = sqr(new.x) + sqr(new.y);
  800.       color = potential(magnitude, color);
  801.    }
  802.    else if (decomp[0] > 0)
  803.       decomposition();
  804.    else if (biomorph != -1)
  805.    {
  806.       if (integerfractal)
  807.       {
  808.      if (labs(lnew.x) < llimit2 || labs(lnew.y) < llimit2)
  809.         color = biomorph;
  810.       }
  811.       else
  812.      if (fabs(new.x) < rqlim2 || fabs(new.y) < rqlim2)
  813.         color = biomorph;
  814.    }
  815.  
  816.    if ((kbdcount -= color) <= 0)
  817.    {
  818.       if (check_key())
  819.      return (-1);
  820.       kbdcount = (cpu == 386) ? 80 : 30;
  821.    }
  822.  
  823.    if (potflag == 0) /* don't adjust color returned by potential routine */
  824.    {
  825.       if (realcolor >= maxit) /* we're "inside" */
  826.       {
  827.      if (caught_a_cycle != 7) /* not showing periodicity */
  828.         if (inside >= 0)      /* set to specified color, ignore logpal */
  829.            color = inside;
  830.         else
  831.         {
  832.            if (inside == -60)
  833.           color = sqrt(min_orbit) * 75;
  834.            else if (inside == -61)
  835.           color = min_index;
  836.            else /* inside == -1 */
  837.           color = maxit;
  838.            if (LogFlag)
  839.           color = LogTable[color];
  840.         }
  841.       }
  842.       else /* not inside */
  843.       {
  844.      if (outside >= 0 && attracted == FALSE) /* merge escape-time stripes */
  845.         color = outside;
  846.      else if (LogFlag)
  847.         color = LogTable[color];
  848.       }
  849.       if (color >= colors) /* don't use color 0 unless from inside/outside */
  850.      if (colors < 16)
  851.         color &= andcolor;
  852.      else
  853.         color = ((color - 1) % andcolor) + 1;  /* skip color zero */
  854.    }
  855.  
  856.    (*plot) (col, row, color);
  857.    return (color);
  858. }
  859.  
  860.  
  861. /**************** standardfractal doodad subroutines *********************/
  862.  
  863. static void decomposition()
  864. {
  865.    static double cos45       = 0.70710678118654750; /* cos 45    degrees */
  866.    static double sin45       = 0.70710678118654750; /* sin 45    degrees */
  867.    static double cos22_5   = 0.92387953251128670; /* cos 22.5    degrees */
  868.    static double sin22_5   = 0.38268343236508980; /* sin 22.5    degrees */
  869.    static double cos11_25  = 0.98078528040323040; /* cos 11.25    degrees */
  870.    static double sin11_25  = 0.19509032201612820; /* sin 11.25    degrees */
  871.    static double cos5_625  = 0.99518472667219690; /* cos 5.625    degrees */
  872.    static double sin5_625  = 0.09801714032956060; /* sin 5.625    degrees */
  873.    static double tan22_5   = 0.41421356237309500; /* tan 22.5    degrees */
  874.    static double tan11_25  = 0.19891236737965800; /* tan 11.25    degrees */
  875.    static double tan5_625  = 0.09849140335716425; /* tan 5.625    degrees */
  876.    static double tan2_8125 = 0.04912684976946725; /* tan 2.8125 degrees */
  877.    static double tan1_4063 = 0.02454862210892544; /* tan 1.4063 degrees */
  878.    static long lcos45      ; /* cos 45      degrees */
  879.    static long lsin45      ; /* sin 45      degrees */
  880.    static long lcos22_5   ; /* cos 22.5   degrees */
  881.    static long lsin22_5   ; /* sin 22.5   degrees */
  882.    static long lcos11_25  ; /* cos 11.25  degrees */
  883.    static long lsin11_25  ; /* sin 11.25  degrees */
  884.    static long lcos5_625  ; /* cos 5.625  degrees */
  885.    static long lsin5_625  ; /* sin 5.625  degrees */
  886.    static long ltan22_5   ; /* tan 22.5   degrees */
  887.    static long ltan11_25  ; /* tan 11.25  degrees */
  888.    static long ltan5_625  ; /* tan 5.625  degrees */
  889.    static long ltan2_8125 ; /* tan 2.8125 degrees */
  890.    static long ltan1_4063 ; /* tan 1.4063 degrees */
  891.    static char start=1;
  892.    int temp = 0;
  893.    int i;
  894.    struct lcomplex lalt;
  895.    struct complex alt;
  896.    if(start & integerfractal)
  897.    {
  898.       start = 0;
  899.       lcos45     = cos45      *fudge;
  900.       lsin45     = sin45      *fudge;
  901.       lcos22_5     = cos22_5    *fudge;
  902.       lsin22_5     = sin22_5    *fudge;
  903.       lcos11_25  = cos11_25   *fudge;
  904.       lsin11_25  = sin11_25   *fudge;
  905.       lcos5_625  = cos5_625   *fudge;
  906.       lsin5_625  = sin5_625   *fudge;
  907.       ltan22_5     = tan22_5    *fudge;
  908.       ltan11_25  = tan11_25   *fudge;
  909.       ltan5_625  = tan5_625   *fudge;
  910.       ltan2_8125 = tan2_8125  *fudge;
  911.       ltan1_4063 = tan1_4063  *fudge;
  912.    }
  913.    color = 0;
  914.    if (integerfractal) /* the only case */
  915.    {
  916.       if (lnew.y < 0)
  917.       {
  918.      temp = 2;
  919.      lnew.y = -lnew.y;
  920.       }
  921.  
  922.       if (lnew.x < 0)
  923.       {
  924.      ++temp;
  925.      lnew.x = -lnew.x;
  926.       }
  927.  
  928.       if (decomp[0] >= 8)
  929.       {
  930.      temp <<= 1;
  931.      if (lnew.x < lnew.y)
  932.      {
  933.         ++temp;
  934.         lalt.x = lnew.x; /* just */
  935.         lnew.x = lnew.y; /* swap */
  936.         lnew.y = lalt.x; /* them */
  937.      }
  938.  
  939.      if (decomp[0] >= 16)
  940.      {
  941.         temp <<= 1;
  942.         if (multiply(lnew.x,ltan22_5,bitshift) < lnew.y)
  943.         {
  944.            ++temp;
  945.            lalt = lnew;
  946.            lnew.x = multiply(lalt.x,lcos45,bitshift) +
  947.            multiply(lalt.y,lsin45,bitshift);
  948.            lnew.y = multiply(lalt.x,lsin45,bitshift) -
  949.            multiply(lalt.y,lcos45,bitshift);
  950.         }
  951.  
  952.         if (decomp[0] >= 32)
  953.         {
  954.            temp <<= 1;
  955.            if (multiply(lnew.x,ltan11_25,bitshift) < lnew.y)
  956.            {
  957.           ++temp;
  958.           lalt = lnew;
  959.           lnew.x = multiply(lalt.x,lcos22_5,bitshift) +
  960.               multiply(lalt.y,lsin22_5,bitshift);
  961.           lnew.y = multiply(lalt.x,lsin22_5,bitshift) -
  962.               multiply(lalt.y,lcos22_5,bitshift);
  963.            }
  964.  
  965.            if (decomp[0] >= 64)
  966.            {
  967.           temp <<= 1;
  968.           if (multiply(lnew.x,ltan5_625,bitshift) < lnew.y)
  969.           {
  970.              ++temp;
  971.              lalt = lnew;
  972.              lnew.x = multiply(lalt.x,lcos11_25,bitshift) +
  973.              multiply(lalt.y,lsin11_25,bitshift);
  974.              lnew.y = multiply(lalt.x,lsin11_25,bitshift) -
  975.              multiply(lalt.y,lcos11_25,bitshift);
  976.           }
  977.  
  978.           if (decomp[0] >= 128)
  979.           {
  980.              temp <<= 1;
  981.              if (multiply(lnew.x,ltan2_8125,bitshift) < lnew.y)
  982.              {
  983.             ++temp;
  984.             lalt = lnew;
  985.             lnew.x = multiply(lalt.x,lcos5_625,bitshift) +
  986.                 multiply(lalt.y,lsin5_625,bitshift);
  987.             lnew.y = multiply(lalt.x,lsin5_625,bitshift) -
  988.                 multiply(lalt.y,lcos5_625,bitshift);
  989.              }
  990.  
  991.              if (decomp[0] == 256)
  992.              {
  993.             temp <<= 1;
  994.             if (multiply(lnew.x,ltan1_4063,bitshift) < lnew.y)
  995.                if ((lnew.x*ltan1_4063 < lnew.y))
  996.                   ++temp;
  997.              }
  998.           }
  999.            }
  1000.         }
  1001.      }
  1002.       }
  1003.    }
  1004.    else /* double case */
  1005.    {
  1006.       if (new.y < 0)
  1007.       {
  1008.      temp = 2;
  1009.      new.y = -new.y;
  1010.       }
  1011.       if (new.x < 0)
  1012.       {
  1013.      ++temp;
  1014.      new.x = -new.x;
  1015.       }
  1016.       if (decomp[0] >= 8)
  1017.       {
  1018.      temp <<= 1;
  1019.      if (new.x < new.y)
  1020.      {
  1021.         ++temp;
  1022.         alt.x = new.x; /* just */
  1023.         new.x = new.y; /* swap */
  1024.         new.y = alt.x; /* them */
  1025.      }
  1026.      if (decomp[0] >= 16)
  1027.      {
  1028.         temp <<= 1;
  1029.         if (new.x*tan22_5 < new.y)
  1030.         {
  1031.            ++temp;
  1032.            alt = new;
  1033.            new.x = alt.x*cos45 + alt.y*sin45;
  1034.            new.y = alt.x*sin45 - alt.y*cos45;
  1035.         }
  1036.  
  1037.         if (decomp[0] >= 32)
  1038.         {
  1039.            temp <<= 1;
  1040.            if (new.x*tan11_25 < new.y)
  1041.            {
  1042.           ++temp;
  1043.           alt = new;
  1044.           new.x = alt.x*cos22_5 + alt.y*sin22_5;
  1045.           new.y = alt.x*sin22_5 - alt.y*cos22_5;
  1046.            }
  1047.  
  1048.            if (decomp[0] >= 64)
  1049.            {
  1050.           temp <<= 1;
  1051.           if (new.x*tan5_625 < new.y)
  1052.           {
  1053.              ++temp;
  1054.              alt = new;
  1055.              new.x = alt.x*cos11_25 + alt.y*sin11_25;
  1056.              new.y = alt.x*sin11_25 - alt.y*cos11_25;
  1057.           }
  1058.  
  1059.           if (decomp[0] >= 128)
  1060.           {
  1061.              temp <<= 1;
  1062.              if (new.x*tan2_8125 < new.y)
  1063.              {
  1064.             ++temp;
  1065.             alt = new;
  1066.             new.x = alt.x*cos5_625 + alt.y*sin5_625;
  1067.             new.y = alt.x*sin5_625 - alt.y*cos5_625;
  1068.              }
  1069.  
  1070.              if (decomp[0] == 256)
  1071.              {
  1072.             temp <<= 1;
  1073.             if ((new.x*tan1_4063 < new.y))
  1074.                ++temp;
  1075.              }
  1076.           }
  1077.            }
  1078.         }
  1079.      }
  1080.       }
  1081.    }
  1082.    for (i = 1; temp > 0; ++i)
  1083.    {
  1084.       if (temp & 1)
  1085.      color = (1 << i) - 1 - color;
  1086.       temp >>= 1;
  1087.    }
  1088.    if (decomp[0] == 2)
  1089.       color &= 1;
  1090.    if (colors > decomp[0])
  1091.       color++;
  1092. }
  1093.  
  1094. /******************************************************************/
  1095. /* Continuous potential calculation for Mandelbrot and Julia      */
  1096. /* Reference: Science of Fractal Images p. 190.           */
  1097. /* Special thanks to Mark Peterson for his "MtMand" program that  */
  1098. /* beautifully approximates plate 25 (same reference) and spurred */
  1099. /* on the inclusion of similar capabilities in FRACTINT.      */
  1100. /*                                  */
  1101. /* The purpose of this function is to calculate a color value      */
  1102. /* for a fractal that varies continuously with the screen pixels  */
  1103. /* locations for better rendering in 3D.              */
  1104. /*                                  */
  1105. /* Here "magnitude" is the modulus of the orbit value at          */
  1106. /* "iterations". The potparms[] are user-entered paramters        */
  1107. /* controlling the level and slope of the continuous potential      */
  1108. /* surface. Returns color.  - Tim Wegner 6/25/89          */
  1109. /*                                  */
  1110. /*               -- Change history --              */
  1111. /*                                  */
  1112. /* 09/12/89   - added floatflag support and fixed float underflow */
  1113. /*                                  */
  1114. /******************************************************************/
  1115.  
  1116. static int potential(double mag, int iterations)
  1117. {
  1118.    float f_mag,f_tmp,pot;
  1119.    double d_tmp;
  1120.    int i_pot;
  1121.    long l_pot;
  1122.    extern char floatflag;
  1123.  
  1124.    if(iterations < maxit)
  1125.    {
  1126.       pot = i_pot = iterations+2;
  1127.       if(i_pot <= 0 || mag <= 1.0)
  1128.      pot = 0.0;
  1129.       else /* pot = log(mag) / pow(2.0, (double)pot); */
  1130.       {
  1131.      if(i_pot < 120 && !floatflag) /* empirically determined limit of fShift */
  1132.      {
  1133.         f_mag = mag;
  1134.         fLog14(f_mag,f_tmp); /* this SHOULD be non-negative */
  1135.         fShift(f_tmp,-i_pot,pot);
  1136.      }
  1137.      else
  1138.      {
  1139.         d_tmp = log(mag)/(double)pow(2.0,(double)pot);
  1140.         if(d_tmp > FLT_MIN) /* prevent float type underflow */
  1141.            pot = d_tmp;
  1142.         else
  1143.            pot = 0.0;
  1144.      }
  1145.       }
  1146.       /* following transformation strictly for aesthetic reasons */
  1147.       /* meaning of parameters:
  1148.         potparam[0] -- zero potential level - highest color -
  1149.         potparam[1] -- slope multiplier -- higher is steeper
  1150.         potparam[2] -- rqlim value if changeable (bailout for modulus) */
  1151.  
  1152.       if(pot > 0.0)
  1153.       {
  1154.      if(floatflag)
  1155.         pot = (float)sqrt((double)pot);
  1156.      else
  1157.      {
  1158.         fSqrt14(pot,f_tmp);
  1159.         pot = f_tmp;
  1160.      }
  1161.      pot = potparam[0] - pot*potparam[1] - 1.0;
  1162.       }
  1163.       else
  1164.      pot = potparam[0] - 1.0;
  1165.       if(pot < 1.0)
  1166.      pot = 1.0; /* avoid color 0 */
  1167.    }
  1168.    else if(inside >= 0)
  1169.       pot = inside;
  1170.    else /* inside < 0 implies inside=maxit, so use 1st pot param instead */
  1171.       pot = potparam[0];
  1172.  
  1173.    i_pot = (l_pot = pot * 256) >> 8;
  1174.    if(i_pot >= colors)
  1175.    {
  1176.       i_pot = colors - 1;
  1177.       l_pot = 255;
  1178.    }
  1179.  
  1180.    if(pot16bit)
  1181.    {
  1182.       if (dotmode != 11) /* if putcolor won't be doing it for us */
  1183.      writedisk(col+sxoffs,row+syoffs,i_pot);
  1184.       writedisk(col+sxoffs,row+sydots+syoffs,(int)l_pot);
  1185.    }
  1186.  
  1187.    return(i_pot);
  1188. }
  1189.  
  1190.  
  1191. /**************** boundary tracing method *********************/
  1192.  
  1193. static int far *LeftX  = (int far *)NULL;
  1194. static int far *RightX = (int far *)NULL;
  1195. static unsigned repeats;
  1196.  
  1197. static int calc_xy(int mx, int my) /* return the color for a pixel */
  1198. {
  1199.  
  1200.    color = getcolor(mx,my); /* see if pixel is black */
  1201.    if (color!=0)        /* pixel is NOT black so we must have already */
  1202.    {              /* calculated its color, so lets skip it    */
  1203.       repeats++;        /* count successive easy ones */
  1204.       return(color);
  1205.    }
  1206.    repeats = 0;     /* we'll have to work for this one wo reset counter */
  1207.  
  1208.    col = mx;
  1209.    row = my;
  1210.    color=(*calctype)();
  1211.    return(color);
  1212. } /* calc_xy function of BTM code */
  1213.  
  1214. static int boundary_trace(int C, int R)   /* BTM main function */
  1215. {
  1216.    enum
  1217.        {
  1218.       North, East, South, West
  1219.    }
  1220.    Dir;
  1221.    int C_first, bcolor, low_row, iters, gcolor;
  1222.    low_row = R;
  1223.    Dir = East;
  1224.    bcolor = color;
  1225.    C_first = C;
  1226.    iters = 0;
  1227.    repeats = 0;
  1228.  
  1229.    /* main loop of BTM inside this loop the boundary is traced on screen! */
  1230.    do
  1231.    {
  1232.       if(--kbdcount<=0)
  1233.       {
  1234.      if(check_key())
  1235.         return(-1);
  1236.      kbdcount=(cpu==386) ? 80 : 30;
  1237.       }
  1238.       iters++;        /* count times thru loop */
  1239.  
  1240.       if (C < LeftX[R])
  1241.      LeftX[R]  = C; /* to aid in filling polygon later */
  1242.       if (C > RightX[R])
  1243.      RightX[R] = C; /* maintain left and right limits */
  1244.       else
  1245.      if (R==low_row)
  1246.         if (C<=C_first) /* works 99.9% of time! */
  1247.            break;
  1248.       switch (Dir)
  1249.       {
  1250.       case North :
  1251.      if (R > low_row)
  1252.         if(calc_xy(C,R-1)==bcolor)
  1253.         {
  1254.            R--;
  1255.            if (C > ixstart)
  1256.           if(calc_xy(C-1,R)==bcolor)
  1257.           {
  1258.              C--;
  1259.              Dir = West;
  1260.           }
  1261.            break;
  1262.         }
  1263.      Dir = East;
  1264.      break;
  1265.       case East :
  1266.      if (C < ixstop)
  1267.         if(calc_xy(C+1,R)==bcolor)
  1268.         {
  1269.            C++;
  1270.            if (R > low_row)
  1271.           if(calc_xy(C,R-1)==bcolor)
  1272.           {
  1273.              R--;
  1274.              Dir = North;
  1275.           }
  1276.            break;
  1277.         }
  1278.      Dir = South;
  1279.      break;
  1280.       case South :
  1281.      if (R < iystop)
  1282.         if(calc_xy(C,R+1)==bcolor)
  1283.         {
  1284.            R++;
  1285.            if (C < ixstop)
  1286.           if(calc_xy(C+1,R)==bcolor)
  1287.           {
  1288.              C++;
  1289.              Dir = East;
  1290.           }
  1291.            break;
  1292.         }
  1293.      Dir = West;
  1294.      break;
  1295.       case West:
  1296.      if (C > ixstart)
  1297.         if(calc_xy(C-1,R)==bcolor)
  1298.         {
  1299.            C--;
  1300.            if (R < iystop)
  1301.           if(calc_xy(C,R+1)== bcolor)
  1302.           {
  1303.              R++;
  1304.              Dir = South;
  1305.           }
  1306.            break;
  1307.         }
  1308.      Dir = North;
  1309.      break;
  1310.       } /* case */
  1311.    }
  1312.    while (repeats<30000); /* emergency backstop, should never be needed */
  1313.    /* PB, made above very high to allow for resumes;  did some checking
  1314.      of code first, and testing, to confirm that it seems unnecessary */
  1315.    if (iters<4)
  1316.    {
  1317.       LeftX[low_row] = 3000;
  1318.       RightX[low_row] = -3000;
  1319.       if (low_row+1 <= iystop)
  1320.       {
  1321.      LeftX[low_row+1] = 3000;
  1322.      RightX[low_row+1] = -3000;
  1323.       }
  1324.       return(0);  /* no need to fill a polygon of 3 points! */
  1325.    }
  1326.  
  1327.    /* Avoid tracing around whole fractal object */
  1328.    if (iystop+1==ydots)
  1329.       if (LeftX[0]==0)
  1330.      if (RightX[0]==ixstop)
  1331.         if (LeftX[iystop]==0)
  1332.            if (RightX[iystop]==ixstop)
  1333.            {
  1334.           /* clean up in this RARE case or next fills will fail! */
  1335.           for (low_row = 0; low_row <= ydots-1; low_row++)
  1336.           {
  1337.              LeftX[low_row] = 3000;
  1338.              RightX[low_row] = -3000;
  1339.           }
  1340.           return(0);
  1341.            }
  1342.    /* fill in the traced polygon, simple but it works darn well */
  1343.    C = 0;
  1344.    for (R = low_row; R<iystop; R++)
  1345.       if (RightX[R] != -3000)
  1346.       {
  1347.      if((kbdcount-=2)<=0)
  1348.      {
  1349.         if(check_key())
  1350.            return(-1);
  1351.         kbdcount=(cpu==386) ? 80 : 30;
  1352.      }
  1353.      if(debugflag==1946)
  1354.         C = fillseg1(LeftX[R], RightX[R],R, bcolor);
  1355.      else
  1356.         C = fillseg(LeftX[R], RightX[R],R, bcolor);
  1357.  
  1358.      LeftX[R]  =  3000;
  1359.      RightX[R] = -3000; /* reset array element */
  1360.       }
  1361.       else if (C!=0) /* this is why C = 0 above! */
  1362.      return(0);
  1363.    return(0);
  1364. } /* BTM function */
  1365.  
  1366. static int fillseg1(int LeftX, int RightX, int R,  int bcolor)
  1367. {
  1368.    register modeON, C;
  1369.    int    gcolor;
  1370.    modeON = 0;
  1371.    for (C = LeftX; C <= RightX; C++)
  1372.    {
  1373.       gcolor=getcolor(C,R);
  1374.       if (modeON!=0 && gcolor==0)
  1375. /*       (*plot)(C,R,bcolor); */
  1376.      (*plot)(C,R,1); /* show boundary by only filling with color 1 */
  1377.       else
  1378.       {
  1379.      if (gcolor==bcolor) /* TW saved a getcolor here */
  1380.         modeON = 1;
  1381.      else
  1382.         modeON = 0;
  1383.       }
  1384.    }
  1385.    return(C);
  1386. }
  1387.  
  1388. static int fillseg(int LeftX, int RightX, int R,  int bcolor)
  1389. {
  1390.    unsigned char *forwards;
  1391.    unsigned char *backwards;
  1392.    register modeON, C;
  1393.    int    gcolor,i;
  1394.    modeON = 0;
  1395.    forwards  = (unsigned char *)decoderline;
  1396.    backwards = (unsigned char *)dstack;
  1397.    modeON = 0;
  1398.    get_line(R,LeftX,RightX,forwards);
  1399.    for (C = LeftX; C <= RightX; C++)
  1400.    {
  1401.       gcolor=forwards[C-LeftX];
  1402.       if (modeON!=0 && gcolor==0)
  1403.      forwards[C-LeftX]=bcolor;
  1404.       else
  1405.       {
  1406.      if (gcolor==bcolor) /* TW saved a getcolor here */
  1407.         modeON = 1;
  1408.      else
  1409.         modeON = 0;
  1410.       }
  1411.    }
  1412.    if(plot==putcolor) /* no symmetry! easy! */
  1413.       put_line(R,LeftX,RightX,forwards);
  1414.    else if(plot==symplot2) /* X-axis symmetry */
  1415.    {
  1416.       put_line(R,   LeftX,RightX,forwards);
  1417.       if ((i=yystop-(R-yystart)) > iystop)
  1418.      put_line(i,LeftX,RightX,forwards);
  1419.    }
  1420.    else if(plot==symplot2J) /* Origin symmetry */
  1421.    {
  1422.       reverse_string(backwards,forwards,RightX-LeftX+1);
  1423.       put_line(R,   LeftX,            RightX,           forwards);
  1424.       if ((i=yystop-(R-yystart)) > iystop)
  1425.      put_line(i,xxstop-(RightX-ixstart),xxstop-(LeftX-ixstart),backwards);
  1426.    }
  1427.    else if(plot==symplot2Y) /* Y-axis symmetry */
  1428.    {
  1429.       reverse_string(backwards,forwards,RightX-LeftX+1);
  1430.       put_line(R,LeftX,          RightX,        forwards);
  1431.       put_line(R,xxstop-(RightX-ixstart),xxstop-(LeftX-ixstart),backwards);
  1432.    }
  1433.    else if(plot==symplot4) /* X-axis and Y-axis symmetry */
  1434.    {
  1435.       reverse_string(backwards,forwards,RightX-LeftX+1);
  1436.       put_line(R,LeftX,             RightX,           forwards);
  1437.       put_line(R,xxstop-(RightX-ixstart),   xxstop-(LeftX-ixstart),backwards);
  1438.       if ((i=yystop-(R-yystart)) > iystop)
  1439.       {
  1440.      put_line(i,LeftX,            RightX,           forwards);
  1441.      put_line(i,xxstop-(RightX-ixstart),xxstop-(LeftX-ixstart),backwards);
  1442.       }
  1443.    }
  1444.    else  /* the other symmetry types are on their own! */
  1445.    {
  1446.       int i;
  1447.       for(i=LeftX;i<=RightX;i++)
  1448.      (*plot)(i,R,forwards[i-LeftX]);
  1449.    }
  1450.    return(C);
  1451. }
  1452.  
  1453. /* copy a string backwards for symmetry functions */
  1454. static void reverse_string(char *t, char *s, int len)
  1455. {
  1456.    register i;
  1457.    len--;
  1458.    for(i=0;i<=len;i++)
  1459.       t[i] = s[len-i];
  1460. }
  1461.  
  1462. static int bound_trace_main()
  1463. {
  1464.    long maxrow;
  1465.    maxrow = ((long)ydots)*sizeof(int);
  1466.  
  1467.    if (inside == 0) {
  1468.       stopmsg(0,"Sorry, boundary tracing cannot be used with inside=0.");
  1469.       return(-1);
  1470.       }
  1471.    if (colors < 16) {
  1472.       stopmsg(0,"Sorry, boundary tracing cannot be used with < 16 colors.");
  1473.       return(-1);
  1474.       }
  1475.  
  1476.    if((LeftX  = (int far *)farmemalloc(maxrow))==(int far *)NULL)
  1477.       return(-1);
  1478.    else if((RightX  = (int far *)farmemalloc(maxrow))==(int far *)NULL)
  1479.    {
  1480.       farmemfree((char far *)LeftX);
  1481.       return(-1);
  1482.    }
  1483.  
  1484.    for (currow = 0; currow < ydots; currow++)
  1485.    {
  1486.       LeftX[currow] = 3000;
  1487.       RightX[currow] = -3000;
  1488.    }
  1489.  
  1490.    got_status = 2;
  1491.    for (currow = iystart; currow <= iystop; currow++)
  1492.    {
  1493.       for (curcol = ixstart; curcol <= ixstop; curcol++)
  1494.       {
  1495.      if(--kbdcount<=0)
  1496.      {
  1497.         if(check_key())
  1498.         {
  1499.            if (iystop != yystop)
  1500.           iystop = yystop - (currow - yystart); /* allow for sym */
  1501.            add_worklist(xxstart,xxstop,currow,iystop,currow,0,worksym);
  1502.            farmemfree((char far *)LeftX);
  1503.            farmemfree((char far *)RightX);
  1504.            return(-1);
  1505.         }
  1506.         kbdcount=(cpu==386) ? 80 : 30;
  1507.      }
  1508.  
  1509.      /* BTM Hook! */
  1510.      color = getcolor(curcol,currow);
  1511.      /* if pixel is BLACK (0) then we haven't done it yet!
  1512.         so first calculate its color and call the routine
  1513.         that will try and trace a polygon if one exists */
  1514.      if (color==0)
  1515.      {
  1516.         reset_periodicity = 1;
  1517.         row = currow;
  1518.         col = curcol;
  1519.         color=(*calctype)();
  1520.         reset_periodicity = 0;
  1521.         boundary_trace(curcol, currow); /* go trace boundary! WHOOOOOOO! */
  1522.      }
  1523.       }
  1524.    }
  1525.    farmemfree((char far *)LeftX);
  1526.    farmemfree((char far *)RightX);
  1527.    return(0);
  1528. } /* end of bound_trace_main */
  1529.  
  1530.  
  1531. /************************ super solid guessing *****************************/
  1532.  
  1533. /*
  1534.    I, Timothy Wegner, invented this solidguessing idea and implemented it in
  1535.    more or less the overall framework you see here.  I am adding this note
  1536.    now in a possibly vain attempt to secure my place in history, because
  1537.    Pieter Branderhorst has totally rewritten this routine, incorporating
  1538.    a *MUCH* more sophisticated algorithm.  His revised code is not only
  1539.    faster, but is also more accurate. Harrumph!
  1540. */
  1541.  
  1542. static int solidguess()
  1543. {
  1544.    int i,x,y,xlim,ylim,blocksize;
  1545.    unsigned int *pfxp0,*pfxp1;
  1546.    unsigned int u;
  1547.  
  1548.    guessplot=(plot!=putcolor && plot!=symplot2 && plot!=symplot2J);
  1549.    /* check if guessing at bottom & right edges is ok */
  1550.    bottom_guess = (plot == symplot2 || (plot == putcolor && iystop+1 == ydots));
  1551.    right_guess    = (plot == symplot2J
  1552.        || ((plot == putcolor || plot == symplot2) && ixstop+1 == xdots));
  1553.  
  1554.    i = maxblock = blocksize = ssg_blocksize();
  1555.    totpasses = 1;
  1556.    while ((i >>= 1) > 1) ++totpasses;
  1557.  
  1558.    /* ensure window top and left are on required boundary, treat window
  1559.      as larger than it really is if necessary (this is the reason symplot
  1560.      routines must check for > xdots/ydots before plotting sym points) */
  1561.    ixstart &= -1 - (maxblock-1);
  1562.    iystart = yybegin;
  1563.    iystart &= -1 - (maxblock-1);
  1564.  
  1565.    got_status = 1;
  1566.  
  1567.    if (workpass == 0) /* otherwise first pass already done */
  1568.    {
  1569.       /* first pass, calc every blocksize**2 pixel, quarter result & paint it */
  1570.       curpass = 1;
  1571.       if (iystart <= yystart) /* first time for this window, init it */
  1572.       {
  1573.      currow = 0;
  1574.      memset(&prefix[1][0][0],0,maxxblk*maxyblk*2); /* noskip flags off */
  1575.      reset_periodicity = 1;
  1576.      row=iystart;
  1577.      for(col=ixstart; col<=ixstop; col+=maxblock)
  1578.      { /* calc top row */
  1579.         if((*calctype)()==-1)
  1580.         {
  1581.            add_worklist(xxstart,xxstop,yystart,yystop,yybegin,0,worksym);
  1582.            goto exit_solidguess;
  1583.         }
  1584.         reset_periodicity = 0;
  1585.      }
  1586.       }
  1587.       else
  1588.      memset(&prefix[1][0][0],-1,maxxblk*maxyblk*2); /* noskip flags on */
  1589.       for(y=iystart; y<=iystop; y+=blocksize)
  1590.       {
  1591.      currow = y;
  1592.      i = 0;
  1593.      if(y+blocksize<=iystop)
  1594.      { /* calc the row below */
  1595.         row=y+blocksize;
  1596.         reset_periodicity = 1;
  1597.         for(col=ixstart; col<=ixstop; col+=maxblock)
  1598.         {
  1599.            if((i=(*calctype)()) == -1)
  1600.           break;
  1601.            reset_periodicity = 0;
  1602.         }
  1603.      }
  1604.      reset_periodicity = 1;
  1605.      if (i == -1 || guessrow(1,y,blocksize) != 0) /* interrupted? */
  1606.      {
  1607.         if (y < yystart)
  1608.            y = yystart;
  1609.         add_worklist(xxstart,xxstop,yystart,yystop,y,0,worksym);
  1610.         goto exit_solidguess;
  1611.      }
  1612.       }
  1613.  
  1614.       if (num_worklist) /* work list not empty, just do 1st pass */
  1615.       {
  1616.      add_worklist(xxstart,xxstop,yystart,yystop,yystart,1,worksym);
  1617.      goto exit_solidguess;
  1618.       }
  1619.       ++workpass;
  1620.       iystart = yystart & (-1 - (maxblock-1));
  1621.  
  1622.       /* calculate skip flags for skippable blocks */
  1623.       xlim=(ixstop+maxblock)/maxblock+1;
  1624.       ylim=((iystop+maxblock)/maxblock+15)/16+1;
  1625.       if(right_guess==0) /* no right edge guessing, zap border */
  1626.      for(y=0;y<=ylim;++y)
  1627.         prefix[1][y][xlim]=-1;
  1628.       if(bottom_guess==0) /* no bottom edge guessing, zap border */
  1629.       {
  1630.      i=(iystop+maxblock)/maxblock+1;
  1631.      y=i/16+1;
  1632.      i=1<<(i&15);
  1633.      for(x=0;x<=xlim;++x)
  1634.         prefix[1][y][x]|=i;
  1635.       }
  1636.       /* set each bit in prefix[0] to OR of it & surrounding 8 in prefix[1] */
  1637.       for(y=0;++y<ylim;)
  1638.       {
  1639.      pfxp0=&prefix[0][y][0];
  1640.      pfxp1=&prefix[1][y][0];
  1641.      for(x=0;++x<xlim;)
  1642.      {
  1643.         ++pfxp1;
  1644.         u=*(pfxp1-1)|*pfxp1|*(pfxp1+1);
  1645.         *(++pfxp0)=u|(u>>1)|(u<<1)
  1646.            |((*(pfxp1-(maxxblk+1))|*(pfxp1-maxxblk)|*(pfxp1-(maxxblk-1)))>>15)
  1647.           |((*(pfxp1+(maxxblk-1))|*(pfxp1+maxxblk)|*(pfxp1+(maxxblk+1)))<<15);
  1648.      }
  1649.       }
  1650.    }
  1651.    else /* first pass already done */
  1652.       memset(&prefix[0][0][0],-1,maxxblk*maxyblk*2); /* noskip flags on */
  1653.  
  1654.    /* remaining pass(es), halve blocksize & quarter each blocksize**2 */
  1655.    i = workpass;
  1656.    while (--i > 0) /* allow for already done passes */
  1657.       blocksize = blocksize>>1;
  1658.    reset_periodicity = 1;
  1659.    while((blocksize=blocksize>>1)>=2)
  1660.    {
  1661.       curpass = workpass + 1;
  1662.       for(y=iystart; y<=iystop; y+=blocksize)
  1663.       {
  1664.      currow = y;
  1665.      if(guessrow(0,y,blocksize)!=0)
  1666.      {
  1667.         if (y < yystart)
  1668.            y = yystart;
  1669.         add_worklist(xxstart,xxstop,yystart,yystop,y,workpass,worksym);
  1670.         goto exit_solidguess;
  1671.      }
  1672.       }
  1673.       ++workpass;
  1674.       if (num_worklist /* work list not empty, do one pass at a time */
  1675.       && blocksize>2) /* if 2, we just did last pass */
  1676.       {
  1677.      add_worklist(xxstart,xxstop,yystart,yystop,yystart,workpass,worksym);
  1678.      goto exit_solidguess;
  1679.       }
  1680.       iystart = yystart & (-1 - (maxblock-1));
  1681.    }
  1682.  
  1683.    exit_solidguess:
  1684.    return(0);
  1685. }
  1686.  
  1687. #define calcadot(c,x,y) { col=x; row=y; if((c=(*calctype)())==-1) return -1; }
  1688.  
  1689. static int guessrow(int firstpass,int y,int blocksize)
  1690. {
  1691.    int x,i,j,color;
  1692.    int xplushalf,xplusblock;
  1693.    int ylessblock,ylesshalf,yplushalf,yplusblock;
  1694.    int       c21,c31,c41;     /* cxy is the color of pixel at (x,y) */
  1695.    int c12,c22,c32,c42;     /* where c22 is the topleft corner of */
  1696.    int c13,c23,c33;        /* the block being handled in current */
  1697.    int       c24,    c44;     /* iteration                  */
  1698.    int guessed23,guessed32,guessed33,guessed12,guessed13;
  1699.    int orig23,orig32,orig33;
  1700.    int prev11,fix21,fix31;
  1701.    unsigned int *pfxptr,pfxmask;
  1702.  
  1703.    halfblock=blocksize>>1;
  1704.    i=y/maxblock;
  1705.    pfxptr=&prefix[firstpass][(i>>4)+1][ixstart/maxblock];
  1706.    pfxmask=1<<(i&15);
  1707.    ylesshalf=y-halfblock;
  1708.    ylessblock=y-blocksize; /* constants, for speed */
  1709.    yplushalf=y+halfblock;
  1710.    yplusblock=y+blocksize;
  1711.    prev11=-1;
  1712.    c24=c12=c13=c22=getcolor(ixstart,y);
  1713.    c31=c21=getcolor(ixstart,(y>0)?ylesshalf:0);
  1714.    if(yplusblock<=iystop)
  1715.       c24=getcolor(ixstart,yplusblock);
  1716.    else if(bottom_guess==0)
  1717.       c24=-1;
  1718.    guessed12=guessed13=0;
  1719.  
  1720.    for(x=ixstart; x<=ixstop;)  /* increment at end, or when doing continue */
  1721.    {
  1722.       if((x&(maxblock-1))==0)  /* time for skip flag stuff */
  1723.       {
  1724.      ++pfxptr;
  1725.      if(firstpass==0 && (*pfxptr&pfxmask)==0)  /* check for fast skip */
  1726.      {
  1727.         /* next useful in testing to make skips visible */
  1728.         /*
  1729.                if(halfblock==1)
  1730.                {
  1731.                   (*plot)(x+1,y,0); (*plot)(x,y+1,0); (*plot)(x+1,y+1,0);
  1732.                   }
  1733.              */
  1734.         x+=maxblock;
  1735.         prev11=c31=c21=c24=c12=c13=c22;
  1736.         guessed12=guessed13=0;
  1737.         continue;
  1738.      }
  1739.       }
  1740.  
  1741.       if(firstpass)  /* 1st pass, paint topleft corner */
  1742.      plotblock(0,x,y,c22);
  1743.       /* setup variables */
  1744.       xplusblock=(xplushalf=x+halfblock)+halfblock;
  1745.       if(xplushalf>ixstop)
  1746.       {
  1747.      if(right_guess==0)
  1748.         c31=-1;
  1749.       }
  1750.       else if(y>0)
  1751.      c31=getcolor(xplushalf,ylesshalf);
  1752.       if(xplusblock<=ixstop)
  1753.       {
  1754.      if(yplusblock<=iystop)
  1755.         c44=getcolor(xplusblock,yplusblock);
  1756.      c41=getcolor(xplusblock,(y>0)?ylesshalf:0);
  1757.      c42=getcolor(xplusblock,y);
  1758.       }
  1759.       else if(right_guess==0)
  1760.      c41=c42=c44=-1;
  1761.       if(yplusblock>iystop)
  1762.      c44=(bottom_guess)?c42:-1;
  1763.  
  1764.       /* guess or calc the remaining 3 quarters of current block */
  1765.       guessed23=guessed32=guessed33=1;
  1766.       c23=c32=c33=c22;
  1767.       if(yplushalf>iystop)
  1768.       {
  1769.      if(bottom_guess==0)
  1770.         c23=c33=-1;
  1771.      guessed23=guessed33=-1;
  1772.       }
  1773.       if(xplushalf>ixstop)
  1774.       {
  1775.      if(right_guess==0)
  1776.         c32=c33=-1;
  1777.      guessed32=guessed33=-1;
  1778.       }
  1779.       while(1) /* go around till none of 23,32,33 change anymore */
  1780.       {
  1781.      if(guessed33>0
  1782.          && (c33!=c44 || c33!=c42 || c33!=c24 || c33!=c32 || c33!=c23))
  1783.      {
  1784.         calcadot(c33,xplushalf,yplushalf);
  1785.         guessed33=0;
  1786.      }
  1787.      if(guessed32>0
  1788.          && (c32!=c33 || c32!=c42 || c32!=c31 || c32!=c21
  1789.          || c32!=c41 || c32!=c23))
  1790.      {
  1791.         calcadot(c32,xplushalf,y);
  1792.         guessed32=0;
  1793.         continue;
  1794.      }
  1795.      if(guessed23>0
  1796.          && (c23!=c33 || c23!=c24 || c23!=c13 || c23!=c12 || c23!=c32))
  1797.      {
  1798.         calcadot(c23,x,yplushalf);
  1799.         guessed23=0;
  1800.         continue;
  1801.      }
  1802.      break;
  1803.       }
  1804.  
  1805.       if(firstpass) /* note whether any of block's contents were calculated */
  1806.      if(guessed23==0 || guessed32==0 || guessed33==0)
  1807.         *pfxptr|=pfxmask;
  1808.  
  1809.       if(halfblock>1) /* not last pass, check if something to display */
  1810.      if(firstpass)    /* display guessed corners, fill in block */
  1811.      {
  1812.         if(guessplot)
  1813.         {
  1814.            if(guessed23>0)
  1815.           (*plot)(x,yplushalf,c23);
  1816.            if(guessed32>0)
  1817.           (*plot)(xplushalf,y,c32);
  1818.            if(guessed33>0)
  1819.           (*plot)(xplushalf,yplushalf,c33);
  1820.         }
  1821.         plotblock(1,x,yplushalf,c23);
  1822.         plotblock(0,xplushalf,y,c32);
  1823.         plotblock(1,xplushalf,yplushalf,c33);
  1824.      }
  1825.      else  /* repaint changed blocks */
  1826.      {
  1827.         if(c23!=c22)
  1828.            plotblock(-1,x,yplushalf,c23);
  1829.         if(c32!=c22)
  1830.            plotblock(-1,xplushalf,y,c32);
  1831.         if(c33!=c22)
  1832.            plotblock(-1,xplushalf,yplushalf,c33);
  1833.      }
  1834.  
  1835.       /* check if some calcs in this block mean earlier guesses need fixing */
  1836.       fix21=((c22!=c12 || c22!=c32)
  1837.       && c21==c22 && c21==c31 && c21==prev11
  1838.       && y>0
  1839.       && (x==ixstart || c21==getcolor(x-halfblock,ylessblock))
  1840.       && (xplushalf>ixstop || c21==getcolor(xplushalf,ylessblock))
  1841.       && c21==getcolor(x,ylessblock));
  1842.       fix31=(c22!=c32
  1843.       && c31==c22 && c31==c42 && c31==c21 && c31==c41
  1844.       && y>0 && xplushalf<=ixstop
  1845.       && c31==getcolor(xplushalf,ylessblock)
  1846.       && (xplusblock>ixstop || c31==getcolor(xplusblock,ylessblock))
  1847.       && c31==getcolor(x,ylessblock));
  1848.       prev11=c31; /* for next time around */
  1849.       if(fix21)
  1850.       {
  1851.      calcadot(c21,x,ylesshalf);
  1852.      if(halfblock>1 && c21!=c22)
  1853.         plotblock(-1,x,ylesshalf,c21);
  1854.       }
  1855.       if(fix31)
  1856.       {
  1857.      calcadot(c31,xplushalf,ylesshalf);
  1858.      if(halfblock>1 && c31!=c22)
  1859.         plotblock(-1,xplushalf,ylesshalf,c31);
  1860.       }
  1861.       if(c23!=c22)
  1862.       {
  1863.      if(guessed12)
  1864.      {
  1865.         calcadot(c12,x-halfblock,y);
  1866.         if(halfblock>1 && c12!=c22)
  1867.            plotblock(-1,x-halfblock,y,c12);
  1868.      }
  1869.      if(guessed13)
  1870.      {
  1871.         calcadot(c13,x-halfblock,yplushalf);
  1872.         if(halfblock>1 && c13!=c22)
  1873.            plotblock(-1,x-halfblock,yplushalf,c13);
  1874.      }
  1875.       }
  1876.       c22=c42;
  1877.       c24=c44;
  1878.       c13=c33;
  1879.       c31=c21=c41;
  1880.       c12=c32;
  1881.       guessed12=guessed32;
  1882.       guessed13=guessed33;
  1883.       x+=blocksize;
  1884.    } /* end x loop */
  1885.  
  1886.    if(firstpass==0 || guessplot) return 0;
  1887.  
  1888.    /* paint rows the fast way */
  1889.    for(i=0;i<halfblock;++i)
  1890.    {
  1891.       if((j=y+i)<=iystop)
  1892.      put_line(j,xxstart,ixstop,&dstack[xxstart]);
  1893.       if((j=y+i+halfblock)<=iystop)
  1894.      put_line(j,xxstart,ixstop,&dstack[xxstart+2048]);
  1895.       if(keypressed()) return -1;
  1896.    }
  1897.    if(plot!=putcolor)  /* symmetry, just vertical & origin the fast way */
  1898.    {
  1899.       if(plot==symplot2J) /* origin sym, reverse lines */
  1900.      for(i=(ixstop+xxstart+1)/2;--i>=xxstart;)
  1901.      {
  1902.         color=dstack[i];
  1903.         dstack[i]=dstack[j=ixstop-(i-xxstart)];
  1904.         dstack[j]=color;
  1905.         j+=2048;
  1906.         color=dstack[i+2048];
  1907.         dstack[i+2048]=dstack[j];
  1908.         dstack[j]=color;
  1909.      }
  1910.       for(i=0;i<halfblock;++i)
  1911.       {
  1912.      if((j=yystop-(y+i-yystart))>iystop && j<ydots)
  1913.         put_line(j,xxstart,ixstop,&dstack[xxstart]);
  1914.      if((j=yystop-(y+i+halfblock-yystart))>iystop && j<ydots)
  1915.         put_line(j,xxstart,ixstop,&dstack[xxstart+2048]);
  1916.      if(keypressed()) return -1;
  1917.       }
  1918.    }
  1919.    return 0;
  1920. }
  1921.  
  1922. static void plotblock(int buildrow,int x,int y,int color)
  1923. {
  1924.    int i,xlim,ylim;
  1925.    if((xlim=x+halfblock)>ixstop)
  1926.       xlim=ixstop+1;
  1927.    if(buildrow>=0 && guessplot==0) /* save it for later put_line */
  1928.    {
  1929.       if(buildrow==0)
  1930.      for(i=x;i<xlim;++i)
  1931.         dstack[i]=color;
  1932.       else
  1933.      for(i=x;i<xlim;++i)
  1934.         dstack[i+2048]=color;
  1935.       if (x>=xxstart) /* when x reduced for alignment, paint those dots too */
  1936.      return; /* the usual case */
  1937.    }
  1938.    /* paint it */
  1939.    if((ylim=y+halfblock)>iystop)
  1940.    {
  1941.       if(y>iystop)
  1942.      return;
  1943.       ylim=iystop+1;
  1944.    }
  1945.    for(i=x;++i<xlim;)
  1946.       (*plot)(i,y,color); /* skip 1st dot on 1st row */
  1947.    while(++y<ylim)
  1948.       for(i=x;i<xlim;++i)
  1949.      (*plot)(i,y,color);
  1950. }
  1951.  
  1952.  
  1953. /************************* symmetry plot setup ************************/
  1954.  
  1955. static int xsym_split(int xaxis_row,int xaxis_between)
  1956. {
  1957.    int i;
  1958.    if ((worksym&0x11) == 0x10) /* already decided not sym */
  1959.       return(1);
  1960.    if ((worksym&1) != 0) /* already decided on sym */
  1961.       iystop = (yystart+yystop)/2;
  1962.    else /* new window, decide */
  1963.    {
  1964.       worksym |= 0x10;
  1965.       if (xaxis_row <= yystart || xaxis_row >= yystop)
  1966.      return(1); /* axis not in window */
  1967.       i = xaxis_row + (xaxis_row - yystart);
  1968.       if (xaxis_between)
  1969.      ++i;
  1970.       if (i > yystop) /* split into 2 pieces, bottom has the symmetry */
  1971.       {
  1972.      if (num_worklist >= MAXCALCWORK-1) /* no room to split */
  1973.         return(1);
  1974.      iystop = xaxis_row - (yystop - xaxis_row);
  1975.      if (!xaxis_between)
  1976.         --iystop;
  1977.      add_worklist(xxstart,xxstop,iystop+1,yystop,iystop+1,workpass,0);
  1978.      yystop = iystop;
  1979.      return(1); /* tell set_symmetry no sym for current window */
  1980.       }
  1981.       if (i < yystop) /* split into 2 pieces, top has the symmetry */
  1982.       {
  1983.      if (num_worklist >= MAXCALCWORK-1) /* no room to split */
  1984.         return(1);
  1985.      add_worklist(xxstart,xxstop,i+1,yystop,i+1,workpass,0);
  1986.      yystop = i;
  1987.       }
  1988.       iystop = xaxis_row;
  1989.       worksym |= 1;
  1990.    }
  1991.    symmetry = 0;
  1992.    return(0); /* tell set_symmetry its a go */
  1993. }
  1994.  
  1995. static int ysym_split(int yaxis_col,int yaxis_between)
  1996. {
  1997.    int i;
  1998.    if ((worksym&0x22) == 0x20) /* already decided not sym */
  1999.       return(1);
  2000.    if ((worksym&2) != 0) /* already decided on sym */
  2001.       ixstop = (xxstart+xxstop)/2;
  2002.    else /* new window, decide */
  2003.    {
  2004.       worksym |= 0x20;
  2005.       if (yaxis_col <= xxstart || yaxis_col >= xxstop)
  2006.      return(1); /* axis not in window */
  2007.       i = yaxis_col + (yaxis_col - xxstart);
  2008.       if (yaxis_between)
  2009.      ++i;
  2010.       if (i > xxstop) /* split into 2 pieces, right has the symmetry */
  2011.       {
  2012.      if (num_worklist >= MAXCALCWORK-1) /* no room to split */
  2013.         return(1);
  2014.      ixstop = yaxis_col - (xxstop - yaxis_col);
  2015.      if (!yaxis_between)
  2016.         --ixstop;
  2017.      add_worklist(ixstop+1,xxstop,yystart,yystop,yystart,workpass,0);
  2018.      xxstop = ixstop;
  2019.      return(1); /* tell set_symmetry no sym for current window */
  2020.       }
  2021.       if (i < xxstop) /* split into 2 pieces, left has the symmetry */
  2022.       {
  2023.      if (num_worklist >= MAXCALCWORK-1) /* no room to split */
  2024.         return(1);
  2025.      add_worklist(i+1,xxstop,yystart,yystop,yystart,workpass,0);
  2026.      xxstop = i;
  2027.       }
  2028.       ixstop = yaxis_col;
  2029.       worksym |= 2;
  2030.    }
  2031.    symmetry = 0;
  2032.    return(0); /* tell set_symmetry its a go */
  2033. }
  2034.  
  2035. static void setsymmetry(int sym, int uselist) /* set up proper symmetrical plot functions */
  2036. {
  2037.    extern int forcesymmetry;
  2038.    int i;
  2039.    int parmszero;
  2040.    int xaxis_row, yaxis_col;         /* pixel number for origin */
  2041.    int xaxis_between, yaxis_between; /* if axis between 2 pixels, not on one */
  2042.    double ftemp;
  2043.    symmetry = 1;
  2044.    if(sym == NOPLOT && forcesymmetry == 999)
  2045.    {
  2046.       plot = noplot;
  2047.       return;
  2048.    }
  2049.    /* NOTE: 16-bit potential disables symmetry */
  2050.    /* also any decomp= option and any inversion not about the origin */
  2051.    /* also any rotation other than 180deg and any off-axis stretch */
  2052.    if ((potflag && pot16bit) || (invert && inversion[2] != 0.0)
  2053.        || decomp[0] != 0
  2054.        || xxmin!=xx3rd || yymin!=yy3rd)
  2055.       return;
  2056.    if(sym != XAXIS && sym != XAXIS_NOPARM && inversion[1] != 0.0 && forcesymmetry == 999)
  2057.       return;
  2058.    if(forcesymmetry != 999)
  2059.       sym = forcesymmetry;
  2060.    parmszero = (parm.x == 0.0 && parm.y == 0.0 && useinitorbit != 1);
  2061.    xaxis_row = yaxis_col = -1;
  2062.    if (fabs(yymin+yymax) < fabs(yymin)+fabs(yymax)) /* axis is on screen */
  2063.    {
  2064.       ftemp = (0.0-yymax) / (yymin-yymax) * (ydots-1) + 0.25;
  2065.       xaxis_row = ftemp;
  2066.       xaxis_between = (ftemp - xaxis_row >= 0.5);
  2067.       if (uselist == 0 && (!xaxis_between || (xaxis_row+1)*2 != ydots))
  2068.      xaxis_row = -1; /* can't split screen, so dead center or not at all */
  2069.    }
  2070.    if (fabs(xxmin+xxmax) < fabs(xxmin)+fabs(xxmax)) /* axis is on screen */
  2071.    {
  2072.       ftemp = (0.0-xxmin) / (xxmax-xxmin) * (xdots-1) + 0.25;
  2073.       yaxis_col = ftemp;
  2074.       yaxis_between = (ftemp - yaxis_col >= 0.5);
  2075.       if (uselist == 0 && (!yaxis_between || (yaxis_col+1)*2 != xdots))
  2076.      yaxis_col = -1; /* can't split screen, so dead center or not at all */
  2077.    }
  2078.    switch(sym)         /* symmetry switch */
  2079.    {
  2080.    case XAXIS_NOREAL:     /* X-axis Symmetry (no real param) */
  2081.       if (parm.x != 0.0) break;
  2082.       goto xsym;
  2083.    case XAXIS_NOIMAG:     /* X-axis Symmetry (no imag param) */
  2084.       if (parm.y != 0.0) break;
  2085.       goto xsym;
  2086.    case XAXIS_NOPARM:                 /* X-axis Symmetry  (no params)*/
  2087.       if (!parmszero)
  2088.      break;
  2089.       xsym:
  2090.    case XAXIS:                 /* X-axis Symmetry */
  2091.       if (xsym_split(xaxis_row,xaxis_between) == 0)
  2092.      if(basin)
  2093.         plot = symplot2basin;
  2094.      else
  2095.         plot = symplot2;
  2096.       break;
  2097.    case YAXIS_NOPARM:                 /* Y-axis Symmetry (No Parms)*/
  2098.       if (!parmszero)
  2099.      break;
  2100.    case YAXIS:                 /* Y-axis Symmetry */
  2101.       if (ysym_split(yaxis_col,yaxis_between) == 0)
  2102.      plot = symplot2Y;
  2103.       break;
  2104.    case XYAXIS_NOPARM:                 /* X-axis AND Y-axis Symmetry (no parms)*/
  2105.       if(!parmszero)
  2106.      break;
  2107.    case XYAXIS:              /* X-axis AND Y-axis Symmetry */
  2108.       xsym_split(xaxis_row,xaxis_between);
  2109.       ysym_split(yaxis_col,yaxis_between);
  2110.       switch (worksym & 3)
  2111.       {
  2112.       case 1: /* just xaxis symmetry */
  2113.      if(basin)
  2114.         plot = symplot2basin;
  2115.      else
  2116.         plot = symplot2 ;
  2117.      break;
  2118.       case 2: /* just yaxis symmetry */
  2119.      if (basin) /* got no routine for this case */
  2120.      {
  2121.         ixstop = xxstop; /* fix what split should not have done */
  2122.         symmetry = 1;
  2123.      }
  2124.      else
  2125.         plot = symplot2Y;
  2126.      break;
  2127.       case 3: /* both axes */
  2128.      if(basin)
  2129.         plot = symplot4basin;
  2130.      else
  2131.         plot = symplot4 ;
  2132.       }
  2133.       break;
  2134.    case ORIGIN_NOPARM:                 /* Origin Symmetry (no parms)*/
  2135.       if (!parmszero)
  2136.      break;
  2137.    case ORIGIN:              /* Origin Symmetry */
  2138.       originsym:
  2139.       if ( xsym_split(xaxis_row,xaxis_between) == 0
  2140.       && ysym_split(yaxis_col,yaxis_between) == 0)
  2141.       {
  2142.      plot = symplot2J;
  2143.      ixstop = xxstop; /* didn't want this changed */
  2144.       }
  2145.       else
  2146.       {
  2147.      iystop = yystop; /* in case first split worked */
  2148.      symmetry = 1;
  2149.      worksym = 0x30; /* let it recombine with others like it */
  2150.       }
  2151.       break;
  2152.    case PI_SYM_NOPARM:
  2153.       if (!parmszero)
  2154.      break;
  2155.    case PI_SYM:              /* PI symmetry */
  2156.       if(invert && forcesymmetry == 999)
  2157.      goto originsym;
  2158.       plot = symPIplot ;
  2159.       symmetry = 0;
  2160.       if ( xsym_split(xaxis_row,xaxis_between) == 0
  2161.       && ysym_split(yaxis_col,yaxis_between) == 0)
  2162.      if(parm.y == 0.0)
  2163.         plot = symPIplot4J; /* both axes */
  2164.      else
  2165.         plot = symPIplot2J; /* origin */
  2166.       else
  2167.       {
  2168.      iystop = yystop; /* in case first split worked */
  2169.      worksym = 0x30;  /* don't mark pisym as ysym, just do it unmarked */
  2170.       }
  2171.       pixelpi = (PI/fabs(xxmax-xxmin))*xdots; /* PI in pixels */
  2172.       if ( (ixstop = xxstart+pixelpi-1 ) > xxstop)
  2173.      ixstop = xxstop;
  2174.       if (plot == symPIplot4J && ixstop > (i = (xxstart+xxstop)/2))
  2175.      ixstop = i;
  2176.       break;
  2177.    default:             /* no symmetry */
  2178.       break;
  2179.    }
  2180. }
  2181.  
  2182.  
  2183. /***************** standalone engine for "test" ********************/
  2184.  
  2185. int test()
  2186. {
  2187.    int startrow,startpass,numpasses;
  2188.    startrow = startpass = 0;
  2189.    if (resuming)
  2190.    {
  2191.       start_resume();
  2192.       get_resume(sizeof(int),&startrow,sizeof(int),&startpass,0);
  2193.       end_resume();
  2194.    }
  2195.    teststart();
  2196.    numpasses = (stdcalcmode == '1') ? 0 : 1;
  2197.    for (passes=startpass; passes <= numpasses ; passes++)
  2198.    {
  2199.       for (row = startrow; row <= iystop; row=row+1+numpasses)
  2200.       {
  2201.      register int col;
  2202.      for (col = 0; col <= ixstop; col++)       /* look at each point on screen */
  2203.      {
  2204.         register color;
  2205.         init.y = dy0[row]+dy1[col];
  2206.         init.x = dx0[col]+dx1[row];
  2207.         if(check_key())
  2208.         {
  2209.            testend();
  2210.            alloc_resume(20,1);
  2211.            put_resume(sizeof(int),&row,sizeof(int),&passes,0);
  2212.            return(-1);
  2213.         }
  2214.         color = testpt(init.x,init.y,parm.x,parm.y,maxit,inside);
  2215.         if (color >= colors) /* avoid trouble if color is 0 */
  2216.            if (colors < 16)
  2217.           color &= andcolor;
  2218.            else
  2219.           color = ((color-1) % andcolor) + 1; /* skip color zero */
  2220.         (*plot)(col,row,color);
  2221.         if(numpasses && (passes == 0))
  2222.            (*plot)(col,row+1,color);
  2223.      }
  2224.       }
  2225.       startrow = passes + 1;
  2226.    }
  2227.    testend();
  2228.    return(0);
  2229. }
  2230.  
  2231.  
  2232. /***************** standalone engine for "plasma" ********************/
  2233.  
  2234. static int iparmx;                /* iparmx = parm.x * 16 */
  2235. static int shiftvalue;                /* shift based on #colors */
  2236.  
  2237. typedef struct palett
  2238. {
  2239.    unsigned char red;
  2240.    unsigned char green;
  2241.    unsigned char blue;
  2242. }
  2243. Palettetype;
  2244.  
  2245. extern Palettetype dacbox[256];
  2246. static int plasma_check;            /* to limit kbd checking */
  2247.  
  2248. static void adjust(int xa,int ya,int x,int y,int xb,int yb)
  2249. {
  2250.    long pseudorandom;
  2251.    if(getcolor(x,y) != 0)
  2252.       return;
  2253.  
  2254.    pseudorandom = ((long)iparmx)*((rand()-16383));
  2255.    pseudorandom = pseudorandom*(abs(xa-xb)+abs(ya-yb));
  2256.    pseudorandom = pseudorandom >> shiftvalue;
  2257.    pseudorandom = ((getcolor(xa,ya)+getcolor(xb,yb)+1)>>1)+pseudorandom;
  2258.    if (pseudorandom <    1) pseudorandom =   1;
  2259.    if (pseudorandom >= colors) pseudorandom = colors-1;
  2260.    putcolor(x,y,(int)pseudorandom);
  2261. }
  2262.  
  2263. static void subDivide(int x1,int y1,int x2,int y2)
  2264. {
  2265.    int x,y;
  2266.    int v;
  2267.    if ((++plasma_check & 0x7f) == 1)
  2268.       if(check_key())
  2269.       {
  2270.      plasma_check--;
  2271.      return;
  2272.       }
  2273.    if(x2-x1<2 && y2-y1<2)
  2274.       return;
  2275.    x = (x1+x2)>>1;
  2276.    y = (y1+y2)>>1;
  2277.    adjust(x1,y1,x ,y1,x2,y1);
  2278.    adjust(x2,y1,x2,y ,x2,y2);
  2279.    adjust(x1,y2,x ,y2,x2,y2);
  2280.    adjust(x1,y1,x1,y ,x1,y2);
  2281.    if(getcolor(x,y) == 0)
  2282.    {
  2283.       v = (getcolor(x1,y1)+getcolor(x2,y1)+getcolor(x2,y2)+
  2284.       getcolor(x1,y2)+2)>>2;
  2285.       putcolor(x,y,v);
  2286.    }
  2287.    subDivide(x1,y1,x ,y);
  2288.    subDivide(x ,y1,x2,y);
  2289.    subDivide(x ,y ,x2,y2);
  2290.    subDivide(x1,y ,x ,y2);
  2291. }
  2292.  
  2293. int plasma()
  2294. {
  2295.    plasma_check = 0;
  2296.  
  2297.    if(colors < 4) {
  2298. static char far plasmamsg[]={"\
  2299. Plasma Clouds can currently only be run in a 4-or-more-color video\n\
  2300. mode (and color-cycled only on VGA adapters [or EGA adapters in their\n\
  2301. 640x350x16 mode])."};
  2302.       stopmsg(0,plasmamsg);
  2303.       return(-1);
  2304.    }
  2305.    iparmx = param[0] * 8;
  2306.    if (parm.x <= 0.0) iparmx = 16;
  2307.    if (parm.x >= 100) iparmx = 800;
  2308.  
  2309.    srand(rseed);
  2310.    if (!rflag) ++rseed;
  2311.  
  2312.    if (colors == 256)            /* set the (256-color) palette */
  2313.       set_Plasma_palette();        /* skip this if < 256 colors */
  2314.  
  2315.    if (colors > 16)            /* adjust based on # of colors */
  2316.       shiftvalue = 18;
  2317.    else {
  2318.       if (colors > 4)
  2319.      shiftvalue = 22;
  2320.       else {
  2321.      if (colors > 2)
  2322.         shiftvalue = 24;
  2323.      else
  2324.         shiftvalue = 25;
  2325.       }
  2326.    }
  2327.  
  2328.    putcolor(      0,      0,1+(((rand()/colors)*(colors-1))>>(shiftvalue-11)));
  2329.    putcolor(xdots-1,      0,1+(((rand()/colors)*(colors-1))>>(shiftvalue-11)));
  2330.    putcolor(xdots-1,ydots-1,1+(((rand()/colors)*(colors-1))>>(shiftvalue-11)));
  2331.    putcolor(      0,ydots-1,1+(((rand()/colors)*(colors-1))>>(shiftvalue-11)));
  2332.  
  2333.    subDivide(0,0,xdots-1,ydots-1);
  2334.    if (! check_key())
  2335.       return(0);
  2336.    else
  2337.       return(1);
  2338. }
  2339.  
  2340. static void set_Plasma_palette()
  2341. {
  2342.    extern char loadPalette;
  2343.    static Palettetype Red    = {
  2344.       63, 0, 0     };
  2345.    static Palettetype Green  = {
  2346.       0,63, 0    };
  2347.    static Palettetype Blue   = {
  2348.       0, 0,63    };
  2349.    int i;
  2350.  
  2351.    if(loadPalette) return;        /* TARGA 3 June 89 j mclain */
  2352.  
  2353.    dacbox[0].red  = 0 ;
  2354.    dacbox[0].green= 0 ;
  2355.    dacbox[0].blue = 0 ;
  2356.    for(i=1;i<=85;i++)
  2357.    {
  2358.       dacbox[i].red   = (i*Green.red   + (86-i)*Blue.red)/85;
  2359.       dacbox[i].green = (i*Green.green + (86-i)*Blue.green)/85;
  2360.       dacbox[i].blue  = (i*Green.blue  + (86-i)*Blue.blue)/85;
  2361.  
  2362.       dacbox[i+85].red     = (i*Red.red    + (86-i)*Green.red)/85;
  2363.       dacbox[i+85].green = (i*Red.green + (86-i)*Green.green)/85;
  2364.       dacbox[i+85].blue  = (i*Red.blue    + (86-i)*Green.blue)/85;
  2365.  
  2366.       dacbox[i+170].red   = (i*Blue.red   + (86-i)*Red.red)/85;
  2367.       dacbox[i+170].green = (i*Blue.green + (86-i)*Red.green)/85;
  2368.       dacbox[i+170].blue  = (i*Blue.blue  + (86-i)*Red.blue)/85;
  2369.    }
  2370.    SetTgaColors();    /* TARGA 3 June 89  j mclain */
  2371.    if (dotmode != 11)
  2372.       spindac(0,1);
  2373. }
  2374.  
  2375.  
  2376. /***************** standalone engine for "diffusion" ********************/
  2377.  
  2378. #define RANDOM(x)  (rand()%(x))
  2379.  
  2380. /* This constant assumes that rand() returns a value from 0 to 32676 */
  2381. #define FOURPI    (long)(4*PI*(double)(1L << 16))
  2382.  
  2383. int diffusion()
  2384. {
  2385.    int xmax,ymax,xmin,ymin;    /* Current maximum coordinates */
  2386.    int border;     /* Distance between release point and fractal */
  2387.    int i;
  2388.    double cosine,sine,angle;
  2389.    long lcosine,lsine;
  2390.    register int x,y;
  2391.    extern char floatflag;
  2392.  
  2393.    if (diskvideo)
  2394.    {
  2395.       notdiskmsg();
  2396.       return(-1);
  2397.    }
  2398.  
  2399.    bitshift = 16;
  2400.    fudge = 1L << 16;
  2401.  
  2402.    border = param[0];
  2403.  
  2404.    if (border <= 0)
  2405.       border = 10;
  2406.  
  2407.    srand(rseed);
  2408.    if (!rflag) ++rseed;
  2409.  
  2410.    xmax = xdots / 2 + border;  /* Initial box */
  2411.    xmin = xdots / 2 - border;
  2412.    ymax = ydots / 2 + border;
  2413.    ymin = ydots / 2 - border;
  2414.  
  2415.    if (resuming) /* restore worklist, if we can't the above will stay in place */
  2416.    {
  2417.       start_resume();
  2418.       get_resume(sizeof(int),&xmax,sizeof(int),&xmin,sizeof(int),&ymax,
  2419.          sizeof(int),&ymin,0);
  2420.       end_resume();
  2421.    }
  2422.  
  2423.    putcolor(xdots / 2, ydots / 2,RANDOM(colors-1)+1);  /* Seed point */
  2424.  
  2425.    while (1)
  2426.    {
  2427.       /* Release new point on circle just inside the box */
  2428.  
  2429.       if (floatflag)
  2430.       {
  2431.      angle=2*(double)rand()/(RAND_MAX/PI);
  2432.      FPUsincos(&angle,&sine,&cosine);
  2433.      x = cosine*(xmax-xmin) + xdots;
  2434.      y = sine  *(ymax-ymin) + ydots;
  2435.       }
  2436.       else
  2437.       {
  2438.      SinCos086(multiply((long)rand(),FOURPI,16),&lsine,&lcosine);
  2439.      x = (lcosine*(long)(xmax-xmin) >> 16) + xdots;
  2440.      y = (lsine  *(long)(ymax-ymin) >> 16) + ydots;
  2441.       }
  2442.  
  2443.       x = x >> 1; /* divide by 2 */
  2444.       y = y >> 1;
  2445.  
  2446.       /* Loop as long as the point (x,y) is surrounded by color 0 */
  2447.       /* on all eight sides                      */
  2448.       while((getcolor(x+1,y+1) == 0) && (getcolor(x+1,y) == 0) &&
  2449.       (getcolor(x+1,y-1) == 0) && (getcolor(x  ,y+1) == 0) &&
  2450.       (getcolor(x  ,y-1) == 0) && (getcolor(x-1,y+1) == 0) &&
  2451.       (getcolor(x-1,y) == 0) && (getcolor(x-1,y-1) == 0))
  2452.       {
  2453.      /* Erase moving point */
  2454.      if (show_orbit)
  2455.         putcolor(x,y,0);
  2456.  
  2457.      /* Make sure point is inside the box */
  2458.      if (x==xmax)
  2459.         x--;
  2460.      else if (x==xmin)
  2461.         x++;
  2462.      if (y==ymax)
  2463.         y--;
  2464.      else if (y==ymin)
  2465.         y++;
  2466.  
  2467.      /* Take one random step */
  2468.      x += RANDOM(3) - 1;
  2469.      y += RANDOM(3) - 1;
  2470.  
  2471.      /* Check keyboard */
  2472.      if ((++plasma_check & 0x7f) == 1)
  2473.         if(check_key())
  2474.         {
  2475.            alloc_resume(20,1);
  2476.            put_resume(sizeof(int),&xmax,sizeof(int),&xmin, sizeof(int),&ymax,
  2477.          sizeof(int),&ymin,0);
  2478.            plasma_check--;
  2479.            return 1;
  2480.         }
  2481.  
  2482.      /* Show the moving point */
  2483.      if (show_orbit)
  2484.         putcolor(x,y,RANDOM(colors-1)+1);
  2485.  
  2486.       }
  2487.       putcolor(x,y,RANDOM(colors-1)+1);
  2488.  
  2489.       /* Is point to close to the edge? */
  2490.       if (((x+border)>xmax) || ((x-border)<xmin)
  2491.       || ((y-border)<ymin) || ((y+border)>ymax))
  2492.       {
  2493.      /* Increase box size, but not past the edge of the screen */
  2494.      if (ymin != 1)
  2495.      {
  2496.         ymin--;
  2497.         ymax++;
  2498.      }
  2499.      if (xmin != 1)
  2500.      {
  2501.         xmin--;
  2502.         xmax++;
  2503.      }
  2504.      if ((ymin==1) || (xmin==1))
  2505.         return 0;
  2506.       }
  2507.    }
  2508. }
  2509.  
  2510.  
  2511. /************ standalone engine for "bifurcation" types ***************/
  2512.  
  2513. /***************************************************************/
  2514. /* The following code now forms a generalised Fractal Engine   */
  2515. /* for Bifurcation fractal typeS.  By rights it now belongs in */
  2516. /* CALCFRACT.C, but it's easier for me to leave it here !      */
  2517.  
  2518. /* Original code by Phil Wilson, hacked around by Kev Allen.   */
  2519.  
  2520. /* Besides generalisation, enhancements include Periodicity    */
  2521. /* Checking during the plotting phase (AND halfway through the */
  2522. /* filter cycle, if possible, to halve calc times), quicker    */
  2523. /* floating-point calculations for the standard Verhulst type, */
  2524. /* and new bifurcation types (integer bifurcation, f.p & int   */
  2525. /* biflambda - the real equivalent of complex Lambda sets -    */
  2526. /* and f.p renditions of bifurcations of r*sin(Pi*p), which    */
  2527. /* spurred Mitchel Feigenbaum on to discover his Number).      */
  2528.  
  2529. /* To add further types, extend the fractalspecific[] array in */
  2530. /* usual way, with Bifurcation as the engine, and the name of  */
  2531. /* the routine that calculates the next bifucation generation  */
  2532. /* as the "orbitcalc" routine in the fractalspecific[] entry.  */
  2533.  
  2534. /* Bifurcation "orbitcalc" routines get called once per screen */
  2535. /* pixel column.  They should calculate the next generation    */
  2536. /* from the doubles Rate & Population (or the longs lRate &    */
  2537. /* lPopulation if they use integer math), placing the result   */
  2538. /* back in Population (or lPopulation).  They should return 0  */
  2539. /* if all is ok, or any non-zero value if calculation bailout  */
  2540. /* is desirable (eg in case of errors, or the series tending   */
  2541. /* to infinity).        Have fun !               */
  2542. /***************************************************************/
  2543.  
  2544. #define BIG 100000.0
  2545. #define DEFAULTFILTER 1000     /* "Beauty of Fractals" recommends using 5000
  2546.                    (p.25), but that seems unnecessary. Can
  2547.                    override this value with a nonzero param1 */
  2548.  
  2549. #define SEED 0.66        /* starting value for population */
  2550.  
  2551. static int far *verhulst_array;
  2552. static unsigned int filter_cycles;
  2553. static unsigned int half_time_check;
  2554. static long   lPopulation, lRate;
  2555. static double Population,  Rate;
  2556. static int    mono, outside_x;
  2557.  
  2558. int Bifurcation(void)
  2559. {
  2560.    unsigned long array_size;
  2561.    int row, column;
  2562.    column = 0;
  2563.    if (resuming)
  2564.    {
  2565.       start_resume();
  2566.       get_resume(sizeof(int),&column,0);
  2567.       end_resume();
  2568.    }
  2569.  
  2570.    array_size =  (iystop + 2) * sizeof(int);
  2571.    if ((verhulst_array = (int far *) farmemalloc(array_size)) == NULL)
  2572.    {
  2573.       stopmsg(0,"Insufficient free memory for calculation.");
  2574.       return(-1);
  2575.    }
  2576.  
  2577.    for (row = 0; row <= iystop+1; row++)
  2578.       verhulst_array[row] = 0;
  2579.  
  2580.    mono = 0;
  2581.    if (colors == 2)
  2582.       mono = 1;
  2583.    if (mono)
  2584.    {
  2585.       if (inside)
  2586.       {
  2587.      outside_x = 0;
  2588.      inside = 1;
  2589.       }
  2590.       else
  2591.      outside_x = 1;
  2592.    }
  2593.  
  2594.    if ((filter_cycles = parm.x) == 0)
  2595.       filter_cycles = DEFAULTFILTER;
  2596.    half_time_check = FALSE;
  2597.    if (periodicitycheck && maxit < filter_cycles)
  2598.    {
  2599.       filter_cycles = (filter_cycles - maxit + 1) / 2;
  2600.       half_time_check = TRUE;
  2601.    }
  2602.  
  2603.    if (integerfractal)
  2604.       linit.y = ly0[iystop];   /* Y-value of    */
  2605.    else
  2606.       init.y = dy0[iystop];   /* bottom pixels */
  2607.  
  2608.    while (column <= ixstop)
  2609.    {
  2610.       if(check_key())
  2611.       {
  2612.      farmemfree((char far *)verhulst_array);
  2613.      alloc_resume(10,1);
  2614.      put_resume(sizeof(int),&column,0);
  2615.      return(-1);
  2616.       }
  2617.       if (integerfractal)
  2618.      lRate = lx0[column];
  2619.       else
  2620.      Rate = dx0[column];
  2621.       verhulst();     /* calculate array once per column */
  2622.  
  2623.       for (row = iystop+1; row > 0; row--)
  2624.       {
  2625.      int color;
  2626.      color = verhulst_array[row];
  2627.      if(color && mono)
  2628.         color = inside;
  2629.      else if((!color) && mono)
  2630.         color = outside_x;
  2631.      verhulst_array[row] = 0;
  2632.      (*plot)(column,row,color);
  2633.       }
  2634.       column++;
  2635.    }
  2636.    farmemfree((char far *)verhulst_array);
  2637.    return(0);
  2638. }
  2639.  
  2640. static void verhulst()        /* P. F. Verhulst (1845) */
  2641. {
  2642.    unsigned int pixel_row, counter, errors;
  2643.  
  2644.    if (integerfractal)
  2645.       lPopulation = SEED * fudge;
  2646.    else
  2647.       Population = SEED;
  2648.  
  2649.    errors = overflow = FALSE;
  2650.  
  2651.    for (counter=0 ; counter < filter_cycles ; counter++)
  2652.    {
  2653.       errors = (*fractalspecific[fractype].orbitcalc)();
  2654.       if (errors)
  2655.      return;
  2656.    }
  2657.  
  2658.    if (half_time_check) /* check for periodicity at half-time */
  2659.    {
  2660.       Bif_Period_Init();
  2661.       for (counter=0 ; counter < maxit ; counter++)
  2662.       {
  2663.      errors = (*fractalspecific[fractype].orbitcalc)();
  2664.      if (errors) return;
  2665.      if (periodicitycheck && Bif_Periodic(counter)) break;
  2666.       }
  2667.       if (counter >= maxit)   /* if not periodic, go the distance */
  2668.       {
  2669.      for (counter=0 ; counter < filter_cycles ; counter++)
  2670.      {
  2671.         errors = (*fractalspecific[fractype].orbitcalc)();
  2672.         if (errors) return;
  2673.      }
  2674.       }
  2675.    }
  2676.  
  2677.    if (periodicitycheck) Bif_Period_Init();
  2678.  
  2679.    for (counter=0 ; counter < maxit ; counter++)
  2680.    {
  2681.       errors = (*fractalspecific[fractype].orbitcalc)();
  2682.       if (errors) return;
  2683.  
  2684.       /* assign population value to Y coordinate in pixels */
  2685.       if (integerfractal)
  2686.      pixel_row = iystop + 1 - (lPopulation - linit.y) / dely;
  2687.       else
  2688.      pixel_row = iystop + 1 - (int)((Population - init.y) / deltaY);
  2689.  
  2690.       /* if it's visible on the screen, save it in the column array */
  2691.       if ((pixel_row >= 0) && (pixel_row <= iystop + 1))
  2692.      verhulst_array[ pixel_row ] ++;
  2693.  
  2694.       if (periodicitycheck && Bif_Periodic(counter))
  2695.       {
  2696.      if ((pixel_row >= 0) && (pixel_row <= iystop + 1))
  2697.         verhulst_array[ pixel_row ] --;
  2698.      break;
  2699.       }
  2700.    }
  2701. }
  2702. static    long    lBif_closenuf, lBif_savedpop;    /* poss future use  */
  2703. static    double     Bif_closenuf,    Bif_savedpop;
  2704. static    int     Bif_savedinc,    Bif_savedand;
  2705.  
  2706. static void Bif_Period_Init()
  2707. {
  2708.    Bif_savedinc = 1;
  2709.    Bif_savedand = 1;
  2710.    if (integerfractal)
  2711.    {
  2712.       lBif_savedpop = -1;
  2713.       lBif_closenuf = dely / 8;
  2714.    }
  2715.    else
  2716.    {
  2717.       Bif_savedpop = -1.0;
  2718.       Bif_closenuf = deltaY / 8.0;
  2719.    }
  2720. }
  2721.  
  2722. static int Bif_Periodic (time)    /* Bifurcation Population Periodicity Check */
  2723. int time;        /* Returns : 1 if periodicity found, else 0 */
  2724. {
  2725.    if ((time & Bif_savedand) == 0)    /* time to save a new value */
  2726.    {
  2727.       if (integerfractal) lBif_savedpop = lPopulation;
  2728.       else             Bif_savedpop =  Population;
  2729.       if (--Bif_savedinc == 0)
  2730.       {
  2731.      Bif_savedand = (Bif_savedand << 1) + 1;
  2732.      Bif_savedinc = 4;
  2733.       }
  2734.    }
  2735.    else             /* check against an old save */
  2736.    {
  2737.       if (integerfractal)
  2738.       {
  2739.      if (labs(lBif_savedpop-lPopulation) <= lBif_closenuf)
  2740.         return(1);
  2741.       }
  2742.       else
  2743.       {
  2744.      if (fabs(Bif_savedpop-Population) <= Bif_closenuf)
  2745.         return(1);
  2746.       }
  2747.    }
  2748.    return(0);
  2749. }
  2750.  
  2751. /**********************************************************************/
  2752. /*                                                      */
  2753. /* The following are Bifurcation "orbitcalc" routines...              */
  2754. /*                                                      */
  2755. /**********************************************************************/
  2756.  
  2757. int BifurcVerhulst()
  2758.   {
  2759.     Population += Rate * Population * (1 - Population);
  2760.     return (fabs(Population) > BIG);
  2761.   }
  2762.  
  2763. int LongBifurcVerhulst()
  2764.   {
  2765.     ltmp.y = lPopulation - multiply(lPopulation,lPopulation,bitshift);
  2766.     lPopulation += multiply(lRate,ltmp.y,bitshift);
  2767.     return (overflow);
  2768.   }
  2769.  
  2770. int BifurcLambda()
  2771.   {
  2772.     Population = Rate * Population * (1 - Population);
  2773.     return (fabs(Population) > BIG);
  2774.   }
  2775.  
  2776. int LongBifurcLambda()
  2777.   {
  2778.     ltmp.y = lPopulation - multiply(lPopulation,lPopulation,bitshift);
  2779.     lPopulation = multiply(lRate,ltmp.y,bitshift);
  2780.     return (overflow);
  2781.   }
  2782.  
  2783. int BifurcAddSinPi()
  2784.   {
  2785.     Population += Rate * sin(PI*Population);
  2786.     return (fabs(Population) > BIG);
  2787.   }
  2788.  
  2789. int BifurcSetSinPi()
  2790.   {
  2791.     Population = Rate * sin(PI*Population);
  2792.     return (fabs(Population) > BIG);
  2793.   }
  2794.  
  2795. /* Here Endeth the Generalised Bifurcation Fractal Engine   */
  2796.  
  2797. /* END Phil Wilson's Code (modified slightly by Kev Allen et. al. !) */
  2798.  
  2799.  
  2800. /******************* standalone engine for "popcorn" ********************/
  2801.  
  2802. int popcorn()    /* subset of std engine */
  2803. {
  2804.    int start_row;
  2805.    start_row = 0;
  2806.    if (resuming)
  2807.    {
  2808.       start_resume();
  2809.       get_resume(sizeof(int),&start_row,0);
  2810.       end_resume();
  2811.    }
  2812.    kbdcount=(cpu==386) ? 80 : 30;
  2813.    plot = noplot;
  2814.    tempsqrx = ltempsqrx = 0; /* PB added this to cover weird BAILOUTs */
  2815.    for (row = start_row; row <= iystop; row++)
  2816.    {
  2817.       reset_periodicity = 1;
  2818.       for (col = 0; col <= ixstop; col++)
  2819.       {
  2820.      if (StandardFractal() == -1) /* interrupted */
  2821.      {
  2822.         alloc_resume(10,1);
  2823.         put_resume(sizeof(int),&row,0);
  2824.         return(-1);
  2825.      }
  2826.      reset_periodicity = 0;
  2827.       }
  2828.    }
  2829.    calc_status = 4;
  2830.    return(0);
  2831. }
  2832.  
  2833.  
  2834.