Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(147)

Side by Side Diff: win_toolchain/toolchain2013.py

Issue 135933002: Automatic Windows toolchain (Closed) Base URL: https://chromium.googlesource.com/chromium/tools/depot_tools.git@master
Patch Set: +x Created 6 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
1 #!/usr/bin/env python
2
M-A Ruel 2014/01/13 18:50:32 no empty line
scottmg 2014/01/13 18:58:36 Done.
3 # Copyright 2013 The Chromium Authors. All rights reserved.
4 # Use of this source code is governed by a BSD-style license that can be
5 # found in the LICENSE file.
6
7 """Extracts a Windows VS2013 toolchain from various downloadable pieces."""
8
9
10 import ctypes
11 from optparse import OptionParser
M-A Ruel 2014/01/13 18:50:32 import optparse it's only used once anyway so it'
scottmg 2014/01/13 18:58:36 Done.
12 import os
13 import shutil
14 import subprocess
15 import sys
16 import tempfile
17 import urllib2
18
19
20 BASEDIR = os.path.dirname(os.path.abspath(__file__))
21 g_temp_dirs = []
22
23
24 def GetLongPathName(path):
25 """Converts any 8dot3 names in the path to the full name."""
26 buf = ctypes.create_unicode_buffer(260)
27 size = ctypes.windll.kernel32.GetLongPathNameW(unicode(path), buf, 260)
28 if (size > 260):
29 raise SystemExit('Long form of path longer than 260 chars: %s' % path)
30 return buf.value
31
32
33 def RunOrDie(command):
34 rc = subprocess.call(command, shell=True)
35 if rc != 0:
36 raise SystemExit('%s failed.' % command)
37
38
39 def TempDir():
40 """Generates a temporary directory (for downloading or extracting to) and keep
41 track of the directory that's created for cleaning up later.
42 """
43 global g_temp_dirs
44 temp = tempfile.mkdtemp()
45 g_temp_dirs.append(temp)
46 return temp
47
48
49 def DeleteAllTempDirs():
50 """Removes all temporary directories created by |TempDir()|."""
51 global g_temp_dirs
52 if g_temp_dirs:
53 sys.stdout.write('Cleaning up temporaries...\n')
54 for temp in g_temp_dirs:
55 # shutil.rmtree errors out on read only attributes.
56 RunOrDie('rmdir /s/q "%s"' % temp)
57 g_temp_dirs = []
58
59
60 def GetIsoUrl(pro):
61 """Gets the .iso URL.
62
63 If |pro| is False, downloads the Express edition.
64 """
65 prefix = 'http://download.microsoft.com/download/'
66 if pro:
67 return (prefix +
68 'A/F/1/AF128362-A6A8-4DB3-A39A-C348086472CC/VS2013_RTM_PRO_ENU.iso')
69 else:
70 return (prefix +
71 '7/2/E/72E0F986-D247-4289-B9DC-C4FB07374894/VS2013_RTM_DskExp_ENU.iso')
72
73
74 def Download(url, local_path):
75 """Downloads a large-ish binary file and print some status information while
76 doing so.
77 """
78 sys.stdout.write('Downloading %s...\n' % url)
79 req = urllib2.urlopen(url)
80 content_length = int(req.headers.get('Content-Length', 0))
81 bytes_read = 0L
82 terminator = '\r' if sys.stdout.isatty() else '\n'
83 with open(local_path, 'wb') as file:
84 while True:
85 chunk = req.read(1024 * 1024)
86 if not chunk:
87 break
88 bytes_read += len(chunk)
89 file.write(chunk)
90 sys.stdout.write('... %d/%d%s' % (bytes_read, content_length, terminator))
91 sys.stdout.flush()
92 sys.stdout.write('\n')
93 if content_length and content_length != bytes_read:
94 raise SystemExit('Got incorrect number of bytes downloading %s' % url)
95
96
97 def ExtractIso(iso_path):
98 """Uses 7zip to extract the contents of the given .iso (or self-extracting
99 .exe).
100 """
101 target_path = TempDir()
102 sys.stdout.write('Extracting %s...\n' % iso_path)
103 sys.stdout.flush()
104 # TODO(scottmg): Do this (and exe) manually with python code.
105 # Note that at the beginning of main() we set the working directory to 7z's
106 # location so that 7z can find its codec dll.
107 RunOrDie('7z x "%s" -y "-o%s" >nul' % (iso_path, target_path))
108 return target_path
109
110
111 def ExtractMsi(msi_path):
112 """Uses msiexec to extract the contents of the given .msi file."""
113 sys.stdout.write('Extracting %s...\n' % msi_path)
114 target_path = TempDir()
115 RunOrDie('msiexec /a "%s" /qn TARGETDIR="%s"' % (msi_path, target_path))
116 return target_path
117
118
119 def DownloadMainIso(url):
120 temp_dir = TempDir()
121 target_path = os.path.join(temp_dir, os.path.basename(url))
122 Download(url, target_path)
123 return target_path
124
125
126 def GetSourceImage(local_dir, pro):
127 url = GetIsoUrl(pro)
128 if local_dir:
129 return os.path.join(local_dir, os.path.basename(url))
130 else:
131 return DownloadMainIso(url)
132
133
134 def ExtractMsiList(iso_dir, packages):
135 """Extracts the contents of a list of .msi files from an already extracted
136 .iso file.
137
138 |packages| is a list of pairs (msi, required). If required is not True, the
139 msi is optional (this is set for packages that are in Pro but not Express).
140 """
141 results = []
142 for (package, required) in packages:
143 path_to_package = os.path.join(iso_dir, 'packages', package)
144 if not os.path.exists(path_to_package) and not required:
145 continue
146 results.append(ExtractMsi(path_to_package))
147 return results
148
149
150 def ExtractComponents(image):
151 packages = [
152 (r'vcRuntimeAdditional_amd64\vc_runtimeAdditional_x64.msi', True),
153 (r'vcRuntimeAdditional_x86\vc_runtimeAdditional_x86.msi', True),
154 (r'vcRuntimeDebug_amd64\vc_runtimeDebug_x64.msi', True),
155 (r'vcRuntimeDebug_x86\vc_runtimeDebug_x86.msi', True),
156 (r'vcRuntimeMinimum_amd64\vc_runtimeMinimum_x64.msi', True),
157 (r'vcRuntimeMinimum_x86\vc_runtimeMinimum_x86.msi', True),
158 (r'vc_compilerCore86\vc_compilerCore86.msi', True),
159 (r'vc_compilerCore86res\vc_compilerCore86res.msi', True),
160 (r'vc_compilerx64nat\vc_compilerx64nat.msi', False),
161 (r'vc_compilerx64natres\vc_compilerx64natres.msi', False),
162 (r'vc_compilerx64x86\vc_compilerx64x86.msi', False),
163 (r'vc_compilerx64x86res\vc_compilerx64x86res.msi', False),
164 (r'vc_librarycore86\vc_librarycore86.msi', True),
165 (r'vc_libraryDesktop\x64\vc_LibraryDesktopX64.msi', True),
166 (r'vc_libraryDesktop\x86\vc_LibraryDesktopX86.msi', True),
167 (r'vc_libraryextended\vc_libraryextended.msi', False),
168 (r'Windows_SDK\Windows Software Development Kit-x86_en-us.msi', True),
169 ('Windows_SDK\\'
170 r'Windows Software Development Kit for Metro style Apps-x86_en-us.msi',
171 True),
172 ]
173 extracted_iso = ExtractIso(image)
174 return ExtractMsiList(extracted_iso, packages)
175
176
177 def CopyToFinalLocation(extracted_dirs, target_dir):
178 sys.stdout.write('Copying to final location...\n')
179 mappings = {
180 'Program Files\\Microsoft Visual Studio 12.0\\': '.\\',
181 'System64\\': 'sys64\\',
182 'System\\': 'sys32\\',
183 'Windows Kits\\8.0\\': 'win8sdk\\',
184 }
185 matches = []
186 for extracted_dir in extracted_dirs:
187 for root, dirnames, filenames in os.walk(extracted_dir):
188 for filename in filenames:
189 matches.append((extracted_dir, os.path.join(root, filename)))
190
191 copies = []
192 for prefix, full_path in matches:
193 # +1 for trailing \.
194 partial_path = full_path[len(prefix) + 1:]
195 for map_from, map_to in mappings.iteritems():
196 if partial_path.startswith(map_from):
197 target_path = os.path.join(map_to, partial_path[len(map_from):])
198 copies.append((full_path, os.path.join(target_dir, target_path)))
199
200 for full_source, full_target in copies:
201 target_dir = os.path.dirname(full_target)
202 if not os.path.isdir(target_dir):
203 os.makedirs(target_dir)
204 shutil.copy2(full_source, full_target)
205
206
207 def GenerateSetEnvCmd(target_dir, pro):
208 """Generate a batch file that gyp expects to exist to set up the compiler
209 environment.
210
211 This is normally generated by a full install of the SDK, but we
212 do it here manually since we do not do a full install."""
213 with open(os.path.join(
214 target_dir, r'win8sdk\bin\SetEnv.cmd'), 'w') as f:
215 f.write('@echo off\n'
216 ':: Generated by win_toolchain\\toolchain2013.py.\n'
217 # Common to x86 and x64
218 'set PATH=%~dp0..\\..\\Common7\\IDE;%PATH%\n'
219 'set INCLUDE=%~dp0..\\..\\win8sdk\\Include\\um;'
220 '%~dp0..\\..\\win8sdk\\Include\\shared;'
221 '%~dp0..\\..\\VC\\include;'
222 '%~dp0..\\..\\VC\\atlmfc\\include\n'
223 'if "%1"=="/x64" goto x64\n')
224
225 # x86. If we're Pro, then use the amd64_x86 cross (we don't support x86
226 # host at all).
227 if pro:
228 f.write('set PATH=%~dp0..\\..\\win8sdk\\bin\\x86;'
229 '%~dp0..\\..\\VC\\bin\\amd64_x86;'
230 '%~dp0..\\..\\VC\\bin\\amd64;' # Needed for mspdb120.dll.
231 '%PATH%\n')
232 else:
233 f.write('set PATH=%~dp0..\\..\\win8sdk\\bin\\x86;'
234 '%~dp0..\\..\\VC\\bin;%PATH%\n')
235 f.write('set LIB=%~dp0..\\..\\VC\\lib;'
236 '%~dp0..\\..\\win8sdk\\Lib\\win8\\um\\x86;'
237 '%~dp0..\\..\\VC\\atlmfc\\lib\n'
238 'goto :EOF\n')
239
240 # Express does not include a native 64 bit compiler, so we have to use
241 # the x86->x64 cross.
242 if not pro:
243 # x86->x64 cross.
244 f.write(':x64\n'
245 'set PATH=%~dp0..\\..\\win8sdk\\bin\\x64;'
246 '%~dp0..\\..\\VC\\bin\\x86_amd64;'
247 '%PATH%\n')
248 else:
249 # x64 native.
250 f.write(':x64\n'
251 'set PATH=%~dp0..\\..\\win8sdk\\bin\\x64;'
252 '%~dp0..\\..\\VC\\bin\\amd64;'
253 '%PATH%\n')
254 f.write('set LIB=%~dp0..\\..\\VC\\lib\\amd64;'
255 '%~dp0..\\..\\win8sdk\\Lib\\win8\\um\\x64;'
256 '%~dp0..\\..\\VC\\atlmfc\\lib\\amd64\n')
257
258
259 def main():
260 parser = OptionParser(description=sys.modules[__name__].__doc__)
261 parser.add_option('--targetdir', metavar='DIR',
262 help='put toolchain into DIR',
263 default=os.path.join(BASEDIR, 'win_toolchain_2013'))
264 parser.add_option('--noclean', action='store_false', dest='clean',
265 help='do not remove temp files',
266 default=True)
267 parser.add_option('--local', metavar='DIR',
268 help='use downloaded files from DIR')
269 parser.add_option('--express',
270 help='use VS Express instead of Pro', action='store_true')
271 options, args = parser.parse_args()
272 try:
273 target_dir = os.path.abspath(options.targetdir)
274 if os.path.exists(target_dir):
275 parser.error('%s already exists. Please [re]move it or use '
276 '--targetdir to select a different target.\n' %
277 target_dir)
278 # Set the working directory to 7z subdirectory. 7-zip doesn't find its
279 # codec dll very well, so this is the simplest way to make sure it runs
280 # correctly, as we don't otherwise care about working directory.
281 os.chdir(os.path.join(BASEDIR, '7z'))
282 image = GetSourceImage(options.local, not options.express)
283 extracted = ExtractComponents(image)
284 CopyToFinalLocation(extracted, target_dir)
285
286 GenerateSetEnvCmd(target_dir, not options.express)
287 finally:
288 if options.clean:
289 DeleteAllTempDirs()
290
291
292 if __name__ == '__main__':
293 sys.exit(main())
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698