forked from sysprog21/rv32emu
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmandelbrot.c
78 lines (67 loc) · 2.72 KB
/
mandelbrot.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
/*
* ASCII rendered Mandelbrot Set fractal using 16 bits of fixed point integer
* arithmetic with a selectable fractional precision in bits.
*
* The fixed point implementation uses "8.8" encoding in 16 bit "short"
* integers, where for example decimal 3.5 is hex 0x380 and decimal 2.25 is
* hex 0x240 (0x40 being 1/4 of 0x100).
*
* Originally written by John Lonergan: https://github.com/Johnlon/mandelbrot
*/
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
/* convert decimal value to a fixed point value in the given precision */
#if 0
short toPrec(double f, int bitsPrecision)
{
short whole = ((short) floor(f) << (bitsPrecision));
short part = (f - floor(f)) * (pow(2, bitsPrecision));
short ret = whole + part;
printf("%x\n", ret);
return ret;
}
#endif
static const int width = 64; /* basic width */
static const int height = 32; /* basic width */
static const int zoom = 1; /* leave at 1 for 32x22 */
int main(int argc, char *argv[])
{
const short bitsPrecision = 6;
const short X1 = 0xE0; /* toPrec(3.5, bitsPrecision) / zoom */
const short X2 = 0x90; /* toPrec(2.25, bitsPrecision) */
const short Y1 = 0xC0; /* toPrec(3, bitsPrecision) / zoom : horiz pos */
const short Y2 = 0x60; /* toPrec(1.5, bitsPrecision) : vert pos */
const short LIMIT = 0x100; /* toPrec(4, bitsPrecision) */
/* fractal */
char charset[] = ".:-=X$#@ ";
const short max_iter = sizeof(charset) - 1;
for (short py = 0; py < height * zoom; py++) {
for (short px = 0; px < width * zoom; px++) {
short x0 = ((px * X1) / width) - X2;
short y0 = ((py * Y1) / height) - Y2;
short x = 0, y = 0;
short i;
for (i = 0; i < max_iter; i++) {
short xSqr = (x * x) >> bitsPrecision;
short ySqr = (y * y) >> bitsPrecision;
/* Breakout if sum is > the limit OR breakout also if sum is
* negative which indicates overflow of the addition has
* occurred The overflow check is only needed for precisions of
* over 6 bits because for 7 and above the sums come out
* overflowed and negative therefore we always run to max_iter
* and we see nothing. By including the overflow break out we
* can see the fractal again though with noise.
*/
if ((xSqr + ySqr) >= LIMIT)
break;
short xt = xSqr - ySqr + x0;
y = (((x * y) >> bitsPrecision) * 2) + y0;
x = xt;
}
printf("\033[48;05;%dm%c\033[0m", i, charset[i - 1]);
}
printf("\n");
}
}