mbrot-mt

changeset 0:e9ae6289e14f

initial commit
author John Tsiombikas <nuclear@member.fsf.org>
date Fri, 07 Mar 2014 07:42:48 +0200 (2014-03-07)
parents
children 3a893f9831ac
files Makefile src/mbrot-mt.c
diffstat 2 files changed, 172 insertions(+), 0 deletions(-) [+]
line diff
     1.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     1.2 +++ b/Makefile	Fri Mar 07 07:42:48 2014 +0200
     1.3 @@ -0,0 +1,13 @@
     1.4 +src = $(wildcard src/*.c)
     1.5 +obj = $(src:.c=.o)
     1.6 +bin = mbrot
     1.7 +
     1.8 +CFLAGS = -pedantic -Wall -g `pkg-config --cflags sdl`
     1.9 +LDFLAGS = `pkg-config --libs sdl`
    1.10 +
    1.11 +$(bin): $(obj)
    1.12 +	$(CC) -o $@ $(obj) $(LDFLAGS)
    1.13 +
    1.14 +.PHONY: clean
    1.15 +clean:
    1.16 +	rm -f $(obj) $(bin)
     2.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     2.2 +++ b/src/mbrot-mt.c	Fri Mar 07 07:42:48 2014 +0200
     2.3 @@ -0,0 +1,159 @@
     2.4 +#include <stdio.h>
     2.5 +#include <complex.h>
     2.6 +#include <SDL/SDL.h>
     2.7 +
     2.8 +void display(void);
     2.9 +int mandelbrot(float x, float y, int max_iter, int *iter);
    2.10 +int handle_event(SDL_Event *ev);
    2.11 +
    2.12 +static SDL_Surface *fbsurf;
    2.13 +static int win_width, win_height;
    2.14 +static float win_aspect;
    2.15 +static float xoffs, yoffs, zoom = 1.0;
    2.16 +static int max_iter = 64;
    2.17 +
    2.18 +int main(void)
    2.19 +{
    2.20 +	win_width = 800;
    2.21 +	win_height = 600;
    2.22 +	win_aspect = (float)win_width / (float)win_height;
    2.23 +
    2.24 +	SDL_Init(SDL_INIT_VIDEO | SDL_INIT_TIMER);
    2.25 +	if(!(fbsurf = SDL_SetVideoMode(win_width, win_height, 32, SDL_SWSURFACE | SDL_RESIZABLE))) {
    2.26 +		fprintf(stderr, "failed to create framebuffer\n");
    2.27 +		return 1;
    2.28 +	}
    2.29 +
    2.30 +	SDL_WM_SetCaption("Mandelbrot", 0);
    2.31 +
    2.32 +	for(;;) {
    2.33 +		SDL_Event ev;
    2.34 +
    2.35 +		while(SDL_PollEvent(&ev)) {
    2.36 +			if(handle_event(&ev) == -1) {
    2.37 +				goto done;
    2.38 +			}
    2.39 +		}
    2.40 +
    2.41 +		display();
    2.42 +	}
    2.43 +
    2.44 +done:
    2.45 +	SDL_Quit();
    2.46 +	return 0;
    2.47 +}
    2.48 +
    2.49 +void display(void)
    2.50 +{
    2.51 +	unsigned char *fbuf, *pixptr;
    2.52 +	int i, j, xsz, ysz;
    2.53 +
    2.54 +	if(SDL_MUSTLOCK(fbsurf)) {
    2.55 +		SDL_LockSurface(fbsurf);
    2.56 +	}
    2.57 +	fbuf = fbsurf->pixels;
    2.58 +	xsz = fbsurf->w;
    2.59 +	ysz = fbsurf->h;
    2.60 +
    2.61 +	pixptr = fbuf;
    2.62 +	for(i=0; i<ysz; i++) {
    2.63 +		float y = 2.0 * (float)i / (float)ysz - 1.0;
    2.64 +		for(j=0; j<xsz; j++) {
    2.65 +			float x = (2.0 * (float)j / (float)xsz - 1.0) * win_aspect;
    2.66 +
    2.67 +			int iter, r, g, b;
    2.68 +			int inset = mandelbrot((x - xoffs) * zoom, (y - yoffs) * zoom, max_iter, &iter);
    2.69 +			if(!inset) {
    2.70 +				r = iter * 256 / max_iter;
    2.71 +				g = 0;
    2.72 +				b = 0;
    2.73 +			} else {
    2.74 +				r = g = b = 0;
    2.75 +			}
    2.76 +
    2.77 +			pixptr[0] = b;
    2.78 +			pixptr[1] = g;
    2.79 +			pixptr[2] = r;
    2.80 +			pixptr[3] = 255;
    2.81 +			pixptr += 4;
    2.82 +		}
    2.83 +	}
    2.84 +
    2.85 +	if(SDL_MUSTLOCK(fbsurf)) {
    2.86 +		SDL_UnlockSurface(fbsurf);
    2.87 +	}
    2.88 +	SDL_Flip(fbsurf);
    2.89 +}
    2.90 +
    2.91 +int mandelbrot(float x, float y, int max_iter, int *iter)
    2.92 +{
    2.93 +	int i;
    2.94 +	complex float c = x + y * I;
    2.95 +	complex float z = c;
    2.96 +
    2.97 +	for(i=0; i<max_iter; i++) {
    2.98 +		float re = creal(z);
    2.99 +		float im = cimag(z);
   2.100 +		if(re * re + im * im > 4) {
   2.101 +			if(iter) *iter = i;
   2.102 +			return 0;
   2.103 +		}
   2.104 +
   2.105 +		z = z * z + c;
   2.106 +	}
   2.107 +
   2.108 +	return 1;
   2.109 +}
   2.110 +
   2.111 +int handle_event(SDL_Event *ev)
   2.112 +{
   2.113 +	static int prev_x, prev_y;
   2.114 +	static int bnstate[32];
   2.115 +
   2.116 +	switch(ev->type) {
   2.117 +	case SDL_KEYDOWN:
   2.118 +		if(ev->key.keysym.sym == SDLK_ESCAPE) {
   2.119 +			return -1;
   2.120 +		}
   2.121 +
   2.122 +	case SDL_MOUSEBUTTONDOWN:
   2.123 +		prev_x = ev->button.x;
   2.124 +		prev_y = ev->button.y;
   2.125 +		bnstate[ev->button.button - SDL_BUTTON_LEFT] = 1;
   2.126 +		break;
   2.127 +
   2.128 +	case SDL_MOUSEBUTTONUP:
   2.129 +		bnstate[ev->button.button - SDL_BUTTON_LEFT] = 0;
   2.130 +		break;
   2.131 +
   2.132 +	case SDL_MOUSEMOTION:
   2.133 +		{
   2.134 +			int dx = ev->button.x - prev_x;
   2.135 +			int dy = ev->button.y - prev_y;
   2.136 +			prev_x = ev->button.x;
   2.137 +			prev_y = ev->button.y;
   2.138 +
   2.139 +			if(!dx && !dy) break;
   2.140 +
   2.141 +			if(bnstate[0]) {
   2.142 +				xoffs += win_aspect * 2.0 * (float)dx / (float)win_width;
   2.143 +				yoffs += 2.0 * (float)dy / (float)win_height;
   2.144 +			}
   2.145 +			if(bnstate[2]) {
   2.146 +				zoom += (float)dy / (float)win_height;
   2.147 +				if(zoom < 1e-6) zoom = 1e-6;
   2.148 +			}
   2.149 +		}
   2.150 +		break;
   2.151 +
   2.152 +	case SDL_VIDEORESIZE:
   2.153 +		win_width = ev->resize.w;
   2.154 +		win_height = ev->resize.h;
   2.155 +		win_aspect = (float)win_width / (float)win_height;
   2.156 +		break;
   2.157 +
   2.158 +	default:
   2.159 +		break;
   2.160 +	}
   2.161 +	return 0;
   2.162 +}