blob: 596a47e377c4fbf728fe069f32da306890a1a69d [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 image zipfile suitable for
19use with 'fastboot update'.
20
21Usage: img_from_target_files [flags] input_target_files output_image_zip
22
23 -b (--board_config) <file>
Doug Zongkerfdd8e692009-08-03 17:27:48 -070024 Deprecated.
Doug Zongkereef39442009-04-02 12:14:19 -070025
Doug Zongker55d93282011-01-25 17:03:34 -080026 -z (--bootable_zip)
27 Include only the bootable images (eg 'boot' and 'recovery') in
28 the output.
29
Doug Zongkereef39442009-04-02 12:14:19 -070030"""
31
32import sys
33
Doug Zongkercf6d5a92014-02-18 10:57:07 -080034if sys.hexversion < 0x02070000:
35 print >> sys.stderr, "Python 2.7 or newer is required."
Doug Zongkereef39442009-04-02 12:14:19 -070036 sys.exit(1)
37
Mike Rittere44fade2009-09-15 11:18:31 -070038import errno
Doug Zongkereef39442009-04-02 12:14:19 -070039import os
40import re
41import shutil
42import subprocess
43import tempfile
44import zipfile
45
46# missing in Python 2.4 and before
47if not hasattr(os, "SEEK_SET"):
48 os.SEEK_SET = 0
49
Ying Wangbd93d422011-10-28 17:02:30 -070050import build_image
Doug Zongkereef39442009-04-02 12:14:19 -070051import common
52
53OPTIONS = common.OPTIONS
54
Ying Wanga0febe52013-03-20 11:02:05 -070055
Doug Zongker01ce19c2014-02-04 13:48:15 -080056def AddSystem(output_zip, sparse=True):
Ying Wanga0febe52013-03-20 11:02:05 -070057 """Turn the contents of SYSTEM into a system image and store it in
58 output_zip."""
Geremy Condra36bd3652014-02-06 19:45:10 -080059 data = BuildSystem(OPTIONS.input_tmp, OPTIONS.info_dict, sparse=sparse)
60 common.ZipWriteStr(output_zip, "system.img", data)
Ying Wanga0febe52013-03-20 11:02:05 -070061
Geremy Condra36bd3652014-02-06 19:45:10 -080062
Doug Zongker5fad2032014-02-24 08:13:45 -080063def BuildSystem(input_dir, info_dict, sparse=True, map_file=None):
Ying Wanga0febe52013-03-20 11:02:05 -070064 print "creating system.img..."
65
66 img = tempfile.NamedTemporaryFile()
67
68 # The name of the directory it is making an image out of matters to
69 # mkyaffs2image. It wants "system" but we have a directory named
70 # "SYSTEM", so create a symlink.
71 try:
Geremy Condra36bd3652014-02-06 19:45:10 -080072 os.symlink(os.path.join(input_dir, "SYSTEM"),
73 os.path.join(input_dir, "system"))
Ying Wanga0febe52013-03-20 11:02:05 -070074 except OSError, e:
75 # bogus error on my mac version?
76 # File "./build/tools/releasetools/img_from_target_files", line 86, in AddSystem
77 # os.path.join(OPTIONS.input_tmp, "system"))
78 # OSError: [Errno 17] File exists
79 if (e.errno == errno.EEXIST):
80 pass
81
Geremy Condra36bd3652014-02-06 19:45:10 -080082 image_props = build_image.ImagePropFromGlobalDict(info_dict, "system")
83 fstab = info_dict["fstab"]
Ying Wanga0febe52013-03-20 11:02:05 -070084 if fstab:
85 image_props["fs_type" ] = fstab["/system"].fs_type
Geremy Condra36bd3652014-02-06 19:45:10 -080086 succ = build_image.BuildImage(os.path.join(input_dir, "system"),
Ying Wanga0febe52013-03-20 11:02:05 -070087 image_props, img.name)
88 assert succ, "build system.img image failed"
89
Doug Zongker5fad2032014-02-24 08:13:45 -080090 mapdata = None
91
Doug Zongker01ce19c2014-02-04 13:48:15 -080092 if sparse:
93 img.seek(os.SEEK_SET, 0)
94 data = img.read()
95 img.close()
96 else:
97 success, name = build_image.UnsparseImage(img.name, replace=False)
98 if not success:
99 assert False, "unsparsing system.img failed"
Doug Zongker5fad2032014-02-24 08:13:45 -0800100
101 if map_file:
102 mmap = tempfile.NamedTemporaryFile()
103 mimg = tempfile.NamedTemporaryFile(delete=False)
104 success = build_image.MappedUnsparseImage(
105 img.name, name, mmap.name, mimg.name)
106 if not success:
107 assert False, "creating sparse map failed"
108 os.unlink(name)
109 name = mimg.name
110
111 with open(mmap.name) as f:
112 mapdata = f.read()
113
Doug Zongker01ce19c2014-02-04 13:48:15 -0800114 try:
115 with open(name) as f:
116 data = f.read()
117 finally:
118 os.unlink(name)
Ying Wanga0febe52013-03-20 11:02:05 -0700119
Doug Zongker5fad2032014-02-24 08:13:45 -0800120 if mapdata is None:
121 return data
122 else:
123 return mapdata, data
Ying Wanga0febe52013-03-20 11:02:05 -0700124
125
126def AddVendor(output_zip):
127 """Turn the contents of VENDOR into vendor.img and store it in
128 output_zip."""
129
130 image_props = build_image.ImagePropFromGlobalDict(OPTIONS.info_dict,
131 "vendor")
132 # The build system has to explicitly request for vendor.img.
133 if "fs_type" not in image_props:
134 return
135
136 print "creating vendor.img..."
137
138 img = tempfile.NamedTemporaryFile()
139
140 # The name of the directory it is making an image out of matters to
141 # mkyaffs2image. It wants "vendor" but we have a directory named
142 # "VENDOR", so create a symlink or an empty directory if VENDOR does not
143 # exist.
144 if not os.path.exists(os.path.join(OPTIONS.input_tmp, "vendor")):
145 if os.path.exists(os.path.join(OPTIONS.input_tmp, "VENDOR")):
146 os.symlink(os.path.join(OPTIONS.input_tmp, "VENDOR"),
147 os.path.join(OPTIONS.input_tmp, "vendor"))
148 else:
149 os.mkdir(os.path.join(OPTIONS.input_tmp, "vendor"))
150
151 img = tempfile.NamedTemporaryFile()
152
153 fstab = OPTIONS.info_dict["fstab"]
154 if fstab:
155 image_props["fs_type" ] = fstab["/vendor"].fs_type
156 succ = build_image.BuildImage(os.path.join(OPTIONS.input_tmp, "vendor"),
157 image_props, img.name)
158 assert succ, "build vendor.img image failed"
159
160 common.CheckSize(img.name, "vendor.img", OPTIONS.info_dict)
161 output_zip.write(img.name, "vendor.img")
162 img.close()
163
164
Doug Zongkereef39442009-04-02 12:14:19 -0700165def AddUserdata(output_zip):
166 """Create an empty userdata image and store it in output_zip."""
167
Ying Wang4e3f44f2012-11-19 10:26:00 -0800168 image_props = build_image.ImagePropFromGlobalDict(OPTIONS.info_dict,
169 "data")
Ying Wangd7321d32013-03-15 10:32:29 -0700170 # If no userdata_size is provided for extfs, skip userdata.img.
171 if (image_props.get("fs_type", "").startswith("ext") and
172 not image_props.get("partition_size")):
Ying Wang4e3f44f2012-11-19 10:26:00 -0800173 return
174
Doug Zongkereef39442009-04-02 12:14:19 -0700175 print "creating userdata.img..."
176
177 # The name of the directory it is making an image out of matters to
178 # mkyaffs2image. So we create a temp dir, and within it we create an
179 # empty dir named "data", and build the image from that.
180 temp_dir = tempfile.mkdtemp()
181 user_dir = os.path.join(temp_dir, "data")
182 os.mkdir(user_dir)
183 img = tempfile.NamedTemporaryFile()
184
Doug Zongker33a4b082010-09-21 16:12:55 -0700185 fstab = OPTIONS.info_dict["fstab"]
Ying Wangbd93d422011-10-28 17:02:30 -0700186 if fstab:
187 image_props["fs_type" ] = fstab["/data"].fs_type
188 succ = build_image.BuildImage(user_dir, image_props, img.name)
189 assert succ, "build userdata.img image failed"
Doug Zongkereef39442009-04-02 12:14:19 -0700190
Doug Zongker37974732010-09-16 17:44:38 -0700191 common.CheckSize(img.name, "userdata.img", OPTIONS.info_dict)
Doug Zongkereef39442009-04-02 12:14:19 -0700192 output_zip.write(img.name, "userdata.img")
193 img.close()
194 os.rmdir(user_dir)
195 os.rmdir(temp_dir)
196
197
Ying Wang9f8e8db2011-11-04 11:37:01 -0700198def AddCache(output_zip):
199 """Create an empty cache image and store it in output_zip."""
200
201 image_props = build_image.ImagePropFromGlobalDict(OPTIONS.info_dict,
202 "cache")
203 # The build system has to explicitly request for cache.img.
204 if "fs_type" not in image_props:
205 return
206
207 print "creating cache.img..."
208
209 # The name of the directory it is making an image out of matters to
210 # mkyaffs2image. So we create a temp dir, and within it we create an
211 # empty dir named "cache", and build the image from that.
212 temp_dir = tempfile.mkdtemp()
213 user_dir = os.path.join(temp_dir, "cache")
214 os.mkdir(user_dir)
215 img = tempfile.NamedTemporaryFile()
216
217 fstab = OPTIONS.info_dict["fstab"]
218 if fstab:
219 image_props["fs_type" ] = fstab["/cache"].fs_type
220 succ = build_image.BuildImage(user_dir, image_props, img.name)
221 assert succ, "build cache.img image failed"
222
223 common.CheckSize(img.name, "cache.img", OPTIONS.info_dict)
224 output_zip.write(img.name, "cache.img")
225 img.close()
226 os.rmdir(user_dir)
227 os.rmdir(temp_dir)
228
229
Doug Zongkereef39442009-04-02 12:14:19 -0700230def CopyInfo(output_zip):
231 """Copy the android-info.txt file from the input to the output."""
232 output_zip.write(os.path.join(OPTIONS.input_tmp, "OTA", "android-info.txt"),
233 "android-info.txt")
234
235
236def main(argv):
Doug Zongker55d93282011-01-25 17:03:34 -0800237 bootable_only = [False]
Doug Zongkereef39442009-04-02 12:14:19 -0700238
239 def option_handler(o, a):
240 if o in ("-b", "--board_config"):
Doug Zongkerfdd8e692009-08-03 17:27:48 -0700241 pass # deprecated
Doug Zongker55d93282011-01-25 17:03:34 -0800242 if o in ("-z", "--bootable_zip"):
243 bootable_only[0] = True
Doug Zongkereef39442009-04-02 12:14:19 -0700244 else:
245 return False
Doug Zongkerfdd8e692009-08-03 17:27:48 -0700246 return True
Doug Zongkereef39442009-04-02 12:14:19 -0700247
248 args = common.ParseOptions(argv, __doc__,
Doug Zongker55d93282011-01-25 17:03:34 -0800249 extra_opts="b:z",
250 extra_long_opts=["board_config=",
251 "bootable_zip"],
Doug Zongkereef39442009-04-02 12:14:19 -0700252 extra_option_handler=option_handler)
253
Doug Zongker55d93282011-01-25 17:03:34 -0800254 bootable_only = bootable_only[0]
255
Doug Zongkereef39442009-04-02 12:14:19 -0700256 if len(args) != 2:
257 common.Usage(__doc__)
258 sys.exit(1)
259
Doug Zongker55d93282011-01-25 17:03:34 -0800260 OPTIONS.input_tmp, input_zip = common.UnzipTemp(args[0])
Doug Zongker37974732010-09-16 17:44:38 -0700261 OPTIONS.info_dict = common.LoadInfoDict(input_zip)
Ying Wangd421f572010-08-25 20:39:41 -0700262
Kenny Roote2e9f612013-05-29 12:59:35 -0700263 # If this image was originally labelled with SELinux contexts, make sure we
264 # also apply the labels in our new image. During building, the "file_contexts"
265 # is in the out/ directory tree, but for repacking from target-files.zip it's
266 # in the root directory of the ramdisk.
267 if "selinux_fc" in OPTIONS.info_dict:
268 OPTIONS.info_dict["selinux_fc"] = os.path.join(OPTIONS.input_tmp, "BOOT", "RAMDISK",
269 "file_contexts")
270
Doug Zongkereef39442009-04-02 12:14:19 -0700271 output_zip = zipfile.ZipFile(args[1], "w", compression=zipfile.ZIP_DEFLATED)
272
Doug Zongker55d93282011-01-25 17:03:34 -0800273 common.GetBootableImage(
274 "boot.img", "boot.img", OPTIONS.input_tmp, "BOOT").AddToZip(output_zip)
275 common.GetBootableImage(
276 "recovery.img", "recovery.img", OPTIONS.input_tmp,
277 "RECOVERY").AddToZip(output_zip)
278
279 if not bootable_only:
280 AddSystem(output_zip)
Ying Wanga0febe52013-03-20 11:02:05 -0700281 AddVendor(output_zip)
Doug Zongker55d93282011-01-25 17:03:34 -0800282 AddUserdata(output_zip)
Ying Wang9f8e8db2011-11-04 11:37:01 -0700283 AddCache(output_zip)
Doug Zongker55d93282011-01-25 17:03:34 -0800284 CopyInfo(output_zip)
Doug Zongkereef39442009-04-02 12:14:19 -0700285
286 print "cleaning up..."
287 output_zip.close()
288 shutil.rmtree(OPTIONS.input_tmp)
289
290 print "done."
291
292
293if __name__ == '__main__':
294 try:
Ying Wang7e6d4e42010-12-13 16:25:36 -0800295 common.CloseInheritedPipes()
Doug Zongkereef39442009-04-02 12:14:19 -0700296 main(sys.argv[1:])
297 except common.ExternalError, e:
298 print
299 print " ERROR: %s" % (e,)
300 print
301 sys.exit(1)