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, re, subprocess, threading 6import common 7from autotest_lib.client.bin import site_utils, test, utils 8from autotest_lib.client.common_lib import error 9 10 11class hardware_Badblocks(test.test): 12 """ 13 Runs badblocks on the root partition that is not being used. 14 15 """ 16 17 version = 1 18 19 # Define output that is expected from a successful badblocks run. 20 _EXPECTED_BADBLOCKS_OUTPUT = ( 21 'Pass completed, 0 bad blocks found. (0/0/0 errors)') 22 23 # Define Linux badblocks utility name. 24 _BADBLOCKS = 'badblocks' 25 26 # Define variables to store some statistics of the runs. 27 _pass_count = 0 28 _fail_count = 0 29 _longest_runtime = 0 30 31 32 def _get_sector_size(self, dev): 33 """ 34 Finds block device's sector size in bytes. 35 36 @return the sector size. 37 38 """ 39 40 argv = ('parted ' + dev + ' print | grep "Sector size" | awk -F ' + 41 '"/" \'{print $3}\' | sed \'$s/.$//\'') 42 43 return utils.system_output(argv) 44 45 46 def _timeout(self, badblocks_proc): 47 """ 48 Timeout callback for the badblocks process. 49 50 Kills badblocks process if still running and fails test. 51 52 """ 53 54 # Kill badblocks, report if not killed, log any exceptions. 55 if badblocks_proc.poll() == None: 56 try: 57 logging.info('badblocks taking too long---sending SIGKILL') 58 badblocks_proc.kill() 59 except Exception as e: 60 logging.info('%s', e) 61 finally: 62 # name of the kernel function in which the process is sleeping. 63 argv = ('ps eopid,fname,wchan | grep ' + self._BADBLOCKS + 64 ' | awk \'{print $3}\'') 65 waiton = utils.system_output(argv) 66 if waiton: 67 logging.info('badblocks is waiting on %s', waiton) 68 raise error.TestError('Error: badblocks timed out.') 69 70 71 def _run_badblocks(self, dev, sector_size, tmout): 72 """ 73 Runs badblocks. 74 75 """ 76 77 # Run badblocks on the selected partition, with parameters: 78 # -s = show progress 79 # -v = verbose (print error count) 80 # -w = destructive write+read test 81 # -b = block size (set equal to sector size) 82 argv = [self._BADBLOCKS, '-svw', '-d', str(sector_size), dev] 83 msg = 'Running: ' + ' '.join(map(str, argv)) 84 logging.info(msg) 85 badblocks_proc = subprocess.Popen( 86 argv, 87 shell=False, 88 stderr=subprocess.STDOUT, # Combine stderr with stdout. 89 stdout=subprocess.PIPE) 90 91 # Start timeout timer thread. 92 t = threading.Timer(tmout, self._timeout, [badblocks_proc]) 93 t.start() 94 95 # Get badblocks output. 96 stdout, _ = badblocks_proc.communicate() 97 98 # Stop timer if badblocks has finished. 99 t.cancel() 100 101 # Check badblocks exit status. 102 if badblocks_proc.returncode != 0: 103 raise error.TestError('badblocks returned with code: %s', 104 badblocks_proc.returncode) 105 106 # Parse and log badblocks output. 107 logging.info('badblocks output:') 108 lines = stdout.split('\n') 109 del lines[-1] # Remove blank line at end. 110 logging.info(lines[0]) 111 logging.info(lines[1]) 112 # Log the progress of badblocks (line 2 onwards, minus last line). 113 for line in lines[2:-1]: 114 # replace backspace characters with a newline character. 115 line = re.sub(r'[\b]+', '\n', line) 116 # Log test pattern info. 117 pattern_info = line[:line.find(':') + 1] 118 logging.info('%s', pattern_info) 119 sublines = line[line.find(':') + 2:].split('\n') 120 for subline in sublines: 121 logging.info('%s', subline) 122 # Log result (last line). 123 logging.info(lines[-1]) 124 125 # Get run time in seconds. 126 min_sec = re.match(r'(\w+):(\w+)', lines[-2].split()[-4]) 127 runtime = int(min_sec.group(1)) * 60 + int(min_sec.group(2)) 128 129 # Update longest run time. 130 if self._longest_runtime < runtime: 131 self._longest_runtime = runtime 132 133 # Check badblocks result. 134 result = lines[-1].strip() 135 if result != self._EXPECTED_BADBLOCKS_OUTPUT: 136 self._fail_count += 1 137 return 138 self._pass_count += 1 139 140 141 def run_once(self, iters=1, tmout=60 * 60): 142 """ 143 Executes test. 144 145 @param iters: Number of times to run badblocks. 146 @param tmout: Time allowed badblocks to run before killing it. 147 (Default time is 60 minutes.) 148 149 """ 150 151 # Log starting message. 152 logging.info('Statring hardware_Badblocks Test.') 153 logging.info('Iterations: %d', iters) 154 logging.info('badblocks Timeout (sec): %d', tmout) 155 156 # Determine which device and partition to use. 157 logging.info('Determine unused root partition to test on:') 158 dev = site_utils.get_free_root_partition() 159 logging.info('Testing on ' + dev) 160 161 # Get block device's sector size. 162 logging.info('Determine block device sector size:') 163 sector_size = self._get_sector_size(site_utils.get_root_device()) 164 logging.info('Sector size (bytes): ' + sector_size) 165 166 # Get partition size. 167 logging.info('Determine partition size:') 168 part_size = utils.get_disk_size(dev) 169 logging.info('Partition size (bytes): %s', part_size) 170 171 # Run badblocks. 172 for i in range(iters): 173 logging.info('Starting iteration %d', i) 174 self._run_badblocks(dev, sector_size, tmout) 175 176 # Report statistics. 177 logging.info('Total pass: %d', self._pass_count) 178 logging.info('Total fail: %d', self._fail_count) 179 stats = {} 180 stats['ea_badblocks_runs'] = iters 181 stats['ea_passed_count'] = self._pass_count 182 stats['ea_failed_count'] = self._fail_count 183 stats['sec_longest_run'] = self._longest_runtime 184 # TODO: change write_perf_keyval() to output_perf_value() as soon as 185 # autotest is ready for it. 186 self.write_perf_keyval(stats) 187 188 # Report test pass/fail. 189 if self._pass_count != iters: 190 raise error.TestFail('One or more runs found bad blocks on' 191 ' storage device.') 192