This repo contains code to play the game of Mastermind with various numbers of colors and pins using various well-known algorithms. There are scalar, vectorized, and GPU variants of both the scoring functions and the gameplay including a novel algorithm for playing all games at once on the GPU, implemented in CUDA and described in Mastermind on the GPU.
I was motivated to dig into Mastermind through volunteer work with APCS high school students taught by Michael Miyoshi at Cedarcrest High School. They are given an assignment to write the scoring function for Mastermind, which is quite a challenge for new students to get right. I had never heard of the game before, and I found attempting to play all games efficiently to be a pretty fun challenge myself.
This repo holds most of my work on playing all games of various sizes quickly:
- A GPU implementation using CUDA which runs all games at once. This has been tailored for compute capability 8.9, CUDA 12.2, running on an NVIDIA GeForce RTX 4090.
- A simple reference CPU implementation, and a more optimized CPU variant to compare with the GPU version.
- Multiple versions of the scoring function, including a vectorized version using SSE2. See the comments in codeword.inl for details, and Compiler Explorer links showing the generated code.
- The following algorithms for playing all games. See algos.hpp
and solver.hpp for more details.
- Knuth
- Most Parts
- Expected Size
- Entropy
- Various gameplay optimizations from Ville[2].
- Mastermind on the GPU
- Symmetry and Case Equivalence
- Efficient Scoring Functions
- Packed Indices for Mastermind Scores
- Score Caching Considered Harmful
- Initial Guesses
More results and data are in the results directory. Using the GPU for larger games is much faster, as you would expect. 5p8c is a nice example, SolverCPUFaster vs. SolverCUDA:
Strategy | Initial Guess | Max Turns | Average Turns | CPU-only (s) | GPU-only (s) |
---|---|---|---|---|---|
Knuth | 11223 | 7 | 5.6084 | 2.4373 | 0.0177 |
Most Parts | 11223 | 8 | 5.5491 | 1.0209 | 0.0175 |
Expected Size | 11223 | 7 | 5.4997 | 2.5254 | 0.0175 |
Entropy | 11223 | 7 | 5.4856 | 3.1512 | 0.0182 |
Results are post-processed into a "strategy" for playing any game of a given size. These strategies are output as GraphViz graphs which can be rendered with any of the standard GraphViz engines. See the results directory for examples.
I originally wrote similar code in Java, since I was first motivated by the APCS students. I've kept the Java version pretty straightforward, and it is an easier introduction to the problem than the optimized C++ code you'll find here. See https://github.com/mikemag/CS-Education/tree/master/APCS/Mastermind
This C++ version was originally an evolution of the Java version, based on playing one game at a time. It grew to include a GPU implementation written in Apple's Metal API for AMD GPUs, which I then ported to CUDA for Nvidia. However, playing one game at a time isn't the best way to take full advantage of available GPU resources, so I made a large switch to playing all games concurrently. The older version is interesting for a) the Metal impl, and b) the caching approach to growing a gameplay strategy over time. I've kept it on the game_at_a_time branch in this repo for reference.
Most recently built and tested on the following systems:
- Ubuntu 22.04.2 LTS, g++ 11.4.0, CUDA Toolkit 12.2, w/ various Nvidia Ampere and Lovelace GPUs.
- macOS Ventura 13.3.1, Apple clang version 14.0.3. (cmake build only.)
If you build with CMake without the Nvidia CUDA Toolkit installed then the GPU support is turned off. Building on Ubuntu w/ the CUDA toolkit installed turns on the GPU support automatically.
There is a CLion workspace included in the repo, so if you use CLion you can just clone and go. This is the way I develop 95% of the code. Works with CLion 2023.2 and CMake 3.24.4.
This isn't an easy-to-use program; there are no command line options. Basic config is done by editing settings at the top of main.cpp. Multiple runs over different algorithms are done by changing the config vars in main.cpp to include the specific instantiations you wish to test. Various experiment control flags are scattered throughout the Strategy classes. A JSON file with stats and run parameters is produced at the end.
The program will tell you about the GPUs on your system and which one it selected when you run. System, CPU, and GPU details are recorded in the .json files.
Code is licensed under the MIT license except as otherwise noted. See LICENSE for details.
Documents and images are copyright by Michael Magruder and licensed under a Creative Commons Attribution-ShareAlike 4.0 International License.
[1] D.E. Knuth. The computer as Master Mind. Journal of Recreational Mathematics, 9(1):1–6, 1976.
[2] Geoffroy Ville, An Optimal Mastermind (4,7) Strategy and More Results in the Expected Case, March 2013, arXiv: 1305.1010 [cs.GT]. https://arxiv.org/abs/1305.1010
[3] Barteld Kooi, Yet another mastermind strategy. International Computer Games Association Journal, 28(1):13–20, 2005. https://www.researchgate.net/publication/30485793_Yet_another_Mastermind_strategy