1# Copyright (c) 2013 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 5import logging 6import re 7 8from autotest_lib.client.common_lib import error 9from autotest_lib.server.cros.faft.firmware_test import FirmwareTest 10 11 12class firmware_ECHash(FirmwareTest): 13 """ 14 Servo based EC hash recompute test. 15 16 This test ensures that the AP will ask the EC to recompute the hash if 17 the current hash isn't the right size/offset. Use the 'echash' command 18 of EC tool to request the hash of some other part of EC EEPROM, then 19 warm-reboot the AP and check what hash the EC has after booting. 20 AP-RW should have requested the EC recompute the hash of EC-RW. 21 """ 22 version = 1 23 24 def initialize(self, host, cmdline_args): 25 super(firmware_ECHash, self).initialize(host, cmdline_args) 26 self.backup_firmware() 27 self.switcher.setup_mode('normal') 28 self.setup_usbkey(usbkey=False) 29 self.setup_rw_boot() 30 31 def cleanup(self): 32 try: 33 if self.is_firmware_saved(): 34 self.restore_firmware() 35 except Exception as e: 36 logging.error("Caught exception: %s", str(e)) 37 super(firmware_ECHash, self).cleanup() 38 39 def get_echash(self): 40 """Get the current EC hash via ectool/fwtool.""" 41 if self.faft_client.system.has_host(): 42 command = 'fwtool ec echash' 43 else: 44 command = 'ectool echash' 45 lines = self.faft_client.system.run_shell_command_get_output(command) 46 pattern = re.compile('hash: ([0-9a-f]{64})') 47 for line in lines: 48 matched = pattern.match(line) 49 if matched: 50 return matched.group(1) 51 raise error.TestError("Wrong output of '%s': \n%s" % 52 (command, '\n'.join(lines))) 53 54 def invalidate_echash(self): 55 """Invalidate the EC hash by requesting hashing some other part.""" 56 if self.faft_client.system.has_host(): 57 command = 'fwtool ec echash recalc 0 4' 58 else: 59 command = 'ectool echash recalc 0 4' 60 self.faft_client.system.run_shell_command(command) 61 62 def save_echash_and_invalidate(self): 63 """Save the current EC hash and invalidate it.""" 64 self.original_echash = self.get_echash() 65 logging.info("Original EC hash: %s", self.original_echash) 66 self.invalidate_echash() 67 invalid_echash = self.get_echash() 68 logging.info("Invalid EC hash: %s", invalid_echash) 69 if invalid_echash == self.original_echash: 70 raise error.TestFail("Failed to invalidate EC hash") 71 72 def compare_echashes(self): 73 """Compare the current EC with the original one.""" 74 recomputed_echash = self.get_echash() 75 logging.info("Recomputed EC hash: %s", recomputed_echash) 76 return recomputed_echash == self.original_echash 77 78 def run_once(self): 79 if not self.check_ec_capability(): 80 raise error.TestNAError("Nothing needs to be tested on this device") 81 logging.info("Save the EC hash, invalidate it, and warm reboot.") 82 self.save_echash_and_invalidate() 83 self.switcher.mode_aware_reboot() 84 85 logging.info("Compare the recomputed EC hash with the original one.") 86 self.check_state(self.compare_echashes) 87