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        self.restore_firmware()
33        super(firmware_ECHash, self).cleanup()
34
35    def get_echash(self):
36        """Get the current EC hash via ectool/fwtool."""
37        if self.faft_client.system.has_host():
38            command = 'fwtool ec echash'
39        else:
40            command = 'ectool echash'
41        lines = self.faft_client.system.run_shell_command_get_output(command)
42        pattern = re.compile('hash:    ([0-9a-f]{64})')
43        for line in lines:
44            matched = pattern.match(line)
45            if matched:
46                return matched.group(1)
47        raise error.TestError("Wrong output of '%s': \n%s" %
48                              (command, '\n'.join(lines)))
49
50    def invalidate_echash(self):
51        """Invalidate the EC hash by requesting hashing some other part."""
52        if self.faft_client.system.has_host():
53            command = 'fwtool ec echash recalc 0 4'
54        else:
55            command = 'ectool echash recalc 0 4'
56        self.faft_client.system.run_shell_command(command)
57
58    def save_echash_and_invalidate(self):
59        """Save the current EC hash and invalidate it."""
60        self.original_echash = self.get_echash()
61        logging.info("Original EC hash: %s", self.original_echash)
62        self.invalidate_echash()
63        invalid_echash = self.get_echash()
64        logging.info("Invalid EC hash: %s", invalid_echash)
65        if invalid_echash == self.original_echash:
66            raise error.TestFail("Failed to invalidate EC hash")
67
68    def compare_echashes(self):
69        """Compare the current EC with the original one."""
70        recomputed_echash = self.get_echash()
71        logging.info("Recomputed EC hash: %s", recomputed_echash)
72        return recomputed_echash == self.original_echash
73
74    def run_once(self):
75        if not self.check_ec_capability():
76            raise error.TestNAError("Nothing needs to be tested on this device")
77        logging.info("Save the EC hash, invalidate it, and warm reboot.")
78        self.save_echash_and_invalidate()
79        self.switcher.mode_aware_reboot()
80
81        logging.info("Compare the recomputed EC hash with the original one.")
82        self.check_state(self.compare_echashes)
83