1# Copyright (c) 2011 The Chromium OS Authors. All rights reserved. 2# Use of this source code is governed by a BSD-style license that can be 3# found in the LICENSE file. 4 5""" 6Factory install VM tests. 7 8This test supports the flags documented in FactoryInstallTest, plus: 9 10 debug_vnc: whether to run VNC on the KVM (for debugging only) 11 debug_save_hda: if specified, path to save the hda.bin to after 12 running the factory install shim (for debugging only) 13 debug_reuse_hda: if specified, path to an existing hda.bin image 14 to reuse (for debugging only) 15""" 16 17 18import os, re 19 20from autotest_lib.client.bin import utils as client_utils 21from autotest_lib.server.cros.factory_install_test import FactoryInstallTest 22from autotest_lib.server.hosts import ssh_host 23 24 25# How long to wait after killing KVMs. 26_KILL_KVM_WAIT_SEC = 5 27 28# The size of the "SSD" in the KVM. 29_HDA_SIZE_MB = 8192 30 31# The name of the image used for hda. This should be unique to this test, 32# since we will kill all stray KVM processes using this disk image. 33_HDA_FILENAME = "factory_InstallVM_hda.bin" 34 35 36class factory_InstallVM(FactoryInstallTest): 37 """ 38 Factory install VM tests. 39 40 See file-level docstring for more information. 41 """ 42 43 def _get_kvm_command(self, kvm_args=[]): 44 """ 45 Returns the command to run KVM. 46 47 @param kvm_args: A list of extra args to pass to KVM. 48 """ 49 kvm_base_args = [ 50 "kvm", 51 "-m", "2048", 52 "-net", "nic,model=virtio", 53 "-net", "user,hostfwd=tcp::%d-:22" % self.ssh_tunnel_port, 54 "-vga", "vmware", # Because -vga std is slow 55 ] 56 57 if self.vnc: 58 # Without nographic, we need to explicitly add "-serial stdio" 59 # (or output will go to vc). Use 127.0.0.1 to ensure that kvm 60 # listens with IPv4. 61 kvm_base_args.extend(["-serial", "stdio", "-vnc", "127.0.0.1:1"]) 62 else: 63 kvm_base_args.append("-nographic") 64 65 return " ".join(kvm_base_args + kvm_args) 66 67 def _kill_kvm(self): 68 """ 69 Kills the KVM on the client machine. 70 71 This will kill any KVM whose command line contains _HDA_FILENAME 72 (which is specific to this test). 73 """ 74 def try_kill_kvm(): 75 pattern = "^kvm.*%s" % _HDA_FILENAME, 76 if (self.client.run("pgrep -f '%s'" % pattern, ignore_status=True) 77 .exit_status == 1): 78 return True 79 self.client.run("pkill -f '%s'" % (pattern)) 80 return False 81 82 client_utils.poll_for_condition( 83 try_kill_kvm, timeout=_KILL_KVM_WAIT_SEC, desc="Kill KVM") 84 85 def get_hwid_cfg(self): 86 """ 87 Overridden from superclass. 88 """ 89 return "vm" 90 91 def get_dut_client(self): 92 """ 93 Overridden from superclass. 94 """ 95 return ssh_host.SSHHost("localhost", port=self.ssh_tunnel_port) 96 97 def run_factory_install(self, local_hdb): 98 """ 99 Overridden from superclass. 100 """ 101 self.hda = os.path.join(self.client.get_tmp_dir(), _HDA_FILENAME) 102 103 if self.reuse_hda is not None: 104 self.client.run("cp %s %s" % (self.reuse_hda, self.hda)) 105 else: 106 # Mount partition 12 the image and modify it to enable serial 107 # logging. 108 mount = self._mount_partition(local_hdb, 12) 109 self._modify_file( 110 os.path.join(mount, "syslinux/usb.A.cfg"), 111 lambda contents: re.sub(r"console=\w+", "console=ttyS0", 112 contents)) 113 self._umount_partition(mount) 114 115 # On the client, create a nice big sparse file for hda 116 # (a.k.a. the SSD). 117 self.client.run("truncate -s %dM %s" % (_HDA_SIZE_MB, self.hda)) 118 hdb = os.path.join(self.client.get_tmp_dir(), "hdb.bin") 119 self.client.send_file(local_hdb, hdb) 120 121 # Fire up the KVM and wait for the factory install to complete. 122 self._kill_kvm() # Just in case 123 self.client.run_grep( 124 self._get_kvm_command( 125 ["-drive", "file=%s,boot=off" % self.hda, 126 "-drive", "file=%s,boot=on" % hdb, 127 "-no-reboot"]), 128 timeout=FactoryInstallTest.FACTORY_INSTALL_TIMEOUT_SEC, 129 stdout_ok_regexp="Factory Installer Complete") 130 self._kill_kvm() 131 132 if self.save_hda is not None: 133 self.client.run("cp %s %s" % (self.hda, self.save_hda)) 134 135 # Run KVM again (the factory tests should now come up). 136 kvm = self.client.run(self._get_kvm_command([ 137 "-hda", self.hda, 138 "-daemonize", 139 "-no-reboot"])) 140 141 def reboot_for_wipe(self): 142 """ 143 Overridden from superclass. 144 """ 145 # Use halt instead of reboot; reboot doesn't work consistently in KVM. 146 self.get_dut_client().halt() 147 self._kill_kvm() 148 149 # Start KVM again. The ChromeOS test image should now come up. 150 kvm = self.client.run( 151 self._get_kvm_command(["-hda", self.hda, "-daemonize"])) 152 153 def run_once(self, host, debug_reuse_hda=None, debug_save_hda=None, 154 debug_vnc=False, **args): 155 self.client = host 156 self.reuse_hda = debug_reuse_hda 157 self.save_hda = debug_save_hda 158 self.vnc = self.parse_boolean(debug_vnc) 159 160 self.cleanup_tasks.append(self._kill_kvm) 161 162 super(factory_InstallVM, self).run_once(**args) 163