1 # Copyright 2014 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 import os 6 import time 7 8 from autotest_lib.client.common_lib import error 9 from autotest_lib.client.cros.chameleon import chameleon_port_finder 10 from autotest_lib.client.cros.video import method_logger 11 12 13 class ChameleonScreenshotCapturer(object): 14 """ 15 Provides an interface to capture a dut screenshot using a Chameleon Board. 16 17 Example use: 18 with ChameleonScreenshotCapturer(board, 'HDMI', dutil, '/tmp', 10) as c: 19 c.capture(filename) 20 """ 21 22 23 def __init__(self, chameleon_board, interface, display_facade, dest_dir, 24 timeout_video_input_s, box=None): 25 """ 26 @param chameleon_board: object representing the ChameleonBoard. 27 @param interface: string, display interface to use. eg.: HDMI 28 @param display_facade: display facade object to interact with DUT 29 @param dest_dir: path, full path to the dest dir to put the screenshot. 30 @param timeout_video_input_s: int, max time to wait for chameleon video 31 input to become stable. 32 @box: int tuple, left, upper, right, lower pixel coordinates 33 defining a desired image region 34 35 """ 36 self.chameleon_board = chameleon_board 37 self.display_facade = display_facade 38 self.interface = interface.lower() 39 self.dest_dir = dest_dir 40 self.port = None 41 self.box = box 42 self.timeout_video_input_s = timeout_video_input_s 43 self.was_plugged = False 44 self._find_connected_port() 45 46 47 @method_logger.log 48 def __enter__(self): 49 50 if not self.was_plugged: 51 self.port.plug() 52 53 self.port.wait_video_input_stable(self.timeout_video_input_s) 54 55 self.display_facade.set_mirrored(True) 56 time.sleep(self.timeout_video_input_s) 57 58 return self 59 60 61 @method_logger.log 62 def _find_connected_port(self): 63 """ 64 Gets a connected port of the pre-specified interface. 65 66 @raises TestError if desired port was not detected. 67 68 """ 69 self.chameleon_board.reset() 70 finder = chameleon_port_finder.ChameleonVideoInputFinder( 71 self.chameleon_board, self.display_facade) 72 73 connected_port = finder.find_port(self.interface) 74 75 if connected_port is None: 76 msg = 'No %s port found.\n' % self.interface 77 raise error.TestError(msg + str(finder)) 78 79 self.port = connected_port 80 self.was_plugged = connected_port.plugged 81 82 83 @method_logger.log 84 def capture(self, filename, box=None): 85 """ 86 Captures a screenshot using provided chameleon board. 87 88 We save to a file because comparers like bp take files. 89 90 @param filename: string, filename of the image to save to. 91 @param box: int tuple, left, upper, right, lower pixel coordinates 92 defining a box region of what the image should be. 93 @returns a fullpath to the image just captured. 94 95 """ 96 97 fullpath = os.path.join(self.dest_dir, filename) 98 99 if not box: 100 box = self.box 101 102 img = self.port.capture_screen() 103 img.crop(box).save(fullpath) 104 105 return fullpath 106 107 108 @method_logger.log 109 def __exit__(self, exc_type, exc_val, exc_tb): 110 if not self.was_plugged: 111 self.port.unplug() 112