Created
January 17, 2025 03:29
-
-
Save scruss/c85cfd98dd8c6884b2ea6cb91bb6e658 to your computer and use it in GitHub Desktop.
benchmark Mandelbrot set (aka Brooks-Matelski set) on OLED - MicroPython
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
# benchmark Mandelbrot set (aka Brooks-Matelski set) on OLED | |
# scruss, 2025-01 | |
# MicroPython | |
# -*- coding: utf-8 -*- | |
from machine import Pin, I2C, idle, reset, freq | |
# from os import uname | |
from sys import implementation | |
from ssd1306 import SSD1306_I2C | |
from time import ticks_ms, ticks_diff | |
# %%% These are the only things you should edit %%% | |
startpin = 16 # pin for trigger configured with external pulldown | |
# I2C connection for display | |
i2c = machine.I2C(1, freq=400000, scl=19, sda=18, timeout=50000) | |
# %%% Stop editing here - I mean it!!!1! %%% | |
# maps value between istart..istop to range ostart..ostop | |
def valmap(value, istart, istop, ostart, ostop): | |
return ostart + (ostop - ostart) * ( | |
(value - istart) / (istop - istart) | |
) | |
WIDTH = 128 | |
HEIGHT = 64 | |
TEXTSIZE = 8 # 16x8 text chars | |
maxit = 120 # DO NOT CHANGE! | |
# value of 120 gives roughly 10 second run time for Pico 2W | |
# get some information about the board | |
# thanks to projectgus for the sys.implementation tip | |
if type(freq()) is int: | |
f_mhz = freq() // 1_000_000 | |
else: | |
# STM32 has freq return a tuple | |
f_mhz = freq()[0] // 1_000_000 | |
sys_id = ( | |
implementation.name, | |
".".join([str(x) for x in implementation.version]).rstrip( | |
"." | |
), # version | |
implementation._machine.split()[-1], # processor | |
"%d MHz" % (f_mhz), # frequency | |
"%d*%d; %d" % (WIDTH, HEIGHT, maxit), # run parameters | |
) | |
p = Pin(startpin, Pin.IN) | |
# displays I have are yellow/blue, have no pull-up resistors | |
# and have a confusing I2C address on the silkscreen | |
oled = SSD1306_I2C(WIDTH, HEIGHT, i2c) | |
oled.contrast(31) | |
oled.fill(0) | |
# display system info | |
ypos = (HEIGHT - TEXTSIZE * len(sys_id)) // 2 | |
for s in sys_id: | |
ts = s[: WIDTH // TEXTSIZE] | |
xpos = (WIDTH - TEXTSIZE * len(ts)) // 2 | |
oled.text(ts, xpos, ypos) | |
ypos = ypos + TEXTSIZE | |
oled.show() | |
while p.value() == 0: | |
# wait for button press | |
idle() | |
oled.fill(0) | |
oled.show() | |
start = ticks_ms() | |
# NB: oled.pixel() is *slow*, so only refresh once per row | |
for y in range(HEIGHT): | |
# complex range reversed because display axes wrong way up | |
cc = valmap(float(y + 1), 1.0, float(HEIGHT), 1.2, -1.2) | |
for x in range(WIDTH): | |
cr = valmap(float(x + 1), 1.0, float(WIDTH), -2.8, 2.0) | |
# can't use complex type as small boards don't have it dammit) | |
zr = 0.0 | |
zc = 0.0 | |
for k in range(maxit): | |
t = zr | |
zr = zr * zr - zc * zc + cr | |
zc = 2 * t * zc + cc | |
if zr * zr + zc * zc > 4.0: | |
oled.pixel(x, y, k % 2) # set pixel if escaped | |
break | |
oled.show() | |
elapsed = ticks_diff(ticks_ms(), start) / 1000 | |
elapsed_str = "%.1f s" % elapsed | |
# oled.text(" " * len(elapsed_str), 0, HEIGHT - TEXTSIZE) | |
oled.rect( | |
0, HEIGHT - TEXTSIZE, TEXTSIZE * len(elapsed_str), TEXTSIZE, 0, True | |
) | |
oled.text(elapsed_str, 0, HEIGHT - TEXTSIZE) | |
oled.show() | |
# we're done, so clear screen and reset after the button is pressed | |
while p.value() == 0: | |
idle() | |
oled.fill(0) | |
oled.show() | |
reset() |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
see Parallel MicroPython Benchmarking – We Saw a Chicken … for more details