forked from virtio-win/virtio-win-pkg-scripts
-
Notifications
You must be signed in to change notification settings - Fork 0
/
make-driver-dir.py
executable file
·270 lines (223 loc) · 9.89 KB
/
make-driver-dir.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
#!/usr/bin/env python3
#
# Copyright 2015 Red Hat, Inc.
#
# This work is licensed under the terms of the GNU GPL, version 2 or later.
# See the COPYING file in the top-level directory.
import argparse
import glob
import os
import re
import shutil
import sys
import textwrap
from util import filemap
from util.utils import fail
######################
# Functional helpers #
######################
def copy_license(input_dir, output_dir):
srcfile = os.path.join(input_dir, "LICENSE")
destfile = os.path.join(output_dir, "virtio-win_license.txt")
shutil.copy(srcfile, destfile)
return [srcfile]
def _update_copymap_for_driver(input_dir, ostuple, drivername, copymap):
destdirs = filemap.DRIVER_OS_MAP[drivername][ostuple]
missing_patterns = []
for destdir in destdirs:
dest_os = destdir.split("/")[0]
filelist = filemap.FILELISTS.get(drivername + ":" + dest_os, None)
if filelist is None:
filelist = filemap.FILELISTS.get(drivername)
for pattern in filelist:
files = glob.glob(os.path.abspath(
os.path.join(input_dir, ostuple, pattern)))
if not files:
strpattern = os.path.join(ostuple, pattern)
if strpattern not in missing_patterns:
missing_patterns.append(strpattern)
continue
for f in files:
if f not in copymap:
copymap[f] = []
copymap[f].append(os.path.join(drivername, destdir))
return missing_patterns
def copy_virtio_drivers(input_dir, output_dir, flavor):
# Create a flat list of every leaf directory in the virtio-win directory
alldirs = []
for dirpath, dirnames, files in os.walk(input_dir):
dummy = files
if dirnames:
continue
ostuple = dirpath[len(input_dir) + 1:]
if ostuple not in alldirs:
alldirs.append(ostuple)
drivers = list(filemap.DRIVER_OS_MAP.keys())[:]
copymap = {}
missing_patterns = []
qemupciserial_ostuple = "./rhel" if flavor == "rhel" else "./"
for drivername in drivers:
for ostuple in sorted(filemap.DRIVER_OS_MAP[drivername]):
# ./rhel is only used on RHEL builds for the qemupciserial
# driver, so if it's not present on public builds, ignore it
# Similarly, if we're asked to create a RHEL build, don't
# look for the Fedora qemupciserial in ./
if (drivername == "qemupciserial" and
ostuple != qemupciserial_ostuple):
continue
if os.path.normpath(ostuple) not in alldirs and ostuple != "./":
fail("driver=%s ostuple=%s not found in input=%s" %
(drivername, ostuple, input_dir))
# We know that the ostuple dir contains bits for this driver,
# figure out what files we want to copy.
ret = _update_copymap_for_driver(input_dir,
ostuple, drivername, copymap)
missing_patterns.extend(ret)
if missing_patterns:
msg = ("\nDid not find any files matching these patterns:\n %s\n\n"
% "\n ".join(missing_patterns))
msg += textwrap.fill("This means we expected to find that file in the "
"virtio-win-prewhql archive, but it wasn't found. This means the "
"build output changed. Assuming this file was intentionally "
"removed, you'll need to update the file whitelists in "
"filemap.py to accurately reflect the current new file layout.")
msg += "\n\n"
fail(msg)
# Actually copy the files, and track the ones we've seen
for srcfile, dests in list(copymap.items()):
for d in dests:
d = os.path.join(output_dir, d)
if not os.path.exists(d):
os.makedirs(d)
shutil.copy2(srcfile, d)
# The keys here are all a list of files we actually copied
return list(copymap.keys())
def check_remaining_files(input_dir, seenfiles, flavor):
# Expected files that we want to skip. The reason we are so strict here
# is to make sure that we don't forget to ship important files that appear
# in new virtio-win builds. If a new file appears, we probably need to ask
# the driver developers whether to ship it or not.
whitelist = [
# vadim confirmed these files should _not_ be shipped
# (private mail May 2015)
r".*DVL\.XML",
".*vioser-test.*",
# Added in 171 build in May 2019, similar to above XML so I
# presume it shouldn't be shipped
r".*DVL-compat\.XML",
# These are files that are needed for the vfd build process. They
# were added to the prewhql sources in July 2015.
# See: https://bugzilla.redhat.com/show_bug.cgi?id=1217799
#
# We could possibly use them in this repo, but it's a bit
# difficult because of the RHEL build process sharing.
# Bug: https://bugzilla.redhat.com/show_bug.cgi?id=1251770
".*/disk1",
".*/txtsetup-i386.oem",
".*/txtsetup-amd64.oem",
# qxlwddm changelogs
".*/spice-qxl-wddm-dod/w10/Changelog",
".*/spice-qxl-wddm-dod-8.1-compatible/Changelog",
# virtio-win build system unconditionally builds every driver
# for every windows platform that supports it. However, depending
# on the driver, functionally identical binaries might be
# generated. In those cases, we ship only one build of the driver
# for every windows version it will work on (see filemap.py
# DRIVER_OS_MAP)
#
# This also simplifies the WHQL submission process, one submission
# can cover multiple windows versions.
#
# In those cases, we end up with unused virtio-win build output.
# That's what the below drivers cover.
#
# If you add to this list, be sure it's not a newly introduced
# driver that you are ignoring! Everything listed here needs
# be covered by a mapping in DRIVER_OS_MAP
]
if flavor == "rhel":
whitelist.extend([
"/qemupciserial.cat",
"/qemupciserial.inf",
])
else:
whitelist.extend([
# Added in virtio-win build 137, for rhel only
"/rhel/qemupciserial.cat",
"/rhel/qemupciserial.inf",
])
remaining = []
for dirpath, dirnames, files in os.walk(input_dir):
dummy = dirnames
for f in files:
remaining.append(os.path.join(dirpath, f))
notseen = [f for f in remaining if f not in seenfiles]
seenpatterns = []
for pattern in whitelist:
for f in notseen[:]:
if not re.match(pattern, f[len(input_dir):]):
continue
notseen.remove(f)
if pattern not in seenpatterns:
seenpatterns.append(pattern)
if notseen:
msg = ("\nUnhandled virtio-win files:\n %s\n\n" %
"\n ".join([f[len(input_dir):] for f in sorted(notseen)]))
msg += textwrap.fill("This means the above files were not tracked "
"in filemap.py _and_ not tracked in the internal whitelist "
"in this script. This probably means that there is new build "
"output. You need to determine if it's something we should "
"be shipping (add it to filemap.py) or something we should "
"ignore (add it to the whitelist).")
fail(msg)
if len(seenpatterns) != len(whitelist):
msg = ("\nDidn't match some whitelist entries:\n %s\n\n" %
"\n ".join([p for p in whitelist if p not in seenpatterns]))
msg += textwrap.fill("This means that the above pattern did not "
"match anything in the build output. That pattern comes from "
"the internal whitelist tracked as part of this script: they "
"are files that we expect to see in the build output, but "
"that we deliberately do _not_ ship as part of the RPM. If "
"the whitelist entry didn't match, it likely means that the "
"files are no longer output by the driver build, so you can "
"just remove the explicit whitelist entry.")
fail(msg)
###################
# main() handling #
###################
def parse_args():
parser = argparse.ArgumentParser(
description="Copy built windows drivers to --output_dir "
"with the file layout expected by "
"make-virtio-win-rpm-archive.py. "
"See README.md for details.")
parser.add_argument("input_dir", help="Directory containing "
"virtio-win and qxl-win build output")
default_output_dir = os.path.join(os.getcwd(), "drivers_output")
parser.add_argument("--output-dir", "--outdir",
help="Directory to output the organized drivers. "
"Default=%s" % default_output_dir, default=default_output_dir)
parser.add_argument("--flavor", help="Flavor of drivers to use if more "
"than one is available. Default is fedora.", default="fedora")
return parser.parse_args()
def main():
options = parse_args()
output_dir = options.output_dir
flavor = options.flavor
if not os.path.exists(output_dir):
os.mkdir(output_dir)
if os.listdir(output_dir):
fail("%s is not empty." % output_dir)
if flavor != "fedora" and flavor != "rhel":
fail("%s is not a known flavor." % flavor)
options.input_dir = os.path.abspath(os.path.expanduser(options.input_dir))
# Actually move the files
seenfiles = []
seenfiles += copy_virtio_drivers(options.input_dir, output_dir, flavor)
seenfiles += copy_license(options.input_dir, output_dir)
# Verify that there is nothing left over that we missed
check_remaining_files(options.input_dir, seenfiles, flavor)
print("Generated %s" % output_dir)
return 0
if __name__ == '__main__':
sys.exit(main())