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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
|
"""This file contains BUILD extensions for generating source code from LLVM's table definition files using the TableGen tool.
See http://llvm.org/cmds/tblgen.html for more information on the TableGen
tool.
TODO(chandlerc): Currently this expresses include-based dependencies as
"sources", and has no transitive understanding due to these files not being
correctly understood by the build system.
"""
def _dict_add(*dictionaries):
"""Returns a new `dict` that has all the entries of the given dictionaries.
If the same key is present in more than one of the input dictionaries, the
last of them in the argument list overrides any earlier ones.
This function is designed to take zero or one arguments as well as multiple
dictionaries, so that it follows arithmetic identities and callers can avoid
special cases for their inputs: the sum of zero dictionaries is the empty
dictionary, and the sum of a single dictionary is a copy of itself.
Re-implemented here to avoid adding a dependency on skylib.
Args:
*dictionaries: Zero or more dictionaries to be added.
Returns:
A new `dict` that has all the entries of the given dictionaries.
"""
result = {}
for d in dictionaries:
result.update(d)
return result
def gentbl(name, tblgen, td_file, td_srcs, tbl_outs, library = True, **kwargs):
"""gentbl() generates tabular code from a table definition file.
Args:
name: The name of the build rule for use in dependencies.
tblgen: The binary used to produce the output.
td_file: The primary table definitions file.
td_srcs: A list of table definition files included transitively.
tbl_outs: A list of tuples (opts, out), where each opts is a string of
options passed to tblgen, and the out is the corresponding output file
produced.
library: Whether to bundle the generated files into a library.
**kwargs: Keyword arguments to pass to subsidiary cc_library() rule.
"""
if td_file not in td_srcs:
td_srcs += [td_file]
includes = []
for (opts, out) in tbl_outs:
outdir = out[:out.rindex("/")]
if outdir not in includes:
includes.append(outdir)
rule_suffix = "_".join(opts.replace("-", "_").replace("=", "_").split(" "))
native.genrule(
name = "%s_%s_genrule" % (name, rule_suffix),
srcs = td_srcs,
outs = [out],
tools = [tblgen],
message = "Generating code from table: %s" % td_file,
cmd = (("$(location %s) " + "-I external/llvm/include " +
"-I external/llvm/tools/clang/include " +
"-I $$(dirname $(location %s)) " + "%s $(location %s) -o $@") % (
tblgen,
td_file,
opts,
td_file,
)),
)
# For now, all generated files can be assumed to comprise public interfaces.
# If this is not true, you should specify library = False
# and list the generated '.inc' files in "srcs".
if library:
native.cc_library(
name = name,
textual_hdrs = [f for (_, f) in tbl_outs],
includes = includes,
**kwargs
)
def llvm_target_cmake_vars(native_arch, target_triple):
return {
"LLVM_HOST_TRIPLE": target_triple,
"LLVM_DEFAULT_TARGET_TRIPLE": target_triple,
"LLVM_NATIVE_ARCH": native_arch,
}
def _quote(s):
"""Quotes the given string for use in a shell command.
This function double-quotes the given string (in case it contains spaces or
other special characters) and escapes any special characters (dollar signs,
double-quotes, and backslashes) that may be present.
Args:
s: The string to quote.
Returns:
An escaped and quoted version of the string that can be passed to a shell
command.
"""
return ('"' +
s.replace("\\", "\\\\").replace("$", "\\$").replace('"', '\\"') +
'"')
def cmake_var_string(cmake_vars):
"""Converts a dictionary to an input suitable for expand_cmake_vars.
Ideally we would jist stringify in the expand_cmake_vars() rule, but select()
interacts badly with genrules.
TODO(phawkins): replace the genrule() with native rule and delete this rule.
Args:
cmake_vars: a dictionary with string keys and values that are convertable to
strings.
Returns:
cmake_vars in a form suitable for passing to expand_cmake_vars.
"""
return " ".join([
_quote("{}={}".format(k, str(v)))
for (k, v) in cmake_vars.items()
])
def expand_cmake_vars(name, src, dst, cmake_vars):
"""Expands #cmakedefine, #cmakedefine01, and CMake variables in a text file.
Args:
name: the name of the rule
src: the input of the rule
dst: the output of the rule
cmake_vars: a string containing the CMake variables, as generated by
cmake_var_string.
"""
expand_cmake_vars_tool = Label("@org_tensorflow//third_party/llvm:expand_cmake_vars")
native.genrule(
name = name,
srcs = [src],
tools = [expand_cmake_vars_tool],
outs = [dst],
cmd = ("$(location {}) ".format(expand_cmake_vars_tool) + cmake_vars +
"< $< > $@"),
)
# TODO(phawkins): the set of CMake variables was hardcoded for expediency.
# However, we should really detect many of these via configure-time tests.
# The set of CMake variables common to all targets.
cmake_vars = {
# Headers
"HAVE_DIRENT_H": 1,
"HAVE_DLFCN_H": 1,
"HAVE_ERRNO_H": 1,
"HAVE_EXECINFO_H": 1,
"HAVE_FCNTL_H": 1,
"HAVE_INTTYPES_H": 1,
"HAVE_PTHREAD_H": 1,
"HAVE_SIGNAL_H": 1,
"HAVE_STDINT_H": 1,
"HAVE_SYS_IOCTL_H": 1,
"HAVE_SYS_MMAN_H": 1,
"HAVE_SYS_PARAM_H": 1,
"HAVE_SYS_RESOURCE_H": 1,
"HAVE_SYS_STAT_H": 1,
"HAVE_SYS_TIME_H": 1,
"HAVE_SYS_TYPES_H": 1,
"HAVE_TERMIOS_H": 1,
"HAVE_UNISTD_H": 1,
"HAVE_ZLIB_H": 1,
# Features
"HAVE_BACKTRACE": 1,
"BACKTRACE_HEADER": "execinfo.h",
"HAVE_DLOPEN": 1,
"HAVE_FUTIMES": 1,
"HAVE_GETCWD": 1,
"HAVE_GETPAGESIZE": 1,
"HAVE_GETRLIMIT": 1,
"HAVE_GETRUSAGE": 1,
"HAVE_GETTIMEOFDAY": 1,
"HAVE_INT64_T": 1,
"HAVE_ISATTY": 1,
"HAVE_LIBEDIT": 1,
"HAVE_LIBPTHREAD": 1,
"HAVE_LIBZ": 1,
"HAVE_MKDTEMP": 1,
"HAVE_MKSTEMP": 1,
"HAVE_MKTEMP": 1,
"HAVE_PREAD": 1,
"HAVE_PTHREAD_GETSPECIFIC": 1,
"HAVE_PTHREAD_MUTEX_LOCK": 1,
"HAVE_PTHREAD_RWLOCK_INIT": 1,
"HAVE_REALPATH": 1,
"HAVE_SBRK": 1,
"HAVE_SETENV": 1,
"HAVE_SETRLIMIT": 1,
"HAVE_SIGALTSTACK": 1,
"HAVE_STRERROR": 1,
"HAVE_STRERROR_R": 1,
"HAVE_STRTOLL": 1,
"HAVE_SYSCONF": 1,
"HAVE_UINT64_T": 1,
"HAVE__UNWIND_BACKTRACE": 1,
# LLVM features
"ENABLE_BACKTRACES": 1,
"LLVM_BINDIR": "/dev/null",
"LLVM_DISABLE_ABI_BREAKING_CHECKS_ENFORCING": 0,
"LLVM_ENABLE_ABI_BREAKING_CHECKS": 0,
"LLVM_ENABLE_THREADS": 1,
"LLVM_ENABLE_ZLIB": 1,
"LLVM_HAS_ATOMICS": 1,
"LLVM_INCLUDEDIR": "/dev/null",
"LLVM_INFODIR": "/dev/null",
"LLVM_MANDIR": "/dev/null",
"LLVM_NATIVE_TARGET": 1,
"LLVM_NATIVE_TARGETINFO": 1,
"LLVM_NATIVE_TARGETMC": 1,
"LLVM_NATIVE_ASMPRINTER": 1,
"LLVM_NATIVE_ASMPARSER": 1,
"LLVM_NATIVE_DISASSEMBLER": 1,
"LLVM_ON_UNIX": 1,
"LLVM_PREFIX": "/dev/null",
"LLVM_VERSION_MAJOR": 0,
"LLVM_VERSION_MINOR": 0,
"LLVM_VERSION_PATCH": 0,
"LTDL_SHLIB_EXT": ".so",
"PACKAGE_NAME": "llvm",
"PACKAGE_STRING": "llvm tensorflow-trunk",
"PACKAGE_VERSION": "tensorflow-trunk",
"RETSIGTYPE": "void",
}
# CMake variables specific to the Linux platform
linux_cmake_vars = {
"HAVE_MALLOC_H": 1,
"HAVE_LINK_H": 1,
"HAVE_MALLINFO": 1,
"HAVE_FUTIMENS": 1,
}
# CMake variables specific to the Darwin (Mac OS X) platform.
darwin_cmake_vars = {
"HAVE_MALLOC_MALLOC_H": 1,
}
# Select a set of CMake variables based on the platform.
# TODO(phawkins): use a better method to select the right host triple, rather
# than hardcoding x86_64.
llvm_all_cmake_vars = select({
"@org_tensorflow//tensorflow:darwin": cmake_var_string(
_dict_add(
cmake_vars,
llvm_target_cmake_vars("X86", "x86_64-apple-darwin"),
darwin_cmake_vars,
),
),
"@org_tensorflow//tensorflow:linux_ppc64le": cmake_var_string(
_dict_add(
cmake_vars,
llvm_target_cmake_vars("PowerPC", "powerpc64le-unknown-linux_gnu"),
linux_cmake_vars,
),
),
"//conditions:default": cmake_var_string(
_dict_add(
cmake_vars,
llvm_target_cmake_vars("X86", "x86_64-unknown-linux_gnu"),
linux_cmake_vars,
),
),
})
llvm_linkopts = ["-ldl", "-lm", "-lpthread"]
llvm_defines = [
"LLVM_ENABLE_STATS",
"__STDC_LIMIT_MACROS",
"__STDC_CONSTANT_MACROS",
"__STDC_FORMAT_MACROS",
"_DEBUG",
"LLVM_BUILD_GLOBAL_ISEL",
]
llvm_copts = []
# Platform specific sources for libSupport.
def llvm_support_platform_specific_srcs_glob():
return select({
"//conditions:default": native.glob([
"lib/Support/Unix/*.inc",
"lib/Support/Unix/*.h",
]),
})
|