@@ -32,6 +32,7 @@ use std::char;
32
32
use std:: collections:: HashSet ;
33
33
use std:: io:: { fs, TempDir , Command } ;
34
34
use std:: io;
35
+ use std:: mem;
35
36
use std:: ptr;
36
37
use std:: str;
37
38
use std:: string:: String ;
@@ -45,6 +46,36 @@ use syntax::attr::AttrMetaMethods;
45
46
use syntax:: codemap:: Span ;
46
47
use syntax:: parse:: token;
47
48
49
+ // RLIB LLVM-BYTECODE OBJECT LAYOUT
50
+ // Version 1
51
+ // Bytes Data
52
+ // 0..10 "RUST_OBJECT" encoded in ASCII
53
+ // 11..14 format version as little-endian u32
54
+ // 15..22 size in bytes of deflate compressed LLVM bitcode as
55
+ // little-endian u64
56
+ // 23.. compressed LLVM bitcode
57
+
58
+ // This is the "magic number" expected at the beginning of a LLVM bytecode
59
+ // object in an rlib.
60
+ pub static RLIB_BYTECODE_OBJECT_MAGIC : & ' static [ u8 ] = b"RUST_OBJECT" ;
61
+
62
+ // The version number this compiler will write to bytecode objects in rlibs
63
+ pub static RLIB_BYTECODE_OBJECT_VERSION : u32 = 1 ;
64
+
65
+ // The offset in bytes the bytecode object format version number can be found at
66
+ pub static RLIB_BYTECODE_OBJECT_VERSION_OFFSET : uint = 11 ;
67
+
68
+ // The offset in bytes the size of the compressed bytecode can be found at in
69
+ // format version 1
70
+ pub static RLIB_BYTECODE_OBJECT_V1_DATASIZE_OFFSET : uint =
71
+ RLIB_BYTECODE_OBJECT_VERSION_OFFSET + 4 ;
72
+
73
+ // The offset in bytes the compressed LLVM bytecode can be found at in format
74
+ // version 1
75
+ pub static RLIB_BYTECODE_OBJECT_V1_DATA_OFFSET : uint =
76
+ RLIB_BYTECODE_OBJECT_V1_DATASIZE_OFFSET + 8 ;
77
+
78
+
48
79
#[ deriving( Clone , PartialEq , PartialOrd , Ord , Eq ) ]
49
80
pub enum OutputType {
50
81
OutputTypeBitcode ,
@@ -1103,28 +1134,44 @@ fn link_rlib<'a>(sess: &'a Session,
1103
1134
// is never exactly 16 bytes long by adding a 16 byte extension to
1104
1135
// it. This is to work around a bug in LLDB that would cause it to
1105
1136
// crash if the name of a file in an archive was exactly 16 bytes.
1106
- let bc = obj_filename. with_extension ( "bc" ) ;
1107
- let bc_deflated = obj_filename. with_extension ( "bytecode.deflate" ) ;
1108
- match fs:: File :: open ( & bc) . read_to_end ( ) . and_then ( |data| {
1109
- fs:: File :: create ( & bc_deflated)
1110
- . write ( match flate:: deflate_bytes ( data. as_slice ( ) ) {
1111
- Some ( compressed) => compressed,
1112
- None => sess. fatal ( "failed to compress bytecode" )
1113
- } . as_slice ( ) )
1114
- } ) {
1137
+ let bc_filename = obj_filename. with_extension ( "bc" ) ;
1138
+ let bc_deflated_filename = obj_filename. with_extension ( "bytecode.deflate" ) ;
1139
+
1140
+ let bc_data = match fs:: File :: open ( & bc_filename) . read_to_end ( ) {
1141
+ Ok ( buffer) => buffer,
1142
+ Err ( e) => sess. fatal ( format ! ( "failed to read bytecode: {}" ,
1143
+ e) . as_slice ( ) )
1144
+ } ;
1145
+
1146
+ let bc_data_deflated = match flate:: deflate_bytes ( bc_data. as_slice ( ) ) {
1147
+ Some ( compressed) => compressed,
1148
+ None => sess. fatal ( format ! ( "failed to compress bytecode from {}" ,
1149
+ bc_filename. display( ) ) . as_slice ( ) )
1150
+ } ;
1151
+
1152
+ let mut bc_file_deflated = match fs:: File :: create ( & bc_deflated_filename) {
1153
+ Ok ( file) => file,
1154
+ Err ( e) => {
1155
+ sess. fatal ( format ! ( "failed to create compressed bytecode \
1156
+ file: {}", e) . as_slice ( ) )
1157
+ }
1158
+ } ;
1159
+
1160
+ match write_rlib_bytecode_object_v1 ( & mut bc_file_deflated,
1161
+ bc_data_deflated. as_slice ( ) ) {
1115
1162
Ok ( ( ) ) => { }
1116
1163
Err ( e) => {
1117
1164
sess. err ( format ! ( "failed to write compressed bytecode: \
1118
- {}",
1119
- e) . as_slice ( ) ) ;
1165
+ {}", e) . as_slice ( ) ) ;
1120
1166
sess. abort_if_errors ( )
1121
1167
}
1122
- }
1123
- ab. add_file ( & bc_deflated) . unwrap ( ) ;
1124
- remove ( sess, & bc_deflated) ;
1168
+ } ;
1169
+
1170
+ ab. add_file ( & bc_deflated_filename) . unwrap ( ) ;
1171
+ remove ( sess, & bc_deflated_filename) ;
1125
1172
if !sess. opts . cg . save_temps &&
1126
1173
!sess. opts . output_types . contains ( & OutputTypeBitcode ) {
1127
- remove ( sess, & bc ) ;
1174
+ remove ( sess, & bc_filename ) ;
1128
1175
}
1129
1176
}
1130
1177
@@ -1134,6 +1181,32 @@ fn link_rlib<'a>(sess: &'a Session,
1134
1181
ab
1135
1182
}
1136
1183
1184
+ fn write_rlib_bytecode_object_v1 < T : Writer > ( writer : & mut T ,
1185
+ bc_data_deflated : & [ u8 ] )
1186
+ -> :: std:: io:: IoResult < ( ) > {
1187
+ let bc_data_deflated_size: u64 = bc_data_deflated. as_slice ( ) . len ( ) as u64 ;
1188
+
1189
+ try! { writer. write ( RLIB_BYTECODE_OBJECT_MAGIC ) } ;
1190
+ try! { writer. write_le_u32 ( 1 ) } ;
1191
+ try! { writer. write_le_u64 ( bc_data_deflated_size) } ;
1192
+ try! { writer. write ( bc_data_deflated. as_slice ( ) ) } ;
1193
+
1194
+ let number_of_bytes_written_so_far =
1195
+ RLIB_BYTECODE_OBJECT_MAGIC . len ( ) + // magic id
1196
+ mem:: size_of_val ( & RLIB_BYTECODE_OBJECT_VERSION ) + // version
1197
+ mem:: size_of_val ( & bc_data_deflated_size) + // data size field
1198
+ bc_data_deflated_size as uint ; // actual data
1199
+
1200
+ // If the number of bytes written to the object so far is odd, add a
1201
+ // padding byte to make it even. This works around a crash bug in LLDB
1202
+ // (see issue #15950)
1203
+ if number_of_bytes_written_so_far % 2 == 1 {
1204
+ try! { writer. write_u8 ( 0 ) } ;
1205
+ }
1206
+
1207
+ return Ok ( ( ) ) ;
1208
+ }
1209
+
1137
1210
// Create a static archive
1138
1211
//
1139
1212
// This is essentially the same thing as an rlib, but it also involves adding
0 commit comments