1import os, logging 2 3from autotest_lib.client.bin import utils 4from autotest_lib.client.common_lib import error, autotemp 5from autotest_lib.client.cros import power_status 6from autotest_lib.client.cros import storage as storage_mod 7 8 9class hardware_MultiReaderPowerConsumption(storage_mod.StorageTester): 10 version = 1 11 _files_to_delete = [] 12 _ramdisk_path = None 13 _storage = None 14 15 16 def initialize(self): 17 super(hardware_MultiReaderPowerConsumption, self).initialize() 18 19 # Make sure we're not on AC power 20 self.status = power_status.get_status() 21 if self.status.on_ac(): 22 raise error.TestNAError( 23 'This test needs to be run with the AC power offline') 24 25 26 def cleanup(self): 27 # Remove intermediate files 28 for path in self._files_to_delete: 29 utils.system('rm -f %s' % path) 30 31 if self._storage and os.path.ismount(self._storage['mountpoint']): 32 self.scanner.umount_volume(storage_dict=self._storage) 33 34 if self._ramdisk_path and os.path.ismount(self._ramdisk_path.name): 35 umount_ramdisk(self._ramdisk_path.name) 36 self._ramdisk_path.clean() 37 38 super(hardware_MultiReaderPowerConsumption, self).cleanup() 39 40 41 def readwrite_test(self, path, size, delete_file=False): 42 """Heavy-duty random read/write test. Run `dd` & `tail -f` in parallel 43 44 The random write is done by writing a file from /dev/urandom into the 45 given location, while the random read is done by concurrently reading 46 that file. 47 48 @param path: The directory that will create the test file. 49 @param size: Size of the test file, in MiB. 50 @param delete_file: Flag the file to be deleted on test exit. 51 Otherwise file deletion won't be performed. 52 """ 53 # Calculate the parameters for dd 54 size = 1024*1024*size 55 blocksize = 8192 56 57 # Calculate the filename and full path, flag to delete if needed 58 filename = 'tempfile.%d.delete-me' % size 59 pathfile = os.path.join(path, filename) 60 if delete_file: 61 self._files_to_delete.append(pathfile) 62 63 pid = os.fork() # We need to run two processes in parallel 64 if pid: 65 # parent 66 utils.BgJob('tail -f %s --pid=%s > /dev/null' 67 % (pathfile, pid)) 68 # Reap the dd child so that tail does not wait for the zombie 69 os.waitpid(pid, 0) 70 else: 71 # child 72 utils.system('dd if=/dev/urandom of=%s bs=%d count=%s' 73 % (pathfile, blocksize, (size//blocksize))) 74 # A forked child is exiting here, so we really do want os._exit: 75 os._exit(0) 76 77 78 def run_once(self, ramdisk_size=513, file_size=512, drain_limit=1.05, 79 volume_filter={'bus': 'usb'}): 80 """Test card reader CPU power consumption to be within acceptable 81 range while performing random r/w 82 83 The random r/w is performed in the readwrite_test() method, by 84 concurrently running `dd if=/dev/urandom` and `tail -f`. It is run once 85 on a ramdisk with the SD card mounted, then on the SD card with the 86 ramdisk unmounted, and then on the SD card with the ramdisk unmounted. 87 The measured values are then reported. 88 89 @param ramdisk_size: Size of ramdisk (in MiB). 90 @param file_size: Size of test file (in MiB). 91 @param volume_filter: Where to find the card reader. 92 @param drain_limit: maximum ratio between the card reader 93 energy consumption and each of the two 94 ramdisk read/write test energy consumption 95 values. 1.00 means the card reader test may 96 not consume more energy than either ramdisk 97 test, 0.9 means it may consume no more than 98 90% of the ramdisk value, and so forth. 99 """ 100 # Switch to VT2 so the screen turns itself off automatically instead of 101 # dimming, in order to reduce the battery consuption caused by other 102 # variables. 103 utils.system('chvt 2') 104 105 logging.debug('STEP 1: ensure SD card is inserted and mounted') 106 self._storage = self.wait_for_device(volume_filter, cycles=1, 107 mount_volume=True)[0] 108 109 logging.debug('STEP 2: mount the ramdisk') 110 self._ramdisk_path = autotemp.tempdir(unique_id='ramdisk', 111 dir=self.tmpdir) 112 mount_ramdisk(self._ramdisk_path.name, ramdisk_size) 113 114 # Read current charge, as well as maximum charge. 115 self.status.refresh() 116 max_charge = self.status.battery[0].charge_full_design 117 initial_charge = self.status.battery[0].charge_now 118 119 logging.debug('STEP 3: perform heavy-duty read-write test on ramdisk') 120 self.readwrite_test(self._ramdisk_path.name, file_size) 121 # Read current charge (reading A) 122 self.status.refresh() 123 charge_A = self.status.battery[0].charge_now 124 125 logging.debug('STEP 4: unmount ramdisk') 126 umount_ramdisk(self._ramdisk_path.name) 127 128 logging.debug('STEP 5: perform identical read write test on SD card') 129 self.readwrite_test(self._storage['mountpoint'], file_size, 130 delete_file=True) 131 # Read current charge (reading B) 132 self.status.refresh() 133 charge_B = self.status.battery[0].charge_now 134 135 logging.debug('STEP 6: unmount card') 136 self.scanner.umount_volume(storage_dict=self._storage, args='-f -l') 137 138 logging.debug('STEP 7: perform ramdisk test again') 139 mount_ramdisk(self._ramdisk_path.name, ramdisk_size) 140 self.readwrite_test(self._ramdisk_path.name, file_size) 141 # Read current charge (reading C) 142 self.status.refresh() 143 charge_C = self.status.battery[0].charge_now 144 145 # Compute the results 146 ramdisk_plus = initial_charge - charge_A 147 sd_card_solo = charge_A - charge_B 148 ramdisk_solo = charge_B - charge_C 149 150 sd_card_drain_ratio_a = (sd_card_solo / ramdisk_plus) 151 sd_card_drain_ratio_b = (sd_card_solo / ramdisk_solo) 152 153 msg = None 154 if sd_card_drain_ratio_a > drain_limit: 155 msg = ('Card reader drain exceeds mounted baseline by > %f (%f)' 156 % (drain_limit, sd_card_drain_ratio_a)) 157 elif sd_card_drain_ratio_b > drain_limit: 158 msg = ('Card reader drain exceeds unmounted baseline by > %f (%f)' 159 % (drain_limit, sd_card_drain_ratio_b)) 160 161 if msg: 162 raise error.TestError(msg) 163 else: 164 fmt = 'Card reader drain ratio Ok: mounted %f; unmounted %f' 165 logging.info(fmt % (sd_card_drain_ratio_a, sd_card_drain_ratio_b)) 166 167 168def mount_ramdisk(path, size): 169 utils.system('mount -t tmpfs none %s -o size=%sm' % (path, size)) 170 171 172def umount_ramdisk(path): 173 """Umount ramdisk mounted at |path| 174 175 @param path: the mountpoint for the mountd RAM disk 176 """ 177 utils.system('rm -rf %s/*' % path) 178 utils.system('umount -f -l %s' % path) 179