1# Copyright (c) 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 5import logging 6import operator 7import os 8 9 10# TODO: This is a quick workaround; some of our arm devices so far only 11# support the HDMI EDIDs and the DP one at 1680x1050. A more proper 12# solution is to build a database of supported resolutions and pixel 13# clocks for each model and check if the EDID is in the supported list. 14def is_edid_supported(host, interface, width, height): 15 """Check whether the EDID is supported by DUT 16 17 @param host: A CrosHost object. 18 @param interface: The display interface, like 'HDMI'. 19 @param width: The screen width 20 @param height: The screen height 21 22 @return: True if the check passes; False otherwise. 23 """ 24 # TODO: Support client test that the host is not a CrosHost. 25 platform = host.get_platform() 26 prefix = platform.lower().split('_')[0] 27 if prefix in ('snow', 'spring', 'skate', 'peach', 'veyron'): 28 if interface == 'DP': 29 return width == 1680 and height == 1050 30 return True 31 32 33class Edid(object): 34 """Edid is an abstraction of EDID (Extended Display Identification Data). 35 36 It provides methods to get the properties, manipulate the structure, 37 import from a file, export to a file, etc. 38 39 """ 40 41 BLOCK_SIZE = 128 42 43 44 def __init__(self, data, skip_verify=False): 45 """Construct an Edid. 46 47 @param data: A byte-array of EDID data. 48 @param skip_verify: True to skip the correctness check. 49 """ 50 if not Edid.verify(data) and not skip_verify: 51 raise ValueError('Not a valid EDID.') 52 self.data = data 53 54 55 @staticmethod 56 def verify(data): 57 """Verify the correctness of EDID. 58 59 @param data: A byte-array of EDID data. 60 61 @return True if the EDID is correct; False otherwise. 62 """ 63 data_len = len(data) 64 if data_len % Edid.BLOCK_SIZE != 0: 65 logging.debug('EDID has an invalid length: %d', data_len) 66 return False 67 68 for start in xrange(0, data_len, Edid.BLOCK_SIZE): 69 # Each block (128-byte) has a checksum at the last byte. 70 checksum = reduce(operator.add, 71 map(ord, data[start:start+Edid.BLOCK_SIZE])) 72 if checksum % 256 != 0: 73 logging.debug('Wrong checksum in the block %d of EDID', 74 start / Edid.BLOCK_SIZE) 75 return False 76 77 return True 78 79 80 @classmethod 81 def from_file(cls, filename, skip_verify=False): 82 """Construct an Edid from a file. 83 84 @param filename: A string of filename. 85 @param skip_verify: True to skip the correctness check. 86 """ 87 if not os.path.exists(filename): 88 raise ValueError('EDID file %r does not exist' % filename) 89 90 if filename.upper().endswith('.TXT'): 91 # Convert the EDID text format, returning from xrandr. 92 data = reduce(operator.add, 93 map(lambda s: s.strip().decode('hex'), 94 open(filename).readlines())) 95 else: 96 data = open(filename).read() 97 return cls(data, skip_verify) 98 99 100 def to_file(self, filename): 101 """Export the EDID to a file. 102 103 @param filename: A string of filename. 104 """ 105 with open(filename, 'w+') as f: 106 f.write(self.data) 107 108 109# A constant object to represent no EDID. 110NO_EDID = Edid('', skip_verify=True) 111