xref: /openbmc/linux/tools/testing/selftests/tpm2/tpm2.py (revision 9a87ffc99ec8eb8d35eed7c4f816d75f5cc9662e)
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