1
1
use std:: collections:: { HashSet , HashMap } ;
2
2
use std:: path:: { Path , PathBuf } ;
3
- use std:: str;
3
+ use std:: str:: { self , FromStr } ;
4
4
use std:: sync:: Arc ;
5
5
6
6
use regex:: Regex ;
7
7
8
8
use core:: { SourceMap , Package , PackageId , PackageSet , Resolve , Target , Profile } ;
9
9
use core:: { TargetKind , LibKind , Profiles , Metadata , Dependency } ;
10
10
use core:: dependency:: Kind as DepKind ;
11
- use util:: { self , CargoResult , ChainError , internal, Config , profile} ;
11
+ use util:: { self , CargoResult , ChainError , internal, Config , profile, Cfg , human } ;
12
12
13
13
use super :: TargetConfig ;
14
14
use super :: custom_build:: { BuildState , BuildScripts } ;
@@ -41,16 +41,20 @@ pub struct Context<'a, 'cfg: 'a> {
41
41
host : Layout ,
42
42
target : Option < Layout > ,
43
43
target_triple : String ,
44
- host_dylib : Option < ( String , String ) > ,
45
- host_staticlib : Option < ( String , String ) > ,
46
- host_exe : String ,
44
+ target_info : TargetInfo ,
45
+ host_info : TargetInfo ,
47
46
package_set : & ' a PackageSet ,
48
- target_dylib : Option < ( String , String ) > ,
49
- target_staticlib : Option < ( String , String ) > ,
50
- target_exe : String ,
51
47
profiles : & ' a Profiles ,
52
48
}
53
49
50
+ #[ derive( Clone ) ]
51
+ struct TargetInfo {
52
+ dylib : Option < ( String , String ) > ,
53
+ staticlib : Option < ( String , String ) > ,
54
+ exe : String ,
55
+ cfg : Option < Vec < Cfg > > ,
56
+ }
57
+
54
58
impl < ' a , ' cfg > Context < ' a , ' cfg > {
55
59
pub fn new ( resolve : & ' a Resolve ,
56
60
sources : & ' a SourceMap < ' cfg > ,
@@ -62,12 +66,11 @@ impl<'a, 'cfg> Context<'a, 'cfg> {
62
66
profiles : & ' a Profiles ) -> CargoResult < Context < ' a , ' cfg > > {
63
67
let target = build_config. requested_target . clone ( ) ;
64
68
let target = target. as_ref ( ) . map ( |s| & s[ ..] ) ;
65
- let ( target_dylib, target_staticlib, target_exe) = try!( Context :: filename_parts ( target,
66
- config) ) ;
67
- let ( host_dylib, host_staticlib, host_exe) = if build_config. requested_target . is_none ( ) {
68
- ( target_dylib. clone ( ) , target_staticlib. clone ( ) , target_exe. clone ( ) )
69
+ let target_info = try!( Context :: target_info ( target, config) ) ;
70
+ let host_info = if build_config. requested_target . is_none ( ) {
71
+ target_info. clone ( )
69
72
} else {
70
- try!( Context :: filename_parts ( None , config) )
73
+ try!( Context :: target_info ( None , config) )
71
74
} ;
72
75
let target_triple = target. unwrap_or_else ( || {
73
76
& config. rustc_info ( ) . host [ ..]
@@ -83,12 +86,8 @@ impl<'a, 'cfg> Context<'a, 'cfg> {
83
86
sources : sources,
84
87
package_set : deps,
85
88
config : config,
86
- target_dylib : target_dylib,
87
- target_staticlib : target_staticlib,
88
- target_exe : target_exe,
89
- host_dylib : host_dylib,
90
- host_staticlib : host_staticlib,
91
- host_exe : host_exe,
89
+ target_info : target_info,
90
+ host_info : host_info,
92
91
compilation : Compilation :: new ( config) ,
93
92
build_state : Arc :: new ( BuildState :: new ( & build_config, deps) ) ,
94
93
build_config : build_config,
@@ -103,8 +102,8 @@ impl<'a, 'cfg> Context<'a, 'cfg> {
103
102
104
103
/// Run `rustc` to discover the dylib prefix/suffix for the target
105
104
/// specified as well as the exe suffix
106
- fn filename_parts ( target : Option < & str > , cfg : & Config )
107
- -> CargoResult < ( Option < ( String , String ) > , Option < ( String , String ) > , String ) > {
105
+ fn target_info ( target : Option < & str > , cfg : & Config )
106
+ -> CargoResult < TargetInfo > {
108
107
let mut process = util:: process ( cfg. rustc ( ) ) ;
109
108
process. arg ( "-" )
110
109
. arg ( "--crate-name" ) . arg ( "_" )
@@ -116,7 +115,18 @@ impl<'a, 'cfg> Context<'a, 'cfg> {
116
115
if let Some ( s) = target {
117
116
process. arg ( "--target" ) . arg ( s) ;
118
117
} ;
119
- let output = try!( process. exec_with_output ( ) ) ;
118
+
119
+ let mut with_cfg = process. clone ( ) ;
120
+ with_cfg. arg ( "--print=cfg" ) ;
121
+
122
+ let mut has_cfg = true ;
123
+ let output = try!( with_cfg. exec_with_output ( ) . or_else ( |_| {
124
+ has_cfg = false ;
125
+ process. exec_with_output ( )
126
+ } ) . chain_error ( || {
127
+ human ( format ! ( "failed to run `rustc` to learn about \
128
+ target-specific information") )
129
+ } ) ) ;
120
130
121
131
let error = str:: from_utf8 ( & output. stderr ) . unwrap ( ) ;
122
132
let output = str:: from_utf8 ( & output. stdout ) . unwrap ( ) ;
@@ -143,13 +153,25 @@ impl<'a, 'cfg> Context<'a, 'cfg> {
143
153
Some ( ( staticlib_parts[ 0 ] . to_string ( ) , staticlib_parts[ 1 ] . to_string ( ) ) )
144
154
} ;
145
155
146
- let exe_suffix = if nobin. is_match ( error) {
156
+ let exe = if nobin. is_match ( error) {
147
157
String :: new ( )
148
158
} else {
149
159
lines. next ( ) . unwrap ( ) . trim ( )
150
160
. split ( '_' ) . skip ( 1 ) . next ( ) . unwrap ( ) . to_string ( )
151
161
} ;
152
- Ok ( ( dylib, staticlib, exe_suffix) )
162
+
163
+ let cfg = if has_cfg {
164
+ Some ( try!( lines. map ( Cfg :: from_str) . collect ( ) ) )
165
+ } else {
166
+ None
167
+ } ;
168
+
169
+ Ok ( TargetInfo {
170
+ dylib : dylib,
171
+ staticlib : staticlib,
172
+ exe : exe,
173
+ cfg : cfg,
174
+ } )
153
175
}
154
176
155
177
/// Prepare this context, ensuring that all filesystem directories are in
@@ -207,9 +229,9 @@ impl<'a, 'cfg> Context<'a, 'cfg> {
207
229
/// otherwise it corresponds to the target platform.
208
230
fn dylib ( & self , kind : Kind ) -> CargoResult < ( & str , & str ) > {
209
231
let ( triple, pair) = if kind == Kind :: Host {
210
- ( & self . config . rustc_info ( ) . host , & self . host_dylib )
232
+ ( & self . config . rustc_info ( ) . host , & self . host_info . dylib )
211
233
} else {
212
- ( & self . target_triple , & self . target_dylib )
234
+ ( & self . target_triple , & self . target_info . dylib )
213
235
} ;
214
236
match * pair {
215
237
None => bail ! ( "dylib outputs are not supported for {}" , triple) ,
@@ -223,9 +245,9 @@ impl<'a, 'cfg> Context<'a, 'cfg> {
223
245
/// otherwise it corresponds to the target platform.
224
246
pub fn staticlib ( & self , kind : Kind ) -> CargoResult < ( & str , & str ) > {
225
247
let ( triple, pair) = if kind == Kind :: Host {
226
- ( & self . config . rustc_info ( ) . host , & self . host_staticlib )
248
+ ( & self . config . rustc_info ( ) . host , & self . host_info . staticlib )
227
249
} else {
228
- ( & self . target_triple , & self . target_staticlib )
250
+ ( & self . target_triple , & self . target_info . staticlib )
229
251
} ;
230
252
match * pair {
231
253
None => bail ! ( "staticlib outputs are not supported for {}" , triple) ,
@@ -284,9 +306,9 @@ impl<'a, 'cfg> Context<'a, 'cfg> {
284
306
pub fn target_filenames ( & self , unit : & Unit ) -> CargoResult < Vec < String > > {
285
307
let stem = self . file_stem ( unit) ;
286
308
let suffix = if unit. target . for_host ( ) {
287
- & self . host_exe
309
+ & self . host_info . exe
288
310
} else {
289
- & self . target_exe
311
+ & self . target_info . exe
290
312
} ;
291
313
292
314
let mut ret = Vec :: new ( ) ;
@@ -532,15 +554,15 @@ impl<'a, 'cfg> Context<'a, 'cfg> {
532
554
fn dep_platform_activated ( & self , dep : & Dependency , kind : Kind ) -> bool {
533
555
// If this dependency is only available for certain platforms,
534
556
// make sure we're only enabling it for that platform.
535
- match ( dep. only_for_platform ( ) , kind ) {
536
- ( Some ( ref platform ) , Kind :: Host ) => {
537
- * platform == self . config . rustc_info ( ) . host
538
- } ,
539
- ( Some ( ref platform ) , Kind :: Target ) => {
540
- * platform == self . target_triple
541
- } ,
542
- ( None , _ ) => true
543
- }
557
+ let platform = match dep. platform ( ) {
558
+ Some ( p ) => p ,
559
+ None => return true ,
560
+ } ;
561
+ let ( name , info ) = match kind {
562
+ Kind :: Host => ( & self . config . rustc_info ( ) . host , & self . host_info ) ,
563
+ Kind :: Target => ( & self . target_triple , & self . target_info ) ,
564
+ } ;
565
+ platform . matches ( name , info . cfg . as_ref ( ) . map ( |cfg| & cfg [ .. ] ) )
544
566
}
545
567
546
568
/// Gets a package for the given package id.
0 commit comments