Skip to content

Commit 4bcf25a

Browse files
Add Polly support. Use can be triggered via -Z polly, when rustc uses an LLVM which includes polly.
Force LLVM rebuild on buildbots.
1 parent 5a662bf commit 4bcf25a

File tree

13 files changed

+152
-13
lines changed

13 files changed

+152
-13
lines changed

config.toml.example

+8
Original file line numberDiff line numberDiff line change
@@ -303,6 +303,10 @@
303303
#optimize-tests = true
304304
#debuginfo-tests = true
305305

306+
# Flag indicating whether tests are optimized with Polly. If optimize-tests is false,
307+
# polly-tests will be false regardless of its value here.
308+
#polly-tests = false
309+
306310
# Flag indicating whether codegen tests will be run or not. If you get an error
307311
# saying that the FileCheck executable is missing, you may want to disable this.
308312
#codegen-tests = true
@@ -346,6 +350,10 @@
346350
# Whether to deny warnings in crates
347351
#deny-warnings = true
348352

353+
# Use Polly on the rust compiler itself. If optimize is false, this will be
354+
# false as well.
355+
#polly-self = false
356+
349357
# =============================================================================
350358
# Options for specific targets
351359
#

src/bootstrap/bin/rustc.rs

+10-1
Original file line numberDiff line numberDiff line change
@@ -91,6 +91,7 @@ fn main() {
9191
("RUSTC_REAL", "RUSTC_LIBDIR")
9292
};
9393
let stage = env::var("RUSTC_STAGE").expect("RUSTC_STAGE was not set");
94+
let stage = usize::from_str(stage.as_str()).expect("RUSTC_STAGE not a usize");
9495
let sysroot = env::var_os("RUSTC_SYSROOT").expect("RUSTC_SYSROOT was not set");
9596
let on_fail = env::var_os("RUSTC_ON_FAIL").map(|of| Command::new(of));
9697

@@ -152,7 +153,7 @@ fn main() {
152153
// workaround undefined references to `rust_eh_unwind_resume` generated
153154
// otherwise, see issue https://github.com/rust-lang/rust/issues/43095.
154155
if crate_name == "panic_abort" ||
155-
crate_name == "compiler_builtins" && stage != "0" {
156+
crate_name == "compiler_builtins" && stage != 0 {
156157
cmd.arg("-C").arg("panic=abort");
157158
}
158159

@@ -267,6 +268,14 @@ fn main() {
267268
cmd.arg("--cfg").arg("parallel_queries");
268269
}
269270

271+
let use_polly = match env::var("RUSTC_USE_POLLY") {
272+
Ok(v) => v != "0",
273+
Err(_) => false,
274+
};
275+
if use_polly && stage >= 1 {
276+
cmd.arg("-Z").arg("polly");
277+
}
278+
270279
let color = match env::var("RUSTC_COLOR") {
271280
Ok(s) => usize::from_str(&s).expect("RUSTC_COLOR should be an integer"),
272281
Err(_) => 0,

src/bootstrap/bootstrap.py

+4
Original file line numberDiff line numberDiff line change
@@ -690,6 +690,10 @@ def update_submodules(self):
690690
recorded_submodules[data[3]] = data[2]
691691
for module in filtered_submodules:
692692
self.update_submodule(module[0], module[1], recorded_submodules)
693+
polly_path = "src/llvm/tools/polly"
694+
if not os.path.exists(os.path.join(self.rust_root, polly_path)):
695+
run(["git", "clone", "https://github.com/llvm-mirror/polly.git",
696+
"src/llvm/tools/polly", "-b", "release_60"])
693697
print("Submodules updated in %.2f seconds" % (time() - start_time))
694698

695699
def set_dev_environment(self):

src/bootstrap/compile.rs

+13-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@ use native;
3636
use tool;
3737

3838
use cache::{INTERNER, Interned};
39-
use builder::{Step, RunConfig, ShouldRun, Builder};
39+
use builder::{Step, RunConfig, ShouldRun, Builder, Kind};
4040

4141
#[derive(Debug, PartialOrd, Ord, Copy, Clone, PartialEq, Eq, Hash)]
4242
pub struct Std {
@@ -553,6 +553,18 @@ pub fn rustc_cargo_env(builder: &Builder, cargo: &mut Command) {
553553
if builder.config.rustc_parallel_queries {
554554
cargo.env("RUSTC_PARALLEL_QUERIES", "1");
555555
}
556+
557+
let use_polly = match builder.kind {
558+
Kind::Test | Kind::Check => {
559+
builder.config.rust_polly_tests
560+
},
561+
_ => builder.config.rust_polly_self
562+
};
563+
if use_polly {
564+
cargo.env("RUSTC_USE_POLLY", "1");
565+
} else {
566+
cargo.env("RUSTC_USE_POLLY", "0");
567+
}
556568
}
557569

558570
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]

src/bootstrap/config.rs

+16
Original file line numberDiff line numberDiff line change
@@ -99,10 +99,12 @@ pub struct Config {
9999
pub rustc_parallel_queries: bool,
100100
pub rustc_default_linker: Option<String>,
101101
pub rust_optimize_tests: bool,
102+
pub rust_polly_tests: bool,
102103
pub rust_debuginfo_tests: bool,
103104
pub rust_dist_src: bool,
104105
pub rust_codegen_backends: Vec<Interned<String>>,
105106
pub rust_codegen_backends_dir: String,
107+
pub rust_polly_self: bool,
106108

107109
pub build: Interned<String>,
108110
pub hosts: Vec<Interned<String>>,
@@ -294,6 +296,7 @@ struct Rust {
294296
rpath: Option<bool>,
295297
optimize_tests: Option<bool>,
296298
debuginfo_tests: Option<bool>,
299+
polly_tests: Option<bool>,
297300
codegen_tests: Option<bool>,
298301
ignore_git: Option<bool>,
299302
debug: Option<bool>,
@@ -306,6 +309,7 @@ struct Rust {
306309
wasm_syscall: Option<bool>,
307310
lld: Option<bool>,
308311
deny_warnings: Option<bool>,
312+
polly_self: Option<bool>,
309313
}
310314

311315
/// TOML representation of how each build target is configured.
@@ -515,6 +519,10 @@ impl Config {
515519
ignore_git = rust.ignore_git;
516520
debug_jemalloc = rust.debug_jemalloc;
517521
set(&mut config.rust_optimize_tests, rust.optimize_tests);
522+
set(&mut config.rust_polly_tests, rust.polly_tests);
523+
if !config.rust_optimize_tests {
524+
config.rust_polly_tests = false;
525+
}
518526
set(&mut config.rust_debuginfo_tests, rust.debuginfo_tests);
519527
set(&mut config.codegen_tests, rust.codegen_tests);
520528
set(&mut config.rust_rpath, rust.rpath);
@@ -545,6 +553,10 @@ impl Config {
545553
Some(n) => config.rust_codegen_units = Some(n),
546554
None => {}
547555
}
556+
557+
config.rust_polly_self = rust
558+
.polly_self
559+
.unwrap_or(false);
548560
}
549561

550562
if let Some(ref t) = toml.target {
@@ -602,6 +614,10 @@ impl Config {
602614
config.rust_debug_assertions = debug_assertions.unwrap_or(default);
603615
config.rust_optimize = optimize.unwrap_or(!default);
604616

617+
if !config.rust_optimize {
618+
config.rust_polly_self = false;
619+
}
620+
605621
let default = config.channel == "dev";
606622
config.ignore_git = ignore_git.unwrap_or(default);
607623

src/bootstrap/test.rs

+3
Original file line numberDiff line numberDiff line change
@@ -960,6 +960,9 @@ impl Step for Compiletest {
960960
}
961961
flags.push("-Zunstable-options".to_string());
962962
flags.push(builder.config.cmd.rustc_args().join(" "));
963+
if builder.config.rust_polly_self {
964+
flags.push("-Zpolly".into());
965+
}
963966

964967
if let Some(linker) = builder.linker(target) {
965968
cmd.arg("--linker").arg(linker);

src/librustc/session/config.rs

+2
Original file line numberDiff line numberDiff line change
@@ -1293,6 +1293,8 @@ options! {DebuggingOptions, DebuggingSetter, basic_debugging_options,
12931293
"make the current crate share its generic instantiations"),
12941294
chalk: bool = (false, parse_bool, [TRACKED],
12951295
"enable the experimental Chalk-based trait solving engine"),
1296+
polly: bool = (false, parse_bool, [UNTRACKED], "Run the Polly polyhedral \
1297+
model optimization passes."),
12961298
}
12971299

12981300
pub fn default_lib_output() -> CrateType {

src/librustc_llvm/build.rs

+56
Original file line numberDiff line numberDiff line change
@@ -157,11 +157,53 @@ fn main() {
157157
cfg.define("LLVM_RUSTLLVM", None);
158158
}
159159

160+
let (enable_polly, polly_link_kind, polly_link_isl) = {
161+
let mut cmd = Command::new(&llvm_config);
162+
cmd.arg("--libdir");
163+
let libdir = output(&mut cmd);
164+
let libdir = libdir.lines().next().unwrap();
165+
let libdir = Path::new(&libdir);
166+
assert!(libdir.exists());
167+
168+
// We can't specify the full libname to rust, so the linker will always expect (on unix)
169+
// LLVMPolly to be libLLVMPolly, which won't be present. I didn't realize this fact until
170+
// after I wrote the following, but maybe this issue will be resolved in the future.
171+
let allow_shared = false;
172+
let mut found_static = false;
173+
let mut found_shared = false;
174+
for entry in libdir.read_dir().unwrap() {
175+
if let Ok(entry) = entry {
176+
if let Some(name) = entry.path().file_name() {
177+
let name = name.to_str().unwrap();
178+
if name.contains("Polly") {
179+
if !found_static {
180+
found_static = !name.contains("LLVM");
181+
}
182+
if !found_shared {
183+
found_shared = name.contains("LLVM");
184+
}
185+
}
186+
}
187+
}
188+
}
189+
190+
let found_static = found_static;
191+
let found_shared = allow_shared && found_shared;
192+
let enabled = found_static || found_shared;
193+
let (kind, isl) = match (found_static, found_shared) {
194+
(false, false) => ("", false),
195+
(true, _) => ("static", true),
196+
(false, true) => ("dylib", false),
197+
};
198+
(enabled, kind, isl)
199+
};
200+
160201
build_helper::rerun_if_changed_anything_in_dir(Path::new("../rustllvm"));
161202
cfg.file("../rustllvm/PassWrapper.cpp")
162203
.file("../rustllvm/RustWrapper.cpp")
163204
.file("../rustllvm/ArchiveWrapper.cpp")
164205
.file("../rustllvm/Linker.cpp")
206+
.define("ENABLE_POLLY", if enable_polly { "1" } else { "0" })
165207
.cpp(true)
166208
.cpp_link_stdlib(None) // we handle this below
167209
.compile("rustllvm");
@@ -214,6 +256,20 @@ fn main() {
214256
println!("cargo:rustc-link-lib={}={}", kind, name);
215257
}
216258

259+
if enable_polly {
260+
match polly_link_kind {
261+
"dylib" => {
262+
panic!("dynamically linking polly is not possible :(");
263+
//println!("cargo:rustc-flags=-l:LLVMPolly")
264+
},
265+
_ => println!("cargo:rustc-link-lib={}=Polly", polly_link_kind),
266+
}
267+
268+
if polly_link_isl {
269+
println!("cargo:rustc-link-lib={}=PollyISL", polly_link_kind);
270+
}
271+
}
272+
217273
// LLVM ldflags
218274
//
219275
// If we're a cross-compile of LLVM then unfortunately we can't trust these

src/librustc_llvm/ffi.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -1642,7 +1642,8 @@ extern "C" {
16421642
Singlethread: bool)
16431643
-> TargetMachineRef;
16441644
pub fn LLVMRustDisposeTargetMachine(T: TargetMachineRef);
1645-
pub fn LLVMRustAddAnalysisPasses(T: TargetMachineRef, PM: PassManagerRef, M: ModuleRef);
1645+
pub fn LLVMRustAddAnalysisPasses(T: TargetMachineRef, PM: PassManagerRef, M: ModuleRef,
1646+
Polly: bool);
16461647
pub fn LLVMRustAddBuilderLibraryInfo(PMB: PassManagerBuilderRef,
16471648
M: ModuleRef,
16481649
DisableSimplifyLibCalls: bool);

src/librustc_trans/back/lto.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -460,7 +460,7 @@ fn run_pass_manager(cgcx: &CodegenContext,
460460
debug!("running the pass manager");
461461
unsafe {
462462
let pm = llvm::LLVMCreatePassManager();
463-
llvm::LLVMRustAddAnalysisPasses(tm, pm, llmod);
463+
llvm::LLVMRustAddAnalysisPasses(tm, pm, llmod, config.polly);
464464
let pass = llvm::LLVMRustFindAndCreatePass("verify\0".as_ptr() as *const _);
465465
assert!(!pass.is_null());
466466
llvm::LLVMRustAddPass(pm, pass);

src/librustc_trans/back/write.rs

+15-7
Original file line numberDiff line numberDiff line change
@@ -250,6 +250,7 @@ pub struct ModuleConfig {
250250
no_integrated_as: bool,
251251
embed_bitcode: bool,
252252
embed_bitcode_marker: bool,
253+
pub polly: bool,
253254
}
254255

255256
impl ModuleConfig {
@@ -281,7 +282,8 @@ impl ModuleConfig {
281282
vectorize_loop: false,
282283
vectorize_slp: false,
283284
merge_functions: false,
284-
inline_threshold: None
285+
inline_threshold: None,
286+
polly: false,
285287
}
286288
}
287289

@@ -319,6 +321,8 @@ impl ModuleConfig {
319321

320322
self.merge_functions = sess.opts.optimize == config::OptLevel::Default ||
321323
sess.opts.optimize == config::OptLevel::Aggressive;
324+
self.polly = sess.opts.debugging_opts.polly && !self.no_prepopulate_passes &&
325+
!sess.target.target.options.is_like_emscripten;
322326
}
323327
}
324328

@@ -546,8 +550,8 @@ unsafe fn optimize(cgcx: &CodegenContext,
546550

547551
if !config.no_verify { assert!(addpass("verify")); }
548552
if !config.no_prepopulate_passes {
549-
llvm::LLVMRustAddAnalysisPasses(tm, fpm, llmod);
550-
llvm::LLVMRustAddAnalysisPasses(tm, mpm, llmod);
553+
llvm::LLVMRustAddAnalysisPasses(tm, fpm, llmod, config.polly);
554+
llvm::LLVMRustAddAnalysisPasses(tm, mpm, llmod, config.polly);
551555
let opt_level = config.opt_level.unwrap_or(llvm::CodeGenOptLevel::None);
552556
with_llvm_pmb(llmod, &config, opt_level, &mut |b| {
553557
llvm::LLVMPassManagerBuilderPopulateFunctionPassManager(b, fpm);
@@ -645,11 +649,12 @@ unsafe fn codegen(cgcx: &CodegenContext,
645649
unsafe fn with_codegen<F, R>(tm: TargetMachineRef,
646650
llmod: ModuleRef,
647651
no_builtins: bool,
652+
polly: bool,
648653
f: F) -> R
649654
where F: FnOnce(PassManagerRef) -> R,
650655
{
651656
let cpm = llvm::LLVMCreatePassManager();
652-
llvm::LLVMRustAddAnalysisPasses(tm, cpm, llmod);
657+
llvm::LLVMRustAddAnalysisPasses(tm, cpm, llmod, polly);
653658
llvm::LLVMRustAddLibraryInfo(cpm, llmod, no_builtins);
654659
f(cpm)
655660
}
@@ -744,7 +749,8 @@ unsafe fn codegen(cgcx: &CodegenContext,
744749
cursor.position() as size_t
745750
}
746751

747-
with_codegen(tm, llmod, config.no_builtins, |cpm| {
752+
with_codegen(tm, llmod, config.no_builtins, config.polly,
753+
|cpm| {
748754
llvm::LLVMRustPrintModule(cpm, llmod, out.as_ptr(), demangle_callback);
749755
llvm::LLVMDisposePassManager(cpm);
750756
});
@@ -762,7 +768,8 @@ unsafe fn codegen(cgcx: &CodegenContext,
762768
} else {
763769
llmod
764770
};
765-
with_codegen(tm, llmod, config.no_builtins, |cpm| {
771+
with_codegen(tm, llmod, config.no_builtins, config.polly,
772+
|cpm| {
766773
write_output_file(diag_handler, tm, cpm, llmod, &path,
767774
llvm::FileType::AssemblyFile)
768775
})?;
@@ -773,7 +780,8 @@ unsafe fn codegen(cgcx: &CodegenContext,
773780
}
774781

775782
if write_obj {
776-
with_codegen(tm, llmod, config.no_builtins, |cpm| {
783+
with_codegen(tm, llmod, config.no_builtins, config.polly,
784+
|cpm| {
777785
write_output_file(diag_handler, tm, cpm, llmod, &obj_out,
778786
llvm::FileType::ObjectFile)
779787
})?;

src/rustllvm/PassWrapper.cpp

+21-1
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,13 @@ DEFINE_STDCXX_CONVERSION_FUNCTIONS(TargetMachine, LLVMTargetMachineRef)
6161
DEFINE_STDCXX_CONVERSION_FUNCTIONS(PassManagerBuilder,
6262
LLVMPassManagerBuilderRef)
6363

64+
#if ENABLE_POLLY
65+
namespace polly {
66+
void initializePollyPasses(llvm::PassRegistry &Registry);
67+
void registerPollyPasses(llvm::legacy::PassManagerBase &PM);
68+
}
69+
#endif
70+
6471
extern "C" void LLVMInitializePasses() {
6572
PassRegistry &Registry = *PassRegistry::getPassRegistry();
6673
initializeCore(Registry);
@@ -73,6 +80,10 @@ extern "C" void LLVMInitializePasses() {
7380
initializeInstCombine(Registry);
7481
initializeInstrumentation(Registry);
7582
initializeTarget(Registry);
83+
84+
#if ENABLE_POLLY
85+
polly::initializePollyPasses(Registry);
86+
#endif
7687
}
7788

7889
enum class LLVMRustPassKind {
@@ -420,10 +431,19 @@ extern "C" void LLVMRustDisposeTargetMachine(LLVMTargetMachineRef TM) {
420431
// this function.
421432
extern "C" void LLVMRustAddAnalysisPasses(LLVMTargetMachineRef TM,
422433
LLVMPassManagerRef PMR,
423-
LLVMModuleRef M) {
434+
LLVMModuleRef M,
435+
bool Polly) {
424436
PassManagerBase *PM = unwrap(PMR);
425437
PM->add(
426438
createTargetTransformInfoWrapperPass(unwrap(TM)->getTargetIRAnalysis()));
439+
440+
#if ENABLE_POLLY
441+
if(Polly) {
442+
polly::registerPollyPasses(*PM);
443+
}
444+
#else
445+
(void)Polly;
446+
#endif
427447
}
428448

429449
extern "C" void LLVMRustConfigurePassManagerBuilder(

0 commit comments

Comments
 (0)