blob: 2c3b8302e6e4a6b738cc8a64c87f03b3163dad51 [file] [log] [blame]
asharif252df0f2013-02-15 04:46:28 +00001#!/usr/bin/python2.6
2#
3# Copyright 2010 Google Inc. All Rights Reserved.
4
5"""Script to enter the ChromeOS chroot with mounted sources.
6
7This script enters the chroot with mounted sources.
8"""
9
10__author__ = "asharif@google.com (Ahmad Sharif)"
11
12import getpass
13import optparse
14import os
asharif556f4ff2013-02-15 04:50:35 +000015import pwd
asharifc0f71932013-02-15 04:56:18 +000016import stat
asharif252df0f2013-02-15 04:46:28 +000017import sys
raymes01959ae2013-02-15 04:50:07 +000018from utils import command_executer
19from utils import logger
asharif252df0f2013-02-15 04:46:28 +000020from utils import utils
21
asharifda9ac652013-02-15 04:50:09 +000022class MountPoint:
23 def __init__(self, external_dir, mount_dir, owner, options=None):
asharif6c619132013-02-15 21:55:28 +000024 self.external_dir = os.path.realpath(external_dir)
25 self.mount_dir = os.path.realpath(mount_dir)
asharifda9ac652013-02-15 04:50:09 +000026 self.owner = owner
27 self.options = options
28
29
asharif556f4ff2013-02-15 04:50:35 +000030 def CreateAndOwnDir(self, dir_name):
31 retval = 0
32 if not os.path.exists(dir_name):
33 command = "mkdir -p " + dir_name
34 command += " || sudo mkdir -p " + dir_name
asharif8a873872013-02-15 04:56:52 +000035 retval = command_executer.GetCommandExecuter().RunCommand(command)
asharif556f4ff2013-02-15 04:50:35 +000036 if retval != 0:
37 return retval
38 pw = pwd.getpwnam(self.owner)
39 if os.stat(dir_name).st_uid != pw.pw_uid:
40 command = "sudo chown -f " + self.owner + " " + dir_name
asharif8a873872013-02-15 04:56:52 +000041 retval = command_executer.GetCommandExecuter().RunCommand(command)
asharifda9ac652013-02-15 04:50:09 +000042 return retval
43
44
45 def DoMount(self):
asharif6c619132013-02-15 21:55:28 +000046 ce = command_executer.GetCommandExecuter()
47 mount_signature = "%s on %s" % (self.external_dir, self.mount_dir)
48 command = "mount"
49 retval, out, err = ce.RunCommand(command, return_output=True)
50 if mount_signature not in out:
51 retval = self.CreateAndOwnDir(self.mount_dir)
52 logger.GetLogger().LogFatalIf(retval, "Cannot create mount_dir!")
53 retval = self.CreateAndOwnDir(self.external_dir)
54 logger.GetLogger().LogFatalIf(retval, "Cannot create external_dir!")
55 retval = self.MountDir()
56 logger.GetLogger().LogFatalIf(retval, "Cannot mount!")
57 return retval
58 else:
59 return 0
asharifda9ac652013-02-15 04:50:09 +000060
61
62 def MountDir(self):
63 command = "sudo mount --bind " + self.external_dir + " " + self.mount_dir
64 if self.options == "ro":
65 command += " && sudo mount --bind -oremount,ro " + self.mount_dir
asharif8a873872013-02-15 04:56:52 +000066 retval = command_executer.GetCommandExecuter().RunCommand(command)
asharifda9ac652013-02-15 04:50:09 +000067 return retval
68
69
70 def __str__(self):
71 ret = ""
72 ret += self.external_dir + "\n"
73 ret += self.mount_dir + "\n"
74 if self.owner:
75 ret += self.owner + "\n"
76 if self.options:
77 ret += self.options + "\n"
78 return ret
79
80
81def Main(argv, return_output=False):
asharif252df0f2013-02-15 04:46:28 +000082 """The main function."""
83 parser = optparse.OptionParser()
84 parser.add_option("-c", "--chromeos_root", dest="chromeos_root",
asharif0b2f0402013-02-15 04:50:25 +000085 default="../..",
asharif252df0f2013-02-15 04:46:28 +000086 help="ChromeOS root checkout directory.")
87 parser.add_option("-t", "--toolchain_root", dest="toolchain_root",
88 help="Toolchain root directory.")
asharif541b6392013-02-15 04:50:38 +000089 parser.add_option("-o", "--output", dest="output",
90 help="Toolchain output directory")
asharif52284c62013-02-15 19:59:08 +000091 parser.add_option("--sudo", dest="sudo",
92 action="store_true",
93 default=False,
94 help="Run the command with sudo.")
asharif8697d4e2013-02-15 09:18:09 +000095 parser.add_option("-r", "--third_party", dest="third_party",
96 help="The third_party directory to mount.")
asharif541b6392013-02-15 04:50:38 +000097 parser.add_option("-m", "--other_mounts", dest="other_mounts",
raymesa7d219c2013-02-15 04:56:23 +000098 help="Other mount points in the form: " +
asharifda9ac652013-02-15 04:50:09 +000099 "dir:mounted_dir:options")
asharif1755b432013-02-15 04:55:29 +0000100 parser.add_option("-s", "--mount-scripts-only",
101 dest="mount_scripts_only",
102 action="store_true",
103 default=False,
104 help="Mount only the scripts dir, and not the sources.")
asharif252df0f2013-02-15 04:46:28 +0000105
raymes01959ae2013-02-15 04:50:07 +0000106 passthrough_argv = []
asharif1755b432013-02-15 04:55:29 +0000107 (options, passthrough_argv) = parser.parse_args(argv)
asharif252df0f2013-02-15 04:46:28 +0000108
asharif0b2f0402013-02-15 04:50:25 +0000109 chromeos_root = options.chromeos_root
asharif17621302013-02-15 04:46:35 +0000110
111 chromeos_root = os.path.expanduser(chromeos_root)
asharifda9ac652013-02-15 04:50:09 +0000112 if options.toolchain_root:
113 options.toolchain_root = os.path.expanduser(options.toolchain_root)
asharif17621302013-02-15 04:46:35 +0000114
asharif252df0f2013-02-15 04:46:28 +0000115 chromeos_root = os.path.abspath(chromeos_root)
116
asharif8697d4e2013-02-15 09:18:09 +0000117 tc_dirs = []
asharif642509c2013-02-15 09:19:32 +0000118 if options.toolchain_root is None or options.mount_scripts_only:
asharif8697d4e2013-02-15 09:18:09 +0000119 m = "toolchain_root not specified. Will not mount toolchain dirs."
120 logger.GetLogger().LogWarning(m)
121 else:
122 tc_dirs = [options.toolchain_root + "/google_vendor_src_branch/gcc",
123 options.toolchain_root + "/google_vendor_src_branch/binutils"]
asharif252df0f2013-02-15 04:46:28 +0000124
asharif8697d4e2013-02-15 09:18:09 +0000125 for tc_dir in tc_dirs:
126 if not os.path.exists(tc_dir):
127 logger.GetLogger().LogError("toolchain path " +
128 tc_dir + " does not exist!")
129 parser.print_help()
130 sys.exit(1)
asharif0b2f0402013-02-15 04:50:25 +0000131
132 if not os.path.exists(chromeos_root):
133 logger.GetLogger().LogError("chromeos_root " + options.chromeos_root +
134 " does not exist!")
135 parser.print_help()
136 sys.exit(1)
137
asharifc8e8e122013-02-15 21:19:59 +0000138 if not os.path.exists(chromeos_root + "/src/scripts/build_packages"):
raymesa7d219c2013-02-15 04:56:23 +0000139 logger.GetLogger().LogError(options.chromeos_root +
asharifc8e8e122013-02-15 21:19:59 +0000140 "/src/scripts/build_packages"
asharif0b2f0402013-02-15 04:50:25 +0000141 " not found!")
142 parser.print_help()
143 sys.exit(1)
144
asharifb225e792013-02-15 21:20:11 +0000145 version_dir = os.path.realpath(os.path.expanduser(os.path.dirname(__file__)))
asharif252df0f2013-02-15 04:46:28 +0000146
asharif252df0f2013-02-15 04:46:28 +0000147 mounted_tc_root = "/usr/local/toolchain_root"
148 full_mounted_tc_root = chromeos_root + "/chroot/" + mounted_tc_root
149 full_mounted_tc_root = os.path.abspath(full_mounted_tc_root)
raymesa7d219c2013-02-15 04:56:23 +0000150
asharifda9ac652013-02-15 04:50:09 +0000151 mount_points = []
asharif8697d4e2013-02-15 09:18:09 +0000152 for tc_dir in tc_dirs:
153 last_dir = utils.GetRoot(tc_dir)[1]
154 mount_point = MountPoint(tc_dir, full_mounted_tc_root + "/" + last_dir,
155 getpass.getuser(), "ro")
156 mount_points.append(mount_point)
asharif252df0f2013-02-15 04:46:28 +0000157
asharif8697d4e2013-02-15 09:18:09 +0000158 # Add the third_party mount point if it exists
159 if options.third_party:
160 third_party_dir = options.third_party
kbaclawski6999ada2013-02-15 19:57:09 +0000161 logger.GetLogger().LogFatalIf(not os.path.isdir(third_party_dir),
162 "--third_party option is not a valid dir.")
asharif8697d4e2013-02-15 09:18:09 +0000163 else:
164 third_party_dir = os.path.abspath("%s/../../../third_party" %
165 os.path.dirname(__file__))
166
167 if os.path.isdir(third_party_dir):
168 mount_point = MountPoint(third_party_dir,
169 ("%s/%s" %
170 (full_mounted_tc_root,
171 os.path.basename(third_party_dir))),
172 getpass.getuser())
173 mount_points.append(mount_point)
kbaclawski6999ada2013-02-15 19:57:09 +0000174
asharif541b6392013-02-15 04:50:38 +0000175 output = options.output
asharif8697d4e2013-02-15 09:18:09 +0000176 if output is None and options.toolchain_root:
asharif8697d4e2013-02-15 09:18:09 +0000177 # Mount the output directory at /usr/local/toolchain_root/output
asharif01410cc2013-02-15 09:19:31 +0000178 output = options.toolchain_root + "/output"
179
180 if output:
asharif8697d4e2013-02-15 09:18:09 +0000181 mount_points.append(MountPoint(output, full_mounted_tc_root + "/output",
182 getpass.getuser()))
183
184 # Mount the other mount points
raymesa7d219c2013-02-15 04:56:23 +0000185 mount_points += CreateMountPointsFromString(options.other_mounts,
asharifda9ac652013-02-15 04:50:09 +0000186 chromeos_root + "/chroot/")
187
asharif252df0f2013-02-15 04:46:28 +0000188 last_dir = utils.GetRoot(version_dir)[1]
asharif8697d4e2013-02-15 09:18:09 +0000189
190 # Mount the version dir (v14) at /usr/local/toolchain_root/v14
asharifda9ac652013-02-15 04:50:09 +0000191 mount_point = MountPoint(version_dir, full_mounted_tc_root + "/" + last_dir,
192 getpass.getuser())
193 mount_points.append(mount_point)
194
195 for mount_point in mount_points:
asharif556f4ff2013-02-15 04:50:35 +0000196 retval = mount_point.DoMount()
197 if retval != 0:
198 return retval
asharif252df0f2013-02-15 04:46:28 +0000199
200 # Finally, create the symlink to build-gcc.
asharifda9ac652013-02-15 04:50:09 +0000201 command = "sudo chown " + getpass.getuser() + " " + full_mounted_tc_root
asharif8a873872013-02-15 04:56:52 +0000202 retval = command_executer.GetCommandExecuter().RunCommand(command)
asharifda9ac652013-02-15 04:50:09 +0000203
asharif252df0f2013-02-15 04:46:28 +0000204 try:
asharif8a873872013-02-15 04:56:52 +0000205 CreateSymlink(last_dir + "/build-gcc", full_mounted_tc_root + "/build-gcc")
206 CreateSymlink(last_dir + "/build-binutils", full_mounted_tc_root + "/build-binutils")
asharif252df0f2013-02-15 04:46:28 +0000207 except Exception as e:
asharif0b2f0402013-02-15 04:50:25 +0000208 logger.GetLogger().LogError(str(e))
asharif252df0f2013-02-15 04:46:28 +0000209
asharif36666532013-02-15 21:08:14 +0000210 # Now call cros_sdk --enter with the rest of the arguments.
211 command = "cd %s/src/scripts && cros_sdk --enter" % chromeos_root
asharif252df0f2013-02-15 04:46:28 +0000212
asharif0269d462013-02-15 04:46:31 +0000213 if len(passthrough_argv) > 1:
asharif51516da2013-02-15 04:56:12 +0000214 inner_command = " ".join(passthrough_argv[1:])
215 inner_command = inner_command.strip()
216 if inner_command.startswith("-- "):
217 inner_command = inner_command[3:]
asharifc0f71932013-02-15 04:56:18 +0000218 command_file = "tc_enter_chroot.cmd"
219 command_file_path = chromeos_root + "/src/scripts/" + command_file
asharif8a873872013-02-15 04:56:52 +0000220 retval = command_executer.GetCommandExecuter().RunCommand("sudo rm -f " + command_file_path)
asharifc0f71932013-02-15 04:56:18 +0000221 if retval != 0:
222 return retval
223 f = open(command_file_path, "w")
224 f.write(inner_command)
225 f.close()
raymesa7d219c2013-02-15 04:56:23 +0000226 logger.GetLogger().LogCmd(inner_command)
asharif8a873872013-02-15 04:56:52 +0000227 retval = command_executer.GetCommandExecuter().RunCommand("chmod +x " + command_file_path)
asharifc0f71932013-02-15 04:56:18 +0000228 if retval != 0:
229 return retval
asharif52284c62013-02-15 19:59:08 +0000230
231 if options.sudo:
232 command += " sudo ./" + command_file
233 else:
234 command += " ./" + command_file
asharif8a873872013-02-15 04:56:52 +0000235 retval = command_executer.GetCommandExecuter().RunCommand(command, return_output)
asharif252df0f2013-02-15 04:46:28 +0000236 return retval
237 else:
asharif36666532013-02-15 21:08:14 +0000238 os.chdir("%s/src/scripts" % chromeos_root)
239 ce = command_executer.GetCommandExecuter()
240 [ret, out, err] = ce.RunCommand("which cros_sdk", return_output=True)
241 cros_sdk_binary = out.split()[0]
242 return os.execv(cros_sdk_binary, ["", "--enter"])
asharif252df0f2013-02-15 04:46:28 +0000243
244
asharifda9ac652013-02-15 04:50:09 +0000245def CreateMountPointsFromString(mount_strings, chroot_dir):
246 # String has options in the form dir:mount:options
247 mount_points = []
248 if not mount_strings:
249 return mount_points
250 mount_list = mount_strings.split()
251 for mount_string in mount_list:
252 mount_values = mount_string.split(":")
253 external_dir = mount_values[0]
254 mount_dir = mount_values[1]
raymesa7d219c2013-02-15 04:56:23 +0000255 if len(mount_values) > 2:
asharifda9ac652013-02-15 04:50:09 +0000256 options = mount_values[2]
257 else:
258 options = None
raymesa7d219c2013-02-15 04:56:23 +0000259 mount_point = MountPoint(external_dir, chroot_dir + "/" + mount_dir,
asharifda9ac652013-02-15 04:50:09 +0000260 getpass.getuser(), options)
261 mount_points.append(mount_point)
262 return mount_points
asharif252df0f2013-02-15 04:46:28 +0000263
264
asharif8a873872013-02-15 04:56:52 +0000265def CreateSymlink(target, link_name):
kbaclawski6999ada2013-02-15 19:57:09 +0000266 logger.GetLogger().LogFatalIf(target.startswith("/"),
267 "Can't create symlink to absolute path!")
asharif8a873872013-02-15 04:56:52 +0000268 real_from_file = utils.GetRoot(link_name)[0] + "/" + target
269 if os.path.realpath(real_from_file) != os.path.realpath(link_name):
270 if os.path.exists(link_name):
271 command = "rm -rf " + link_name
272 command_executer.GetCommandExecuter().RunCommand(command)
273 os.symlink(target, link_name)
274
275
asharif252df0f2013-02-15 04:46:28 +0000276if __name__ == "__main__":
asharif2198c512013-02-15 09:21:35 +0000277 retval = Main(sys.argv)
278 sys.exit(retval)