blob: 78f37dbd3cd04bb72ec17e68d8c482e669759200 [file] [log] [blame]
Doug Zongkereef39442009-04-02 12:14:19 -07001#!/usr/bin/env python
2#
3# Copyright (C) 2008 The Android Open Source Project
4#
5# Licensed under the Apache License, Version 2.0 (the "License");
6# you may not use this file except in compliance with the License.
7# You may obtain a copy of the License at
8#
9# http://www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an "AS IS" BASIS,
13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14# See the License for the specific language governing permissions and
15# limitations under the License.
16
17"""
18Given a target-files zipfile, produces an OTA package that installs
19that build. An incremental OTA is produced if -i is given, otherwise
20a full OTA is produced.
21
22Usage: ota_from_target_files [flags] input_target_files output_ota_package
23
24 -b (--board_config) <file>
Doug Zongkerfdd8e692009-08-03 17:27:48 -070025 Deprecated.
Doug Zongkereef39442009-04-02 12:14:19 -070026
27 -k (--package_key) <key>
28 Key to use to sign the package (default is
29 "build/target/product/security/testkey").
30
31 -i (--incremental_from) <file>
32 Generate an incremental OTA using the given target-files zip as
33 the starting build.
34
Doug Zongkerdbfaae52009-04-21 17:12:54 -070035 -w (--wipe_user_data)
36 Generate an OTA package that will wipe the user data partition
37 when installed.
38
Doug Zongker962069c2009-04-23 11:41:58 -070039 -n (--no_prereq)
40 Omit the timestamp prereq check normally included at the top of
41 the build scripts (used for developer OTA packages which
42 legitimately need to go back and forth).
43
Doug Zongker1c390a22009-05-14 19:06:36 -070044 -e (--extra_script) <file>
45 Insert the contents of file at the end of the update script.
46
Doug Zongkereef39442009-04-02 12:14:19 -070047"""
48
49import sys
50
51if sys.hexversion < 0x02040000:
52 print >> sys.stderr, "Python 2.4 or newer is required."
53 sys.exit(1)
54
55import copy
Doug Zongkerc18736b2009-09-30 09:20:32 -070056import errno
Doug Zongkereef39442009-04-02 12:14:19 -070057import os
58import re
Doug Zongkereef39442009-04-02 12:14:19 -070059import subprocess
60import tempfile
61import time
62import zipfile
63
davidcad0bb92011-03-15 14:21:38 +000064try:
65 from hashlib import sha1 as sha1
66except ImportError:
67 from sha import sha as sha1
68
Doug Zongkereef39442009-04-02 12:14:19 -070069import common
Doug Zongkerc494d7c2009-06-18 08:43:44 -070070import edify_generator
Doug Zongkereef39442009-04-02 12:14:19 -070071
72OPTIONS = common.OPTIONS
73OPTIONS.package_key = "build/target/product/security/testkey"
74OPTIONS.incremental_source = None
75OPTIONS.require_verbatim = set()
76OPTIONS.prohibit_verbatim = set(("system/build.prop",))
77OPTIONS.patch_threshold = 0.95
Doug Zongkerdbfaae52009-04-21 17:12:54 -070078OPTIONS.wipe_user_data = False
Doug Zongker962069c2009-04-23 11:41:58 -070079OPTIONS.omit_prereq = False
Doug Zongker1c390a22009-05-14 19:06:36 -070080OPTIONS.extra_script = None
Doug Zongker761e6422009-09-25 10:45:39 -070081OPTIONS.worker_threads = 3
Doug Zongkereef39442009-04-02 12:14:19 -070082
83def MostPopularKey(d, default):
84 """Given a dict, return the key corresponding to the largest
85 value. Returns 'default' if the dict is empty."""
86 x = [(v, k) for (k, v) in d.iteritems()]
87 if not x: return default
88 x.sort()
89 return x[-1][1]
90
91
92def IsSymlink(info):
93 """Return true if the zipfile.ZipInfo object passed in represents a
94 symlink."""
95 return (info.external_attr >> 16) == 0120777
96
97
Doug Zongkereef39442009-04-02 12:14:19 -070098class Item:
99 """Items represent the metadata (user, group, mode) of files and
100 directories in the system image."""
101 ITEMS = {}
102 def __init__(self, name, dir=False):
103 self.name = name
104 self.uid = None
105 self.gid = None
106 self.mode = None
107 self.dir = dir
108
109 if name:
110 self.parent = Item.Get(os.path.dirname(name), dir=True)
111 self.parent.children.append(self)
112 else:
113 self.parent = None
114 if dir:
115 self.children = []
116
117 def Dump(self, indent=0):
118 if self.uid is not None:
119 print "%s%s %d %d %o" % (" "*indent, self.name, self.uid, self.gid, self.mode)
120 else:
121 print "%s%s %s %s %s" % (" "*indent, self.name, self.uid, self.gid, self.mode)
122 if self.dir:
123 print "%s%s" % (" "*indent, self.descendants)
124 print "%s%s" % (" "*indent, self.best_subtree)
125 for i in self.children:
126 i.Dump(indent=indent+1)
127
128 @classmethod
129 def Get(cls, name, dir=False):
130 if name not in cls.ITEMS:
131 cls.ITEMS[name] = Item(name, dir=dir)
132 return cls.ITEMS[name]
133
134 @classmethod
Doug Zongker283e2a12010-03-15 17:52:32 -0700135 def GetMetadata(cls, input_zip):
136
137 try:
138 # See if the target_files contains a record of what the uid,
139 # gid, and mode is supposed to be.
140 output = input_zip.read("META/filesystem_config.txt")
141 except KeyError:
142 # Run the external 'fs_config' program to determine the desired
143 # uid, gid, and mode for every Item object. Note this uses the
144 # one in the client now, which might not be the same as the one
145 # used when this target_files was built.
146 p = common.Run(["fs_config"], stdin=subprocess.PIPE,
147 stdout=subprocess.PIPE, stderr=subprocess.PIPE)
148 suffix = { False: "", True: "/" }
149 input = "".join(["%s%s\n" % (i.name, suffix[i.dir])
150 for i in cls.ITEMS.itervalues() if i.name])
Doug Zongker3475d362010-03-17 16:39:30 -0700151 output, error = p.communicate(input)
Doug Zongker283e2a12010-03-15 17:52:32 -0700152 assert not error
Doug Zongkereef39442009-04-02 12:14:19 -0700153
154 for line in output.split("\n"):
155 if not line: continue
156 name, uid, gid, mode = line.split()
Doug Zongker283e2a12010-03-15 17:52:32 -0700157 i = cls.ITEMS.get(name, None)
158 if i is not None:
159 i.uid = int(uid)
160 i.gid = int(gid)
161 i.mode = int(mode, 8)
162 if i.dir:
163 i.children.sort(key=lambda i: i.name)
164
165 # set metadata for the files generated by this script.
166 i = cls.ITEMS.get("system/recovery-from-boot.p", None)
167 if i: i.uid, i.gid, i.mode = 0, 0, 0644
168 i = cls.ITEMS.get("system/etc/install-recovery.sh", None)
169 if i: i.uid, i.gid, i.mode = 0, 0, 0544
Doug Zongkereef39442009-04-02 12:14:19 -0700170
171 def CountChildMetadata(self):
172 """Count up the (uid, gid, mode) tuples for all children and
173 determine the best strategy for using set_perm_recursive and
174 set_perm to correctly chown/chmod all the files to their desired
175 values. Recursively calls itself for all descendants.
176
177 Returns a dict of {(uid, gid, dmode, fmode): count} counting up
178 all descendants of this node. (dmode or fmode may be None.) Also
179 sets the best_subtree of each directory Item to the (uid, gid,
180 dmode, fmode) tuple that will match the most descendants of that
181 Item.
182 """
183
184 assert self.dir
185 d = self.descendants = {(self.uid, self.gid, self.mode, None): 1}
186 for i in self.children:
187 if i.dir:
188 for k, v in i.CountChildMetadata().iteritems():
189 d[k] = d.get(k, 0) + v
190 else:
191 k = (i.uid, i.gid, None, i.mode)
192 d[k] = d.get(k, 0) + 1
193
194 # Find the (uid, gid, dmode, fmode) tuple that matches the most
195 # descendants.
196
197 # First, find the (uid, gid) pair that matches the most
198 # descendants.
199 ug = {}
200 for (uid, gid, _, _), count in d.iteritems():
201 ug[(uid, gid)] = ug.get((uid, gid), 0) + count
202 ug = MostPopularKey(ug, (0, 0))
203
204 # Now find the dmode and fmode that match the most descendants
205 # with that (uid, gid), and choose those.
206 best_dmode = (0, 0755)
207 best_fmode = (0, 0644)
208 for k, count in d.iteritems():
209 if k[:2] != ug: continue
210 if k[2] is not None and count >= best_dmode[0]: best_dmode = (count, k[2])
211 if k[3] is not None and count >= best_fmode[0]: best_fmode = (count, k[3])
212 self.best_subtree = ug + (best_dmode[1], best_fmode[1])
213
214 return d
215
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700216 def SetPermissions(self, script):
Doug Zongkereef39442009-04-02 12:14:19 -0700217 """Append set_perm/set_perm_recursive commands to 'script' to
218 set all permissions, users, and groups for the tree of files
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700219 rooted at 'self'."""
Doug Zongkereef39442009-04-02 12:14:19 -0700220
221 self.CountChildMetadata()
222
223 def recurse(item, current):
224 # current is the (uid, gid, dmode, fmode) tuple that the current
225 # item (and all its children) have already been set to. We only
226 # need to issue set_perm/set_perm_recursive commands if we're
227 # supposed to be something different.
228 if item.dir:
229 if current != item.best_subtree:
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700230 script.SetPermissionsRecursive("/"+item.name, *item.best_subtree)
Doug Zongkereef39442009-04-02 12:14:19 -0700231 current = item.best_subtree
232
233 if item.uid != current[0] or item.gid != current[1] or \
234 item.mode != current[2]:
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700235 script.SetPermissions("/"+item.name, item.uid, item.gid, item.mode)
Doug Zongkereef39442009-04-02 12:14:19 -0700236
237 for i in item.children:
238 recurse(i, current)
239 else:
240 if item.uid != current[0] or item.gid != current[1] or \
241 item.mode != current[3]:
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700242 script.SetPermissions("/"+item.name, item.uid, item.gid, item.mode)
Doug Zongkereef39442009-04-02 12:14:19 -0700243
244 recurse(self, (-1, -1, -1, -1))
245
246
247def CopySystemFiles(input_zip, output_zip=None,
248 substitute=None):
249 """Copies files underneath system/ in the input zip to the output
250 zip. Populates the Item class with their metadata, and returns a
251 list of symlinks. output_zip may be None, in which case the copy is
252 skipped (but the other side effects still happen). substitute is an
253 optional dict of {output filename: contents} to be output instead of
254 certain input files.
255 """
256
257 symlinks = []
258
259 for info in input_zip.infolist():
260 if info.filename.startswith("SYSTEM/"):
261 basefilename = info.filename[7:]
262 if IsSymlink(info):
263 symlinks.append((input_zip.read(info.filename),
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700264 "/system/" + basefilename))
Doug Zongkereef39442009-04-02 12:14:19 -0700265 else:
266 info2 = copy.copy(info)
267 fn = info2.filename = "system/" + basefilename
268 if substitute and fn in substitute and substitute[fn] is None:
269 continue
270 if output_zip is not None:
271 if substitute and fn in substitute:
272 data = substitute[fn]
273 else:
274 data = input_zip.read(info.filename)
275 output_zip.writestr(info2, data)
276 if fn.endswith("/"):
277 Item.Get(fn[:-1], dir=True)
278 else:
279 Item.Get(fn, dir=False)
280
281 symlinks.sort()
282 return symlinks
283
284
Doug Zongkereef39442009-04-02 12:14:19 -0700285def SignOutput(temp_zip_name, output_zip_name):
286 key_passwords = common.GetKeyPasswords([OPTIONS.package_key])
287 pw = key_passwords[OPTIONS.package_key]
288
Doug Zongker951495f2009-08-14 12:44:19 -0700289 common.SignFile(temp_zip_name, output_zip_name, OPTIONS.package_key, pw,
290 whole_file=True)
Doug Zongkereef39442009-04-02 12:14:19 -0700291
292
Doug Zongkereef39442009-04-02 12:14:19 -0700293def AppendAssertions(script, input_zip):
Doug Zongkereef39442009-04-02 12:14:19 -0700294 device = GetBuildProp("ro.product.device", input_zip)
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700295 script.AssertDevice(device)
Doug Zongkereef39442009-04-02 12:14:19 -0700296
Doug Zongkereef39442009-04-02 12:14:19 -0700297
Doug Zongker73ef8252009-07-23 15:12:53 -0700298def MakeRecoveryPatch(output_zip, recovery_img, boot_img):
299 """Generate a binary patch that creates the recovery image starting
300 with the boot image. (Most of the space in these images is just the
301 kernel, which is identical for the two, so the resulting patch
302 should be efficient.) Add it to the output zip, along with a shell
303 script that is run from init.rc on first boot to actually do the
304 patching and install the new recovery image.
305
306 recovery_img and boot_img should be File objects for the
307 corresponding images.
308
309 Returns an Item for the shell script, which must be made
310 executable.
311 """
312
Doug Zongkerea5d7a92010-09-12 15:26:16 -0700313 d = common.Difference(recovery_img, boot_img)
Doug Zongker761e6422009-09-25 10:45:39 -0700314 _, _, patch = d.ComputePatch()
Doug Zongkercfd7db62009-10-07 11:35:53 -0700315 common.ZipWriteStr(output_zip, "recovery/recovery-from-boot.p", patch)
Doug Zongker73ef8252009-07-23 15:12:53 -0700316 Item.Get("system/recovery-from-boot.p", dir=False)
317
Doug Zongker96a57e72010-09-26 14:57:41 -0700318 boot_type, boot_device = common.GetTypeAndDevice("/boot", OPTIONS.info_dict)
319 recovery_type, recovery_device = common.GetTypeAndDevice("/recovery", OPTIONS.info_dict)
Doug Zongker780c2372010-09-22 10:12:54 -0700320
Doug Zongker73ef8252009-07-23 15:12:53 -0700321 # Images with different content will have a different first page, so
322 # we check to see if this recovery has already been installed by
323 # testing just the first 2k.
324 HEADER_SIZE = 2048
davidcad0bb92011-03-15 14:21:38 +0000325 header_sha1 = sha1(recovery_img.data[:HEADER_SIZE]).hexdigest()
Doug Zongker73ef8252009-07-23 15:12:53 -0700326 sh = """#!/system/bin/sh
Doug Zongker780c2372010-09-22 10:12:54 -0700327if ! applypatch -c %(recovery_type)s:%(recovery_device)s:%(header_size)d:%(header_sha1)s; then
Doug Zongker73ef8252009-07-23 15:12:53 -0700328 log -t recovery "Installing new recovery image"
Doug Zongker780c2372010-09-22 10:12:54 -0700329 applypatch %(boot_type)s:%(boot_device)s:%(boot_size)d:%(boot_sha1)s %(recovery_type)s:%(recovery_device)s %(recovery_sha1)s %(recovery_size)d %(boot_sha1)s:/system/recovery-from-boot.p
Doug Zongker73ef8252009-07-23 15:12:53 -0700330else
331 log -t recovery "Recovery image already installed"
332fi
333""" % { 'boot_size': boot_img.size,
334 'boot_sha1': boot_img.sha1,
335 'header_size': HEADER_SIZE,
336 'header_sha1': header_sha1,
337 'recovery_size': recovery_img.size,
Doug Zongker780c2372010-09-22 10:12:54 -0700338 'recovery_sha1': recovery_img.sha1,
339 'boot_type': boot_type,
340 'boot_device': boot_device,
341 'recovery_type': recovery_type,
342 'recovery_device': recovery_device,
343 }
Doug Zongkercfd7db62009-10-07 11:35:53 -0700344 common.ZipWriteStr(output_zip, "recovery/etc/install-recovery.sh", sh)
Doug Zongker73ef8252009-07-23 15:12:53 -0700345 return Item.Get("system/etc/install-recovery.sh", dir=False)
346
347
Doug Zongkerc77a9ad2010-09-16 11:28:43 -0700348def WriteFullOTAPackage(input_zip, output_zip):
Doug Zongkerc637db12010-04-21 14:08:44 -0700349 # TODO: how to determine this? We don't know what version it will
350 # be installed on top of. For now, we expect the API just won't
351 # change very often.
Doug Zongkerc77a9ad2010-09-16 11:28:43 -0700352 script = edify_generator.EdifyGenerator(3, OPTIONS.info_dict)
Doug Zongkereef39442009-04-02 12:14:19 -0700353
Doug Zongker2ea21062010-04-28 16:05:21 -0700354 metadata = {"post-build": GetBuildProp("ro.build.fingerprint", input_zip),
355 "pre-device": GetBuildProp("ro.product.device", input_zip),
Doug Zongker3b852692010-06-21 15:30:45 -0700356 "post-timestamp": GetBuildProp("ro.build.date.utc", input_zip),
Doug Zongker2ea21062010-04-28 16:05:21 -0700357 }
358
Doug Zongker05d3dea2009-06-22 11:32:31 -0700359 device_specific = common.DeviceSpecificParams(
360 input_zip=input_zip,
Doug Zongker37974732010-09-16 17:44:38 -0700361 input_version=OPTIONS.info_dict["recovery_api_version"],
Doug Zongker05d3dea2009-06-22 11:32:31 -0700362 output_zip=output_zip,
363 script=script,
Doug Zongker2ea21062010-04-28 16:05:21 -0700364 input_tmp=OPTIONS.input_tmp,
Doug Zongker96a57e72010-09-26 14:57:41 -0700365 metadata=metadata,
366 info_dict=OPTIONS.info_dict)
Doug Zongker05d3dea2009-06-22 11:32:31 -0700367
Doug Zongker962069c2009-04-23 11:41:58 -0700368 if not OPTIONS.omit_prereq:
369 ts = GetBuildProp("ro.build.date.utc", input_zip)
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700370 script.AssertOlderBuild(ts)
Doug Zongkereef39442009-04-02 12:14:19 -0700371
372 AppendAssertions(script, input_zip)
Doug Zongker05d3dea2009-06-22 11:32:31 -0700373 device_specific.FullOTA_Assertions()
Doug Zongker171f1cd2009-06-15 22:36:37 -0700374
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700375 script.ShowProgress(0.5, 0)
Doug Zongkereef39442009-04-02 12:14:19 -0700376
Doug Zongkerdbfaae52009-04-21 17:12:54 -0700377 if OPTIONS.wipe_user_data:
Doug Zongker258bf462010-09-20 18:04:41 -0700378 script.FormatPartition("/data")
Doug Zongkerdbfaae52009-04-21 17:12:54 -0700379
Doug Zongker258bf462010-09-20 18:04:41 -0700380 script.FormatPartition("/system")
381 script.Mount("/system")
Doug Zongkercfd7db62009-10-07 11:35:53 -0700382 script.UnpackPackageDir("recovery", "/system")
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700383 script.UnpackPackageDir("system", "/system")
Doug Zongkereef39442009-04-02 12:14:19 -0700384
385 symlinks = CopySystemFiles(input_zip, output_zip)
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700386 script.MakeSymlinks(symlinks)
Doug Zongkereef39442009-04-02 12:14:19 -0700387
Doug Zongkerea5d7a92010-09-12 15:26:16 -0700388 boot_img = common.File("boot.img", common.BuildBootableImage(
Doug Zongker73ef8252009-07-23 15:12:53 -0700389 os.path.join(OPTIONS.input_tmp, "BOOT")))
Doug Zongkerea5d7a92010-09-12 15:26:16 -0700390 recovery_img = common.File("recovery.img", common.BuildBootableImage(
Doug Zongker73ef8252009-07-23 15:12:53 -0700391 os.path.join(OPTIONS.input_tmp, "RECOVERY")))
Doug Zongker283e2a12010-03-15 17:52:32 -0700392 MakeRecoveryPatch(output_zip, recovery_img, boot_img)
Doug Zongkereef39442009-04-02 12:14:19 -0700393
Doug Zongker283e2a12010-03-15 17:52:32 -0700394 Item.GetMetadata(input_zip)
Doug Zongker73ef8252009-07-23 15:12:53 -0700395 Item.Get("system").SetPermissions(script)
396
Doug Zongker37974732010-09-16 17:44:38 -0700397 common.CheckSize(boot_img.data, "boot.img", OPTIONS.info_dict)
Doug Zongker73ef8252009-07-23 15:12:53 -0700398 common.ZipWriteStr(output_zip, "boot.img", boot_img.data)
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700399 script.ShowProgress(0.2, 0)
400
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700401 script.ShowProgress(0.2, 10)
Doug Zongker258bf462010-09-20 18:04:41 -0700402 script.WriteRawImage("/boot", "boot.img")
Doug Zongker05d3dea2009-06-22 11:32:31 -0700403
404 script.ShowProgress(0.1, 0)
405 device_specific.FullOTA_InstallEnd()
Doug Zongkereef39442009-04-02 12:14:19 -0700406
Doug Zongker1c390a22009-05-14 19:06:36 -0700407 if OPTIONS.extra_script is not None:
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700408 script.AppendExtra(OPTIONS.extra_script)
Doug Zongker1c390a22009-05-14 19:06:36 -0700409
Doug Zongker14833602010-02-02 13:12:04 -0800410 script.UnmountAll()
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700411 script.AddToZip(input_zip, output_zip)
Doug Zongker2ea21062010-04-28 16:05:21 -0700412 WriteMetadata(metadata, output_zip)
413
414
415def WriteMetadata(metadata, output_zip):
416 common.ZipWriteStr(output_zip, "META-INF/com/android/metadata",
417 "".join(["%s=%s\n" % kv
418 for kv in sorted(metadata.iteritems())]))
Doug Zongkereef39442009-04-02 12:14:19 -0700419
420
Doug Zongkereef39442009-04-02 12:14:19 -0700421
422
423def LoadSystemFiles(z):
424 """Load all the files from SYSTEM/... in a given target-files
425 ZipFile, and return a dict of {filename: File object}."""
426 out = {}
427 for info in z.infolist():
428 if info.filename.startswith("SYSTEM/") and not IsSymlink(info):
429 fn = "system/" + info.filename[7:]
430 data = z.read(info.filename)
Doug Zongkerea5d7a92010-09-12 15:26:16 -0700431 out[fn] = common.File(fn, data)
Doug Zongkereef39442009-04-02 12:14:19 -0700432 return out
433
434
Doug Zongkereef39442009-04-02 12:14:19 -0700435def GetBuildProp(property, z):
436 """Return the fingerprint of the build of a given target-files
437 ZipFile object."""
438 bp = z.read("SYSTEM/build.prop")
439 if not property:
440 return bp
441 m = re.search(re.escape(property) + r"=(.*)\n", bp)
442 if not m:
Doug Zongker9fc74c72009-06-23 16:27:38 -0700443 raise common.ExternalError("couldn't find %s in build.prop" % (property,))
Doug Zongkereef39442009-04-02 12:14:19 -0700444 return m.group(1).strip()
445
446
Doug Zongkerc77a9ad2010-09-16 11:28:43 -0700447def WriteIncrementalOTAPackage(target_zip, source_zip, output_zip):
Doug Zongker37974732010-09-16 17:44:38 -0700448 source_version = OPTIONS.source_info_dict["recovery_api_version"]
449 target_version = OPTIONS.target_info_dict["recovery_api_version"]
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700450
Doug Zongkerc637db12010-04-21 14:08:44 -0700451 if source_version == 0:
452 print ("WARNING: generating edify script for a source that "
453 "can't install it.")
Doug Zongkerc77a9ad2010-09-16 11:28:43 -0700454 script = edify_generator.EdifyGenerator(source_version, OPTIONS.info_dict)
Doug Zongkereef39442009-04-02 12:14:19 -0700455
Doug Zongker2ea21062010-04-28 16:05:21 -0700456 metadata = {"pre-device": GetBuildProp("ro.product.device", source_zip),
Doug Zongker3b852692010-06-21 15:30:45 -0700457 "post-timestamp": GetBuildProp("ro.build.date.utc", target_zip),
Doug Zongker2ea21062010-04-28 16:05:21 -0700458 }
459
Doug Zongker05d3dea2009-06-22 11:32:31 -0700460 device_specific = common.DeviceSpecificParams(
461 source_zip=source_zip,
Doug Zongker14833602010-02-02 13:12:04 -0800462 source_version=source_version,
Doug Zongker05d3dea2009-06-22 11:32:31 -0700463 target_zip=target_zip,
Doug Zongker14833602010-02-02 13:12:04 -0800464 target_version=target_version,
Doug Zongker05d3dea2009-06-22 11:32:31 -0700465 output_zip=output_zip,
Doug Zongker2ea21062010-04-28 16:05:21 -0700466 script=script,
Doug Zongker96a57e72010-09-26 14:57:41 -0700467 metadata=metadata,
468 info_dict=OPTIONS.info_dict)
Doug Zongker05d3dea2009-06-22 11:32:31 -0700469
Doug Zongkereef39442009-04-02 12:14:19 -0700470 print "Loading target..."
471 target_data = LoadSystemFiles(target_zip)
472 print "Loading source..."
473 source_data = LoadSystemFiles(source_zip)
474
475 verbatim_targets = []
476 patch_list = []
Doug Zongker761e6422009-09-25 10:45:39 -0700477 diffs = []
Doug Zongkereef39442009-04-02 12:14:19 -0700478 largest_source_size = 0
479 for fn in sorted(target_data.keys()):
480 tf = target_data[fn]
Doug Zongker761e6422009-09-25 10:45:39 -0700481 assert fn == tf.name
Doug Zongkereef39442009-04-02 12:14:19 -0700482 sf = source_data.get(fn, None)
483
484 if sf is None or fn in OPTIONS.require_verbatim:
485 # This file should be included verbatim
486 if fn in OPTIONS.prohibit_verbatim:
Doug Zongker9fc74c72009-06-23 16:27:38 -0700487 raise common.ExternalError("\"%s\" must be sent verbatim" % (fn,))
Doug Zongkereef39442009-04-02 12:14:19 -0700488 print "send", fn, "verbatim"
489 tf.AddToZip(output_zip)
490 verbatim_targets.append((fn, tf.size))
491 elif tf.sha1 != sf.sha1:
492 # File is different; consider sending as a patch
Doug Zongkerea5d7a92010-09-12 15:26:16 -0700493 diffs.append(common.Difference(tf, sf))
Doug Zongkereef39442009-04-02 12:14:19 -0700494 else:
495 # Target file identical to source.
496 pass
497
Doug Zongkerea5d7a92010-09-12 15:26:16 -0700498 common.ComputeDifferences(diffs)
Doug Zongker761e6422009-09-25 10:45:39 -0700499
500 for diff in diffs:
501 tf, sf, d = diff.GetPatch()
502 if d is None or len(d) > tf.size * OPTIONS.patch_threshold:
503 # patch is almost as big as the file; don't bother patching
504 tf.AddToZip(output_zip)
505 verbatim_targets.append((tf.name, tf.size))
506 else:
507 common.ZipWriteStr(output_zip, "patch/" + tf.name + ".p", d)
davidcad0bb92011-03-15 14:21:38 +0000508 patch_list.append((tf.name, tf, sf, tf.size, sha1(d).hexdigest()))
Doug Zongker761e6422009-09-25 10:45:39 -0700509 largest_source_size = max(largest_source_size, sf.size)
Doug Zongkereef39442009-04-02 12:14:19 -0700510
511 source_fp = GetBuildProp("ro.build.fingerprint", source_zip)
512 target_fp = GetBuildProp("ro.build.fingerprint", target_zip)
Doug Zongker2ea21062010-04-28 16:05:21 -0700513 metadata["pre-build"] = source_fp
514 metadata["post-build"] = target_fp
Doug Zongkereef39442009-04-02 12:14:19 -0700515
Doug Zongker258bf462010-09-20 18:04:41 -0700516 script.Mount("/system")
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700517 script.AssertSomeFingerprint(source_fp, target_fp)
Doug Zongkereef39442009-04-02 12:14:19 -0700518
Doug Zongkerea5d7a92010-09-12 15:26:16 -0700519 source_boot = common.File("/tmp/boot.img",
520 common.BuildBootableImage(
521 os.path.join(OPTIONS.source_tmp, "BOOT")))
522 target_boot = common.File("/tmp/boot.img",
523 common.BuildBootableImage(
524 os.path.join(OPTIONS.target_tmp, "BOOT")))
Doug Zongker5da317e2009-06-02 13:38:17 -0700525 updating_boot = (source_boot.data != target_boot.data)
Doug Zongkereef39442009-04-02 12:14:19 -0700526
Doug Zongkerea5d7a92010-09-12 15:26:16 -0700527 source_recovery = common.File("system/recovery.img",
528 common.BuildBootableImage(
529 os.path.join(OPTIONS.source_tmp, "RECOVERY")))
530 target_recovery = common.File("system/recovery.img",
531 common.BuildBootableImage(
532 os.path.join(OPTIONS.target_tmp, "RECOVERY")))
Doug Zongkerf6a8bad2009-05-29 11:41:21 -0700533 updating_recovery = (source_recovery.data != target_recovery.data)
Doug Zongkereef39442009-04-02 12:14:19 -0700534
Doug Zongker881dd402009-09-20 14:03:55 -0700535 # Here's how we divide up the progress bar:
536 # 0.1 for verifying the start state (PatchCheck calls)
537 # 0.8 for applying patches (ApplyPatch calls)
538 # 0.1 for unpacking verbatim files, symlinking, and doing the
539 # device-specific commands.
Doug Zongkereef39442009-04-02 12:14:19 -0700540
541 AppendAssertions(script, target_zip)
Doug Zongker05d3dea2009-06-22 11:32:31 -0700542 device_specific.IncrementalOTA_Assertions()
Doug Zongkereef39442009-04-02 12:14:19 -0700543
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700544 script.Print("Verifying current system...")
545
Doug Zongker881dd402009-09-20 14:03:55 -0700546 script.ShowProgress(0.1, 0)
547 total_verify_size = float(sum([i[2].size for i in patch_list]) + 1)
548 if updating_boot:
549 total_verify_size += source_boot.size
550 so_far = 0
Doug Zongkereef39442009-04-02 12:14:19 -0700551
Doug Zongker5a482092010-02-17 16:09:18 -0800552 for fn, tf, sf, size, patch_sha in patch_list:
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700553 script.PatchCheck("/"+fn, tf.sha1, sf.sha1)
Doug Zongker881dd402009-09-20 14:03:55 -0700554 so_far += sf.size
555 script.SetProgress(so_far / total_verify_size)
Doug Zongkereef39442009-04-02 12:14:19 -0700556
Doug Zongker5da317e2009-06-02 13:38:17 -0700557 if updating_boot:
Doug Zongkerea5d7a92010-09-12 15:26:16 -0700558 d = common.Difference(target_boot, source_boot)
Doug Zongker761e6422009-09-25 10:45:39 -0700559 _, _, d = d.ComputePatch()
Doug Zongker5da317e2009-06-02 13:38:17 -0700560 print "boot target: %d source: %d diff: %d" % (
561 target_boot.size, source_boot.size, len(d))
562
Doug Zongker048e7ca2009-06-15 14:31:53 -0700563 common.ZipWriteStr(output_zip, "patch/boot.img.p", d)
Doug Zongker5da317e2009-06-02 13:38:17 -0700564
Doug Zongker96a57e72010-09-26 14:57:41 -0700565 boot_type, boot_device = common.GetTypeAndDevice("/boot", OPTIONS.info_dict)
Doug Zongker780c2372010-09-22 10:12:54 -0700566
567 script.PatchCheck("%s:%s:%d:%s:%d:%s" %
568 (boot_type, boot_device,
569 source_boot.size, source_boot.sha1,
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700570 target_boot.size, target_boot.sha1))
Doug Zongker881dd402009-09-20 14:03:55 -0700571 so_far += source_boot.size
572 script.SetProgress(so_far / total_verify_size)
Doug Zongker5da317e2009-06-02 13:38:17 -0700573
574 if patch_list or updating_recovery or updating_boot:
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700575 script.CacheFreeSpaceCheck(largest_source_size)
Doug Zongker5a482092010-02-17 16:09:18 -0800576
Doug Zongker05d3dea2009-06-22 11:32:31 -0700577 device_specific.IncrementalOTA_VerifyEnd()
578
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700579 script.Comment("---- start making changes here ----")
Doug Zongkereef39442009-04-02 12:14:19 -0700580
Doug Zongkerdbfaae52009-04-21 17:12:54 -0700581 if OPTIONS.wipe_user_data:
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700582 script.Print("Erasing user data...")
Doug Zongker258bf462010-09-20 18:04:41 -0700583 script.FormatPartition("/data")
Doug Zongkerdbfaae52009-04-21 17:12:54 -0700584
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700585 script.Print("Removing unneeded files...")
Doug Zongker0f3298a2009-06-30 08:16:58 -0700586 script.DeleteFiles(["/"+i[0] for i in verbatim_targets] +
587 ["/"+i for i in sorted(source_data)
Doug Zongker3b949f02009-08-24 10:24:32 -0700588 if i not in target_data] +
589 ["/system/recovery.img"])
Doug Zongkereef39442009-04-02 12:14:19 -0700590
Doug Zongker881dd402009-09-20 14:03:55 -0700591 script.ShowProgress(0.8, 0)
592 total_patch_size = float(sum([i[1].size for i in patch_list]) + 1)
593 if updating_boot:
594 total_patch_size += target_boot.size
595 so_far = 0
596
597 script.Print("Patching system files...")
Doug Zongker5a482092010-02-17 16:09:18 -0800598 for fn, tf, sf, size, _ in patch_list:
Doug Zongkerc8d446b2010-02-22 15:41:53 -0800599 script.ApplyPatch("/"+fn, "-", tf.size, tf.sha1, sf.sha1, "patch/"+fn+".p")
Doug Zongker881dd402009-09-20 14:03:55 -0700600 so_far += tf.size
601 script.SetProgress(so_far / total_patch_size)
602
Doug Zongkereef39442009-04-02 12:14:19 -0700603 if updating_boot:
Doug Zongker5da317e2009-06-02 13:38:17 -0700604 # Produce the boot image by applying a patch to the current
605 # contents of the boot partition, and write it back to the
606 # partition.
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700607 script.Print("Patching boot image...")
Doug Zongker780c2372010-09-22 10:12:54 -0700608 script.ApplyPatch("%s:%s:%d:%s:%d:%s"
609 % (boot_type, boot_device,
610 source_boot.size, source_boot.sha1,
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700611 target_boot.size, target_boot.sha1),
612 "-",
613 target_boot.size, target_boot.sha1,
Doug Zongkerc8d446b2010-02-22 15:41:53 -0800614 source_boot.sha1, "patch/boot.img.p")
Doug Zongker881dd402009-09-20 14:03:55 -0700615 so_far += target_boot.size
616 script.SetProgress(so_far / total_patch_size)
Doug Zongkereef39442009-04-02 12:14:19 -0700617 print "boot image changed; including."
618 else:
619 print "boot image unchanged; skipping."
620
621 if updating_recovery:
Doug Zongker73ef8252009-07-23 15:12:53 -0700622 # Is it better to generate recovery as a patch from the current
623 # boot image, or from the previous recovery image? For large
624 # updates with significant kernel changes, probably the former.
625 # For small updates where the kernel hasn't changed, almost
626 # certainly the latter. We pick the first option. Future
627 # complicated schemes may let us effectively use both.
628 #
629 # A wacky possibility: as long as there is room in the boot
630 # partition, include the binaries and image files from recovery in
631 # the boot image (though not in the ramdisk) so they can be used
632 # as fodder for constructing the recovery image.
Doug Zongker283e2a12010-03-15 17:52:32 -0700633 MakeRecoveryPatch(output_zip, target_recovery, target_boot)
Doug Zongker42265392010-02-12 10:21:00 -0800634 script.DeleteFiles(["/system/recovery-from-boot.p",
635 "/system/etc/install-recovery.sh"])
Doug Zongker73ef8252009-07-23 15:12:53 -0700636 print "recovery image changed; including as patch from boot."
Doug Zongkereef39442009-04-02 12:14:19 -0700637 else:
638 print "recovery image unchanged; skipping."
639
Doug Zongker881dd402009-09-20 14:03:55 -0700640 script.ShowProgress(0.1, 10)
Doug Zongkereef39442009-04-02 12:14:19 -0700641
642 target_symlinks = CopySystemFiles(target_zip, None)
643
644 target_symlinks_d = dict([(i[1], i[0]) for i in target_symlinks])
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700645 temp_script = script.MakeTemporary()
Doug Zongker283e2a12010-03-15 17:52:32 -0700646 Item.GetMetadata(target_zip)
Doug Zongker73ef8252009-07-23 15:12:53 -0700647 Item.Get("system").SetPermissions(temp_script)
Doug Zongkereef39442009-04-02 12:14:19 -0700648
649 # Note that this call will mess up the tree of Items, so make sure
650 # we're done with it.
651 source_symlinks = CopySystemFiles(source_zip, None)
652 source_symlinks_d = dict([(i[1], i[0]) for i in source_symlinks])
653
654 # Delete all the symlinks in source that aren't in target. This
655 # needs to happen before verbatim files are unpacked, in case a
656 # symlink in the source is replaced by a real file in the target.
657 to_delete = []
658 for dest, link in source_symlinks:
659 if link not in target_symlinks_d:
660 to_delete.append(link)
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700661 script.DeleteFiles(to_delete)
Doug Zongkereef39442009-04-02 12:14:19 -0700662
663 if verbatim_targets:
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700664 script.Print("Unpacking new files...")
665 script.UnpackPackageDir("system", "/system")
666
Doug Zongker42265392010-02-12 10:21:00 -0800667 if updating_recovery:
668 script.Print("Unpacking new recovery...")
669 script.UnpackPackageDir("recovery", "/system")
670
Doug Zongker05d3dea2009-06-22 11:32:31 -0700671 script.Print("Symlinks and permissions...")
Doug Zongkereef39442009-04-02 12:14:19 -0700672
673 # Create all the symlinks that don't already exist, or point to
674 # somewhere different than what we want. Delete each symlink before
675 # creating it, since the 'symlink' command won't overwrite.
676 to_create = []
677 for dest, link in target_symlinks:
678 if link in source_symlinks_d:
679 if dest != source_symlinks_d[link]:
680 to_create.append((dest, link))
681 else:
682 to_create.append((dest, link))
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700683 script.DeleteFiles([i[1] for i in to_create])
684 script.MakeSymlinks(to_create)
Doug Zongkereef39442009-04-02 12:14:19 -0700685
686 # Now that the symlinks are created, we can set all the
687 # permissions.
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700688 script.AppendScript(temp_script)
Doug Zongkereef39442009-04-02 12:14:19 -0700689
Doug Zongker881dd402009-09-20 14:03:55 -0700690 # Do device-specific installation (eg, write radio image).
Doug Zongker05d3dea2009-06-22 11:32:31 -0700691 device_specific.IncrementalOTA_InstallEnd()
692
Doug Zongker1c390a22009-05-14 19:06:36 -0700693 if OPTIONS.extra_script is not None:
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700694 scirpt.AppendExtra(OPTIONS.extra_script)
Doug Zongker1c390a22009-05-14 19:06:36 -0700695
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700696 script.AddToZip(target_zip, output_zip)
Doug Zongker2ea21062010-04-28 16:05:21 -0700697 WriteMetadata(metadata, output_zip)
Doug Zongkereef39442009-04-02 12:14:19 -0700698
699
700def main(argv):
701
702 def option_handler(o, a):
703 if o in ("-b", "--board_config"):
Doug Zongkerfdd8e692009-08-03 17:27:48 -0700704 pass # deprecated
Doug Zongkereef39442009-04-02 12:14:19 -0700705 elif o in ("-k", "--package_key"):
706 OPTIONS.package_key = a
Doug Zongkereef39442009-04-02 12:14:19 -0700707 elif o in ("-i", "--incremental_from"):
708 OPTIONS.incremental_source = a
Doug Zongkerdbfaae52009-04-21 17:12:54 -0700709 elif o in ("-w", "--wipe_user_data"):
710 OPTIONS.wipe_user_data = True
Doug Zongker962069c2009-04-23 11:41:58 -0700711 elif o in ("-n", "--no_prereq"):
712 OPTIONS.omit_prereq = True
Doug Zongker1c390a22009-05-14 19:06:36 -0700713 elif o in ("-e", "--extra_script"):
714 OPTIONS.extra_script = a
Doug Zongker761e6422009-09-25 10:45:39 -0700715 elif o in ("--worker_threads"):
716 OPTIONS.worker_threads = int(a)
Doug Zongkereef39442009-04-02 12:14:19 -0700717 else:
718 return False
Doug Zongkerdbfaae52009-04-21 17:12:54 -0700719 return True
Doug Zongkereef39442009-04-02 12:14:19 -0700720
721 args = common.ParseOptions(argv, __doc__,
Doug Zongkerc637db12010-04-21 14:08:44 -0700722 extra_opts="b:k:i:d:wne:",
Doug Zongkereef39442009-04-02 12:14:19 -0700723 extra_long_opts=["board_config=",
724 "package_key=",
Doug Zongkerdbfaae52009-04-21 17:12:54 -0700725 "incremental_from=",
Doug Zongker962069c2009-04-23 11:41:58 -0700726 "wipe_user_data",
Doug Zongker1c390a22009-05-14 19:06:36 -0700727 "no_prereq",
Doug Zongkerc494d7c2009-06-18 08:43:44 -0700728 "extra_script=",
Doug Zongker761e6422009-09-25 10:45:39 -0700729 "worker_threads="],
Doug Zongkereef39442009-04-02 12:14:19 -0700730 extra_option_handler=option_handler)
731
732 if len(args) != 2:
733 common.Usage(__doc__)
734 sys.exit(1)
735
Doug Zongker1c390a22009-05-14 19:06:36 -0700736 if OPTIONS.extra_script is not None:
737 OPTIONS.extra_script = open(OPTIONS.extra_script).read()
738
Doug Zongkereef39442009-04-02 12:14:19 -0700739 print "unzipping target target-files..."
740 OPTIONS.input_tmp = common.UnzipTemp(args[0])
Doug Zongkerfdd8e692009-08-03 17:27:48 -0700741
Doug Zongkereef39442009-04-02 12:14:19 -0700742 OPTIONS.target_tmp = OPTIONS.input_tmp
743 input_zip = zipfile.ZipFile(args[0], "r")
Doug Zongker37974732010-09-16 17:44:38 -0700744 OPTIONS.info_dict = common.LoadInfoDict(input_zip)
745 if OPTIONS.verbose:
746 print "--- target info ---"
747 common.DumpInfoDict(OPTIONS.info_dict)
748
749 if OPTIONS.device_specific is None:
750 OPTIONS.device_specific = OPTIONS.info_dict.get("tool_extensions", None)
751 if OPTIONS.device_specific is not None:
752 OPTIONS.device_specific = os.path.normpath(OPTIONS.device_specific)
753 print "using device-specific extensions in", OPTIONS.device_specific
754
Doug Zongkereef39442009-04-02 12:14:19 -0700755 if OPTIONS.package_key:
756 temp_zip_file = tempfile.NamedTemporaryFile()
757 output_zip = zipfile.ZipFile(temp_zip_file, "w",
758 compression=zipfile.ZIP_DEFLATED)
759 else:
760 output_zip = zipfile.ZipFile(args[1], "w",
761 compression=zipfile.ZIP_DEFLATED)
762
763 if OPTIONS.incremental_source is None:
Doug Zongkerc77a9ad2010-09-16 11:28:43 -0700764 WriteFullOTAPackage(input_zip, output_zip)
Doug Zongkereef39442009-04-02 12:14:19 -0700765 else:
766 print "unzipping source target-files..."
767 OPTIONS.source_tmp = common.UnzipTemp(OPTIONS.incremental_source)
768 source_zip = zipfile.ZipFile(OPTIONS.incremental_source, "r")
Doug Zongker37974732010-09-16 17:44:38 -0700769 OPTIONS.target_info_dict = OPTIONS.info_dict
770 OPTIONS.source_info_dict = common.LoadInfoDict(source_zip)
771 if OPTIONS.verbose:
772 print "--- source info ---"
773 common.DumpInfoDict(OPTIONS.source_info_dict)
Doug Zongkerc77a9ad2010-09-16 11:28:43 -0700774 WriteIncrementalOTAPackage(input_zip, source_zip, output_zip)
Doug Zongkereef39442009-04-02 12:14:19 -0700775
776 output_zip.close()
777 if OPTIONS.package_key:
778 SignOutput(temp_zip_file.name, args[1])
779 temp_zip_file.close()
780
781 common.Cleanup()
782
783 print "done."
784
785
786if __name__ == '__main__':
787 try:
Ying Wangf9bbfb52010-12-13 16:25:36 -0800788 common.CloseInheritedPipes()
Doug Zongkereef39442009-04-02 12:14:19 -0700789 main(sys.argv[1:])
790 except common.ExternalError, e:
791 print
792 print " ERROR: %s" % (e,)
793 print
794 sys.exit(1)