1# 2# Copyright (C) 2013 Intel Corporation 3# 4# SPDX-License-Identifier: MIT 5# 6 7# This module provides a class for starting qemu images using runqemu. 8# It's used by testimage.bbclass. 9 10import subprocess 11import os 12import sys 13import time 14import signal 15import re 16import socket 17import select 18import errno 19import string 20import threading 21import codecs 22import logging 23import tempfile 24from oeqa.utils.dump import HostDumper 25from collections import defaultdict 26import importlib 27 28# Get Unicode non printable control chars 29control_range = list(range(0,32))+list(range(127,160)) 30control_chars = [chr(x) for x in control_range 31 if chr(x) not in string.printable] 32re_control_char = re.compile('[%s]' % re.escape("".join(control_chars))) 33 34class QemuRunner: 35 36 def __init__(self, machine, rootfs, display, tmpdir, deploy_dir_image, logfile, boottime, dump_dir, dump_host_cmds, 37 use_kvm, logger, use_slirp=False, serial_ports=2, boot_patterns = defaultdict(str), use_ovmf=False, workdir=None, tmpfsdir=None): 38 39 # Popen object for runqemu 40 self.runqemu = None 41 self.runqemu_exited = False 42 # pid of the qemu process that runqemu will start 43 self.qemupid = None 44 # target ip - from the command line or runqemu output 45 self.ip = None 46 # host ip - where qemu is running 47 self.server_ip = None 48 # target ip netmask 49 self.netmask = None 50 51 self.machine = machine 52 self.rootfs = rootfs 53 self.display = display 54 self.tmpdir = tmpdir 55 self.deploy_dir_image = deploy_dir_image 56 self.logfile = logfile 57 self.boottime = boottime 58 self.logged = False 59 self.thread = None 60 self.use_kvm = use_kvm 61 self.use_ovmf = use_ovmf 62 self.use_slirp = use_slirp 63 self.serial_ports = serial_ports 64 self.msg = '' 65 self.boot_patterns = boot_patterns 66 self.tmpfsdir = tmpfsdir 67 68 self.runqemutime = 300 69 if not workdir: 70 workdir = os.getcwd() 71 self.qemu_pidfile = workdir + '/pidfile_' + str(os.getpid()) 72 self.host_dumper = HostDumper(dump_host_cmds, dump_dir) 73 self.monitorpipe = None 74 75 self.logger = logger 76 # Whether we're expecting an exit and should show related errors 77 self.canexit = False 78 79 # Enable testing other OS's 80 # Set commands for target communication, and default to Linux ALWAYS 81 # Other OS's or baremetal applications need to provide their 82 # own implementation passing it through QemuRunner's constructor 83 # or by passing them through TESTIMAGE_BOOT_PATTERNS[flag] 84 # provided variables, where <flag> is one of the mentioned below. 85 accepted_patterns = ['search_reached_prompt', 'send_login_user', 'search_login_succeeded', 'search_cmd_finished'] 86 default_boot_patterns = defaultdict(str) 87 # Default to the usual paterns used to communicate with the target 88 default_boot_patterns['search_reached_prompt'] = ' login:' 89 default_boot_patterns['send_login_user'] = 'root\n' 90 default_boot_patterns['search_login_succeeded'] = r"root@[a-zA-Z0-9\-]+:~#" 91 default_boot_patterns['search_cmd_finished'] = r"[a-zA-Z0-9]+@[a-zA-Z0-9\-]+:~#" 92 93 # Only override patterns that were set e.g. login user TESTIMAGE_BOOT_PATTERNS[send_login_user] = "webserver\n" 94 for pattern in accepted_patterns: 95 if not self.boot_patterns[pattern]: 96 self.boot_patterns[pattern] = default_boot_patterns[pattern] 97 98 def create_socket(self): 99 try: 100 sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) 101 sock.setblocking(0) 102 sock.bind(("127.0.0.1",0)) 103 sock.listen(2) 104 port = sock.getsockname()[1] 105 self.logger.debug("Created listening socket for qemu serial console on: 127.0.0.1:%s" % port) 106 return (sock, port) 107 108 except socket.error: 109 sock.close() 110 raise 111 112 def decode_qemulog(self, todecode): 113 # Sanitize the data received from qemu as it may contain control characters 114 msg = todecode.decode("utf-8", errors='ignore') 115 msg = re_control_char.sub('', msg) 116 return msg 117 118 def log(self, msg): 119 if self.logfile: 120 msg = self.decode_qemulog(msg) 121 self.msg += msg 122 with codecs.open(self.logfile, "a", encoding="utf-8") as f: 123 f.write("%s" % msg) 124 125 def getOutput(self, o): 126 import fcntl 127 fl = fcntl.fcntl(o, fcntl.F_GETFL) 128 fcntl.fcntl(o, fcntl.F_SETFL, fl | os.O_NONBLOCK) 129 try: 130 return os.read(o.fileno(), 1000000).decode("utf-8") 131 except BlockingIOError: 132 return "" 133 134 135 def handleSIGCHLD(self, signum, frame): 136 if self.runqemu and self.runqemu.poll(): 137 if self.runqemu.returncode: 138 self.logger.error('runqemu exited with code %d' % self.runqemu.returncode) 139 self.logger.error('Output from runqemu:\n%s' % self.getOutput(self.runqemu.stdout)) 140 self.stop() 141 self._dump_host() 142 143 def start(self, qemuparams = None, get_ip = True, extra_bootparams = None, runqemuparams='', launch_cmd=None, discard_writes=True): 144 env = os.environ.copy() 145 if self.display: 146 env["DISPLAY"] = self.display 147 # Set this flag so that Qemu doesn't do any grabs as SDL grabs 148 # interact badly with screensavers. 149 env["QEMU_DONT_GRAB"] = "1" 150 if not os.path.exists(self.rootfs): 151 self.logger.error("Invalid rootfs %s" % self.rootfs) 152 return False 153 if not os.path.exists(self.tmpdir): 154 self.logger.error("Invalid TMPDIR path %s" % self.tmpdir) 155 return False 156 else: 157 env["OE_TMPDIR"] = self.tmpdir 158 if not os.path.exists(self.deploy_dir_image): 159 self.logger.error("Invalid DEPLOY_DIR_IMAGE path %s" % self.deploy_dir_image) 160 return False 161 else: 162 env["DEPLOY_DIR_IMAGE"] = self.deploy_dir_image 163 164 if self.tmpfsdir: 165 env["RUNQEMU_TMPFS_DIR"] = self.tmpfsdir 166 167 if not launch_cmd: 168 launch_cmd = 'runqemu %s' % ('snapshot' if discard_writes else '') 169 if self.use_kvm: 170 self.logger.debug('Using kvm for runqemu') 171 launch_cmd += ' kvm' 172 else: 173 self.logger.debug('Not using kvm for runqemu') 174 if not self.display: 175 launch_cmd += ' nographic' 176 if self.use_slirp: 177 launch_cmd += ' slirp' 178 if self.use_ovmf: 179 launch_cmd += ' ovmf' 180 launch_cmd += ' %s %s' % (runqemuparams, self.machine) 181 if self.rootfs.endswith('.vmdk'): 182 self.logger.debug('Bypassing VMDK rootfs for runqemu') 183 else: 184 launch_cmd += ' %s' % (self.rootfs) 185 186 return self.launch(launch_cmd, qemuparams=qemuparams, get_ip=get_ip, extra_bootparams=extra_bootparams, env=env) 187 188 def launch(self, launch_cmd, get_ip = True, qemuparams = None, extra_bootparams = None, env = None): 189 # use logfile to determine the recipe-sysroot-native path and 190 # then add in the site-packages path components and add that 191 # to the python sys.path so qmp.py can be found. 192 python_path = os.path.dirname(os.path.dirname(self.logfile)) 193 python_path += "/recipe-sysroot-native/usr/lib/qemu-python" 194 sys.path.append(python_path) 195 importlib.invalidate_caches() 196 try: 197 qmp = importlib.import_module("qmp") 198 except Exception as e: 199 self.logger.error("qemurunner: qmp.py missing, please ensure it's installed (%s)" % str(e)) 200 return False 201 # Path relative to tmpdir used as cwd for qemu below to avoid unix socket path length issues 202 qmp_file = "." + next(tempfile._get_candidate_names()) 203 qmp_param = ' -S -qmp unix:./%s,server,wait' % (qmp_file) 204 qmp_port = self.tmpdir + "/" + qmp_file 205 # Create a second socket connection for debugging use, 206 # note this will NOT cause qemu to block waiting for the connection 207 qmp_file2 = "." + next(tempfile._get_candidate_names()) 208 qmp_param += ' -qmp unix:./%s,server,nowait' % (qmp_file2) 209 qmp_port2 = self.tmpdir + "/" + qmp_file2 210 self.logger.info("QMP Available for connection at %s" % (qmp_port2)) 211 212 try: 213 if self.serial_ports >= 2: 214 self.threadsock, threadport = self.create_socket() 215 self.server_socket, self.serverport = self.create_socket() 216 except socket.error as msg: 217 self.logger.error("Failed to create listening socket: %s" % msg[1]) 218 return False 219 220 bootparams = ' printk.time=1' 221 if extra_bootparams: 222 bootparams = bootparams + ' ' + extra_bootparams 223 224 # Ask QEMU to store the QEMU process PID in file, this way we don't have to parse running processes 225 # and analyze descendents in order to determine it. 226 if os.path.exists(self.qemu_pidfile): 227 os.remove(self.qemu_pidfile) 228 self.qemuparams = 'bootparams="{0}" qemuparams="-pidfile {1} {2}"'.format(bootparams, self.qemu_pidfile, qmp_param) 229 230 if qemuparams: 231 self.qemuparams = self.qemuparams[:-1] + " " + qemuparams + " " + '\"' 232 233 if self.serial_ports >= 2: 234 launch_cmd += ' tcpserial=%s:%s %s' % (threadport, self.serverport, self.qemuparams) 235 else: 236 launch_cmd += ' tcpserial=%s %s' % (self.serverport, self.qemuparams) 237 238 self.origchldhandler = signal.getsignal(signal.SIGCHLD) 239 signal.signal(signal.SIGCHLD, self.handleSIGCHLD) 240 241 self.logger.debug('launchcmd=%s'%(launch_cmd)) 242 243 # FIXME: We pass in stdin=subprocess.PIPE here to work around stty 244 # blocking at the end of the runqemu script when using this within 245 # oe-selftest (this makes stty error out immediately). There ought 246 # to be a proper fix but this will suffice for now. 247 self.runqemu = subprocess.Popen(launch_cmd, shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT, stdin=subprocess.PIPE, preexec_fn=os.setpgrp, env=env, cwd=self.tmpdir) 248 output = self.runqemu.stdout 249 launch_time = time.time() 250 251 # 252 # We need the preexec_fn above so that all runqemu processes can easily be killed 253 # (by killing their process group). This presents a problem if this controlling 254 # process itself is killed however since those processes don't notice the death 255 # of the parent and merrily continue on. 256 # 257 # Rather than hack runqemu to deal with this, we add something here instead. 258 # Basically we fork off another process which holds an open pipe to the parent 259 # and also is setpgrp. If/when the pipe sees EOF from the parent dieing, it kills 260 # the process group. This is like pctrl's PDEATHSIG but for a process group 261 # rather than a single process. 262 # 263 r, w = os.pipe() 264 self.monitorpid = os.fork() 265 if self.monitorpid: 266 os.close(r) 267 self.monitorpipe = os.fdopen(w, "w") 268 else: 269 # child process 270 os.setpgrp() 271 os.close(w) 272 r = os.fdopen(r) 273 x = r.read() 274 os.killpg(os.getpgid(self.runqemu.pid), signal.SIGTERM) 275 os._exit(0) 276 277 self.logger.debug("runqemu started, pid is %s" % self.runqemu.pid) 278 self.logger.debug("waiting at most %s seconds for qemu pid (%s)" % 279 (self.runqemutime, time.strftime("%D %H:%M:%S"))) 280 endtime = time.time() + self.runqemutime 281 while not self.is_alive() and time.time() < endtime: 282 if self.runqemu.poll(): 283 if self.runqemu_exited: 284 self.logger.warning("runqemu during is_alive() test") 285 return False 286 if self.runqemu.returncode: 287 # No point waiting any longer 288 self.logger.warning('runqemu exited with code %d' % self.runqemu.returncode) 289 self._dump_host() 290 self.logger.warning("Output from runqemu:\n%s" % self.getOutput(output)) 291 self.stop() 292 return False 293 time.sleep(0.5) 294 295 if self.runqemu_exited: 296 self.logger.warning("runqemu after timeout") 297 298 if self.runqemu.returncode: 299 self.logger.warning('runqemu exited with code %d' % self.runqemu.returncode) 300 301 if not self.is_alive(): 302 self.logger.error("Qemu pid didn't appear in %s seconds (%s)" % 303 (self.runqemutime, time.strftime("%D %H:%M:%S"))) 304 305 qemu_pid = None 306 if os.path.isfile(self.qemu_pidfile): 307 with open(self.qemu_pidfile, 'r') as f: 308 qemu_pid = f.read().strip() 309 310 self.logger.error("Status information, poll status: %s, pidfile exists: %s, pidfile contents %s, proc pid exists %s" 311 % (self.runqemu.poll(), os.path.isfile(self.qemu_pidfile), str(qemu_pid), os.path.exists("/proc/" + str(qemu_pid)))) 312 313 # Dump all processes to help us to figure out what is going on... 314 ps = subprocess.Popen(['ps', 'axww', '-o', 'pid,ppid,pri,ni,command '], stdout=subprocess.PIPE).communicate()[0] 315 processes = ps.decode("utf-8") 316 self.logger.debug("Running processes:\n%s" % processes) 317 self._dump_host() 318 op = self.getOutput(output) 319 self.stop() 320 if op: 321 self.logger.error("Output from runqemu:\n%s" % op) 322 else: 323 self.logger.error("No output from runqemu.\n") 324 return False 325 326 # Create the client socket for the QEMU Monitor Control Socket 327 # This will allow us to read status from Qemu if the the process 328 # is still alive 329 self.logger.debug("QMP Initializing to %s" % (qmp_port)) 330 # chdir dance for path length issues with unix sockets 331 origpath = os.getcwd() 332 try: 333 os.chdir(os.path.dirname(qmp_port)) 334 try: 335 from qmp.legacy import QEMUMonitorProtocol 336 self.qmp = QEMUMonitorProtocol(os.path.basename(qmp_port)) 337 except OSError as msg: 338 self.logger.warning("Failed to initialize qemu monitor socket: %s File: %s" % (msg, msg.filename)) 339 return False 340 341 self.logger.debug("QMP Connecting to %s" % (qmp_port)) 342 if not os.path.exists(qmp_port) and self.is_alive(): 343 self.logger.debug("QMP Port does not exist waiting for it to be created") 344 endtime = time.time() + self.runqemutime 345 while not os.path.exists(qmp_port) and self.is_alive() and time.time() < endtime: 346 self.logger.info("QMP port does not exist yet!") 347 time.sleep(0.5) 348 if not os.path.exists(qmp_port) and self.is_alive(): 349 self.logger.warning("QMP Port still does not exist but QEMU is alive") 350 return False 351 352 try: 353 # set timeout value for all QMP calls 354 self.qmp.settimeout(self.runqemutime) 355 self.qmp.connect() 356 connect_time = time.time() 357 self.logger.info("QMP connected to QEMU at %s and took %s seconds" % 358 (time.strftime("%D %H:%M:%S"), 359 time.time() - launch_time)) 360 except OSError as msg: 361 self.logger.warning("Failed to connect qemu monitor socket: %s File: %s" % (msg, msg.filename)) 362 return False 363 except qmp.legacy.QMPError as msg: 364 self.logger.warning("Failed to communicate with qemu monitor: %s" % (msg)) 365 return False 366 finally: 367 os.chdir(origpath) 368 369 # We worry that mmap'd libraries may cause page faults which hang the qemu VM for periods 370 # causing failures. Before we "start" qemu, read through it's mapped files to try and 371 # ensure we don't hit page faults later 372 mapdir = "/proc/" + str(self.qemupid) + "/map_files/" 373 try: 374 for f in os.listdir(mapdir): 375 try: 376 linktarget = os.readlink(os.path.join(mapdir, f)) 377 if not linktarget.startswith("/") or linktarget.startswith("/dev") or "deleted" in linktarget: 378 continue 379 with open(linktarget, "rb") as readf: 380 data = True 381 while data: 382 data = readf.read(4096) 383 except FileNotFoundError: 384 continue 385 # Centos7 doesn't allow us to read /map_files/ 386 except PermissionError: 387 pass 388 389 # Release the qemu process to continue running 390 self.run_monitor('cont') 391 self.logger.info("QMP released QEMU at %s and took %s seconds from connect" % 392 (time.strftime("%D %H:%M:%S"), 393 time.time() - connect_time)) 394 395 # We are alive: qemu is running 396 out = self.getOutput(output) 397 netconf = False # network configuration is not required by default 398 self.logger.debug("qemu started in %s seconds - qemu procces pid is %s (%s)" % 399 (time.time() - (endtime - self.runqemutime), 400 self.qemupid, time.strftime("%D %H:%M:%S"))) 401 cmdline = '' 402 if get_ip: 403 with open('/proc/%s/cmdline' % self.qemupid) as p: 404 cmdline = p.read() 405 # It is needed to sanitize the data received 406 # because is possible to have control characters 407 cmdline = re_control_char.sub(' ', cmdline) 408 try: 409 if self.use_slirp: 410 tcp_ports = cmdline.split("hostfwd=tcp:")[1] 411 ip, tcp_ports = tcp_ports.split(":")[:2] 412 host_port = tcp_ports[:tcp_ports.find('-')] 413 self.ip = "%s:%s" % (ip, host_port) 414 else: 415 ips = re.findall(r"((?:[0-9]{1,3}\.){3}[0-9]{1,3})", cmdline.split("ip=")[1]) 416 self.ip = ips[0] 417 self.server_ip = ips[1] 418 self.logger.debug("qemu cmdline used:\n{}".format(cmdline)) 419 except (IndexError, ValueError): 420 # Try to get network configuration from runqemu output 421 match = re.match(r'.*Network configuration: (?:ip=)*([0-9.]+)::([0-9.]+):([0-9.]+).*', 422 out, re.MULTILINE|re.DOTALL) 423 if match: 424 self.ip, self.server_ip, self.netmask = match.groups() 425 # network configuration is required as we couldn't get it 426 # from the runqemu command line, so qemu doesn't run kernel 427 # and guest networking is not configured 428 netconf = True 429 else: 430 self.logger.error("Couldn't get ip from qemu command line and runqemu output! " 431 "Here is the qemu command line used:\n%s\n" 432 "and output from runqemu:\n%s" % (cmdline, out)) 433 self._dump_host() 434 self.stop() 435 return False 436 437 self.logger.debug("Target IP: %s" % self.ip) 438 self.logger.debug("Server IP: %s" % self.server_ip) 439 440 if self.serial_ports >= 2: 441 self.thread = LoggingThread(self.log, self.threadsock, self.logger) 442 self.thread.start() 443 if not self.thread.connection_established.wait(self.boottime): 444 self.logger.error("Didn't receive a console connection from qemu. " 445 "Here is the qemu command line used:\n%s\nand " 446 "output from runqemu:\n%s" % (cmdline, out)) 447 self.stop_thread() 448 return False 449 450 self.logger.debug("Output from runqemu:\n%s", out) 451 self.logger.debug("Waiting at most %d seconds for login banner (%s)" % 452 (self.boottime, time.strftime("%D %H:%M:%S"))) 453 endtime = time.time() + self.boottime 454 socklist = [self.server_socket] 455 reachedlogin = False 456 stopread = False 457 qemusock = None 458 bootlog = b'' 459 data = b'' 460 while time.time() < endtime and not stopread: 461 try: 462 sread, swrite, serror = select.select(socklist, [], [], 5) 463 except InterruptedError: 464 continue 465 for sock in sread: 466 if sock is self.server_socket: 467 qemusock, addr = self.server_socket.accept() 468 qemusock.setblocking(0) 469 socklist.append(qemusock) 470 socklist.remove(self.server_socket) 471 self.logger.debug("Connection from %s:%s" % addr) 472 else: 473 # try to avoid reading only a single character at a time 474 time.sleep(0.1) 475 data = data + sock.recv(1024) 476 if data: 477 bootlog += data 478 if self.serial_ports < 2: 479 # this socket has mixed console/kernel data, log it to logfile 480 self.log(data) 481 482 data = b'' 483 484 decodedlog = self.decode_qemulog(bootlog) 485 if self.boot_patterns['search_reached_prompt'] in decodedlog: 486 self.server_socket = qemusock 487 stopread = True 488 reachedlogin = True 489 self.logger.debug("Reached login banner in %s seconds (%s, %s)" % 490 (time.time() - (endtime - self.boottime), 491 time.strftime("%D %H:%M:%S"), time.time())) 492 else: 493 # no need to check if reachedlogin unless we support multiple connections 494 self.logger.debug("QEMU socket disconnected before login banner reached. (%s)" % 495 time.strftime("%D %H:%M:%S")) 496 socklist.remove(sock) 497 sock.close() 498 stopread = True 499 500 if not reachedlogin: 501 if time.time() >= endtime: 502 self.logger.warning("Target didn't reach login banner in %d seconds (%s)" % 503 (self.boottime, time.strftime("%D %H:%M:%S"))) 504 tail = lambda l: "\n".join(l.splitlines()[-25:]) 505 bootlog = self.decode_qemulog(bootlog) 506 # in case bootlog is empty, use tail qemu log store at self.msg 507 lines = tail(bootlog if bootlog else self.msg) 508 self.logger.warning("Last 25 lines of text (%d):\n%s" % (len(bootlog), lines)) 509 self.logger.warning("Check full boot log: %s" % self.logfile) 510 self._dump_host() 511 self.stop() 512 return False 513 514 # If we are not able to login the tests can continue 515 try: 516 (status, output) = self.run_serial(self.boot_patterns['send_login_user'], raw=True, timeout=120) 517 if re.search(self.boot_patterns['search_login_succeeded'], output): 518 self.logged = True 519 self.logger.debug("Logged as root in serial console") 520 if netconf: 521 # configure guest networking 522 cmd = "ifconfig eth0 %s netmask %s up\n" % (self.ip, self.netmask) 523 output = self.run_serial(cmd, raw=True)[1] 524 if re.search(r"root@[a-zA-Z0-9\-]+:~#", output): 525 self.logger.debug("configured ip address %s", self.ip) 526 else: 527 self.logger.debug("Couldn't configure guest networking") 528 else: 529 self.logger.warning("Couldn't login into serial console" 530 " as root using blank password") 531 self.logger.warning("The output:\n%s" % output) 532 except: 533 self.logger.warning("Serial console failed while trying to login") 534 return True 535 536 def stop(self): 537 if hasattr(self, "origchldhandler"): 538 signal.signal(signal.SIGCHLD, self.origchldhandler) 539 self.stop_thread() 540 self.stop_qemu_system() 541 if self.runqemu: 542 if hasattr(self, "monitorpid"): 543 os.kill(self.monitorpid, signal.SIGKILL) 544 self.logger.debug("Sending SIGTERM to runqemu") 545 try: 546 os.killpg(os.getpgid(self.runqemu.pid), signal.SIGTERM) 547 except OSError as e: 548 if e.errno != errno.ESRCH: 549 raise 550 try: 551 outs, errs = self.runqemu.communicate(timeout = self.runqemutime) 552 if outs: 553 self.logger.info("Output from runqemu:\n%s", outs.decode("utf-8")) 554 if errs: 555 self.logger.info("Stderr from runqemu:\n%s", errs.decode("utf-8")) 556 except TimeoutExpired: 557 self.logger.debug("Sending SIGKILL to runqemu") 558 os.killpg(os.getpgid(self.runqemu.pid), signal.SIGKILL) 559 if not self.runqemu.stdout.closed: 560 self.logger.info("Output from runqemu:\n%s" % self.getOutput(self.runqemu.stdout)) 561 self.runqemu.stdin.close() 562 self.runqemu.stdout.close() 563 self.runqemu_exited = True 564 565 if hasattr(self, 'qmp') and self.qmp: 566 self.qmp.close() 567 self.qmp = None 568 if hasattr(self, 'server_socket') and self.server_socket: 569 self.server_socket.close() 570 self.server_socket = None 571 if hasattr(self, 'threadsock') and self.threadsock: 572 self.threadsock.close() 573 self.threadsock = None 574 self.qemupid = None 575 self.ip = None 576 if os.path.exists(self.qemu_pidfile): 577 try: 578 os.remove(self.qemu_pidfile) 579 except FileNotFoundError as e: 580 # We raced, ignore 581 pass 582 if self.monitorpipe: 583 self.monitorpipe.close() 584 585 def stop_qemu_system(self): 586 if self.qemupid: 587 try: 588 # qemu-system behaves well and a SIGTERM is enough 589 os.kill(self.qemupid, signal.SIGTERM) 590 except ProcessLookupError as e: 591 self.logger.warning('qemu-system ended unexpectedly') 592 593 def stop_thread(self): 594 if self.thread and self.thread.is_alive(): 595 self.thread.stop() 596 self.thread.join() 597 598 def allowexit(self): 599 self.canexit = True 600 if self.thread: 601 self.thread.allowexit() 602 603 def restart(self, qemuparams = None): 604 self.logger.warning("Restarting qemu process") 605 if self.runqemu.poll() is None: 606 self.stop() 607 if self.start(qemuparams): 608 return True 609 return False 610 611 def is_alive(self): 612 if not self.runqemu or self.runqemu.poll() is not None or self.runqemu_exited: 613 return False 614 if os.path.isfile(self.qemu_pidfile): 615 # when handling pidfile, qemu creates the file, stat it, lock it and then write to it 616 # so it's possible that the file has been created but the content is empty 617 pidfile_timeout = time.time() + 3 618 while time.time() < pidfile_timeout: 619 with open(self.qemu_pidfile, 'r') as f: 620 qemu_pid = f.read().strip() 621 # file created but not yet written contents 622 if not qemu_pid: 623 time.sleep(0.5) 624 continue 625 else: 626 if os.path.exists("/proc/" + qemu_pid): 627 self.qemupid = int(qemu_pid) 628 return True 629 return False 630 631 def run_monitor(self, command, args=None, timeout=60): 632 if hasattr(self, 'qmp') and self.qmp: 633 self.qmp.settimeout(timeout) 634 if args is not None: 635 return self.qmp.cmd(command, args) 636 else: 637 return self.qmp.cmd(command) 638 639 def run_serial(self, command, raw=False, timeout=60): 640 # Returns (status, output) where status is 1 on success and 0 on error 641 642 # We assume target system have echo to get command status 643 if not raw: 644 command = "%s; echo $?\n" % command 645 646 data = '' 647 status = 0 648 self.server_socket.sendall(command.encode('utf-8')) 649 start = time.time() 650 end = start + timeout 651 while True: 652 now = time.time() 653 if now >= end: 654 data += "<<< run_serial(): command timed out after %d seconds without output >>>\r\n\r\n" % timeout 655 break 656 try: 657 sread, _, _ = select.select([self.server_socket],[],[], end - now) 658 except InterruptedError: 659 continue 660 if sread: 661 # try to avoid reading single character at a time 662 time.sleep(0.1) 663 answer = self.server_socket.recv(1024) 664 if answer: 665 data += answer.decode('utf-8') 666 # Search the prompt to stop 667 if re.search(self.boot_patterns['search_cmd_finished'], data): 668 break 669 else: 670 if self.canexit: 671 return (1, "") 672 raise Exception("No data on serial console socket, connection closed?") 673 674 if data: 675 if raw: 676 status = 1 677 else: 678 # Remove first line (command line) and last line (prompt) 679 data = data[data.find('$?\r\n')+4:data.rfind('\r\n')] 680 index = data.rfind('\r\n') 681 if index == -1: 682 status_cmd = data 683 data = "" 684 else: 685 status_cmd = data[index+2:] 686 data = data[:index] 687 if (status_cmd == "0"): 688 status = 1 689 return (status, str(data)) 690 691 692 def _dump_host(self): 693 self.host_dumper.create_dir("qemu") 694 self.logger.warning("Qemu ended unexpectedly, dump data from host" 695 " is in %s" % self.host_dumper.dump_dir) 696 self.host_dumper.dump_host() 697 698# This class is for reading data from a socket and passing it to logfunc 699# to be processed. It's completely event driven and has a straightforward 700# event loop. The mechanism for stopping the thread is a simple pipe which 701# will wake up the poll and allow for tearing everything down. 702class LoggingThread(threading.Thread): 703 def __init__(self, logfunc, sock, logger): 704 self.connection_established = threading.Event() 705 self.serversock = sock 706 self.logfunc = logfunc 707 self.logger = logger 708 self.readsock = None 709 self.running = False 710 self.canexit = False 711 712 self.errorevents = select.POLLERR | select.POLLHUP | select.POLLNVAL 713 self.readevents = select.POLLIN | select.POLLPRI 714 715 threading.Thread.__init__(self, target=self.threadtarget) 716 717 def threadtarget(self): 718 try: 719 self.eventloop() 720 finally: 721 self.teardown() 722 723 def run(self): 724 self.logger.debug("Starting logging thread") 725 self.readpipe, self.writepipe = os.pipe() 726 threading.Thread.run(self) 727 728 def stop(self): 729 self.logger.debug("Stopping logging thread") 730 if self.running: 731 os.write(self.writepipe, bytes("stop", "utf-8")) 732 733 def teardown(self): 734 self.logger.debug("Tearing down logging thread") 735 self.close_socket(self.serversock) 736 737 if self.readsock is not None: 738 self.close_socket(self.readsock) 739 740 self.close_ignore_error(self.readpipe) 741 self.close_ignore_error(self.writepipe) 742 self.running = False 743 744 def allowexit(self): 745 self.canexit = True 746 747 def eventloop(self): 748 poll = select.poll() 749 event_read_mask = self.errorevents | self.readevents 750 poll.register(self.serversock.fileno()) 751 poll.register(self.readpipe, event_read_mask) 752 753 breakout = False 754 self.running = True 755 self.logger.debug("Starting thread event loop") 756 while not breakout: 757 events = poll.poll() 758 for event in events: 759 # An error occurred, bail out 760 if event[1] & self.errorevents: 761 raise Exception(self.stringify_event(event[1])) 762 763 # Event to stop the thread 764 if self.readpipe == event[0]: 765 self.logger.debug("Stop event received") 766 breakout = True 767 break 768 769 # A connection request was received 770 elif self.serversock.fileno() == event[0]: 771 self.logger.debug("Connection request received") 772 self.readsock, _ = self.serversock.accept() 773 self.readsock.setblocking(0) 774 poll.unregister(self.serversock.fileno()) 775 poll.register(self.readsock.fileno(), event_read_mask) 776 777 self.logger.debug("Setting connection established event") 778 self.connection_established.set() 779 780 # Actual data to be logged 781 elif self.readsock.fileno() == event[0]: 782 data = self.recv(1024) 783 self.logfunc(data) 784 785 # Since the socket is non-blocking make sure to honor EAGAIN 786 # and EWOULDBLOCK. 787 def recv(self, count): 788 try: 789 data = self.readsock.recv(count) 790 except socket.error as e: 791 if e.errno == errno.EAGAIN or e.errno == errno.EWOULDBLOCK: 792 return b'' 793 else: 794 raise 795 796 if data is None: 797 raise Exception("No data on read ready socket") 798 elif not data: 799 # This actually means an orderly shutdown 800 # happened. But for this code it counts as an 801 # error since the connection shouldn't go away 802 # until qemu exits. 803 if not self.canexit: 804 raise Exception("Console connection closed unexpectedly") 805 return b'' 806 807 return data 808 809 def stringify_event(self, event): 810 val = '' 811 if select.POLLERR == event: 812 val = 'POLLER' 813 elif select.POLLHUP == event: 814 val = 'POLLHUP' 815 elif select.POLLNVAL == event: 816 val = 'POLLNVAL' 817 return val 818 819 def close_socket(self, sock): 820 sock.shutdown(socket.SHUT_RDWR) 821 sock.close() 822 823 def close_ignore_error(self, fd): 824 try: 825 os.close(fd) 826 except OSError: 827 pass 828