An easy, fast, and robust library to parse C/C++ source.
- No pre-processing, and preprocessors are part of the ast.
- Most comments are preserved too.
- Developed from scratch and uses back-tracking yacc (BtYacc) to write C++ grammer, that means no dependency on libclang.
- The result of parsing is an AST where elements of a file are arranged in a tree.
- Minimum dependency. Only external dependency is a lexer which is by default available on unix like platforms and can be easily get on Windows.
- Parsing of multi-file program is supported too.
CppParser can be used to build tools that need parsing of C/C++ files. I am using it to develop cib which implements ABI stable SDK architecture for C++ library.
To begin with we will see an example of parsing a hello-world program and see what is the AST that CppParser
creates:
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
return 0;
}
For the above hello-world program we can expect that when it is parsed the generated AST should look like following:
So, how we are going to access these elements of AST using CppParser
?
Below is the program written as unit-test for validating the correctness of generated AST:
#include <catch/catch.hpp>
#include "cppparser.h"
#include <boost/filesystem.hpp>
namespace fs = boost::filesystem;
TEST_CASE("Parsing hello world program")
{
CppParser parser;
const auto testFilePath = fs::path(__FILE__).parent_path() / "test-files/hello-world.cpp";
const auto ast = parser.parseFile(testFilePath.string());
REQUIRE(ast != nullptr);
const auto& members = ast->members();
REQUIRE(members.size() == 2);
CppIncludeEPtr hashInclude = members[0];
REQUIRE(hashInclude);
CHECK(hashInclude->name_ == "<iostream>");
CppFunctionEPtr func = members[1];
REQUIRE(func);
CHECK(func->name_ == "main");
REQUIRE(func->defn());
const auto& mainBodyMembers = func->defn()->members();
REQUIRE(mainBodyMembers.size() == 2);
CppExprEPtr coutHelloWorld = mainBodyMembers[0];
REQUIRE(coutHelloWorld);
CHECK(coutHelloWorld->oper_ == CppOperator::kInsertion);
}
This example is a real one and is part of actual unit test of CppParser.
git clone https://github.com/agladilin/cpppqqarser.git
Install vcpkg to fetch boost
vcpkg install boost-filesystem boost-program-options boost-system boost-date-time
cd cppparser
mkdir build
cmake -B build -S . -DCMAKE_TOOLCHAIN_FILE="[path to vcpkg]/scripts/buildsystems/vcpkg.cmake"
#cd build
#msbuild cppparser.sln
cmake --build build --config Debug
.\build\Debug\cppparsertest.exe
.\build\Debug\cppparserunittest.exe
cmake --build build --config Debug --target install # requires administrator privileges
vcpkg install cppparser --overlay-ports=c:\dev\custom-overlay
To remove
vcpkg remove cppparser
del $VCPKG_ROOT\buildtrees\cppparser
del $VCPKG_ROOT\packages\cppparser_x64-windows
del $VCPKG_ROOT\downloads\agladilin-cppparser-0.1.0.tar.gz
del C:\Users\<user>\AppData\Local\vcpkg\archives\19 # or whatever corresponding to package ABI hash in install log
For Ubuntu to fetch boost:
sudo apt install libboost-dev libboost-system-dev libboost-filesystem-dev libboost-program-options-dev
cd cppparser
mkdir builds
cd builds
cmake ..
make && make test
Alternatively, if you prefer Ninja
instead of make
:
cd cppparser
mkdir builds
cd builds
cmake -G Ninja ..
ninja && ninja test