CLion 2018.2 has added support for the compilation database project format. This tutorial shows how to create a
compilation database for the ml-cpp
project and integrate it with CLion.
Note: this tutorial assumes that you have installed the correct version of GCC (Linux)
or XCode command line tools (macOS) installed on your system and it can be accessed
by simply calling g++
or clang++
.
For testing and evaluation purposes you can use 30-day trial version of CLion or participate on the CLion early access program.
You need to install the current CLion (version 2018.2 or later) with the following plugins:
- Compilation Database
- File Watchers
- Makefile support
You can install the plugins either during the initialization dialog, when you first start CLion, or later using the menu Settings / Preferences | Plugins.
Install Python module compiledb from PyPi:
pip install compiledb
It is better to install the module on the system level, since it will create an executable compiledb
, which can be
easier integrated into CLion.
This section took a lot of information from CLion help page Managing Makefile projects. Please, refer this page for additional instruction and screenshots. For more information on using compilation database in CLion, see JetBrains CLion Help page
To create the compilation database file compile_commands.json
in your project, simply call
compiledb -n make -j
from the project root directory.
The argument -n
will avoid the complete build of the targets (which somewhat accelerates the process), while
all arguments after make
will simply be passed to the make
command.
To add unittests files to the compilation database, you need to go to the directories of the individual unit tests, e.g.
lib/core/unittest
and run
compiledb -o ../../../compile_commands.json -n make -j
New files are usually added to compile_commands.json
unless you additionally specify -f
for override.
Note that on macOS compiledb
may not correctly populate the compile_commands.json
output file when invoking make
as above. An alternative is to call make
directly and pipe the output to compiledb
. For convenience these commands
can be encapsulated in a shell script compiledb.sh
#!/usr/bin/env bash
make -Bnwk -j7 | compiledb -o $1
Set the executable bit on the script
chmod a+x ./compiledb.sh
The script can then be used to populate the compilation database as follows
./compiledb.sh compile_commands.json
for dir in `find lib -name unittest`; do echo $dir; (cd $dir; $CPP_SRC_HOME/compiledb.sh $CPP_SRC_HOME/compile_commands.json); done
If the command runs suspiciously quickly and the compile_commands.json
file is empty, this means that your project is
already built and make
didn't do anything. Simply run make clean
before running compiledb
.
Now, in CLion navigate to File | Open on the main menu and choose the compile_commands.json
file or a directory
that contains it and click Open as Project. All files and symbols processed during the run of compiledb make
are
visible and accessible from CLion.
You can enable automatic reload of the project for every change in compile_commands.json
by setting th Use
auto-import checkbox in Settings / Preferences | Build, Execution, Deployment | Compilation Database.
To follow up the changes in the Makefiles, we can create File Watchers (you should have installed File Watchers plugin). Navigate to Settings / Preferences | Tools | File Watchers and create a new File Watcher for all files of type GNU Makefile located in the project root and subdirectories:
File type: GNU Makefile
Scope: Project Files
Program: compiledb
Arguments: -n -o $ProjectFileDir$/compile_commands.json make -j
Working directory: $ProjectFileDir$
- Auto-save edited files to trigger the watcher
- Trigger the watcher on external changes
Alternatively, if using the compiledb.sh
wrapper script:
File type: GNU Makefile
Scope: Project Files
Program: $ProjectFileDir$/compiledb.sh
Arguments: $ProjectFileDir$/compile_commands.json
Working directory: $FileDir$
- Auto-save edited files to trigger the watcher
- Trigger the watcher on external changes
Compilation database itself lacks the data required for building, running and debugging an application. However, you can set up the workflow by adding custom build targets for your compilation database project and creating custom Run/Debug configurations for these targets.
Let's create a custom target to build the libraries, which gives you the same behavior as running make
in the
project root. Go to Settings / Preference | Build, Execution, Deployment | Custom Build Targets and click + to
add a new target. Pick the name, in this tutorial we will use the name make build.
In the area Toolchain we have to specify custom tools for building and cleaning the project.
For Build: click on ... to open External Tools window and then on + to create a new external tool.
In the window Edit Tool specify:
Name: build_with_make
Program: make
Arguments: -j
Working directory: $ProjectFileDir$
Advanced Options
- Synchronize files after execution\
- Open console for tool output
Similarly, for Clean: create a new external tool with the following entries in the Edit Tool window:
Name: clean_with_make
Program: make
Arguments: clean
Working directory: $ProjectFileDir$
Advanced Options
- Synchronize files after execution\
- Open console for tool output
Once you are done, your Custom Build Targets window should look similar to this:
Once we created the custom build target, we can use it to build projects and run/debug unit tests withing IDE.
Let's go to menu Run | Edit Configurations... and click on + to create a new configuration.
Since we installed Makefile support plugin, we can add Makefile configuration to build the complete project:
Name: Libraries
Makefile: Makefile
Working Directory:
Arguments: -j ML_DEBUG=1
- Allow parallel run
I assume that you want to build the project with debug symbols activated, but, obviously, you need to remove the
argument ML_DUBUG=1
if you don't.
It is important that you specify the environment variable CPP_SRC_HOME
. If you specified it on the system
level, CLion can pick it up automatically, otherwise you have to specify it explicitly in the Environment variables
field. Click on the document symbol 📄 on the right of the field and either make sure that the checkbox
Include system environment variables is activated and your environment variables are listed in the list below, or
add them to the list User environment variables manually.
You can now build the project manually by selecting the configuration Libraries
from the configurations drop-down menu
and clicking the green play button. Moreover, we will create a configuration for running unit tests and use Libraries
as a build dependency so we ensure that the project is up-to-date every time we run those tests.
Let's create another Run/Debug Configuration for building the core
unit tests. Go to menu
Run | Edit Configurations... and click on + to create a new configuration.
Name: Build test core
Makefile: Makefile
Working Directory: *Navigate to lib/core/unittest/*\ **Arguments:**
-j7 ML_DEBUG=1`
- Allow parallel run
Finally we can create another Run/Debug Configuration for running unit tests for the core
module. Again, go to Run | Edit
Configurations... and click on the + symbol to create a new Custom Build Application:
Name: Run test core
Target: Select the custom build target make build
that we created before
Executable: Navigate to lib/core/unittest/
and select the ml_test
binary
If you cannot find the executable ml-test
, then you don't have one yet. Simply, build the unittests by executing
make
in the lib/core/unittest
directory once to create it.
Working directory: lib/core/unittest
In the area Before launch: Another Configuration, Build, Activate tool window click on + and select Run another configuration and then Libraries. Make sure it is run before Build by using up- and down-arrows.
If you want to run an individual test suite or a test case, you can specify those in the field Program arguments.
Now, you can run and debug your code by selecting the appropriate configuration and using play or debug symbols.
Once build configurations for all unit tests have bee created it is possible to create a Run/Debug Configuration to
invoke them all. Go to Run | Edit Configurations... and click on the + symbol to create a new
Compound configuration named e.g. Test All
. Click on the + symbol repeatedly to add each of the Build test...
configurations.
Recent versions of CLion come with integrated support for clang-format
. To ensure that clang-format
is used
in preference to the built-in formatter navigate to Navigate to Settings / Preferences | Tools | clang-format and
ensure that Clang-format binary is set to clang-format
and that the PATH field is empty. Then navigate to
Settings / Preferences | Editor | Code Style and tick Enable ClangFormat with clangd server.
Alternatively, you may wish to set up a File Watcher to invoke clang-format
from
Settings / Preferences | Tools | File Watchers
File type: C/C++
Scope: Project Files
Program: clang-format
Arguments: -i $FileName$
Output paths to refresh: $FileName$
Working directory $FileDir
- Auto-save edited files to trigger the watcher
- Trigger the watcher on external changes
Clion has full support for running and analyzing code using the valgrind
suite.
Valgrind is readily available on most linux distributions but on macOS an experimental build is required
brew install --HEAD https://raw.githubusercontent.com/sowson/valgrind/master/valgrind.rb
Once installed go to Settings / Preference | Build, Execution, Deployment | Dynamic Analysis Tools | Valgrind
and specify the full path to the valgrind
executable. Valgrind
can now be used to analyze run
configurations from
Run | Run Run Configuration with Valgrind Memcheck
For more information and useful screenshots, please refer to the Clion help page on custom build targets. Here you can also find more information on Creating new Run/Debug Configurations as well as running and debugging your code.