![]() |
P4C
The P4 Compiler
|
This folder contains documentation for the P4_16 prototype compiler. The code and documentation are hosted in the p4c repository.
extensions
sub-folders, and also the following supplied back-ends:Documenting the workings of the compiler is a never-ending (many times overlooked) job. We can always write better documentation!
In P4C, documentation is generated using Doxygen. The generated documentation depends on Doxygen Awesome CSS. The documentation is dynamically updated and deployed on GitHub Pages.
Documentation is generated from two main sources: README files distributed across the repository and comments within the code. The README files are tagged with documentation inclusion notes to indicate their integration into the P4 compiler documentation.
Code comments should capture the main intent of the implementation and the "why", rather than the "how". The how can be read from the code, however, documenting the reasons why a certain implementation was chosen will help other contributors understand the design choices and enable them to reuse your code. Also important in the context of the compiler is to document the invariants for each pass (or groups of passes), since it is likely that other developers will need to insert additional passes, and they should understand the effects that the pass ordering has on the AST.
Documentation in the markup documents is intended for higher level design documentation. The files will be automatically captured in the documentation in the order implied by their naming: XX_my_doc.md where XX is a number between 02-99. Currently, 00_revision_history.md contains the documentation revision history, and 01_overview.md is the overview of the compiler goals and architecture.
///
for documenting functions and classes in files.//
should be used for "internal" comments within functions.//
should be used for inline comment./* ... */
style comments.docs/doxygen/doxygen.cfg
.docs/doxygen/Doxymain.md
:docs/assets/css/card.css
.docs/assets/css/flow.css
.docs/assets/architecture_unanimated.html
, with the editable draw file available at docs/assets/Architecture.drawio
.docs/doxygen/p4c_layout.xml
).[TOC]
command.docs\assets\css\p4c_custom.css
.<!-- ... -->
is used for adding documentation inclusion notes. This content is hidden from both the rendered Markdown and Doxygen, but visible in the raw view on GitHub.\internal
and \endinternal
commands within comments can be used to hide information from Doxygen while still displaying it on GitHub. Happy writing! Should you have any questions, please don't hesitate to ask.
make V=1
.gdbinit
file has some additional pretty-printers. If you start gdb in this folder (p4c), then it should be automatically used. Otherwise you can run at the gdb prompt source path-to-p4c/.gdbinit
.YYDEBUG
to 1IR::Node
methods can be used to print nice representations of compiler data structures:void dbprint(std::ostream& out) const
: this method is used when logging information. It should print useful debug information, intended for consumption by compiler writers.cstring toString() const
: this method is used when reporting error messages to compiler users. It should only display information that is related to the P4 user program, and never internal compiler data structures.dbprint
method on objects that provide it. Here is an example usage: LOG1("Replacing " << id << " with " << newid);
ordered_map
and ordered_set
if you iterate, to keep iteration order deterministic.You can control the logging level per compiler source-file with the -T
compiler command-line flag. The flag is followed by a list of file patterns and a numeric level after a colon :
. This flag enables all logging messages above the specified level for all compiler source files that match the file pattern.
For example, to enable logging in file node.cpp
above level 1, and in file pass_manager.cpp
above level 2, use the following compiler command-line option: -Tnode:1,pass_manager:2
To execute LOG statements in a header file you must supply the complete name of the header file, e.g.: -TfunctionsInlining.h:3
.
The testing infrastructure is based on small python and shell scripts.
make check -j3
make check-PATTERN
. E.g., make check-p4
.make recheck
. Example:
ctest –output-on-failure -R 'psa-switch-expression-without-default'`test/gtest
Test programs with file names ending in -bmv2.p4
or -ebpf.p4
may have an STF (Simple Test Framework) file with file name suffix .stf
associated with them. If the machine on which you are running has a copy of simple_switch
or the EBPF software switch installed, not only will those programs be compiled for those targets, but also table entries optionally specified in the STF file will be installed, and input packets will be sent to the data plane and output packets checked against expected packets in the STF file.
When pull requests are created on the p4c Github repository, the changes are built, and the tests executed via make check
. These tests are run with a "recently built" version of simple_switch
from the p4lang/behavioral-model repository, but it can be several hours old. If you are working on P4C features that rely on newly committed changes to simple_switch
you can find out which simple_switch
version these P4C automated tests are using at the link below:
To add a new input test with a sample P4 code file (under testdata/p4_16_samples/
for example), one needs to:
*.p4
file to the testdata/p4_16_samples/
directory. The file name might determine which test suite this test belongs to. Those are determined by cmake commands.testdata/p4_16_samples/
belongs to the p4
test suite (meaning it will run with make check-p4
).*-bmv2.p4
also belongs to the bmv2
test suite (meaning it will run with make check-bmv2
).../backends/p4test/run-p4-sample.py . -f ../testdata/p4_16_samples/some_name.p4
. Note that this command needs to run under the build/
directory. The test will fail if the test output is missing or does not match with the existing reference outputs. Toggling the -f
flag will force the script to produce new reference outputs, which can, and should be committed, along with the changes that caused the output change.../backends/bmv2/run-bmv2-test.py
.P4TEST_REPLACE=True make check
(or make check-*
) to update all tests.cpplint
and clang-format
and their respective configuration files. We have customized Google's cpplint.py
tool for our purposes. The tool can be invoked with make cpplint
. To be able to run clang-format
on Ubuntu 20.04, install it with pip3 install --user clang-format
. Do not use the Debian package. Both tools run in a git hook and as part of CI.const
; it is very important.override
whenever possible (new GCC versions enforce this).const_cast
and reinterpret_cast
.cstring
for constant strings. For java programmers, cstring
should be used where you would use java.lang.String, and std::string
should be used where you would use StringBuilder or StringBuffer.BUG()
macro to signal an exception. This macro is guaranteed to throw an exception.CHECK_NULL()
to validate that pointers are not nullptr.BUG_CHECK()
instead of assert
, and always supply an informative error message.error()
and warning()
for error reporting. See the guidelines for more details.LOGn()
for log messages – the n
is an integer constant for verbosity level. These can be controlled on a per-source-file basis with the -T option. LOG1 should be used for general messages, so that running with -T*:1 (turning on all LOG1 messages) is not too overwhelming. LOG2 should be used to print information about the results of a module that later passes may need to debug them. Details of what a module or pass is doing and looking at (only of interest when debugging that code) should be at LOG4 or higher.vector
and array
wrappers for std::vector
and std::array
(these do bounds checking on all accesses).ordered_map
and ordered_set
when you need to iterate; they provide deterministic iterators.p4c is a compiler driver. The goal is to provide a consistent user interface across different p4 backends and work flows. The compiler driver is written in Python. It can be extended for custom backends.
The usage of the driver is as follows:
To extend the driver, user needs to create a configuration file and add it to the p4c_PYTHON
makefile variable.
There is an global variable config
in the p4c
compiler driver that stores the build steps for a particular target. By default, the bmv2 and ebpf backends are supported. Each backend is identified with a triplet: target-arch-vendor. For example, the default bmv2 backend is identified as bmv2-ss-p4org
. Users may choose to implement different architectures running on the same target, and they should configure the compilation flow as follows:
After adding the new configuration file, rerun bootstrap.sh
For testing purposes, p4c
will be installed in the build/ directory when executing make
. Users can install p4c
to other system path by running make install