16ea3dfe1SJarkko Sakkinen# SPDX-License-Identifier: (GPL-2.0 OR BSD-3-Clause) 26ea3dfe1SJarkko Sakkinen 36ea3dfe1SJarkko Sakkinenimport hashlib 46ea3dfe1SJarkko Sakkinenimport os 56ea3dfe1SJarkko Sakkinenimport socket 66ea3dfe1SJarkko Sakkinenimport struct 76ea3dfe1SJarkko Sakkinenimport sys 86ea3dfe1SJarkko Sakkinenimport unittest 98f84bddcSTadeusz Strukimport fcntl 108f84bddcSTadeusz Strukimport select 116ea3dfe1SJarkko Sakkinen 126ea3dfe1SJarkko SakkinenTPM2_ST_NO_SESSIONS = 0x8001 136ea3dfe1SJarkko SakkinenTPM2_ST_SESSIONS = 0x8002 146ea3dfe1SJarkko Sakkinen 156ea3dfe1SJarkko SakkinenTPM2_CC_FIRST = 0x01FF 166ea3dfe1SJarkko Sakkinen 176ea3dfe1SJarkko SakkinenTPM2_CC_CREATE_PRIMARY = 0x0131 186ea3dfe1SJarkko SakkinenTPM2_CC_DICTIONARY_ATTACK_LOCK_RESET = 0x0139 196ea3dfe1SJarkko SakkinenTPM2_CC_CREATE = 0x0153 206ea3dfe1SJarkko SakkinenTPM2_CC_LOAD = 0x0157 216ea3dfe1SJarkko SakkinenTPM2_CC_UNSEAL = 0x015E 226ea3dfe1SJarkko SakkinenTPM2_CC_FLUSH_CONTEXT = 0x0165 236ea3dfe1SJarkko SakkinenTPM2_CC_START_AUTH_SESSION = 0x0176 246ea3dfe1SJarkko SakkinenTPM2_CC_GET_CAPABILITY = 0x017A 25f1a0ba6cSTadeusz StrukTPM2_CC_GET_RANDOM = 0x017B 266ea3dfe1SJarkko SakkinenTPM2_CC_PCR_READ = 0x017E 276ea3dfe1SJarkko SakkinenTPM2_CC_POLICY_PCR = 0x017F 286ea3dfe1SJarkko SakkinenTPM2_CC_PCR_EXTEND = 0x0182 296ea3dfe1SJarkko SakkinenTPM2_CC_POLICY_PASSWORD = 0x018C 306ea3dfe1SJarkko SakkinenTPM2_CC_POLICY_GET_DIGEST = 0x0189 316ea3dfe1SJarkko Sakkinen 326ea3dfe1SJarkko SakkinenTPM2_SE_POLICY = 0x01 336ea3dfe1SJarkko SakkinenTPM2_SE_TRIAL = 0x03 346ea3dfe1SJarkko Sakkinen 356ea3dfe1SJarkko SakkinenTPM2_ALG_RSA = 0x0001 366ea3dfe1SJarkko SakkinenTPM2_ALG_SHA1 = 0x0004 376ea3dfe1SJarkko SakkinenTPM2_ALG_AES = 0x0006 386ea3dfe1SJarkko SakkinenTPM2_ALG_KEYEDHASH = 0x0008 396ea3dfe1SJarkko SakkinenTPM2_ALG_SHA256 = 0x000B 406ea3dfe1SJarkko SakkinenTPM2_ALG_NULL = 0x0010 416ea3dfe1SJarkko SakkinenTPM2_ALG_CBC = 0x0042 426ea3dfe1SJarkko SakkinenTPM2_ALG_CFB = 0x0043 436ea3dfe1SJarkko Sakkinen 446ea3dfe1SJarkko SakkinenTPM2_RH_OWNER = 0x40000001 456ea3dfe1SJarkko SakkinenTPM2_RH_NULL = 0x40000007 466ea3dfe1SJarkko SakkinenTPM2_RH_LOCKOUT = 0x4000000A 476ea3dfe1SJarkko SakkinenTPM2_RS_PW = 0x40000009 486ea3dfe1SJarkko Sakkinen 496ea3dfe1SJarkko SakkinenTPM2_RC_SIZE = 0x01D5 506ea3dfe1SJarkko SakkinenTPM2_RC_AUTH_FAIL = 0x098E 516ea3dfe1SJarkko SakkinenTPM2_RC_POLICY_FAIL = 0x099D 526ea3dfe1SJarkko SakkinenTPM2_RC_COMMAND_CODE = 0x0143 536ea3dfe1SJarkko Sakkinen 546ea3dfe1SJarkko SakkinenTSS2_RC_LAYER_SHIFT = 16 556ea3dfe1SJarkko SakkinenTSS2_RESMGR_TPM_RC_LAYER = (11 << TSS2_RC_LAYER_SHIFT) 566ea3dfe1SJarkko Sakkinen 576ea3dfe1SJarkko SakkinenTPM2_CAP_HANDLES = 0x00000001 586ea3dfe1SJarkko SakkinenTPM2_CAP_COMMANDS = 0x00000002 590d060f23SStefan BergerTPM2_CAP_PCRS = 0x00000005 606ea3dfe1SJarkko SakkinenTPM2_CAP_TPM_PROPERTIES = 0x00000006 616ea3dfe1SJarkko Sakkinen 626ea3dfe1SJarkko SakkinenTPM2_PT_FIXED = 0x100 636ea3dfe1SJarkko SakkinenTPM2_PT_TOTAL_COMMANDS = TPM2_PT_FIXED + 41 646ea3dfe1SJarkko Sakkinen 656ea3dfe1SJarkko SakkinenHR_SHIFT = 24 666ea3dfe1SJarkko SakkinenHR_LOADED_SESSION = 0x02000000 676ea3dfe1SJarkko SakkinenHR_TRANSIENT = 0x80000000 686ea3dfe1SJarkko Sakkinen 696ea3dfe1SJarkko SakkinenSHA1_DIGEST_SIZE = 20 706ea3dfe1SJarkko SakkinenSHA256_DIGEST_SIZE = 32 716ea3dfe1SJarkko Sakkinen 726ea3dfe1SJarkko SakkinenTPM2_VER0_ERRORS = { 736ea3dfe1SJarkko Sakkinen 0x000: "TPM_RC_SUCCESS", 746ea3dfe1SJarkko Sakkinen 0x030: "TPM_RC_BAD_TAG", 756ea3dfe1SJarkko Sakkinen} 766ea3dfe1SJarkko Sakkinen 776ea3dfe1SJarkko SakkinenTPM2_VER1_ERRORS = { 786ea3dfe1SJarkko Sakkinen 0x000: "TPM_RC_FAILURE", 796ea3dfe1SJarkko Sakkinen 0x001: "TPM_RC_FAILURE", 806ea3dfe1SJarkko Sakkinen 0x003: "TPM_RC_SEQUENCE", 816ea3dfe1SJarkko Sakkinen 0x00B: "TPM_RC_PRIVATE", 826ea3dfe1SJarkko Sakkinen 0x019: "TPM_RC_HMAC", 836ea3dfe1SJarkko Sakkinen 0x020: "TPM_RC_DISABLED", 846ea3dfe1SJarkko Sakkinen 0x021: "TPM_RC_EXCLUSIVE", 856ea3dfe1SJarkko Sakkinen 0x024: "TPM_RC_AUTH_TYPE", 866ea3dfe1SJarkko Sakkinen 0x025: "TPM_RC_AUTH_MISSING", 876ea3dfe1SJarkko Sakkinen 0x026: "TPM_RC_POLICY", 886ea3dfe1SJarkko Sakkinen 0x027: "TPM_RC_PCR", 896ea3dfe1SJarkko Sakkinen 0x028: "TPM_RC_PCR_CHANGED", 906ea3dfe1SJarkko Sakkinen 0x02D: "TPM_RC_UPGRADE", 916ea3dfe1SJarkko Sakkinen 0x02E: "TPM_RC_TOO_MANY_CONTEXTS", 926ea3dfe1SJarkko Sakkinen 0x02F: "TPM_RC_AUTH_UNAVAILABLE", 936ea3dfe1SJarkko Sakkinen 0x030: "TPM_RC_REBOOT", 946ea3dfe1SJarkko Sakkinen 0x031: "TPM_RC_UNBALANCED", 956ea3dfe1SJarkko Sakkinen 0x042: "TPM_RC_COMMAND_SIZE", 966ea3dfe1SJarkko Sakkinen 0x043: "TPM_RC_COMMAND_CODE", 976ea3dfe1SJarkko Sakkinen 0x044: "TPM_RC_AUTHSIZE", 986ea3dfe1SJarkko Sakkinen 0x045: "TPM_RC_AUTH_CONTEXT", 996ea3dfe1SJarkko Sakkinen 0x046: "TPM_RC_NV_RANGE", 1006ea3dfe1SJarkko Sakkinen 0x047: "TPM_RC_NV_SIZE", 1016ea3dfe1SJarkko Sakkinen 0x048: "TPM_RC_NV_LOCKED", 1026ea3dfe1SJarkko Sakkinen 0x049: "TPM_RC_NV_AUTHORIZATION", 1036ea3dfe1SJarkko Sakkinen 0x04A: "TPM_RC_NV_UNINITIALIZED", 1046ea3dfe1SJarkko Sakkinen 0x04B: "TPM_RC_NV_SPACE", 1056ea3dfe1SJarkko Sakkinen 0x04C: "TPM_RC_NV_DEFINED", 1066ea3dfe1SJarkko Sakkinen 0x050: "TPM_RC_BAD_CONTEXT", 1076ea3dfe1SJarkko Sakkinen 0x051: "TPM_RC_CPHASH", 1086ea3dfe1SJarkko Sakkinen 0x052: "TPM_RC_PARENT", 1096ea3dfe1SJarkko Sakkinen 0x053: "TPM_RC_NEEDS_TEST", 1106ea3dfe1SJarkko Sakkinen 0x054: "TPM_RC_NO_RESULT", 1116ea3dfe1SJarkko Sakkinen 0x055: "TPM_RC_SENSITIVE", 1126ea3dfe1SJarkko Sakkinen 0x07F: "RC_MAX_FM0", 1136ea3dfe1SJarkko Sakkinen} 1146ea3dfe1SJarkko Sakkinen 1156ea3dfe1SJarkko SakkinenTPM2_FMT1_ERRORS = { 1166ea3dfe1SJarkko Sakkinen 0x001: "TPM_RC_ASYMMETRIC", 1176ea3dfe1SJarkko Sakkinen 0x002: "TPM_RC_ATTRIBUTES", 1186ea3dfe1SJarkko Sakkinen 0x003: "TPM_RC_HASH", 1196ea3dfe1SJarkko Sakkinen 0x004: "TPM_RC_VALUE", 1206ea3dfe1SJarkko Sakkinen 0x005: "TPM_RC_HIERARCHY", 1216ea3dfe1SJarkko Sakkinen 0x007: "TPM_RC_KEY_SIZE", 1226ea3dfe1SJarkko Sakkinen 0x008: "TPM_RC_MGF", 1236ea3dfe1SJarkko Sakkinen 0x009: "TPM_RC_MODE", 1246ea3dfe1SJarkko Sakkinen 0x00A: "TPM_RC_TYPE", 1256ea3dfe1SJarkko Sakkinen 0x00B: "TPM_RC_HANDLE", 1266ea3dfe1SJarkko Sakkinen 0x00C: "TPM_RC_KDF", 1276ea3dfe1SJarkko Sakkinen 0x00D: "TPM_RC_RANGE", 1286ea3dfe1SJarkko Sakkinen 0x00E: "TPM_RC_AUTH_FAIL", 1296ea3dfe1SJarkko Sakkinen 0x00F: "TPM_RC_NONCE", 1306ea3dfe1SJarkko Sakkinen 0x010: "TPM_RC_PP", 1316ea3dfe1SJarkko Sakkinen 0x012: "TPM_RC_SCHEME", 1326ea3dfe1SJarkko Sakkinen 0x015: "TPM_RC_SIZE", 1336ea3dfe1SJarkko Sakkinen 0x016: "TPM_RC_SYMMETRIC", 1346ea3dfe1SJarkko Sakkinen 0x017: "TPM_RC_TAG", 1356ea3dfe1SJarkko Sakkinen 0x018: "TPM_RC_SELECTOR", 1366ea3dfe1SJarkko Sakkinen 0x01A: "TPM_RC_INSUFFICIENT", 1376ea3dfe1SJarkko Sakkinen 0x01B: "TPM_RC_SIGNATURE", 1386ea3dfe1SJarkko Sakkinen 0x01C: "TPM_RC_KEY", 1396ea3dfe1SJarkko Sakkinen 0x01D: "TPM_RC_POLICY_FAIL", 1406ea3dfe1SJarkko Sakkinen 0x01F: "TPM_RC_INTEGRITY", 1416ea3dfe1SJarkko Sakkinen 0x020: "TPM_RC_TICKET", 1426ea3dfe1SJarkko Sakkinen 0x021: "TPM_RC_RESERVED_BITS", 1436ea3dfe1SJarkko Sakkinen 0x022: "TPM_RC_BAD_AUTH", 1446ea3dfe1SJarkko Sakkinen 0x023: "TPM_RC_EXPIRED", 1456ea3dfe1SJarkko Sakkinen 0x024: "TPM_RC_POLICY_CC", 1466ea3dfe1SJarkko Sakkinen 0x025: "TPM_RC_BINDING", 1476ea3dfe1SJarkko Sakkinen 0x026: "TPM_RC_CURVE", 1486ea3dfe1SJarkko Sakkinen 0x027: "TPM_RC_ECC_POINT", 1496ea3dfe1SJarkko Sakkinen} 1506ea3dfe1SJarkko Sakkinen 1516ea3dfe1SJarkko SakkinenTPM2_WARN_ERRORS = { 1526ea3dfe1SJarkko Sakkinen 0x001: "TPM_RC_CONTEXT_GAP", 1536ea3dfe1SJarkko Sakkinen 0x002: "TPM_RC_OBJECT_MEMORY", 1546ea3dfe1SJarkko Sakkinen 0x003: "TPM_RC_SESSION_MEMORY", 1556ea3dfe1SJarkko Sakkinen 0x004: "TPM_RC_MEMORY", 1566ea3dfe1SJarkko Sakkinen 0x005: "TPM_RC_SESSION_HANDLES", 1576ea3dfe1SJarkko Sakkinen 0x006: "TPM_RC_OBJECT_HANDLES", 1586ea3dfe1SJarkko Sakkinen 0x007: "TPM_RC_LOCALITY", 1596ea3dfe1SJarkko Sakkinen 0x008: "TPM_RC_YIELDED", 1606ea3dfe1SJarkko Sakkinen 0x009: "TPM_RC_CANCELED", 1616ea3dfe1SJarkko Sakkinen 0x00A: "TPM_RC_TESTING", 1626ea3dfe1SJarkko Sakkinen 0x010: "TPM_RC_REFERENCE_H0", 1636ea3dfe1SJarkko Sakkinen 0x011: "TPM_RC_REFERENCE_H1", 1646ea3dfe1SJarkko Sakkinen 0x012: "TPM_RC_REFERENCE_H2", 1656ea3dfe1SJarkko Sakkinen 0x013: "TPM_RC_REFERENCE_H3", 1666ea3dfe1SJarkko Sakkinen 0x014: "TPM_RC_REFERENCE_H4", 1676ea3dfe1SJarkko Sakkinen 0x015: "TPM_RC_REFERENCE_H5", 1686ea3dfe1SJarkko Sakkinen 0x016: "TPM_RC_REFERENCE_H6", 1696ea3dfe1SJarkko Sakkinen 0x018: "TPM_RC_REFERENCE_S0", 1706ea3dfe1SJarkko Sakkinen 0x019: "TPM_RC_REFERENCE_S1", 1716ea3dfe1SJarkko Sakkinen 0x01A: "TPM_RC_REFERENCE_S2", 1726ea3dfe1SJarkko Sakkinen 0x01B: "TPM_RC_REFERENCE_S3", 1736ea3dfe1SJarkko Sakkinen 0x01C: "TPM_RC_REFERENCE_S4", 1746ea3dfe1SJarkko Sakkinen 0x01D: "TPM_RC_REFERENCE_S5", 1756ea3dfe1SJarkko Sakkinen 0x01E: "TPM_RC_REFERENCE_S6", 1766ea3dfe1SJarkko Sakkinen 0x020: "TPM_RC_NV_RATE", 1776ea3dfe1SJarkko Sakkinen 0x021: "TPM_RC_LOCKOUT", 1786ea3dfe1SJarkko Sakkinen 0x022: "TPM_RC_RETRY", 1796ea3dfe1SJarkko Sakkinen 0x023: "TPM_RC_NV_UNAVAILABLE", 1806ea3dfe1SJarkko Sakkinen 0x7F: "TPM_RC_NOT_USED", 1816ea3dfe1SJarkko Sakkinen} 1826ea3dfe1SJarkko Sakkinen 1836ea3dfe1SJarkko SakkinenRC_VER1 = 0x100 1846ea3dfe1SJarkko SakkinenRC_FMT1 = 0x080 1856ea3dfe1SJarkko SakkinenRC_WARN = 0x900 1866ea3dfe1SJarkko Sakkinen 1876ea3dfe1SJarkko SakkinenALG_DIGEST_SIZE_MAP = { 1886ea3dfe1SJarkko Sakkinen TPM2_ALG_SHA1: SHA1_DIGEST_SIZE, 1896ea3dfe1SJarkko Sakkinen TPM2_ALG_SHA256: SHA256_DIGEST_SIZE, 1906ea3dfe1SJarkko Sakkinen} 1916ea3dfe1SJarkko Sakkinen 1926ea3dfe1SJarkko SakkinenALG_HASH_FUNCTION_MAP = { 1936ea3dfe1SJarkko Sakkinen TPM2_ALG_SHA1: hashlib.sha1, 1946ea3dfe1SJarkko Sakkinen TPM2_ALG_SHA256: hashlib.sha256 1956ea3dfe1SJarkko Sakkinen} 1966ea3dfe1SJarkko Sakkinen 1976ea3dfe1SJarkko SakkinenNAME_ALG_MAP = { 1986ea3dfe1SJarkko Sakkinen "sha1": TPM2_ALG_SHA1, 1996ea3dfe1SJarkko Sakkinen "sha256": TPM2_ALG_SHA256, 2006ea3dfe1SJarkko Sakkinen} 2016ea3dfe1SJarkko Sakkinen 2026ea3dfe1SJarkko Sakkinen 2036ea3dfe1SJarkko Sakkinenclass UnknownAlgorithmIdError(Exception): 2046ea3dfe1SJarkko Sakkinen def __init__(self, alg): 2056ea3dfe1SJarkko Sakkinen self.alg = alg 2066ea3dfe1SJarkko Sakkinen 2076ea3dfe1SJarkko Sakkinen def __str__(self): 2086ea3dfe1SJarkko Sakkinen return '0x%0x' % (alg) 2096ea3dfe1SJarkko Sakkinen 2106ea3dfe1SJarkko Sakkinen 2116ea3dfe1SJarkko Sakkinenclass UnknownAlgorithmNameError(Exception): 2126ea3dfe1SJarkko Sakkinen def __init__(self, name): 2136ea3dfe1SJarkko Sakkinen self.name = name 2146ea3dfe1SJarkko Sakkinen 2156ea3dfe1SJarkko Sakkinen def __str__(self): 2166ea3dfe1SJarkko Sakkinen return name 2176ea3dfe1SJarkko Sakkinen 2186ea3dfe1SJarkko Sakkinen 2196ea3dfe1SJarkko Sakkinenclass UnknownPCRBankError(Exception): 2206ea3dfe1SJarkko Sakkinen def __init__(self, alg): 2216ea3dfe1SJarkko Sakkinen self.alg = alg 2226ea3dfe1SJarkko Sakkinen 2236ea3dfe1SJarkko Sakkinen def __str__(self): 2246ea3dfe1SJarkko Sakkinen return '0x%0x' % (alg) 2256ea3dfe1SJarkko Sakkinen 2266ea3dfe1SJarkko Sakkinen 2276ea3dfe1SJarkko Sakkinenclass ProtocolError(Exception): 2286ea3dfe1SJarkko Sakkinen def __init__(self, cc, rc): 2296ea3dfe1SJarkko Sakkinen self.cc = cc 2306ea3dfe1SJarkko Sakkinen self.rc = rc 2316ea3dfe1SJarkko Sakkinen 2326ea3dfe1SJarkko Sakkinen if (rc & RC_FMT1) == RC_FMT1: 2336ea3dfe1SJarkko Sakkinen self.name = TPM2_FMT1_ERRORS.get(rc & 0x3f, "TPM_RC_UNKNOWN") 2346ea3dfe1SJarkko Sakkinen elif (rc & RC_WARN) == RC_WARN: 2356ea3dfe1SJarkko Sakkinen self.name = TPM2_WARN_ERRORS.get(rc & 0x7f, "TPM_RC_UNKNOWN") 2366ea3dfe1SJarkko Sakkinen elif (rc & RC_VER1) == RC_VER1: 2376ea3dfe1SJarkko Sakkinen self.name = TPM2_VER1_ERRORS.get(rc & 0x7f, "TPM_RC_UNKNOWN") 2386ea3dfe1SJarkko Sakkinen else: 2396ea3dfe1SJarkko Sakkinen self.name = TPM2_VER0_ERRORS.get(rc & 0x7f, "TPM_RC_UNKNOWN") 2406ea3dfe1SJarkko Sakkinen 2416ea3dfe1SJarkko Sakkinen def __str__(self): 2426ea3dfe1SJarkko Sakkinen if self.cc: 2436ea3dfe1SJarkko Sakkinen return '%s: cc=0x%08x, rc=0x%08x' % (self.name, self.cc, self.rc) 2446ea3dfe1SJarkko Sakkinen else: 2456ea3dfe1SJarkko Sakkinen return '%s: rc=0x%08x' % (self.name, self.rc) 2466ea3dfe1SJarkko Sakkinen 2476ea3dfe1SJarkko Sakkinen 2486ea3dfe1SJarkko Sakkinenclass AuthCommand(object): 2496ea3dfe1SJarkko Sakkinen """TPMS_AUTH_COMMAND""" 2506ea3dfe1SJarkko Sakkinen 2510b78c9e8SPengfei Xu def __init__(self, session_handle=TPM2_RS_PW, nonce=bytes(), 2520b78c9e8SPengfei Xu session_attributes=0, hmac=bytes()): 2536ea3dfe1SJarkko Sakkinen self.session_handle = session_handle 2546ea3dfe1SJarkko Sakkinen self.nonce = nonce 2556ea3dfe1SJarkko Sakkinen self.session_attributes = session_attributes 2566ea3dfe1SJarkko Sakkinen self.hmac = hmac 2576ea3dfe1SJarkko Sakkinen 2580b78c9e8SPengfei Xu def __bytes__(self): 2596ea3dfe1SJarkko Sakkinen fmt = '>I H%us B H%us' % (len(self.nonce), len(self.hmac)) 2606ea3dfe1SJarkko Sakkinen return struct.pack(fmt, self.session_handle, len(self.nonce), 2616ea3dfe1SJarkko Sakkinen self.nonce, self.session_attributes, len(self.hmac), 2626ea3dfe1SJarkko Sakkinen self.hmac) 2636ea3dfe1SJarkko Sakkinen 2646ea3dfe1SJarkko Sakkinen def __len__(self): 2656ea3dfe1SJarkko Sakkinen fmt = '>I H%us B H%us' % (len(self.nonce), len(self.hmac)) 2666ea3dfe1SJarkko Sakkinen return struct.calcsize(fmt) 2676ea3dfe1SJarkko Sakkinen 2686ea3dfe1SJarkko Sakkinen 2696ea3dfe1SJarkko Sakkinenclass SensitiveCreate(object): 2706ea3dfe1SJarkko Sakkinen """TPMS_SENSITIVE_CREATE""" 2716ea3dfe1SJarkko Sakkinen 2720b78c9e8SPengfei Xu def __init__(self, user_auth=bytes(), data=bytes()): 2736ea3dfe1SJarkko Sakkinen self.user_auth = user_auth 2746ea3dfe1SJarkko Sakkinen self.data = data 2756ea3dfe1SJarkko Sakkinen 2760b78c9e8SPengfei Xu def __bytes__(self): 2776ea3dfe1SJarkko Sakkinen fmt = '>H%us H%us' % (len(self.user_auth), len(self.data)) 2786ea3dfe1SJarkko Sakkinen return struct.pack(fmt, len(self.user_auth), self.user_auth, 2796ea3dfe1SJarkko Sakkinen len(self.data), self.data) 2806ea3dfe1SJarkko Sakkinen 2816ea3dfe1SJarkko Sakkinen def __len__(self): 2826ea3dfe1SJarkko Sakkinen fmt = '>H%us H%us' % (len(self.user_auth), len(self.data)) 2836ea3dfe1SJarkko Sakkinen return struct.calcsize(fmt) 2846ea3dfe1SJarkko Sakkinen 2856ea3dfe1SJarkko Sakkinen 2866ea3dfe1SJarkko Sakkinenclass Public(object): 2876ea3dfe1SJarkko Sakkinen """TPMT_PUBLIC""" 2886ea3dfe1SJarkko Sakkinen 2896ea3dfe1SJarkko Sakkinen FIXED_TPM = (1 << 1) 2906ea3dfe1SJarkko Sakkinen FIXED_PARENT = (1 << 4) 2916ea3dfe1SJarkko Sakkinen SENSITIVE_DATA_ORIGIN = (1 << 5) 2926ea3dfe1SJarkko Sakkinen USER_WITH_AUTH = (1 << 6) 2936ea3dfe1SJarkko Sakkinen RESTRICTED = (1 << 16) 2946ea3dfe1SJarkko Sakkinen DECRYPT = (1 << 17) 2956ea3dfe1SJarkko Sakkinen 2966ea3dfe1SJarkko Sakkinen def __fmt(self): 2976ea3dfe1SJarkko Sakkinen return '>HHIH%us%usH%us' % \ 2986ea3dfe1SJarkko Sakkinen (len(self.auth_policy), len(self.parameters), len(self.unique)) 2996ea3dfe1SJarkko Sakkinen 3000b78c9e8SPengfei Xu def __init__(self, object_type, name_alg, object_attributes, 3010b78c9e8SPengfei Xu auth_policy=bytes(), parameters=bytes(), 3020b78c9e8SPengfei Xu unique=bytes()): 3036ea3dfe1SJarkko Sakkinen self.object_type = object_type 3046ea3dfe1SJarkko Sakkinen self.name_alg = name_alg 3056ea3dfe1SJarkko Sakkinen self.object_attributes = object_attributes 3066ea3dfe1SJarkko Sakkinen self.auth_policy = auth_policy 3076ea3dfe1SJarkko Sakkinen self.parameters = parameters 3086ea3dfe1SJarkko Sakkinen self.unique = unique 3096ea3dfe1SJarkko Sakkinen 3100b78c9e8SPengfei Xu def __bytes__(self): 3116ea3dfe1SJarkko Sakkinen return struct.pack(self.__fmt(), 3126ea3dfe1SJarkko Sakkinen self.object_type, 3136ea3dfe1SJarkko Sakkinen self.name_alg, 3146ea3dfe1SJarkko Sakkinen self.object_attributes, 3156ea3dfe1SJarkko Sakkinen len(self.auth_policy), 3166ea3dfe1SJarkko Sakkinen self.auth_policy, 3176ea3dfe1SJarkko Sakkinen self.parameters, 3186ea3dfe1SJarkko Sakkinen len(self.unique), 3196ea3dfe1SJarkko Sakkinen self.unique) 3206ea3dfe1SJarkko Sakkinen 3216ea3dfe1SJarkko Sakkinen def __len__(self): 3226ea3dfe1SJarkko Sakkinen return struct.calcsize(self.__fmt()) 3236ea3dfe1SJarkko Sakkinen 3246ea3dfe1SJarkko Sakkinen 3256ea3dfe1SJarkko Sakkinendef get_digest_size(alg): 3266ea3dfe1SJarkko Sakkinen ds = ALG_DIGEST_SIZE_MAP.get(alg) 3276ea3dfe1SJarkko Sakkinen if not ds: 3286ea3dfe1SJarkko Sakkinen raise UnknownAlgorithmIdError(alg) 3296ea3dfe1SJarkko Sakkinen return ds 3306ea3dfe1SJarkko Sakkinen 3316ea3dfe1SJarkko Sakkinen 3326ea3dfe1SJarkko Sakkinendef get_hash_function(alg): 3336ea3dfe1SJarkko Sakkinen f = ALG_HASH_FUNCTION_MAP.get(alg) 3346ea3dfe1SJarkko Sakkinen if not f: 3356ea3dfe1SJarkko Sakkinen raise UnknownAlgorithmIdError(alg) 3366ea3dfe1SJarkko Sakkinen return f 3376ea3dfe1SJarkko Sakkinen 3386ea3dfe1SJarkko Sakkinen 3396ea3dfe1SJarkko Sakkinendef get_algorithm(name): 3406ea3dfe1SJarkko Sakkinen alg = NAME_ALG_MAP.get(name) 3416ea3dfe1SJarkko Sakkinen if not alg: 3426ea3dfe1SJarkko Sakkinen raise UnknownAlgorithmNameError(name) 3436ea3dfe1SJarkko Sakkinen return alg 3446ea3dfe1SJarkko Sakkinen 3456ea3dfe1SJarkko Sakkinen 3466ea3dfe1SJarkko Sakkinendef hex_dump(d): 347*787fccb3STzung-Bi Shih d = [format(x, '02x') for x in d] 3480b78c9e8SPengfei Xu d = [d[i: i + 16] for i in range(0, len(d), 16)] 3496ea3dfe1SJarkko Sakkinen d = [' '.join(x) for x in d] 3506ea3dfe1SJarkko Sakkinen d = os.linesep.join(d) 3516ea3dfe1SJarkko Sakkinen 3526ea3dfe1SJarkko Sakkinen return d 3536ea3dfe1SJarkko Sakkinen 3546ea3dfe1SJarkko Sakkinenclass Client: 3556ea3dfe1SJarkko Sakkinen FLAG_DEBUG = 0x01 3566ea3dfe1SJarkko Sakkinen FLAG_SPACE = 0x02 3578f84bddcSTadeusz Struk FLAG_NONBLOCK = 0x04 3586ea3dfe1SJarkko Sakkinen TPM_IOC_NEW_SPACE = 0xa200 3596ea3dfe1SJarkko Sakkinen 3606ea3dfe1SJarkko Sakkinen def __init__(self, flags = 0): 3616ea3dfe1SJarkko Sakkinen self.flags = flags 3626ea3dfe1SJarkko Sakkinen 3636ea3dfe1SJarkko Sakkinen if (self.flags & Client.FLAG_SPACE) == 0: 3646da70580STadeusz Struk self.tpm = open('/dev/tpm0', 'r+b', buffering=0) 3656ea3dfe1SJarkko Sakkinen else: 3666da70580STadeusz Struk self.tpm = open('/dev/tpmrm0', 'r+b', buffering=0) 3676ea3dfe1SJarkko Sakkinen 3688f84bddcSTadeusz Struk if (self.flags & Client.FLAG_NONBLOCK): 3698f84bddcSTadeusz Struk flags = fcntl.fcntl(self.tpm, fcntl.F_GETFL) 3708f84bddcSTadeusz Struk flags |= os.O_NONBLOCK 3718f84bddcSTadeusz Struk fcntl.fcntl(self.tpm, fcntl.F_SETFL, flags) 3728f84bddcSTadeusz Struk self.tpm_poll = select.poll() 3738f84bddcSTadeusz Struk 3742d869f0bSStefan Berger def __del__(self): 3752d869f0bSStefan Berger if self.tpm: 3762d869f0bSStefan Berger self.tpm.close() 3772d869f0bSStefan Berger 3786ea3dfe1SJarkko Sakkinen def close(self): 3796ea3dfe1SJarkko Sakkinen self.tpm.close() 3806ea3dfe1SJarkko Sakkinen 3816ea3dfe1SJarkko Sakkinen def send_cmd(self, cmd): 3826ea3dfe1SJarkko Sakkinen self.tpm.write(cmd) 3838f84bddcSTadeusz Struk 3848f84bddcSTadeusz Struk if (self.flags & Client.FLAG_NONBLOCK): 3858f84bddcSTadeusz Struk self.tpm_poll.register(self.tpm, select.POLLIN) 3868f84bddcSTadeusz Struk self.tpm_poll.poll(10000) 3878f84bddcSTadeusz Struk 3886ea3dfe1SJarkko Sakkinen rsp = self.tpm.read() 3896ea3dfe1SJarkko Sakkinen 3908f84bddcSTadeusz Struk if (self.flags & Client.FLAG_NONBLOCK): 3918f84bddcSTadeusz Struk self.tpm_poll.unregister(self.tpm) 3928f84bddcSTadeusz Struk 3936ea3dfe1SJarkko Sakkinen if (self.flags & Client.FLAG_DEBUG) != 0: 3946ea3dfe1SJarkko Sakkinen sys.stderr.write('cmd' + os.linesep) 3956ea3dfe1SJarkko Sakkinen sys.stderr.write(hex_dump(cmd) + os.linesep) 3966ea3dfe1SJarkko Sakkinen sys.stderr.write('rsp' + os.linesep) 3976ea3dfe1SJarkko Sakkinen sys.stderr.write(hex_dump(rsp) + os.linesep) 3986ea3dfe1SJarkko Sakkinen 3996ea3dfe1SJarkko Sakkinen rc = struct.unpack('>I', rsp[6:10])[0] 4006ea3dfe1SJarkko Sakkinen if rc != 0: 4016ea3dfe1SJarkko Sakkinen cc = struct.unpack('>I', cmd[6:10])[0] 4026ea3dfe1SJarkko Sakkinen raise ProtocolError(cc, rc) 4036ea3dfe1SJarkko Sakkinen 4046ea3dfe1SJarkko Sakkinen return rsp 4056ea3dfe1SJarkko Sakkinen 4066ea3dfe1SJarkko Sakkinen def read_pcr(self, i, bank_alg = TPM2_ALG_SHA1): 4076ea3dfe1SJarkko Sakkinen pcrsel_len = max((i >> 3) + 1, 3) 4086ea3dfe1SJarkko Sakkinen pcrsel = [0] * pcrsel_len 4096ea3dfe1SJarkko Sakkinen pcrsel[i >> 3] = 1 << (i & 7) 4100b78c9e8SPengfei Xu pcrsel = ''.join(map(chr, pcrsel)).encode() 4116ea3dfe1SJarkko Sakkinen 4126ea3dfe1SJarkko Sakkinen fmt = '>HII IHB%us' % (pcrsel_len) 4136ea3dfe1SJarkko Sakkinen cmd = struct.pack(fmt, 4146ea3dfe1SJarkko Sakkinen TPM2_ST_NO_SESSIONS, 4156ea3dfe1SJarkko Sakkinen struct.calcsize(fmt), 4166ea3dfe1SJarkko Sakkinen TPM2_CC_PCR_READ, 4176ea3dfe1SJarkko Sakkinen 1, 4186ea3dfe1SJarkko Sakkinen bank_alg, 4196ea3dfe1SJarkko Sakkinen pcrsel_len, pcrsel) 4206ea3dfe1SJarkko Sakkinen 4216ea3dfe1SJarkko Sakkinen rsp = self.send_cmd(cmd) 4226ea3dfe1SJarkko Sakkinen 4236ea3dfe1SJarkko Sakkinen pcr_update_cnt, pcr_select_cnt = struct.unpack('>II', rsp[10:18]) 4246ea3dfe1SJarkko Sakkinen assert pcr_select_cnt == 1 4256ea3dfe1SJarkko Sakkinen rsp = rsp[18:] 4266ea3dfe1SJarkko Sakkinen 4276ea3dfe1SJarkko Sakkinen alg2, pcrsel_len2 = struct.unpack('>HB', rsp[:3]) 4286ea3dfe1SJarkko Sakkinen assert bank_alg == alg2 and pcrsel_len == pcrsel_len2 4296ea3dfe1SJarkko Sakkinen rsp = rsp[3 + pcrsel_len:] 4306ea3dfe1SJarkko Sakkinen 4316ea3dfe1SJarkko Sakkinen digest_cnt = struct.unpack('>I', rsp[:4])[0] 4326ea3dfe1SJarkko Sakkinen if digest_cnt == 0: 4336ea3dfe1SJarkko Sakkinen return None 4346ea3dfe1SJarkko Sakkinen rsp = rsp[6:] 4356ea3dfe1SJarkko Sakkinen 4366ea3dfe1SJarkko Sakkinen return rsp 4376ea3dfe1SJarkko Sakkinen 4386ea3dfe1SJarkko Sakkinen def extend_pcr(self, i, dig, bank_alg = TPM2_ALG_SHA1): 4396ea3dfe1SJarkko Sakkinen ds = get_digest_size(bank_alg) 4406ea3dfe1SJarkko Sakkinen assert(ds == len(dig)) 4416ea3dfe1SJarkko Sakkinen 4426ea3dfe1SJarkko Sakkinen auth_cmd = AuthCommand() 4436ea3dfe1SJarkko Sakkinen 4446ea3dfe1SJarkko Sakkinen fmt = '>HII I I%us IH%us' % (len(auth_cmd), ds) 4456ea3dfe1SJarkko Sakkinen cmd = struct.pack( 4466ea3dfe1SJarkko Sakkinen fmt, 4476ea3dfe1SJarkko Sakkinen TPM2_ST_SESSIONS, 4486ea3dfe1SJarkko Sakkinen struct.calcsize(fmt), 4496ea3dfe1SJarkko Sakkinen TPM2_CC_PCR_EXTEND, 4506ea3dfe1SJarkko Sakkinen i, 4516ea3dfe1SJarkko Sakkinen len(auth_cmd), 4520b78c9e8SPengfei Xu bytes(auth_cmd), 4536ea3dfe1SJarkko Sakkinen 1, bank_alg, dig) 4546ea3dfe1SJarkko Sakkinen 4556ea3dfe1SJarkko Sakkinen self.send_cmd(cmd) 4566ea3dfe1SJarkko Sakkinen 4576ea3dfe1SJarkko Sakkinen def start_auth_session(self, session_type, name_alg = TPM2_ALG_SHA1): 4586ea3dfe1SJarkko Sakkinen fmt = '>HII IIH16sHBHH' 4596ea3dfe1SJarkko Sakkinen cmd = struct.pack(fmt, 4606ea3dfe1SJarkko Sakkinen TPM2_ST_NO_SESSIONS, 4616ea3dfe1SJarkko Sakkinen struct.calcsize(fmt), 4626ea3dfe1SJarkko Sakkinen TPM2_CC_START_AUTH_SESSION, 4636ea3dfe1SJarkko Sakkinen TPM2_RH_NULL, 4646ea3dfe1SJarkko Sakkinen TPM2_RH_NULL, 4656ea3dfe1SJarkko Sakkinen 16, 4660b78c9e8SPengfei Xu ('\0' * 16).encode(), 4676ea3dfe1SJarkko Sakkinen 0, 4686ea3dfe1SJarkko Sakkinen session_type, 4696ea3dfe1SJarkko Sakkinen TPM2_ALG_NULL, 4706ea3dfe1SJarkko Sakkinen name_alg) 4716ea3dfe1SJarkko Sakkinen 4726ea3dfe1SJarkko Sakkinen return struct.unpack('>I', self.send_cmd(cmd)[10:14])[0] 4736ea3dfe1SJarkko Sakkinen 4746ea3dfe1SJarkko Sakkinen def __calc_pcr_digest(self, pcrs, bank_alg = TPM2_ALG_SHA1, 4756ea3dfe1SJarkko Sakkinen digest_alg = TPM2_ALG_SHA1): 4766ea3dfe1SJarkko Sakkinen x = [] 4776ea3dfe1SJarkko Sakkinen f = get_hash_function(digest_alg) 4786ea3dfe1SJarkko Sakkinen 4796ea3dfe1SJarkko Sakkinen for i in pcrs: 4806ea3dfe1SJarkko Sakkinen pcr = self.read_pcr(i, bank_alg) 4810b78c9e8SPengfei Xu if pcr is None: 4826ea3dfe1SJarkko Sakkinen return None 4836ea3dfe1SJarkko Sakkinen x += pcr 4846ea3dfe1SJarkko Sakkinen 4856ea3dfe1SJarkko Sakkinen return f(bytearray(x)).digest() 4866ea3dfe1SJarkko Sakkinen 4876ea3dfe1SJarkko Sakkinen def policy_pcr(self, handle, pcrs, bank_alg = TPM2_ALG_SHA1, 4886ea3dfe1SJarkko Sakkinen name_alg = TPM2_ALG_SHA1): 4896ea3dfe1SJarkko Sakkinen ds = get_digest_size(name_alg) 4906ea3dfe1SJarkko Sakkinen dig = self.__calc_pcr_digest(pcrs, bank_alg, name_alg) 4916ea3dfe1SJarkko Sakkinen if not dig: 4926ea3dfe1SJarkko Sakkinen raise UnknownPCRBankError(bank_alg) 4936ea3dfe1SJarkko Sakkinen 4946ea3dfe1SJarkko Sakkinen pcrsel_len = max((max(pcrs) >> 3) + 1, 3) 4956ea3dfe1SJarkko Sakkinen pcrsel = [0] * pcrsel_len 4966ea3dfe1SJarkko Sakkinen for i in pcrs: 4976ea3dfe1SJarkko Sakkinen pcrsel[i >> 3] |= 1 << (i & 7) 4980b78c9e8SPengfei Xu pcrsel = ''.join(map(chr, pcrsel)).encode() 4996ea3dfe1SJarkko Sakkinen 5006ea3dfe1SJarkko Sakkinen fmt = '>HII IH%usIHB3s' % ds 5016ea3dfe1SJarkko Sakkinen cmd = struct.pack(fmt, 5026ea3dfe1SJarkko Sakkinen TPM2_ST_NO_SESSIONS, 5036ea3dfe1SJarkko Sakkinen struct.calcsize(fmt), 5046ea3dfe1SJarkko Sakkinen TPM2_CC_POLICY_PCR, 5056ea3dfe1SJarkko Sakkinen handle, 5060b78c9e8SPengfei Xu len(dig), 5070b78c9e8SPengfei Xu bytes(dig), 5086ea3dfe1SJarkko Sakkinen 1, 5096ea3dfe1SJarkko Sakkinen bank_alg, 5106ea3dfe1SJarkko Sakkinen pcrsel_len, pcrsel) 5116ea3dfe1SJarkko Sakkinen 5126ea3dfe1SJarkko Sakkinen self.send_cmd(cmd) 5136ea3dfe1SJarkko Sakkinen 5146ea3dfe1SJarkko Sakkinen def policy_password(self, handle): 5156ea3dfe1SJarkko Sakkinen fmt = '>HII I' 5166ea3dfe1SJarkko Sakkinen cmd = struct.pack(fmt, 5176ea3dfe1SJarkko Sakkinen TPM2_ST_NO_SESSIONS, 5186ea3dfe1SJarkko Sakkinen struct.calcsize(fmt), 5196ea3dfe1SJarkko Sakkinen TPM2_CC_POLICY_PASSWORD, 5206ea3dfe1SJarkko Sakkinen handle) 5216ea3dfe1SJarkko Sakkinen 5226ea3dfe1SJarkko Sakkinen self.send_cmd(cmd) 5236ea3dfe1SJarkko Sakkinen 5246ea3dfe1SJarkko Sakkinen def get_policy_digest(self, handle): 5256ea3dfe1SJarkko Sakkinen fmt = '>HII I' 5266ea3dfe1SJarkko Sakkinen cmd = struct.pack(fmt, 5276ea3dfe1SJarkko Sakkinen TPM2_ST_NO_SESSIONS, 5286ea3dfe1SJarkko Sakkinen struct.calcsize(fmt), 5296ea3dfe1SJarkko Sakkinen TPM2_CC_POLICY_GET_DIGEST, 5306ea3dfe1SJarkko Sakkinen handle) 5316ea3dfe1SJarkko Sakkinen 5326ea3dfe1SJarkko Sakkinen return self.send_cmd(cmd)[12:] 5336ea3dfe1SJarkko Sakkinen 5346ea3dfe1SJarkko Sakkinen def flush_context(self, handle): 5356ea3dfe1SJarkko Sakkinen fmt = '>HIII' 5366ea3dfe1SJarkko Sakkinen cmd = struct.pack(fmt, 5376ea3dfe1SJarkko Sakkinen TPM2_ST_NO_SESSIONS, 5386ea3dfe1SJarkko Sakkinen struct.calcsize(fmt), 5396ea3dfe1SJarkko Sakkinen TPM2_CC_FLUSH_CONTEXT, 5406ea3dfe1SJarkko Sakkinen handle) 5416ea3dfe1SJarkko Sakkinen 5426ea3dfe1SJarkko Sakkinen self.send_cmd(cmd) 5436ea3dfe1SJarkko Sakkinen 5440b78c9e8SPengfei Xu def create_root_key(self, auth_value = bytes()): 5456ea3dfe1SJarkko Sakkinen attributes = \ 5466ea3dfe1SJarkko Sakkinen Public.FIXED_TPM | \ 5476ea3dfe1SJarkko Sakkinen Public.FIXED_PARENT | \ 5486ea3dfe1SJarkko Sakkinen Public.SENSITIVE_DATA_ORIGIN | \ 5496ea3dfe1SJarkko Sakkinen Public.USER_WITH_AUTH | \ 5506ea3dfe1SJarkko Sakkinen Public.RESTRICTED | \ 5516ea3dfe1SJarkko Sakkinen Public.DECRYPT 5526ea3dfe1SJarkko Sakkinen 5536ea3dfe1SJarkko Sakkinen auth_cmd = AuthCommand() 5546ea3dfe1SJarkko Sakkinen sensitive = SensitiveCreate(user_auth=auth_value) 5556ea3dfe1SJarkko Sakkinen 5566ea3dfe1SJarkko Sakkinen public_parms = struct.pack( 5576ea3dfe1SJarkko Sakkinen '>HHHHHI', 5586ea3dfe1SJarkko Sakkinen TPM2_ALG_AES, 5596ea3dfe1SJarkko Sakkinen 128, 5606ea3dfe1SJarkko Sakkinen TPM2_ALG_CFB, 5616ea3dfe1SJarkko Sakkinen TPM2_ALG_NULL, 5626ea3dfe1SJarkko Sakkinen 2048, 5636ea3dfe1SJarkko Sakkinen 0) 5646ea3dfe1SJarkko Sakkinen 5656ea3dfe1SJarkko Sakkinen public = Public( 5666ea3dfe1SJarkko Sakkinen object_type=TPM2_ALG_RSA, 5676ea3dfe1SJarkko Sakkinen name_alg=TPM2_ALG_SHA1, 5686ea3dfe1SJarkko Sakkinen object_attributes=attributes, 5696ea3dfe1SJarkko Sakkinen parameters=public_parms) 5706ea3dfe1SJarkko Sakkinen 5716ea3dfe1SJarkko Sakkinen fmt = '>HIII I%us H%us H%us HI' % \ 5726ea3dfe1SJarkko Sakkinen (len(auth_cmd), len(sensitive), len(public)) 5736ea3dfe1SJarkko Sakkinen cmd = struct.pack( 5746ea3dfe1SJarkko Sakkinen fmt, 5756ea3dfe1SJarkko Sakkinen TPM2_ST_SESSIONS, 5766ea3dfe1SJarkko Sakkinen struct.calcsize(fmt), 5776ea3dfe1SJarkko Sakkinen TPM2_CC_CREATE_PRIMARY, 5786ea3dfe1SJarkko Sakkinen TPM2_RH_OWNER, 5796ea3dfe1SJarkko Sakkinen len(auth_cmd), 5800b78c9e8SPengfei Xu bytes(auth_cmd), 5816ea3dfe1SJarkko Sakkinen len(sensitive), 5820b78c9e8SPengfei Xu bytes(sensitive), 5836ea3dfe1SJarkko Sakkinen len(public), 5840b78c9e8SPengfei Xu bytes(public), 5856ea3dfe1SJarkko Sakkinen 0, 0) 5866ea3dfe1SJarkko Sakkinen 5876ea3dfe1SJarkko Sakkinen return struct.unpack('>I', self.send_cmd(cmd)[10:14])[0] 5886ea3dfe1SJarkko Sakkinen 5896ea3dfe1SJarkko Sakkinen def seal(self, parent_key, data, auth_value, policy_dig, 5906ea3dfe1SJarkko Sakkinen name_alg = TPM2_ALG_SHA1): 5916ea3dfe1SJarkko Sakkinen ds = get_digest_size(name_alg) 5926ea3dfe1SJarkko Sakkinen assert(not policy_dig or ds == len(policy_dig)) 5936ea3dfe1SJarkko Sakkinen 5946ea3dfe1SJarkko Sakkinen attributes = 0 5956ea3dfe1SJarkko Sakkinen if not policy_dig: 5966ea3dfe1SJarkko Sakkinen attributes |= Public.USER_WITH_AUTH 5970b78c9e8SPengfei Xu policy_dig = bytes() 5986ea3dfe1SJarkko Sakkinen 5996ea3dfe1SJarkko Sakkinen auth_cmd = AuthCommand() 6006ea3dfe1SJarkko Sakkinen sensitive = SensitiveCreate(user_auth=auth_value, data=data) 6016ea3dfe1SJarkko Sakkinen 6026ea3dfe1SJarkko Sakkinen public = Public( 6036ea3dfe1SJarkko Sakkinen object_type=TPM2_ALG_KEYEDHASH, 6046ea3dfe1SJarkko Sakkinen name_alg=name_alg, 6056ea3dfe1SJarkko Sakkinen object_attributes=attributes, 6066ea3dfe1SJarkko Sakkinen auth_policy=policy_dig, 6076ea3dfe1SJarkko Sakkinen parameters=struct.pack('>H', TPM2_ALG_NULL)) 6086ea3dfe1SJarkko Sakkinen 6096ea3dfe1SJarkko Sakkinen fmt = '>HIII I%us H%us H%us HI' % \ 6106ea3dfe1SJarkko Sakkinen (len(auth_cmd), len(sensitive), len(public)) 6116ea3dfe1SJarkko Sakkinen cmd = struct.pack( 6126ea3dfe1SJarkko Sakkinen fmt, 6136ea3dfe1SJarkko Sakkinen TPM2_ST_SESSIONS, 6146ea3dfe1SJarkko Sakkinen struct.calcsize(fmt), 6156ea3dfe1SJarkko Sakkinen TPM2_CC_CREATE, 6166ea3dfe1SJarkko Sakkinen parent_key, 6176ea3dfe1SJarkko Sakkinen len(auth_cmd), 6180b78c9e8SPengfei Xu bytes(auth_cmd), 6196ea3dfe1SJarkko Sakkinen len(sensitive), 6200b78c9e8SPengfei Xu bytes(sensitive), 6216ea3dfe1SJarkko Sakkinen len(public), 6220b78c9e8SPengfei Xu bytes(public), 6236ea3dfe1SJarkko Sakkinen 0, 0) 6246ea3dfe1SJarkko Sakkinen 6256ea3dfe1SJarkko Sakkinen rsp = self.send_cmd(cmd) 6266ea3dfe1SJarkko Sakkinen 6276ea3dfe1SJarkko Sakkinen return rsp[14:] 6286ea3dfe1SJarkko Sakkinen 6296ea3dfe1SJarkko Sakkinen def unseal(self, parent_key, blob, auth_value, policy_handle): 6306ea3dfe1SJarkko Sakkinen private_len = struct.unpack('>H', blob[0:2])[0] 6316ea3dfe1SJarkko Sakkinen public_start = private_len + 2 6326ea3dfe1SJarkko Sakkinen public_len = struct.unpack('>H', blob[public_start:public_start + 2])[0] 6336ea3dfe1SJarkko Sakkinen blob = blob[:private_len + public_len + 4] 6346ea3dfe1SJarkko Sakkinen 6356ea3dfe1SJarkko Sakkinen auth_cmd = AuthCommand() 6366ea3dfe1SJarkko Sakkinen 6376ea3dfe1SJarkko Sakkinen fmt = '>HII I I%us %us' % (len(auth_cmd), len(blob)) 6386ea3dfe1SJarkko Sakkinen cmd = struct.pack( 6396ea3dfe1SJarkko Sakkinen fmt, 6406ea3dfe1SJarkko Sakkinen TPM2_ST_SESSIONS, 6416ea3dfe1SJarkko Sakkinen struct.calcsize(fmt), 6426ea3dfe1SJarkko Sakkinen TPM2_CC_LOAD, 6436ea3dfe1SJarkko Sakkinen parent_key, 6446ea3dfe1SJarkko Sakkinen len(auth_cmd), 6450b78c9e8SPengfei Xu bytes(auth_cmd), 6466ea3dfe1SJarkko Sakkinen blob) 6476ea3dfe1SJarkko Sakkinen 6486ea3dfe1SJarkko Sakkinen data_handle = struct.unpack('>I', self.send_cmd(cmd)[10:14])[0] 6496ea3dfe1SJarkko Sakkinen 6506ea3dfe1SJarkko Sakkinen if policy_handle: 6516ea3dfe1SJarkko Sakkinen auth_cmd = AuthCommand(session_handle=policy_handle, hmac=auth_value) 6526ea3dfe1SJarkko Sakkinen else: 6536ea3dfe1SJarkko Sakkinen auth_cmd = AuthCommand(hmac=auth_value) 6546ea3dfe1SJarkko Sakkinen 6556ea3dfe1SJarkko Sakkinen fmt = '>HII I I%us' % (len(auth_cmd)) 6566ea3dfe1SJarkko Sakkinen cmd = struct.pack( 6576ea3dfe1SJarkko Sakkinen fmt, 6586ea3dfe1SJarkko Sakkinen TPM2_ST_SESSIONS, 6596ea3dfe1SJarkko Sakkinen struct.calcsize(fmt), 6606ea3dfe1SJarkko Sakkinen TPM2_CC_UNSEAL, 6616ea3dfe1SJarkko Sakkinen data_handle, 6626ea3dfe1SJarkko Sakkinen len(auth_cmd), 6630b78c9e8SPengfei Xu bytes(auth_cmd)) 6646ea3dfe1SJarkko Sakkinen 6656ea3dfe1SJarkko Sakkinen try: 6666ea3dfe1SJarkko Sakkinen rsp = self.send_cmd(cmd) 6676ea3dfe1SJarkko Sakkinen finally: 6686ea3dfe1SJarkko Sakkinen self.flush_context(data_handle) 6696ea3dfe1SJarkko Sakkinen 6706ea3dfe1SJarkko Sakkinen data_len = struct.unpack('>I', rsp[10:14])[0] - 2 6716ea3dfe1SJarkko Sakkinen 6726ea3dfe1SJarkko Sakkinen return rsp[16:16 + data_len] 6736ea3dfe1SJarkko Sakkinen 6746ea3dfe1SJarkko Sakkinen def reset_da_lock(self): 6756ea3dfe1SJarkko Sakkinen auth_cmd = AuthCommand() 6766ea3dfe1SJarkko Sakkinen 6776ea3dfe1SJarkko Sakkinen fmt = '>HII I I%us' % (len(auth_cmd)) 6786ea3dfe1SJarkko Sakkinen cmd = struct.pack( 6796ea3dfe1SJarkko Sakkinen fmt, 6806ea3dfe1SJarkko Sakkinen TPM2_ST_SESSIONS, 6816ea3dfe1SJarkko Sakkinen struct.calcsize(fmt), 6826ea3dfe1SJarkko Sakkinen TPM2_CC_DICTIONARY_ATTACK_LOCK_RESET, 6836ea3dfe1SJarkko Sakkinen TPM2_RH_LOCKOUT, 6846ea3dfe1SJarkko Sakkinen len(auth_cmd), 6850b78c9e8SPengfei Xu bytes(auth_cmd)) 6866ea3dfe1SJarkko Sakkinen 6876ea3dfe1SJarkko Sakkinen self.send_cmd(cmd) 6886ea3dfe1SJarkko Sakkinen 6896ea3dfe1SJarkko Sakkinen def __get_cap_cnt(self, cap, pt, cnt): 6906ea3dfe1SJarkko Sakkinen handles = [] 6916ea3dfe1SJarkko Sakkinen fmt = '>HII III' 6926ea3dfe1SJarkko Sakkinen 6936ea3dfe1SJarkko Sakkinen cmd = struct.pack(fmt, 6946ea3dfe1SJarkko Sakkinen TPM2_ST_NO_SESSIONS, 6956ea3dfe1SJarkko Sakkinen struct.calcsize(fmt), 6966ea3dfe1SJarkko Sakkinen TPM2_CC_GET_CAPABILITY, 6976ea3dfe1SJarkko Sakkinen cap, pt, cnt) 6986ea3dfe1SJarkko Sakkinen 6996ea3dfe1SJarkko Sakkinen rsp = self.send_cmd(cmd)[10:] 7006ea3dfe1SJarkko Sakkinen more_data, cap, cnt = struct.unpack('>BII', rsp[:9]) 7016ea3dfe1SJarkko Sakkinen rsp = rsp[9:] 7026ea3dfe1SJarkko Sakkinen 7030b78c9e8SPengfei Xu for i in range(0, cnt): 7046ea3dfe1SJarkko Sakkinen handle = struct.unpack('>I', rsp[:4])[0] 7056ea3dfe1SJarkko Sakkinen handles.append(handle) 7066ea3dfe1SJarkko Sakkinen rsp = rsp[4:] 7076ea3dfe1SJarkko Sakkinen 7086ea3dfe1SJarkko Sakkinen return handles, more_data 7096ea3dfe1SJarkko Sakkinen 7106ea3dfe1SJarkko Sakkinen def get_cap(self, cap, pt): 7116ea3dfe1SJarkko Sakkinen handles = [] 7126ea3dfe1SJarkko Sakkinen 7136ea3dfe1SJarkko Sakkinen more_data = True 7146ea3dfe1SJarkko Sakkinen while more_data: 7156ea3dfe1SJarkko Sakkinen next_handles, more_data = self.__get_cap_cnt(cap, pt, 1) 7166ea3dfe1SJarkko Sakkinen handles += next_handles 7176ea3dfe1SJarkko Sakkinen pt += 1 7186ea3dfe1SJarkko Sakkinen 7196ea3dfe1SJarkko Sakkinen return handles 7200d060f23SStefan Berger 7210d060f23SStefan Berger def get_cap_pcrs(self): 7220d060f23SStefan Berger pcr_banks = {} 7230d060f23SStefan Berger 7240d060f23SStefan Berger fmt = '>HII III' 7250d060f23SStefan Berger 7260d060f23SStefan Berger cmd = struct.pack(fmt, 7270d060f23SStefan Berger TPM2_ST_NO_SESSIONS, 7280d060f23SStefan Berger struct.calcsize(fmt), 7290d060f23SStefan Berger TPM2_CC_GET_CAPABILITY, 7300d060f23SStefan Berger TPM2_CAP_PCRS, 0, 1) 7310d060f23SStefan Berger rsp = self.send_cmd(cmd)[10:] 7320d060f23SStefan Berger _, _, cnt = struct.unpack('>BII', rsp[:9]) 7330d060f23SStefan Berger rsp = rsp[9:] 7340d060f23SStefan Berger 7350d060f23SStefan Berger # items are TPMS_PCR_SELECTION's 7360d060f23SStefan Berger for i in range(0, cnt): 7370d060f23SStefan Berger hash, sizeOfSelect = struct.unpack('>HB', rsp[:3]) 7380d060f23SStefan Berger rsp = rsp[3:] 7390d060f23SStefan Berger 7400d060f23SStefan Berger pcrSelect = 0 7410d060f23SStefan Berger if sizeOfSelect > 0: 7420d060f23SStefan Berger pcrSelect, = struct.unpack('%ds' % sizeOfSelect, 7430d060f23SStefan Berger rsp[:sizeOfSelect]) 7440d060f23SStefan Berger rsp = rsp[sizeOfSelect:] 7450d060f23SStefan Berger pcrSelect = int.from_bytes(pcrSelect, byteorder='big') 7460d060f23SStefan Berger 7470d060f23SStefan Berger pcr_banks[hash] = pcrSelect 7480d060f23SStefan Berger 7490d060f23SStefan Berger return pcr_banks 750