1
- extern crate cmake;
2
1
extern crate cc;
3
2
extern crate pkg_config;
4
3
5
4
use std:: env;
6
- use std:: ffi:: OsString ;
7
- use std:: fs:: { self , File } ;
8
- use std:: io:: prelude:: * ;
5
+ use std:: fs;
9
6
use std:: path:: { Path , PathBuf } ;
10
7
use std:: process:: Command ;
11
8
12
- macro_rules! t {
13
- ( $e: expr) => ( match $e{
14
- Ok ( e) => e,
15
- Err ( e) => panic!( "{} failed with {}" , stringify!( $e) , e) ,
16
- } )
17
- }
18
-
19
9
fn main ( ) {
20
10
let https = env:: var ( "CARGO_FEATURE_HTTPS" ) . is_ok ( ) ;
21
11
let ssh = env:: var ( "CARGO_FEATURE_SSH" ) . is_ok ( ) ;
22
12
let curl = env:: var ( "CARGO_FEATURE_CURL" ) . is_ok ( ) ;
23
- if ssh {
24
- register_dep ( "SSH2" ) ;
25
- }
26
- if https {
27
- register_dep ( "OPENSSL" ) ;
28
- }
29
- if curl {
30
- register_dep ( "CURL" ) ;
31
- }
32
- let has_pkgconfig = Command :: new ( "pkg-config" ) . output ( ) . is_ok ( ) ;
33
13
34
14
if env:: var ( "LIBGIT2_SYS_USE_PKG_CONFIG" ) . is_ok ( ) {
35
15
if pkg_config:: find_library ( "libgit2" ) . is_ok ( ) {
@@ -43,188 +23,158 @@ fn main() {
43
23
}
44
24
45
25
let target = env:: var ( "TARGET" ) . unwrap ( ) ;
46
- let host = env:: var ( "HOST" ) . unwrap ( ) ;
47
26
let windows = target. contains ( "windows" ) ;
48
- let msvc = target. contains ( "msvc" ) ;
49
- let mut cfg = cmake:: Config :: new ( "libgit2" ) ;
50
-
51
- #[ cfg( feature = "ssh_key_from_memory" ) ]
52
- cfg. define ( "GIT_SSH_MEMORY_CREDENTIALS" , "1" ) ;
53
-
54
- if msvc {
55
- // libgit2 passes the /GL flag to enable whole program optimization, but
56
- // this requires that the /LTCG flag is passed to the linker later on,
57
- // and currently the compiler does not do that, so we disable whole
58
- // program optimization entirely.
59
- cfg. cflag ( "/GL-" ) ;
60
-
61
- // Currently liblibc links to msvcrt which apparently is a dynamic CRT,
62
- // so we need to turn this off to get it to link right.
63
- let features = env:: var ( "CARGO_CFG_TARGET_FEATURE" )
64
- . unwrap_or ( String :: new ( ) ) ;
65
- if features. contains ( "crt-static" ) {
66
- cfg. define ( "STATIC_CRT" , "ON" ) ;
67
- } else {
68
- cfg. define ( "STATIC_CRT" , "OFF" ) ;
69
- }
27
+ let dst = PathBuf :: from ( env:: var_os ( "OUT_DIR" ) . unwrap ( ) ) ;
28
+ let include = dst. join ( "include" ) ;
29
+ let mut cfg = cc:: Build :: new ( ) ;
30
+ fs:: create_dir_all ( & include) . unwrap ( ) ;
31
+
32
+ // Copy over all header files
33
+ cp_r ( "libgit2/include" . as_ref ( ) , & include) ;
34
+
35
+ cfg. include ( & include)
36
+ . include ( "libgit2/src" )
37
+ . out_dir ( dst. join ( "build" ) )
38
+ . warnings ( false ) ;
39
+
40
+ // Include all cross-platform C files
41
+ add_c_files ( & mut cfg, "libgit2/src" . as_ref ( ) ) ;
42
+ add_c_files ( & mut cfg, "libgit2/src/xdiff" . as_ref ( ) ) ;
43
+
44
+ // These are activated by feautres, but they're all unconditionally always
45
+ // compiled apparently and have internal #define's to make sure they're
46
+ // compiled correctly.
47
+ add_c_files ( & mut cfg, "libgit2/src/transports" . as_ref ( ) ) ;
48
+ add_c_files ( & mut cfg, "libgit2/src/streams" . as_ref ( ) ) ;
49
+
50
+ // Always use bundled http-parser for now
51
+ cfg. include ( "libgit2/deps/http-parser" )
52
+ . file ( "libgit2/deps/http-parser/http_parser.c" ) ;
53
+
54
+ // Always use bundled regex for now
55
+ cfg. include ( "libgit2/deps/regex" )
56
+ . file ( "libgit2/deps/regex/regex.c" ) ;
57
+
58
+ if windows {
59
+ add_c_files ( & mut cfg, "libgit2/src/win32" . as_ref ( ) ) ;
60
+ cfg. define ( "STRSAFE_NO_DEPRECATE" , None ) ;
61
+ } else {
62
+ add_c_files ( & mut cfg, "libgit2/src/unix" . as_ref ( ) ) ;
63
+ cfg. flag ( "-fvisibility=hidden" ) ;
70
64
}
71
65
72
- // libgit2 uses pkg-config to discover libssh2, but this doesn't work on
73
- // windows as libssh2 doesn't come with a libssh2.pc file in that install
74
- // (or when pkg-config isn't found). As a result we just manually turn on
75
- // SSH support in libgit2 (a little jankily) here...
76
- let mut ssh_forced = false ;
77
- if ssh && ( windows || !has_pkgconfig) {
78
- if let Ok ( libssh2_include) = env:: var ( "DEP_SSH2_INCLUDE" ) {
79
- ssh_forced = true ;
80
- if msvc {
81
- cfg. cflag ( format ! ( "/I{}" , libssh2_include) )
82
- . cflag ( "/DGIT_SSH" ) ;
83
- } else {
84
- cfg. cflag ( format ! ( "-I{}" , sanitize_sh( libssh2_include. as_ref( ) ) ) )
85
- . cflag ( "-DGIT_SSH" ) ;
86
- }
87
- }
66
+ let mut features = String :: new ( ) ;
67
+
68
+ features. push_str ( "#ifndef INCLUDE_features_h\n " ) ;
69
+ features. push_str ( "#define INCLUDE_features_h\n " ) ;
70
+ features. push_str ( "#define GIT_THREADS 1\n " ) ;
71
+ features. push_str ( "#define GIT_USE_NSEC 1\n " ) ;
72
+
73
+ if target. contains ( "apple" ) {
74
+ features. push_str ( "#define GIT_USE_STAT_MTIMESPEC 1\n " ) ;
75
+ } else {
76
+ features. push_str ( "#define GIT_USE_STAT_MTIM 1\n " ) ;
88
77
}
89
78
90
- // When cross-compiling, we're pretty unlikely to find a `dlltool` binary
91
- // lying around, so try to find another if it exists
92
- if windows && !host. contains ( "windows" ) {
93
- let c_compiler = cc:: Build :: new ( ) . cargo_metadata ( false )
94
- . get_compiler ( ) ;
95
- let exe = c_compiler. path ( ) ;
96
- let path = env:: var_os ( "PATH" ) . unwrap_or ( OsString :: new ( ) ) ;
97
- let exe = env:: split_paths ( & path)
98
- . map ( |p| p. join ( & exe) )
99
- . find ( |p| p. exists ( ) ) ;
100
- if let Some ( exe) = exe {
101
- if let Some ( name) = exe. file_name ( ) . and_then ( |e| e. to_str ( ) ) {
102
- let name = name. replace ( "gcc" , "dlltool" ) ;
103
- let dlltool = exe. with_file_name ( name) ;
104
- cfg. define ( "DLLTOOL" , & dlltool) ;
105
- }
106
- }
79
+ if env:: var ( "CARGO_CFG_TARGET_POINTER_WIDTH" ) . unwrap ( ) == "32" {
80
+ features. push_str ( "#define GIT_ARCH_32 1\n " ) ;
81
+ } else {
82
+ features. push_str ( "#define GIT_ARCH_64 1\n " ) ;
107
83
}
108
84
109
85
if ssh {
110
- cfg. register_dep ( "SSH2" ) ;
111
- } else {
112
- cfg. define ( "USE_SSH" , "OFF" ) ;
86
+ if let Some ( path) = env:: var_os ( "DEP_SSH2_INCLUDE" ) {
87
+ cfg. include ( path) ;
88
+ }
89
+ features. push_str ( "#define GIT_SSH 1\n " ) ;
90
+ features. push_str ( "#define GIT_SSH_MEMORY_CREDENTIALS 1\n " ) ;
113
91
}
114
92
if https {
115
- cfg. register_dep ( "OPENSSL" ) ;
93
+ features. push_str ( "#define GIT_HTTPS 1\n " ) ;
94
+
95
+ if windows {
96
+ features. push_str ( "#define GIT_WINHTTP 1\n " ) ;
97
+ features. push_str ( "#define GIT_SHA1_WIN32 1\n " ) ;
98
+ cfg. file ( "libgit2/src/hash/hash_win32.c" ) ;
99
+ } else if target. contains ( "apple" ) {
100
+ features. push_str ( "#define GIT_SECURE_TRANSPORT 1\n " ) ;
101
+ features. push_str ( "#define GIT_SHA1_COMMON_CRYPTO 1\n " ) ;
102
+ } else {
103
+ features. push_str ( "#define GIT_OPENSSL 1\n " ) ;
104
+ features. push_str ( "#define GIT_SHA1_OPENSSL 1\n " ) ;
105
+ if let Some ( path) = env:: var_os ( "DEP_OPENSSL_INCLUDE" ) {
106
+ cfg. include ( path) ;
107
+ }
108
+ }
116
109
} else {
117
- cfg. define ( "USE_OPENSSL" , "OFF" ) ;
118
- cfg. define ( "USE_HTTPS" , "OFF" ) ;
110
+ cfg. file ( "libgit2/src/hash/hash_generic.c" ) ;
119
111
}
112
+
120
113
if curl {
121
- cfg. register_dep ( "CURL" ) ;
122
- } else {
123
- cfg. define ( "CURL" , "OFF" ) ;
114
+ features. push_str ( "#define GIT_CURL 1\n " ) ;
115
+ if let Some ( path) = env:: var_os ( "DEP_CURL_INCLUDE" ) {
116
+ cfg. include ( path) ;
117
+ }
118
+ // Handle dllimport/dllexport on windows by making sure that if we built
119
+ // curl statically (as told to us by the `curl-sys` crate) we define the
120
+ // correct values for curl's header files.
121
+ if env:: var_os ( "DEP_CURL_STATIC" ) . is_some ( ) {
122
+ cfg. define ( "CURL_STATICLIB" , None ) ;
123
+ }
124
124
}
125
-
126
- //Use bundled http-parser if cross compiling
127
- if host != target {
128
- cfg. define ( "USE_EXT_HTTP_PARSER" , "OFF" ) ;
125
+ if let Some ( path) = env:: var_os ( "DEP_Z_INCLUDE" ) {
126
+ cfg. include ( path) ;
129
127
}
130
128
131
- let _ = fs:: remove_dir_all ( env:: var ( "OUT_DIR" ) . unwrap ( ) ) ;
132
- t ! ( fs:: create_dir_all( env:: var( "OUT_DIR" ) . unwrap( ) ) ) ;
133
-
134
- // Unset DESTDIR or libgit2.a ends up in it and cargo can't find it
135
- env:: remove_var ( "DESTDIR" ) ;
136
- let dst = cfg. define ( "BUILD_SHARED_LIBS" , "OFF" )
137
- . define ( "BUILD_CLAR" , "OFF" )
138
- . register_dep ( "Z" )
139
- . build ( ) ;
140
-
141
- // Make sure libssh2 was detected on unix systems, because it definitely
142
- // should have been!
143
- if ssh && !ssh_forced {
144
- let flags = dst. join ( "build/src/git2/sys/features.h" ) ;
145
- let mut contents = String :: new ( ) ;
146
- t ! ( t!( File :: open( flags) ) . read_to_string( & mut contents) ) ;
147
- if !contents. contains ( "#define GIT_SSH 1" ) {
148
- panic ! ( "libgit2 failed to find libssh2, and SSH support is required" ) ;
149
- }
129
+ if target. contains ( "apple" ) {
130
+ features. push_str ( "#define GIT_USE_ICONV 1\n " ) ;
150
131
}
151
132
152
- // libgit2 requires the http_parser library for the HTTP transport to be
153
- // implemented, and it will attempt to use the system http_parser if it's
154
- // available. Detect this situation and report using the system http parser
155
- // the same way in this situation.
156
- //
157
- // Note that other dependencies of libgit2 like openssl, libz, and libssh2
158
- // are tracked via crates instead of this. Ideally this should be a crate as
159
- // well.
160
- let pkgconfig_file = dst. join ( "lib/pkgconfig/libgit2.pc" ) ;
161
- if let Ok ( mut f) = File :: open ( & pkgconfig_file) {
162
- let mut contents = String :: new ( ) ;
163
- t ! ( f. read_to_string( & mut contents) ) ;
164
- if contents. contains ( "-lhttp_parser" ) {
165
- println ! ( "cargo:rustc-link-lib=http_parser" ) ;
166
- }
167
- }
133
+ features. push_str ( "#endif\n " ) ;
134
+ fs:: write ( include. join ( "git2/sys/features.h" ) , features) . unwrap ( ) ;
135
+
136
+ cfg. compile ( "git2" ) ;
137
+
138
+ println ! ( "cargo:root={}" , dst. display( ) ) ;
168
139
169
140
if target. contains ( "windows" ) {
170
141
println ! ( "cargo:rustc-link-lib=winhttp" ) ;
171
142
println ! ( "cargo:rustc-link-lib=rpcrt4" ) ;
172
143
println ! ( "cargo:rustc-link-lib=ole32" ) ;
173
144
println ! ( "cargo:rustc-link-lib=crypt32" ) ;
174
- println ! ( "cargo:rustc-link-lib=static=git2" ) ;
175
- println ! ( "cargo:rustc-link-search=native={}/lib" , dst. display( ) ) ;
176
145
return
177
146
}
178
147
179
- println ! ( "cargo:rustc-link-lib=static=git2" ) ;
180
- println ! ( "cargo:rustc-link-search=native={}" , dst. join( "lib" ) . display( ) ) ;
181
148
if target. contains ( "apple" ) {
182
149
println ! ( "cargo:rustc-link-lib=iconv" ) ;
183
150
println ! ( "cargo:rustc-link-lib=framework=Security" ) ;
184
151
println ! ( "cargo:rustc-link-lib=framework=CoreFoundation" ) ;
185
152
}
186
153
}
187
154
188
- fn register_dep ( dep : & str ) {
189
- if let Some ( s) = env:: var_os ( & format ! ( "DEP_{}_ROOT" , dep) ) {
190
- if !cfg ! ( target_env = "msvc" ) {
191
- prepend ( "PKG_CONFIG_PATH" , Path :: new ( & s) . join ( "lib/pkgconfig" ) ) ;
192
- }
193
- return
194
- }
195
- if let Some ( s) = env:: var_os ( & format ! ( "DEP_{}_INCLUDE" , dep) ) {
196
- let root = Path :: new ( & s) . parent ( ) . unwrap ( ) ;
197
- env:: set_var ( & format ! ( "DEP_{}_ROOT" , dep) , root) ;
198
- let path = root. join ( "lib/pkgconfig" ) ;
199
- if path. exists ( ) {
200
- if !cfg ! ( target_env = "msvc" ) {
201
- prepend ( "PKG_CONFIG_PATH" , path) ;
202
- }
203
- return
155
+ fn cp_r ( from : & Path , to : & Path ) {
156
+ for e in from. read_dir ( ) . unwrap ( ) {
157
+ let e = e. unwrap ( ) ;
158
+ let from = e. path ( ) ;
159
+ let to = to. join ( e. file_name ( ) ) ;
160
+ if e. file_type ( ) . unwrap ( ) . is_dir ( ) {
161
+ fs:: create_dir_all ( & to) . unwrap ( ) ;
162
+ cp_r ( & from, & to) ;
163
+ } else {
164
+ println ! ( "{} => {}" , from. display( ) , to. display( ) ) ;
165
+ fs:: copy ( & from, & to) . unwrap ( ) ;
204
166
}
205
167
}
206
168
}
207
169
208
- fn prepend ( var : & str , val : PathBuf ) {
209
- let prefix = env:: var ( var) . unwrap_or ( String :: new ( ) ) ;
210
- let mut v = vec ! [ val] ;
211
- v. extend ( env:: split_paths ( & prefix) ) ;
212
- env:: set_var ( var, & env:: join_paths ( v) . unwrap ( ) ) ;
213
- }
214
-
215
- fn sanitize_sh ( path : & Path ) -> String {
216
- let path = path. to_str ( ) . unwrap ( ) . replace ( "\\ " , "/" ) ;
217
- return change_drive ( & path) . unwrap_or ( path) ;
218
-
219
- fn change_drive ( s : & str ) -> Option < String > {
220
- let mut ch = s. chars ( ) ;
221
- let drive = ch. next ( ) . unwrap_or ( 'C' ) ;
222
- if ch. next ( ) != Some ( ':' ) {
223
- return None
224
- }
225
- if ch. next ( ) != Some ( '/' ) {
226
- return None
170
+ fn add_c_files ( build : & mut cc:: Build , path : & Path ) {
171
+ for e in path. read_dir ( ) . unwrap ( ) {
172
+ let e = e. unwrap ( ) ;
173
+ let path = e. path ( ) ;
174
+ if e. file_type ( ) . unwrap ( ) . is_dir ( ) {
175
+ // skip dirs for now
176
+ } else if path. extension ( ) . and_then ( |s| s. to_str ( ) ) == Some ( "c" ) {
177
+ build. file ( & path) ;
227
178
}
228
- Some ( format ! ( "/{}/{}" , drive, & s[ drive. len_utf8( ) + 2 ..] ) )
229
179
}
230
180
}
0 commit comments