Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

parser graph generation support for graphs backend #969

Merged
merged 5 commits into from
Oct 30, 2017
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion backends/graphs/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -15,10 +15,11 @@
set (GRAPHS_SRCS
graphs.cpp
controls.cpp
parser.cpp
)

set (GRAPHS_HDRS
controls.h
graphs.h
)

add_cpplint_files(${CMAKE_CURRENT_SOURCE_DIR} "${GRAPHS_SRCS};${GRAPHS_HDRS}")
Expand Down
8 changes: 7 additions & 1 deletion backends/graphs/README.md
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# Graphs Backend

This backend produces visual representations of a P4 program as dot files. For
now it only supports the generation of graphs for top-level control blocks.
now it only supports the generation of graphs for top-level control and parser blocks.

## Dependencies

Expand All @@ -26,6 +26,12 @@ test program:

![Flowlet switching ingress graph](resources/flowlet_switching-bmv2.ingress.png)

Here is the graph generated for the parser block of the
[flowlet_switching-bmv2.p4](../../testdata/p4_16_samples/flowlet_switching-bmv2.p4)
test program:

![Flowlet switching ingress graph](resources/flowlet_switching-bmv2.parser.png)

For a more complex example, you can look at the control graph generated for the
egress control of
[switch.p4](https://github.com/p4lang/switch/tree/f219b4f4e25c2db581f3b91c8da94a7c3ac701a7/p4src)
Expand Down
16 changes: 14 additions & 2 deletions backends/graphs/controls.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -14,12 +14,12 @@ See the License for the specific language governing permissions and
limitations under the License.
*/

#include "controls.h"

#include <boost/graph/graphviz.hpp>

#include <iostream>

#include "graphs.h"

#include "frontends/p4/methodInstance.h"
#include "frontends/p4/tableApply.h"
#include "lib/log.h"
Expand Down Expand Up @@ -67,6 +67,18 @@ class EdgeSwitch : public EdgeTypeIface {
const IR::Expression *labelExpr;
};

class EdgeSelect : public EdgeTypeIface {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is not used in this file, so I'm not sure why it is defined here

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This will be removed. Remnant code from my initial experimentation with code.

public:
explicit EdgeSelect(const cstring labelExpr)
: labelExpr(labelExpr) { }
cstring label() const override {
return labelExpr;
};

private:
const cstring labelExpr;
};

using Graph = ControlGraphs::Graph;

Graph *ControlGraphs::ControlStack::pushBack(Graph &currentSubgraph, const cstring &name) {
Expand Down
7 changes: 6 additions & 1 deletion backends/graphs/graphs.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ limitations under the License.
#include "frontends/p4/evaluator/evaluator.h"
#include "frontends/p4/frontend.h"

#include "controls.h"
#include "graphs.h"

namespace graphs {

Expand Down Expand Up @@ -66,6 +66,7 @@ class Options : public CompilerOptions {
}
};


} // namespace graphs

int main(int argc, char *const argv[]) {
Expand Down Expand Up @@ -117,5 +118,9 @@ int main(int argc, char *const argv[]) {
graphs::ControlGraphs cgen(&midEnd.refMap, &midEnd.typeMap, options.graphsDir);
top->getMain()->apply(cgen);

LOG1("Generating parser graphs");
graphs::ParserGraphs pgen(&midEnd.refMap, &midEnd.typeMap, options.graphsDir);
top->getMain()->apply(pgen);

return ::errorCount() > 0;
}
78 changes: 75 additions & 3 deletions backends/graphs/controls.h → backends/graphs/graphs.h
Original file line number Diff line number Diff line change
Expand Up @@ -14,8 +14,8 @@ See the License for the specific language governing permissions and
limitations under the License.
*/

#ifndef _BACKENDS_GRAPHS_CONTROLS_H_
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it's a little confusing to call this file graphs.h, given that none of the classes it prototypes are implemented in graphs.cpp.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is the common header file for both control.cpp and parser.cpp.
If you have any other suggestion for names, i would appreciate that.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe graph_generators.h if you want to have a single header file for both control and parser

Copy link
Contributor

@ChrisDodd ChrisDodd Oct 13, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If you refactor ParserGraphs and ControlGraphs into a single Graphs (abstract) base class with ParserGraphs and ControlGraphs as (simpler) subclasses, then I think this organization does make a fair amount of sense.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I tried to refactor the classes and created a Graphs base class.

#define _BACKENDS_GRAPHS_CONTROLS_H_
#ifndef _BACKENDS_GRAPHS_GRAPHS_H_
#define _BACKENDS_GRAPHS_GRAPHS_H_

#include "config.h"

Expand Down Expand Up @@ -149,6 +149,78 @@ class ControlGraphs : public Inspector {
boost::optional<cstring> instanceName{};
};


class ParserGraphs : public Inspector {
public:
enum class VertexType {
HEADER,
PARSER,
DEFAULT,
STATEMENTS
};
struct Vertex {
cstring name;
VertexType type;
};
class GraphAttributeSetter;
// The boost graph support for graphviz subgraphs is not very intuitive. In
// particular the write_graphviz code assumes the existence of a lot of
// properties. See
// https://stackoverflow.com/questions/29312444/how-to-write-graphviz-subgraphs-with-boostwrite-graphviz
// for more information.
using GraphvizAttributes = std::map<cstring, cstring>;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it seems you pasted a lot of code from ControlGraphs here, and that these 2 classes would benefit from inheriting from a common base class

using vertexProperties =
boost::property<boost::vertex_attribute_t, GraphvizAttributes,
Vertex>;
using edgeProperties =
boost::property<boost::edge_attribute_t, GraphvizAttributes,
boost::property<boost::edge_name_t, cstring,
boost::property<boost::edge_index_t, int> > >;
using graphProperties =
boost::property<boost::graph_name_t, cstring,
boost::property<boost::graph_graph_attribute_t, GraphvizAttributes,
boost::property<boost::graph_vertex_attribute_t, GraphvizAttributes,
boost::property<boost::graph_edge_attribute_t, GraphvizAttributes> > > >;
using Graph_ = boost::adjacency_list<boost::setS, boost::vecS, boost::directedS,
vertexProperties, edgeProperties,
graphProperties>;
using Graph = boost::subgraph<Graph_>;
using vertex_t = boost::graph_traits<Graph>::vertex_descriptor;

using Parents = std::vector<std::pair<vertex_t, EdgeTypeIface *> >;

ParserGraphs(P4::ReferenceMap *refMap, P4::TypeMap *typeMap, const cstring &graphsDir);

vertex_t add_vertex(const cstring &name, VertexType type);
vertex_t add_and_connect_vertex(const cstring &name, VertexType type);
void add_edge(const vertex_t &from, const vertex_t &to, const cstring &name);

bool preorder(const IR::PackageBlock *block) override;
bool preorder(const IR::ParserBlock *block) override;
bool preorder(const IR::P4Parser *pars) override;
bool preorder(const IR::ParserState *state) override;

void writeGraphToFile(const Graph &g, const cstring &name);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

as underlined in my comment above, duplicating this method in ControlGraphs and ParserGraphs looks like a bad design

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Now the code duplication is minimized.


cstring stringRepr(mpz_class value, unsigned bytes);
void convertSimpleKey(const IR::Expression* keySet,
mpz_class& value, mpz_class& mask);
unsigned combine(const IR::Expression* keySet,
const IR::ListExpression* select,
mpz_class& value, mpz_class& mask);

private:
P4::ReferenceMap *refMap; P4::TypeMap *typeMap;
const cstring graphsDir;
Graph *g{nullptr};
vertex_t begin_v{};
vertex_t end_v{};
vertex_t accept_v{};
Parents parents{};
Parents defParents{};
boost::optional<cstring> instanceName{};
};

} // namespace graphs

#endif // _BACKENDS_GRAPHS_CONTROLS_H_
#endif // _BACKENDS_GRAPHS_GRAPHS_H_
Loading