ftp://ftp.krokus.ru/pub/OpenBSD/distfiles/xscreensaver-5.01.tar.gz
[xscreensaver] / hacks / fireworkx.c
index 7f1002e60812a07a8a154d976256875f1518f52a..33f37178a569ef881a9714f79a4fff89a755bd27 100644 (file)
@@ -1,7 +1,9 @@
 /*
- * Fireworkx 1.3 - pyrotechnics simulation program
- * Copyright (c) 1999-2004 Rony B Chandran <ronybc@asia.com>
+ * Fireworkx 1.5 - pyrotechnics simulation program
+ * Copyright (c) 1999-2005 Rony B Chandran <ronybc@asia.com>
  *
+ * From Kerala, INDIA
+ * 
  * url: http://www.ronybc.8k.com
  *
  * Permission to use, copy, modify, distribute, and sell this software and its
  *
  * Additional programming: 
  * ------------------------
- * Support for different display color modes: 
+ * Support for different dpy color modes: 
  * Jean-Pierre Demailly <Jean-Pierre.Demailly@ujf-grenoble.fr>
  *
+ * Fixed array access problems by beating on it with a large hammer.
+ * Nicholas Miell <nmiell@gmail.com>
+ *
+ * Freed 'free'ing up of memory by adding the forgotten 'XSync's
+ * Renuka S <renuka@local.net>
+ *
  */
 
 #include <math.h>
 #include "screenhack.h"
-#include <X11/Xutil.h>
 
-#define FWXVERSION "1.3"
+#define FWXVERSION "1.5"
 
-#define WIDTH 640
-#define HEIGHT 480
-#define SHELLCOUNT 3
+#define SHELLCOUNT 3                   /* 3 or 5  */
 #define PIXCOUNT 500                   /* 500     */
-#define RNDLIFE0 250                   /* violent */
-#define RNDLIFE1 1200                  /* 1200    */
-#define MINLIFE0 50                    /* violent */
-#define MINLIFE1 500                   /* 500     */
 #define POWER 5                        /* 5       */
 #define FTWEAK 12                      /* 12      */
 
-static int depth;
-static int bigendian;
-static Bool light_on = True;
-static Bool fade_on  = False;
-static Bool shoot    = False;
-static Bool verbose  = False;
-static int delay     = 0;
-static int fsc_width = 0;
-static int fsc_height= 0;
-static int rndlife = RNDLIFE1;
-static int minlife = MINLIFE1;
-static float light_fade = 0.99;
-static unsigned char *palaka1=NULL;
-static unsigned char *palaka2=NULL;
-static XImage *xim=NULL;
-static XColor *colors;
-static int ncolors = 255;
+#if HAVE_X86_MMX
+void mmx_blur(char *a, int b, int c, int d); 
+void mmx_glow(char *a, int b, int c, int d, char *e);
+#endif
+
+#define rnd(x) ((x) ? (int)(random() % (x)) : 0)
 
 typedef struct {
   unsigned int burn;
@@ -62,49 +52,69 @@ typedef struct {
   float yv;}firepix;
 
 typedef struct {
+  unsigned int cx,cy;
   unsigned int life;
   unsigned int color;
-  unsigned int cx,cy;
   unsigned int special;
   unsigned int cshift;
   unsigned int vgn,shy;
-  float air,lum;
+  float air,flash;
   firepix *fpix; }fireshell;
 
-int seed = 2387776;
-int rnd(int x)
-{   /* xscreensaver */
-  if ((seed = seed % 44488 * 48271 - seed / 44488 * 3399) < 0)
-  seed += 2147483647;
-  return (seed-1) % x;
-}
+struct state {
+  Display *dpy;
+  Window window;
+
+  fireshell *fshell, *ffshell;
+  GC gc;
+
+  int depth;
+  int bigendian;
+  Bool flash_on;
+  Bool glow_on;
+  Bool verbose;
+  Bool shoot;
+  int width;
+  int height;
+  int rndlife;
+  int minlife;
+  int delay;
+  float flash_fade;
+  unsigned char *palaka1;
+  unsigned char *palaka2;
+  XImage *xim;
+  XColor *colors;
+  int ncolors;
+};
 
-int explode(fireshell *fs)
+static int explode(struct state *st, fireshell *fs)
 {
-  float air,adg = 0.001;     /* gravity */
+  float air,adg = 0.001;         /* gravity */
   unsigned int n,c;
-  unsigned int h = fsc_height;
-  unsigned int w = fsc_width;
+  unsigned int h = st->height;
+  unsigned int w = st->width;
   unsigned int *prgb;
-  unsigned char *palaka = palaka1;
+  unsigned char *palaka = st->palaka1;
   firepix *fp = fs->fpix;
   if(fs->vgn){
     if(--fs->cy == fs->shy){  
-      fs->vgn = 0;
-      fs->lum = 20000;}    /* millions of jouls..! */
+      fs->vgn   = 0;
+      fs->flash = rnd(30000)+15000;}
     else{  
-      fs->lum = 50+(fs->cy - fs->shy)*2;
+      fs->flash = 50+(fs->cy - fs->shy)*2;
+      prgb=(unsigned int *)(palaka + (fs->cy * w + fs->cx + rnd(5)-2)*4);
+     *prgb=(rnd(8)+8)*0x000f0f10;
       return(1);}}    
   if(fs->cshift) --fs->cshift;
   if((fs->cshift+1)%50==0) fs->color = ~fs->color;
   c = fs->color;
   air = fs->air;
-  fs->lum *= light_fade;
+  fs->flash *= st->flash_fade;
   for(n=PIXCOUNT;n;n--){
   if(fp->burn){ --fp->burn; 
   if(fs->special){
   fp->x += fp->xv = fp->xv * air + (float)(rnd(200)-100)/2000;
-  fp->y += fp->yv = fp->yv * air + (float)(rnd(200)-100)/2000 +adg; }
+  fp->y += fp->yv = fp->yv * air + (float)(rnd(200)-100)/2000 + adg; }
   else{
   fp->x += fp->xv = fp->xv * air + (float)(rnd(200)-100)/20000;
   fp->y += fp->yv = fp->yv * air + adg; }
@@ -115,25 +125,26 @@ int explode(fireshell *fs)
      prgb = (unsigned int *)(palaka + ((int)fp->y * w + (int)fp->x)*4);
     *prgb = c; }
   } fp++;
-  } return(--(fs->life));
-}
+  } return(--fs->life); }
 
-void recycle(fireshell *fs,int x,int y)
+static void recycle(struct state *st, fireshell *fs,int x,int y)
 {
   unsigned int n,pixlife;
   firepix *fp = fs->fpix;
-  fs->vgn = shoot;
+  fs->vgn = st->shoot;
   fs->shy = y;
   fs->cx = x;
-  fs->cy = shoot ? fsc_height : y ;
+  fs->cy = st->shoot ? st->height : y ;
   fs->color = (rnd(155)+100) <<16 |
               (rnd(155)+100) <<8  |
                rnd(255);
-  fs->life = rnd(rndlife)+minlife;
+  fs->life = rnd(st->rndlife)+st->minlife;
   fs->air  = 1-(float)(rnd(200))/10000;
-  fs->lum  = 20000;
+  fs->flash   = rnd(30000)+15000;        /* million jouls */
   fs->cshift  = !rnd(5) ? 120:0; 
   fs->special = !rnd(10) ? 1:0; 
+  if(st->verbose)
+  printf("recycle(): color = %x air = %f life = %d \n",fs->color,fs->air,fs->life);
   pixlife = rnd(fs->life)+fs->life/10+1;    /* ! */
   for(n=0;n<PIXCOUNT;n++){
   fp->burn = rnd(pixlife)+32;
@@ -142,55 +153,77 @@ void recycle(fireshell *fs,int x,int y)
                (float)(rnd(20000)-10000)/10000;
   fp->x = x;
   fp->y = y; 
-  fp++;             }
-}
+  fp++;             }}
 
-void blur_best(void)
+static void glow(struct state *st)
 {
-  unsigned int n;
-  unsigned int w = fsc_width;
-  unsigned int h = fsc_height;
-  unsigned char *pa, *pb, *pm;
-  pm = palaka1;
-  for(n=0;n<w*4;n++) pm[n]=0;    /* clean first line */
-  pm+=n;
-  h-=2; 
+  unsigned int n,q;
+  unsigned int w = st->width;
+  unsigned int h = st->height;
+  unsigned char *pa, *pb, *pm, *po;
+  if (!st->xim) return;
+  pm = st->palaka1;
+  po = st->palaka2;
+  for(n=0;n<w*4;n++) 
+  {pm[n]=0; po[n]=0;}    /* clean first line */
+  pm+=n; po+=n; h-=2; 
   pa = pm-(w*4);
   pb = pm+(w*4);
-  if(fade_on){
-  for(n=0;n<w*h*4;n++){
-  pm[n]=(pm[n-4] + pm[n] + pm[n+4] + 
+  for(n=4;(signed) n< (signed) (w*h*4-4); n++){
+  q    = pm[n-4] + (pm[n]*8) + pm[n+4] + 
          pa[n-4] + pa[n] + pa[n+4] + 
-         pb[n-4] + pb[n] + pb[n+4] +
-         pm[n] + pm[n] + (pm[n]<<2))/ 16;}}
-  else{
-  for(n=0;n<w*h*4;n++){
-  pm[n]=(pm[n-4] + pm[n] + pm[n+4] + 
+         pb[n-4] + pb[n] + pb[n+4];
+  q    -= q>8 ? 8:q;
+  pm[n] = q/16;
+  q     = q/8;
+  if(q>255) q=255;
+  po[n] = q;}
+  pm+=n; po+=n;
+  for(n=0;n<w*4;n++)
+  {pm[n]=0; po[n]=0;}}   /* clean last line */
+
+static void blur(struct state *st)
+{
+  unsigned int n,q;
+  unsigned int w = st->width;
+  unsigned int h = st->height;
+  unsigned char *pa, *pb, *pm;
+  pm = st->palaka1;
+  pm += w*4; h-=2;  /* line 0&h */
+  pa = pm-(w*4);
+  pb = pm+(w*4);
+  for(n=4;n<w*h*4-4;n++){
+  q    = pm[n-4] + (pm[n]*8) + pm[n+4] + 
          pa[n-4] + pa[n] + pa[n+4] + 
-         pb[n-4] + pb[n] + pb[n+4])/ 9;}}
-  pm+=n;
-  for(n=0;n<w*4;n++) pm[n]=0;    /* clean last line */
-}
+         pb[n-4] + pb[n] + pb[n+4];
+  q    -= q>8 ? 8:q;
+  pm[n] = q>>4;}
+  pm += n-4;    /* last line */
+  for(n=0;n<w*4+4;n++) pm[n]=0;
+  pm = st->palaka1; /* first line */
+  for(n=0;n<w*4+4;n++) pm[n]=pm[n+w*4]; }
 
-void light_2x2(fireshell *fss)
+static void light_2x2(struct state *st, fireshell *fss)
 {
   unsigned int l,t,n,x,y;
   float s;
-  int w = fsc_width;
-  int h = fsc_height;
-  unsigned char *dim = palaka2;
-  unsigned char *sim = palaka1;
+  int w = st->width;
+  int h = st->height;
+  unsigned char *dim = st->palaka2;
+  unsigned char *sim = st->palaka1;
   int nl = w*4;
   fireshell *f;
+  if(st->glow_on) sim=dim;
   for(y=0;y<h;y+=2){
   for(x=0;x<w;x+=2){
   f = fss; s = 0;
   for(n=SHELLCOUNT;n;n--,f++){
-  s += f->lum / ( 1 + sqrt((f->cx - x)*(f->cx - x)+(f->cy - y)*(f->cy - y))); }
+  s += f->flash/(sqrt(1+(f->cx - x)*(f->cx - x)+
+                        (f->cy - y)*(f->cy - y)));}
   l = s;
 
   t = l + sim[0];
-  dim[0] = (t > 255 ? 255 : t);         /* cmov's */
+  dim[0] = (t > 255 ? 255 : t);        
   t = l + sim[1];
   dim[1] = (t > 255 ? 255 : t);
   t = l + sim[2];
@@ -217,206 +250,238 @@ void light_2x2(fireshell *fss)
   t = l + sim[nl+6];
   dim[nl+6] = (t > 255 ? 255 : t);
 
-  sim += 8; dim += 8; } sim += nl; dim += nl;}
-}
+  sim += 8; dim += 8; } sim += nl; dim += nl;}}
 
-void resize(Display *display, Window win)
+static void resize(struct state *st)
 {
   XWindowAttributes xwa;
-  XGetWindowAttributes (display, win, &xwa);
-  xwa.width  -= xwa.width % 4;
-  xwa.height -= xwa.height % 4;
-  if(xwa.height != fsc_height || xwa.width != fsc_width) {
-  fsc_width  = xwa.width;
-  fsc_height = xwa.height;
-  if (xim) {
-  if (xim->data==(char *)palaka2) xim->data=NULL;  
-  XDestroyImage(xim);
-  if (palaka2!=palaka1) free(palaka2);
-  free(palaka1); 
+  XGetWindowAttributes (st->dpy, st->window, &xwa);
+  xwa.width  -= xwa.width % 2;
+  xwa.height -= xwa.height % 2;
+  if(xwa.height != st->height || xwa.width != st->width) {
+  st->width  = xwa.width;
+  st->height = xwa.height;
+  if (st->verbose)
+  printf("sky size: %dx%d ------------------------------\n",st->width,st->height);
+  if (st->xim) {
+  if (st->xim->data==(char *)st->palaka2) st->xim->data=NULL;  
+  XDestroyImage(st->xim);
+  if (st->palaka2!=st->palaka1) free(st->palaka2 - 8);
+  free(st->palaka1 - 8); 
   }
-  palaka1 = NULL;     
-  palaka2 = NULL; 
-  xim = XCreateImage(display, xwa.visual, xwa.depth, ZPixmap, 0, 0,
-                    fsc_width, fsc_height, 32, 0);
-  palaka1 = calloc(xim->height,xim->width*4);
-  if(light_on)
-  palaka2 = calloc(xim->height,xim->width*4);
+  st->palaka1 = NULL;
+  st->palaka2 = NULL;
+  st->xim = XCreateImage(st->dpy, xwa.visual, xwa.depth, ZPixmap, 0, 0,
+                    st->width, st->height, 8, 0);
+  if (!st->xim) return;
+  st->palaka1 = (unsigned char *) calloc(st->xim->height+1,st->xim->width*4) + 8;
+  if(st->flash_on|st->glow_on)
+  st->palaka2 = (unsigned char *) calloc(st->xim->height+1,st->xim->width*4) + 8;
   else
-  palaka2 = palaka1;
-  if (depth>=24)
-  xim->data = (char *)palaka2;
+  st->palaka2 = st->palaka1;
+  if (st->depth>=24)
+  st->xim->data = (char *)st->palaka2;
   else
-  xim->data = calloc(xim->height,xim->bytes_per_line);       
- }
-}
+  st->xim->data = calloc(st->xim->height,st->xim->bytes_per_line); }}
 
-void put_image(Display *display, Window win, GC gc, XImage *xim)
+static void put_image(struct state *st)
 {
   int x,y,i,j;
   unsigned char r, g, b;
+  if (!st->xim) return;
   i = 0;
   j = 0;
-  if (depth==16) {
-     if(bigendian)
-     for (y=0;y<xim->height; y++)
-     for (x=0;x<xim->width; x++) {
-     r = palaka2[j++];
-     g = palaka2[j++];
-     b = palaka2[j++];
+  if (st->depth==16) {
+     if(st->bigendian)
+     for (y=0;y<st->xim->height; y++)
+     for (x=0;x<st->xim->width; x++) {
+     r = st->palaka2[j++];
+     g = st->palaka2[j++];
+     b = st->palaka2[j++];
      j++;
-     xim->data[i++] = (g&224)>>5 | (r&248);
-     xim->data[i++] = (b&248)>>3 | (g&28)<<3;
+     st->xim->data[i++] = (g&224)>>5 | (r&248);
+     st->xim->data[i++] = (b&248)>>3 | (g&28)<<3;
      }
      else
-     for (y=0;y<xim->height; y++)
-     for (x=0;x<xim->width; x++) {
-     r = palaka2[j++];
-     g = palaka2[j++];
-     b = palaka2[j++];
+     for (y=0;y<st->xim->height; y++)
+     for (x=0;x<st->xim->width; x++) {
+     r = st->palaka2[j++];
+     g = st->palaka2[j++];
+     b = st->palaka2[j++];
      j++;
-     xim->data[i++] = (b&248)>>3 | (g&28)<<3;
-     xim->data[i++] = (g&224)>>5 | (r&248);
+     st->xim->data[i++] = (b&248)>>3 | (g&28)<<3;
+     st->xim->data[i++] = (g&224)>>5 | (r&248);
      }
   }
-  if (depth==15) {
-     if(bigendian)
-     for (y=0;y<xim->height; y++)
-     for (x=0;x<xim->width; x++) {
-     r = palaka2[j++];
-     g = palaka2[j++];
-     b = palaka2[j++];
+  if (st->depth==15) {
+     if(st->bigendian)
+     for (y=0;y<st->xim->height; y++)
+     for (x=0;x<st->xim->width; x++) {
+     r = st->palaka2[j++];
+     g = st->palaka2[j++];
+     b = st->palaka2[j++];
      j++;
-     xim->data[i++] = (g&192)>>6 | (r&248)>>1;
-     xim->data[i++] = (b&248)>>3 | (g&56)<<2;
+     st->xim->data[i++] = (g&192)>>6 | (r&248)>>1;
+     st->xim->data[i++] = (b&248)>>3 | (g&56)<<2;
      }
      else
-     for (y=0;y<xim->height; y++)
-     for (x=0;x<xim->width; x++) {
-     r = palaka2[j++];
-     g = palaka2[j++];
-     b = palaka2[j++];
+     for (y=0;y<st->xim->height; y++)
+     for (x=0;x<st->xim->width; x++) {
+     r = st->palaka2[j++];
+     g = st->palaka2[j++];
+     b = st->palaka2[j++];
      j++;
-     xim->data[i++] = (b&248)>>3 | (g&56)<<2;
-     xim->data[i++] = (g&192)>>6 | (r&248)>>1;
+     st->xim->data[i++] = (b&248)>>3 | (g&56)<<2;
+     st->xim->data[i++] = (g&192)>>6 | (r&248)>>1;
      }
   }
-  if (depth==8) {
-     for (y=0;y<xim->height; y++)
-     for (x=0;x<xim->width; x++) {
-     r = palaka2[j++];
-     g = palaka2[j++];
-     b = palaka2[j++];
+  if (st->depth==8) {
+     for (y=0;y<st->xim->height; y++)
+     for (x=0;x<st->xim->width; x++) {
+     r = st->palaka2[j++];
+     g = st->palaka2[j++];
+     b = st->palaka2[j++];
      j++;     
-     xim->data[i++] = (((7*g)/256)*36)+(((6*r)/256)*6)+((6*b)/256);
+     st->xim->data[i++] = (((7*g)/256)*36)+(((6*r)/256)*6)+((6*b)/256);
      }
   }
-  XPutImage(display,win,gc,xim,0,0,0,0,xim->width,xim->height); 
-}
-
-void sniff_events(Display *dis, Window win, fireshell *fss)
-{
-  XEvent e;
-  while (XPending(dis)){
-  XNextEvent (dis, &e);
-  if (e.type == ConfigureNotify) resize(dis,win);
-  if (e.type == ButtonPress)     recycle(fss,e.xbutton.x, e.xbutton.y);
-  screenhack_handle_event(dis,&e);}
-}
-
+  XPutImage(st->dpy,st->window,st->gc,st->xim,0,0,0,0,st->xim->width,st->xim->height); }
 
-char *progclass = "Fireworkx";
-
-char *defaults [] = {
-  ".background:        black",
-  ".foreground:        white",
-  "*delay:     5000",
-  "*maxlife:   1200",
-  "*light:     True",
-  "*fade:      False",
-  "*shoot:     False",
-  "*verbose:   False",
-  0
-};
-
-XrmOptionDescRec options [] = {
-  { "-delay",          ".delay",       XrmoptionSepArg, 0 },
-  { "-maxlife",                ".maxlife",     XrmoptionSepArg, 0 },
-  { "-nolight",                ".light",       XrmoptionNoArg, "False" },
-  { "-fade",           ".fade",        XrmoptionNoArg, "True" },
-  { "-shoot",          ".shoot",       XrmoptionNoArg, "True" },
-  { "-verbose",                ".verbose",     XrmoptionNoArg, "True" },
-  { 0, 0, 0, 0 }
-};
 
-void
-screenhack (Display *display, Window win)
+static void *
+fireworkx_init (Display *dpy, Window win)
 {
-  unsigned int n,q;
+  struct state *st = (struct state *) calloc (1, sizeof(*st));
+  unsigned int n;
   Visual *vi;
   Colormap cmap;
   Bool writable;
   XWindowAttributes xwa;
-  GC gc;
   XGCValues gcv;
-  firepix *fpixs, *fpix;
-  fireshell *fshells, *fshell;
-  fade_on  = get_boolean_resource("fade"    , "Boolean");
-  light_on = get_boolean_resource("light"   , "Boolean");
-  shoot    = get_boolean_resource("shoot"   , "Boolean");
-  verbose  = get_boolean_resource("verbose" , "Boolean");
-  rndlife  = get_integer_resource("maxlife" , "Integer");
-  delay    = get_integer_resource("delay"   , "Integer");
-  minlife  = rndlife/4;
-  if(rndlife<1000) light_fade=0.98;
-  if(rndlife<500) light_fade=0.97;
-  if(fade_on) light_fade=0.97;
-  if(verbose){
+  firepix *fpix, *ffpix;
+
+  st->dpy = dpy;
+  st->window = win;
+
+  st->glow_on  = get_boolean_resource(st->dpy, "glow"    , "Boolean");
+  st->flash_on = get_boolean_resource(st->dpy, "flash"   , "Boolean");
+  st->shoot    = get_boolean_resource(st->dpy, "shoot"   , "Boolean");
+  st->verbose  = get_boolean_resource(st->dpy, "verbose" , "Boolean");
+  st->rndlife  = get_integer_resource(st->dpy, "maxlife" , "Integer");
+  st->delay    = get_integer_resource(st->dpy, "delay"   , "Integer");
+  st->minlife  = st->rndlife/4;
+  if(st->rndlife<1000) st->flash_fade=0.98;
+  if(st->rndlife<500) st->flash_fade=0.97;
+  if(st->verbose){
   printf("Fireworkx %s - pyrotechnics simulation program \n", FWXVERSION);
-  printf("Copyright (c) 1999-2004 Rony B Chandran <ronybc@asia.com> \n\n");
+  printf("Copyright (c) 1999-2005 Rony B Chandran <ronybc@asia.com> \n\n");
   printf("url: http://www.ronybc.8k.com \n\n");}
 
-  XGetWindowAttributes(display,win,&xwa);
-  depth     = xwa.depth;
+  XGetWindowAttributes(st->dpy,win,&xwa);
+  st->depth     = xwa.depth;
   vi        = xwa.visual;
   cmap      = xwa.colormap;
-  bigendian = (ImageByteOrder(display) == MSBFirst);
+  st->bigendian = (ImageByteOrder(st->dpy) == MSBFirst);
  
-  if(depth==8){
-  if(verbose){
-  printf("Pseudocolor color: use '-fade' & '-nolight' for better results.\n");}
-  colors = (XColor *) calloc(sizeof(XColor),ncolors+1);
+  if(st->depth==8){
+  if(st->verbose){
+  printf("Pseudocolor color: use '-no-flash' for better results.\n");}
+  st->colors = (XColor *) calloc(sizeof(XColor),st->ncolors+1);
   writable = False;
-  make_smooth_colormap(display, vi, cmap, colors, &ncolors,
+  make_smooth_colormap(st->dpy, vi, cmap, st->colors, &st->ncolors,
                                 False, &writable, True);
   }
-  gc = XCreateGC(display, win, 0, &gcv);
+  st->gc = XCreateGC(st->dpy, win, 0, &gcv);
 
-  resize(display,win);   /* initialize palakas */ 
-  seed += time(0);
+  resize(st);   /* initialize palakas */ 
   
-  fpixs = malloc(sizeof(firepix) * PIXCOUNT * SHELLCOUNT);
-  fshells = malloc(sizeof(fireshell) * SHELLCOUNT);
-  fshell = fshells;
-  fpix = fpixs;
+  ffpix = malloc(sizeof(firepix) * PIXCOUNT * SHELLCOUNT);
+  st->ffshell = malloc(sizeof(fireshell) * SHELLCOUNT);
+  st->fshell = st->ffshell;
+  fpix = ffpix;
   for (n=0;n<SHELLCOUNT;n++){
-  fshell->fpix = fpix;
-  recycle (fshell,rnd(fsc_width),rnd(fsc_height));
-  fshell++; 
+  st->fshell->fpix = fpix;
+  recycle (st, st->fshell,rnd(st->width),rnd(st->height));
+  st->fshell++; 
   fpix += PIXCOUNT; }
   
-  while(1) {
+  return st;
+}
+
+
+static unsigned long
+fireworkx_draw (Display *dpy, Window win, void *closure)
+{
+  struct state *st = (struct state *) closure;
+  int q,n;
   for(q=FTWEAK;q;q--){
-  fshell=fshells;
-  for(n=SHELLCOUNT;n;n--){
-  if (!explode(fshell)){
-       recycle(fshell,rnd(fsc_width),rnd(fsc_height)); }
-       fshell++; }}
-  if(light_on) light_2x2(fshells);
-  put_image(display,win,gc,xim);
-  usleep(delay);
-  XSync(display,0);
-  sniff_events(display, win, fshells);
-  blur_best();}
+    st->fshell=st->ffshell;
+    for(n=SHELLCOUNT;n;n--){
+      if (!explode(st, st->fshell)){
+        recycle(st, st->fshell,rnd(st->width),rnd(st->height)); }
+      st->fshell++; }}
+#if HAVE_X86_MMX
+  if(st->glow_on) mmx_glow(st->palaka1,st->width,st->height,8,st->palaka2);
+#else
+  if(st->glow_on) glow(st);
+#endif
+  if(st->flash_on) light_2x2(st, st->ffshell);
+  put_image(st);
+#if HAVE_X86_MMX
+  if(!st->glow_on) mmx_blur(st->palaka1,st->width,st->height,8);
+#else
+  if(!st->glow_on) blur(st);
+#endif
+
+  return st->delay;
+}
+
+
+static void
+fireworkx_reshape (Display *dpy, Window window, void *closure, 
+                 unsigned int w, unsigned int h)
+{
+  struct state *st = (struct state *) closure;
+  resize(st);
+}
+
+static Bool
+fireworkx_event (Display *dpy, Window window, void *closure, XEvent *event)
+{
+  struct state *st = (struct state *) closure;
+  if (event->type == ButtonPress) {
+    recycle(st, st->fshell, event->xbutton.x, event->xbutton.y);
+    return True;
+  }
+  return False;
+}
 
+static void
+fireworkx_free (Display *dpy, Window window, void *closure)
+{
 }
+
+
+static const char *fireworkx_defaults [] = {
+  ".background:        black",
+  ".foreground:        white",
+  "*delay:     20000",  /* never default to zero! */
+  "*maxlife:   2000",
+  "*flash:     True",
+  "*glow:      True",
+  "*shoot:     False",
+  "*verbose:   False",
+  0
+};
+
+static XrmOptionDescRec fireworkx_options [] = {
+  { "-delay",          ".delay",       XrmoptionSepArg, 0 },
+  { "-maxlife",                ".maxlife",     XrmoptionSepArg, 0 },
+  { "-no-flash",       ".flash",       XrmoptionNoArg, "False" },
+  { "-no-glow",                ".glow",        XrmoptionNoArg, "False" },
+  { "-shoot",          ".shoot",       XrmoptionNoArg, "True" },
+  { "-verbose",                ".verbose",     XrmoptionNoArg, "True" },
+  { 0, 0, 0, 0 }
+};
+
+XSCREENSAVER_MODULE ("Fireworkx", fireworkx)