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/satya-das/common.git
git clone https://github.com/satya-das/CppParser.git
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