- All important tasks you just select in menu
- Works out of the box if system dependencies are once met
- For advanced Medior developers
- C/C++ development in VSCode
- Works on Linux π§, macOS π, and Windows πͺ
- Creates standalone executables and libraries
- Includes modern CMake templates with a built-in debugger
- Compatible with Conan
- Supports cross-compilation
- Compatible with SSH and WSL
- Flexible and highly configurable
- Follows best practices
πππ
This template started as a small C++ project and gradually became larger and larger. From the very beginning, I created the template with the goal of having a flexibly designed solution. As I gradually added more and more important features, the project may seem quite large at first glance. However, every file in the project structure has its justification, and nothing is unnecessarily redundant.
Note: To understand this project, you will need several hours, maybe days. The reward should be that you will subsequently be able to build any large project within 30 minutes, which will stand on solid foundations of the modern world.
πππ
VCMake2025-introduction.mp4
The project contains its own Main.cpp, which is the main entry point for the standalone executable. It also includes one header file and one implementation file for the library. Thus, the project consists of one executable file and one library that is linked to this executable file. Together, they work in tandem.
The project also includes custom Python scripts that manage the entire workflow. These scripts come with preconfigured CMake settings, task configurations, debugger settings, and more. Everything is ready and functional right out of the box.
Note: The only requirement is to have all the dependencies installed that this project needs to function correctly.
- Visual Studio Code: Preconfigured for seamless development.
https://code.visualstudio.com/ - Pyenv: Simplifies Python version management.
https://github.com/pyenv/pyenv - Python 3: Required for the project's internal workflow.
https://www.python.org/downloads/ - Conan 2: Simplifies dependency management.
https://docs.conan.io - CMake: Ideally the latest version.
https://cmake.org/download/ - Formatters: Preconfigured for consistent code formatting.
- CPM.cmake: Lightweight dependency management as an add-in if necessary.
https://github.com/cpm-cmake/CPM.cmake - CPM.license: Automates third-party license management.
https://github.com/cpm-cmake/CPMLicenses.cmake - ModernCppStarter: Inspired by industry best practices for initializing C++ projects.
https://github.com/TheLartians/ModernCppStarter - GitHub Action Workflows: Actively monitors the source code status across all major platforms.
https://github.com/tomasmark79/MarkWareVCMake/actions
As you can see, there is quite a bit that needs to be correctly installed on your system. Additionally, cross-tool tools are missing here, which are used to create a toolchain tailored precisely to your requirements.
Note: Combining Conan 2 and CPM.cmake is generally not recommended, but it is not prohibited either. If you approach the dependency management concept with Conan 2 as the primary dependency manager and CPM.cmake as a supplementary dependency manager, everything will be fine. Just make sure to watch out for conflicts.
-
Update system packages:
sudo apt update && sudo apt upgrade -y sudo apt install cmake python3-pip curl git libssl-dev \ libbz2-dev libcurses-ocaml-dev build-essential gdb libffi-dev \ libsqlite3-dev liblzma-dev libreadline-dev libtk-img-dev
-
Install and configure Pyenv:
curl https://pyenv.run | bash pyenv install 3.9.2 pyenv virtualenv 3.9.2 env392 pip install --upgrade pip
-
Install Conan via Python:
pip install conan conan profile detect --force
-
Install and configure Pyenv in Powershell:
Powershell Installation Stepspyenv install 3.9.2 pyenv global 3.9.2 pyenv local 3.9.2 pip install --upgrade pip
-
Install Conan in Powershell:
pip install conan conan profile detect --force
-
Enable WSL in Powershell and install for ex. Debian based distro:
wsl --install wsl --list --online wsl --install Debian wsl --set-default-version 2 shutdown /r
-
Install required VSCode extensions:
ms-vscode-remote.remote-wsl
ms-vscode.cpptools
-
Connect to WSL and open the project in VSCode
code .
πππ
-
Clone the template:
git clone https://github.com/tomasmark79/MarkWareVCMake cd MarkWareVCMake/
-
You may rename Standalone and Library name:
python SolutionRenamer.py VCMLib MyLibrary VCMStandalone MyStandalone
-
Open the project in VSCode:
code .
-
If you have all the necessary dependencies installed, set up the keyboard shortcuts for convenient control. These shortcuts are already prepared for you in the
keybindings.json
file. You need to copy the content into the systemkeybindings.json
file, which can be found through the VSCode settings. -
Once the keyboard shortcuts are functional, you can start controlling the entire setup.
-
Press
Shift+F7
to display the menu where you can choose from the available actions. -
To simplify things at the beginning, just hit
**π Zero to Build**
. If everything is in order, tarball archives will be created in theBuild/Artefacts
folder at the end of the process. If not, you will see an error that you need to fix to complete the build process.
πππ
If you have installed at least the essential packages for building software, the default compiler of the distribution should be set up in your system. This default compiler will be used if you use the default
architecture in the template. It is more than likely that you will want to modify your compiler settings over time. You can do this in the tasks.json
and SolutionController.py
files, where you can change or add names.
Since Conan profiles are primarily used in the template, you will want to customize the individual configuration. As an example, here is a Conan profile for cross-compiling to Raspberry Pi 4.
[settings]
os=Linux
arch=armv8
compiler=gcc
compiler.cppstd=17
compiler.libcxx=libstdc++11
compiler.version=12
#compiler.threads=posix
build_type=Release
[buildenv]
CC=/home/tomas/x-tools/aarch64-rpi4-linux-gnu/bin/aarch64-rpi4-linux-gnu-gcc
CXX=/home/tomas/x-tools/aarch64-rpi4-linux-gnu/bin/aarch64-rpi4-linux-gnu-g++
LD=/home/tomas/x-tools/aarch64-rpi4-linux-gnu/bin/aarch64-rpi4-linux-gnu-ld
AR=/home/tomas/x-tools/aarch64-rpi4-linux-gnu/bin/aarch64-rpi4-linux-gnu-ar
AS=/home/tomas/x-tools/aarch64-rpi4-linux-gnu/bin/aarch64-rpi4-linux-gnu-as
RANLIB=/home/tomas/x-tools/aarch64-rpi4-linux-gnu/bin/aarch64-rpi4-linux-gnu-ranlib
STRIP=/home/tomas/x-tools/aarch64-rpi4-linux-gnu/bin/aarch64-rpi4-linux-gnu-strip
RC=/home/tomas/x-tools/aarch64-rpi4-linux-gnu/bin/aarch64-rpi4-linux-gnu-windres
CMAKE_SYSROOT=/home/tomas/x-tools/aarch64-rpi4-linux-gnu/aarch64-rpi4-linux-gnu/sysroot
CMAKE_INCLUDE_PATH=/home/tomas/x-tools/aarch64-rpi4-linux-gnu/aarch64-rpi4-linux-gnu/sysroot/usr/include
CMAKE_LIBRARY_PATH=/home/tomas/x-tools/aarch64-rpi4-linux-gnu/aarch64-rpi4-linux-gnu/sysroot/usr/lib/aarch64-rpi4-linux-gnu/
[conf]
tools.system.package_manager:mode=report
tools.build:sysroot=/home/tomas/x-tools/aarch64-rpi4-linux-gnu/aarch64-rpi4-linux-gnu/sysroot
tools.cmake.cmaketoolchain:system_name=Linux
tools.cmake.cmaketoolchain:system_processor=aarch64
F5 |
Debuging | Launch Standalone process with active debugger. |
F7 |
π¨ Build [s] | Just Fast (Re)Build Standalone. |
SHIFT+F7 |
Show Task Menu | Menu with tasks to manage buildings, etc. |
Shift+Ctrl+S |
Permutate scenarios β | Does π Zero to Build for all architectures and build types. May be chanhed on demand to fit your testing needs. |
Note: If you are building sources with the Standalone task, it will also automaticaly build the library because the library source code is part of the Standalone source code. The separate tasks for Standalone [s]
, Library [l]
, and both [sl]
are provided for specific needs and developer convenience.
Task Menu | Description of individual steps |
---|---|
π Zero to Build | Clean β Conan install β CMake configure β Build |
π¦Έ Zero to Hero | Clean β Conan install β CMake configure β Build β Install Artefacts β Release Tarballs |
π§Ή Clean folder | Removes oarticular folder |
π‘οΈ Conan install | Prepare dependencies and CMake generators |
π§ CMake configure | Configures CMake |
πͺ² CMake configure with debugger | Configures CMake with a debugger. Requires CMake > 3.27 & CMake Tools VSCode extension. An awesome feature to debug CMakeLists.txt files. It is simple: add a breakpoint to any line in CMakeLists.txt and start this task. If a message Running with debugger on. and Waiting for debugger client to connect... appears in the console, you have to launch the Debug CMake configuration task defined in launch.json. Or press Ctrl+C to break this action. |
π¨ Build F7 |
Builds the project. |
π Collect licenses | Gathers licenses from CPM dependencies. |
π Install artefacts | Installs to /Build/Installed/. |
π¦ Release tarballs | Creates tarballs in /Build/Artifacts/. |
βοΈ Conan graph.html | Creates an HTML output of Conan dependencies. |
- | - |
π lint Ctrl+Alt+L |
Runs diagnostics on C/C++ files. |
π clang-format Ctrl+Alt+F |
Formats all project C/C++ files. |
π cmake-format Ctrl+Alt+M |
Formats all project CMake files. |
When you clone my template to your local disk at the beginning, one of the first things you will want to do is rename the names of the standalone application and the library. If this had to be done manually, it would be very cumbersome, so I created another Python script. SolutionRenamer.py
changes all the necessary strings, including those that define object classes. It is ideal to use this renaming right at the beginning to avoid manual work later on.
Since the template is continuously evolving, I have also created an additional Python script, SolutionUpgrader.py
, which allows updating the uncommented files defined within this script. Open the script and uncomment the files you want to be updated. The script creates a backup in case you accidentally update a file that was important to you. The update process involves downloading the latest files from the GitHub repository.
Debug
,Release
,RelWithDebInfo
,MinSizeRel
Note: Since this is a custom workflow solution, it is prohibited to use CMake Layout in the conanfile.py
settings. Other options can be used.
# it should remain commented
# def layout(self):
# cmake_layout(self)
option(BUILD_SHARED_LIBS "Build using shared libraries" OFF)
option(USE_STATIC_RUNTIME "Link against static runtime libraries" ON)
When correctly statically linking all source components, the resulting executable file is the only thing needed to run the application correctly, whether on the development platform or another transferred device.
When dynamically linking external libraries to our Standalone, all dynamic libraries must be available. On the native platform where we are currently developing, dynamic libraries are automatically accessible. However, when transferring the compiled Standalone/Library to another device or platform, dynamic libraries cannot be handled automatically. Therefore, the developer must ensure the availability of dynamic libraries on the foreign system. The simplest way to ensure the availability of dynamic libraries from the source platform on a foreign platform is to extract these libraries from the host
folder, which Conan generates during the dependency installation preparation. The host
folder is always located in the Build
folder and is the result of Conan's dependency preparation. The host
content is created thanks to --deployer=full_deploy
argument when conan is installing.
Be aware that Install and Artefacts folders are containing only CMakeLists.txt targets.
Note: You will normally use the Default
setting out of the box because it uses the compiler found in the system. Conan created a profile named default
using the conan profile detect
command, and this profile will be used by this setting.
Other settings are essentially additional profiles for the Conan manager that I have created using existing toolchains and actively use. For this reason, I have left them in the template for you as well.
x86_64-bookworm-linux-gnu
(requires cross-compilation toolchain)x86_64-w64-mingw32
(requires cross-compilation toolchain)aarch64-linux-gnu
(requires cross-compilation toolchain)
It is easy to add a new architecture.
Command line control of projects is also available.
The real intention of this template is to completely avoid the command line unless absolutely necessary.
However, there may be situations where manual work is required. Therefore, I have provided some task scenarios as examples. You can extract these commands yourself from the SolutionController.log
file, which is automatically created in the main directory of the project's file system when using VSCode tasks. In any case, if Visual Studio Code is not available, you can maintain the projects using just command line commands.
MarkWareVCMake | ModernCppStarter | cmake_template | |
---|---|---|---|
Minimal setup required | yes | yes | yes |
CMake configure with Debugger | yes | no | no |
Supports best practices + Modern CMake | yes | yes | yes |
Stanadlone->Library concept | yes | yes | yes |
Highly Customizable | yes | yes | yes |
Github workflow with badges | yes | yes | not sure |
PackageProject.cmake | yes | author | no |
Dependency solution CPM.cmake integrated | yes | author | no |
CPM.license integrated | yes | author | no |
CPM.tools integrated | yes | author | no |
LintC, clang-format, cmake-format | VSCode Tasks + shortcuts + cli | only cli | no |
Solution Renamer | yes | no | no |
Dependency solution by Conan | yes, wrapped by tasks in VSCode | no | no |
CMake generators by Conan preconfigured | yes | no | no |
Cross-Compilation preconfigured | yes, require toolchain | no | no |
Cross-Compilation targets preconfigured | linux, windows, raspberry pi 4/5 | no | no |
VSCode CMake controller | yes | no | no |
VSCode Tasks via Menu and Shortcuts | yes | no | no |
Clean | Removes the entire build folder | no | no |
Conan | Builds Conan dependencies | no | no |
Configure | Configures CMake | only cli | only cli |
Build | Builds the project | only cli | only cli |
Collect Licenses | Gathers licenses from CPM dependencies | no | no |
Install Artefacts | Installs toΒ /Build/Installed/ | no | no |
Release Artefacts | Tarballs toΒ /Build/Artefacts/ | no | no |
Lint C/C++ | C/C++ files diagnostics | only cli | only cli |
Format C/C++ | Recursive formatting for C/C++ files | only cli | only cli |
Format CMake | Recursive formatting for CMake files | only cli | only cli |
Permutate All Tasks | Executes all task scenarios (testing) | no | no |
Loging runned commands | yes, to file | no | no |
C/C++ tests | not yet | yes | yes |
Doxygen | not yet | yes | no |
This template is licensed under MIT License.