about summary refs log tree commit diff
path: root/src
diff options
context:
space:
mode:
authorbors <bors@rust-lang.org>2015-01-19 12:17:07 +0000
committerbors <bors@rust-lang.org>2015-01-19 12:17:07 +0000
commit43f2c199e4e87d7ccd15658c52ad8dc5a1d54fb9 (patch)
tree3da57695cf99738e7df9b11dd924598f3f35ba61 /src
parent7b87900d72cf53037119e3bac1506a9786ca508a (diff)
parent25a4adc62f4a743b6696a248ef365aaaaee06362 (diff)
downloadrust-43f2c199e4e87d7ccd15658c52ad8dc5a1d54fb9.tar.gz
rust-43f2c199e4e87d7ccd15658c52ad8dc5a1d54fb9.zip
Auto merge of #21282 - Aatch:init-memzero, r=alexcrichton
LLVM gets overwhelmed when presented with a zeroinitializer for a large
type. In unoptimised builds, it generates a long sequence of stores to
memory. In optmised builds, it manages to generate a standard memset of
zero values, but takes a long time doing so.

Call out to the `llvm.memset` function to zero out the memory instead.

Fixes #21264
Diffstat (limited to 'src')
-rw-r--r--src/librustc_trans/trans/intrinsic.rs9
-rw-r--r--src/test/run-pass/init-large-type.rs25
2 files changed, 29 insertions, 5 deletions
diff --git a/src/librustc_trans/trans/intrinsic.rs b/src/librustc_trans/trans/intrinsic.rs
index 91c7409182d..6d0f7746cad 100644
--- a/src/librustc_trans/trans/intrinsic.rs
+++ b/src/librustc_trans/trans/intrinsic.rs
@@ -361,12 +361,11 @@ pub fn trans_intrinsic_call<'a, 'blk, 'tcx>(mut bcx: Block<'blk, 'tcx>,
         }
         (_, "init") => {
             let tp_ty = *substs.types.get(FnSpace, 0);
-            let lltp_ty = type_of::arg_type_of(ccx, tp_ty);
-            if return_type_is_void(ccx, tp_ty) {
-                C_nil(ccx)
-            } else {
-                C_null(lltp_ty)
+            if !return_type_is_void(ccx, tp_ty) {
+                // Just zero out the stack slot. (See comment on base::memzero for explaination)
+                zero_mem(bcx, llresult, tp_ty);
             }
+            C_nil(ccx)
         }
         // Effectively no-ops
         (_, "uninit") | (_, "forget") => {
diff --git a/src/test/run-pass/init-large-type.rs b/src/test/run-pass/init-large-type.rs
new file mode 100644
index 00000000000..0534d0c054f
--- /dev/null
+++ b/src/test/run-pass/init-large-type.rs
@@ -0,0 +1,25 @@
+// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
+// file at the top-level directory of this distribution and at
+// http://rust-lang.org/COPYRIGHT.
+//
+// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
+// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
+// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
+// option. This file may not be copied, modified, or distributed
+// except according to those terms.
+
+// Makes sure that zero-initializing large types is reasonably fast,
+// Doing it incorrectly causes massive slowdown in LLVM during
+// optimisation.
+
+#![feature(intrinsics)]
+
+extern "rust-intrinsic" {
+    pub fn init<T>() -> T;
+}
+
+const SIZE: usize = 1024 * 1024;
+
+fn main() {
+    let _memory: [u8; SIZE] = unsafe { init() };
+}