img_from_target_files.py revision 2de68bbbf49473c09d434a75e669345392e03240
1#!/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  -z  (--bootable_zip)
24      Include only the bootable images (eg 'boot' and 'recovery') in
25      the output.
26
27"""
28
29import sys
30
31if sys.hexversion < 0x02070000:
32  print >> sys.stderr, "Python 2.7 or newer is required."
33  sys.exit(1)
34
35import errno
36import os
37import re
38import shutil
39import subprocess
40import tempfile
41import zipfile
42
43# missing in Python 2.4 and before
44if not hasattr(os, "SEEK_SET"):
45  os.SEEK_SET = 0
46
47import common
48
49OPTIONS = common.OPTIONS
50
51
52def CopyInfo(output_zip):
53  """Copy the android-info.txt file from the input to the output."""
54  output_zip.write(os.path.join(OPTIONS.input_tmp, "OTA", "android-info.txt"),
55                   "android-info.txt")
56
57
58def main(argv):
59  bootable_only = [False]
60
61  def option_handler(o, a):
62    if o in ("-z", "--bootable_zip"):
63      bootable_only[0] = True
64    else:
65      return False
66    return True
67
68  args = common.ParseOptions(argv, __doc__,
69                             extra_opts="z",
70                             extra_long_opts=["bootable_zip"],
71                             extra_option_handler=option_handler)
72
73  bootable_only = bootable_only[0]
74
75  if len(args) != 2:
76    common.Usage(__doc__)
77    sys.exit(1)
78
79  OPTIONS.input_tmp, input_zip = common.UnzipTemp(args[0])
80  output_zip = zipfile.ZipFile(args[1], "w", compression=zipfile.ZIP_DEFLATED)
81  CopyInfo(output_zip)
82
83  try:
84    done = False
85    images_path = os.path.join(OPTIONS.input_tmp, "IMAGES")
86    if os.path.exists(images_path):
87      # If this is a new target-files, it already contains the images,
88      # and all we have to do is copy them to the output zip.
89      images = os.listdir(images_path)
90      if images:
91        for i in images:
92          if bootable_only and i not in ("boot.img", "recovery.img"): continue
93          if not i.endswith(".img"): continue
94          with open(os.path.join(images_path, i), "r") as f:
95            common.ZipWriteStr(output_zip, i, f.read())
96        done = True
97
98    if not done:
99      # We have an old target-files that doesn't already contain the
100      # images, so build them.
101      import add_img_to_target_files
102
103      OPTIONS.info_dict = common.LoadInfoDict(input_zip)
104
105      # If this image was originally labelled with SELinux contexts,
106      # make sure we also apply the labels in our new image. During
107      # building, the "file_contexts" is in the out/ directory tree,
108      # but for repacking from target-files.zip it's in the root
109      # directory of the ramdisk.
110      if "selinux_fc" in OPTIONS.info_dict:
111        OPTIONS.info_dict["selinux_fc"] = os.path.join(
112            OPTIONS.input_tmp, "BOOT", "RAMDISK", "file_contexts")
113
114      boot_image = common.GetBootableImage(
115          "boot.img", "boot.img", OPTIONS.input_tmp, "BOOT")
116      if boot_image:
117          boot_image.AddToZip(output_zip)
118      recovery_image = common.GetBootableImage(
119          "recovery.img", "recovery.img", OPTIONS.input_tmp, "RECOVERY")
120      if recovery_image:
121        recovery_image.AddToZip(output_zip)
122
123      def banner(s):
124        print "\n\n++++ " + s + " ++++\n\n"
125
126      if not bootable_only:
127        banner("AddSystem")
128        add_img_to_target_files.AddSystem(output_zip, prefix="")
129        try:
130          input_zip.getinfo("VENDOR/")
131          banner("AddVendor")
132          add_img_to_target_files.AddVendor(output_zip, prefix="")
133        except KeyError:
134          pass   # no vendor partition for this device
135        banner("AddUserdata")
136        add_img_to_target_files.AddUserdata(output_zip, prefix="")
137        banner("AddCache")
138        add_img_to_target_files.AddCache(output_zip, prefix="")
139
140  finally:
141    print "cleaning up..."
142    output_zip.close()
143    shutil.rmtree(OPTIONS.input_tmp)
144
145  print "done."
146
147
148if __name__ == '__main__':
149  try:
150    common.CloseInheritedPipes()
151    main(sys.argv[1:])
152  except common.ExternalError, e:
153    print
154    print "   ERROR: %s" % (e,)
155    print
156    sys.exit(1)
157