A multithreaded C++ application for drawing and exploring Mandelbrot set. Offers multiple color palettes and responsive UI with real-time rendering. Uses Qt for developing GUI.
Short demo:
mandelbrot_demo.mp4
- Install vcpkg (Step 1 from the instruction)
- Add vcpkg installation directory to system variables
- Windows Search -> "variables"
- Edit the system environment variables
- Environment variables
- New user variable
- Variable name: "VCPKG_ROOT" (without quotes)
- Variable value:
cd %VCPKG_DIR% && .\vcpkg.exe install qtbase[core,gui,widgets]:x64-windows-static
- Navigate to the directory where you want to install the project
git clone --recurse-submodules https://github.com/belous-dp/Mandelbrot-Set.git
- Open Visual Studio 2022 (make sure "C++ CMake tools for Windows" marked in Visual Studio installer -> Visual Studio 2022 -> Modify -> Desktop Development with C++)
- Choose "Open folder" and open the project directory
- Choose CMake Preset ("Release" is the best in terms of performance)
- Project -> Configure mandelbrot
- Build -> Build all
- Select Startup Item -> mandelbrot.exe
- Debug -> Start without debugging
You can try to install Qt via vcpkg in manifest mode by renaming .vcpkg.json to
vcpkg.json
Good luck with that.
- Run
sudo apt update && sudo apt install git build-essential ninja-build libgl1-mesa-dev
- (VPN required) Download Qt online installer.
- (VPN required) Run it with mirror argument (e.g.
./qt-online-installer-linux-x64-4.8.0.run --mirror http://www.nic.funet.fi/pub/mirrors/download.qt-project.org
) - (VPN required) In the installation menu you can select only 'Qt6.x.x/desktop' component. Unselect other components for faster and smaller installation.
- Add to environment variables /path/to/Qt/version/compiler as QT_DIR (in my case I appended
export QT_DIR="/home/belous/Qt/6.7.1/gcc_64"
to~/.profile
) - Log out or restart the computer
- Navigate to the directory where you want to install the project
git clone --recurse-submodules https://github.com/belous-dp/Mandelbrot-Set.git
- Open project directory in CLion and choose an appropriate CMake preset or configure and build the project using CLI and CMake.
- main.cpp starts application
- main_window configures menus, status bar and updated them when needed
- picture is responsible for communication with worker thread and displaying image
- worker thread(s) does all hard computations using multithreading and colors the image
The central widget of the window is class picture. User interacts with this widget and rendered images are shown there. As soon as Qt recieves a signal from a user (e.g. mouse move, wheel scroll), Qt calls the appropriate function in the picture. In such functions the necessary data (for example, the image layout) is updated, the signal about new rendering task is sent to the worker thread, and, while the worker thread is rendering new image, the old image is changed according to the user's request and is shown to the user. When the new image is rendered, the worker thread sends it back to the picture, which is shows it.
Almost all interaction between different modules of the application, such as the picture and the worker thread, is implemented using the signals and slots mechanism.
To prevent application 'freezing', the program's logic is devided into multiple threads:
- The main thread maintains the information about the image, shows temporary dummy images (while the real one is rendering), processes events from the event loop.
- The worker thread only renders images that are requested by the main thread.
- To accelerate the rendering, the worker thread creates additional threads and splits a work into equal parts (height of an image is divided by the number of threads).
- Relaxed atomics are used for synchronization between worker threads.
Before reading the next paragraph, make sure you understand escape-time algorithm that is used for computating the Mandelbrot set.
Among the other things, the following optimization is used. The rendering is done progressively: initially, the number of steps in escape-time algorithm is 100. The image is rendered really quickly and is sent to the user. Then the number of steps is increased by 100. It takes longer time to render an image with nsteps=200, but it has higher quality. And so on the number of steps is increased with each iteration. This workflow gives the user the feeling of a progress. In sectuion dynamic scaling this process is described in more details.
Another optimization is to frequently check for new render tasks in the render queue. When such task is discovered, all worker threads stops the current task as soon as possible. This is done to make sure no resources are wasted rendering the image that is no longer needed.
In the result of the above design decisions, the application is responsive and has almost no freezes, and the rendering is done effectively.
todo