forked from p4lang/p4c
-
Notifications
You must be signed in to change notification settings - Fork 0
/
p4_library.bzl
122 lines (112 loc) · 4.08 KB
/
p4_library.bzl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
"""P4 compilation rule."""
load("@bazel_tools//tools/cpp:toolchain_utils.bzl", "find_cpp_toolchain")
def _p4_library_impl(ctx):
p4c = ctx.executable._p4c
p4file = ctx.file.src
p4deps = ctx.files._p4include + ctx.files.deps
target = ctx.attr.target
args = [
p4file.path,
"--std",
ctx.attr.std,
"--target",
(target if target else "bmv2"),
"--arch",
ctx.attr.arch,
]
if ctx.attr.extra_args:
args.append(ctx.attr.extra_args)
include_dirs = {d.dirname: 0 for d in p4deps} # Use dict to express set.
include_dirs["."] = 0 # Enable include paths relative to workspace root.
args += [("-I" + dir) for dir in include_dirs.keys()]
outputs = []
if ctx.outputs.p4info_out:
if target != "" and target != "bmv2":
fail('Must use `target = "bmv2"` when specifying p4info_out.')
args += ["--p4runtime-files", ctx.outputs.p4info_out.path]
outputs.append(ctx.outputs.p4info_out)
if ctx.outputs.target_out:
if not target:
fail("Cannot specify target_out without specifying target explicitly.")
args += ["-o", ctx.outputs.target_out.path]
outputs.append(ctx.outputs.target_out)
if not outputs:
fail("No outputs specified. Must specify p4info_out or target_out or both.")
cpp_toolchain = find_cpp_toolchain(ctx)
ctx.actions.run_shell(
command = """
# p4c invokes cc for preprocessing; we provide it below.
function cc () {{ "{cc}" "$@"; }}
export -f cc
"{p4c}" {p4c_args}
""".format(
cc = cpp_toolchain.compiler_executable,
p4c = p4c.path,
p4c_args = " ".join(args),
),
inputs = p4deps + [p4file],
tools = depset(
direct = [p4c],
transitive = [cpp_toolchain.all_files],
),
outputs = outputs,
progress_message = "Compiling P4 program %s" % p4file.short_path,
use_default_shell_env = True,
)
p4_library = rule(
doc = "Compiles P4 program using the p4c compiler.",
implementation = _p4_library_impl,
attrs = {
"src": attr.label(
doc = "P4 source file to pass to p4c.",
mandatory = True,
allow_single_file = [".p4"],
),
"deps": attr.label_list(
doc = "Additional P4 dependencies (optional). Use for #include-ed files.",
mandatory = False,
allow_files = [".p4", ".h"],
default = [],
),
"p4info_out": attr.output(
mandatory = False,
doc = "The name of the p4info output file.",
),
"target_out": attr.output(
mandatory = False,
doc = "The name of the target output file, passed to p4c via -o.",
),
"target": attr.string(
doc = "The --target argument passed to p4c (default: bmv2).",
mandatory = False,
default = "", # Use "" so we can recognize implicit target.
),
"arch": attr.string(
doc = "The --arch argument passed to p4c (default: v1model).",
mandatory = False,
default = "v1model"
),
"std": attr.string(
doc = "The --std argument passed to p4c (default: p4-16).",
mandatory = False,
default = "p4-16"
),
"extra_args": attr.string(
doc = "String of additional command line arguments to pass to p4c.",
mandatory = False,
default = "",
),
"_p4c": attr.label(
default = Label("@com_github_p4lang_p4c//:p4c_bmv2"),
executable = True,
cfg = "target",
),
"_p4include": attr.label(
default = Label("@com_github_p4lang_p4c//:p4include"),
allow_files = [".p4", ".h"],
),
"_cc_toolchain": attr.label(default = Label("@bazel_tools//tools/cpp:current_cc_toolchain")),
},
incompatible_use_toolchain_transition = True,
toolchains = ["@bazel_tools//tools/cpp:toolchain_type"],
)