mirror of
https://github.com/trezor/trezor-firmware.git
synced 2025-01-14 17:31:04 +00:00
apps.fido_u2f: small changes in msg_register
This commit is contained in:
parent
cd7ee79c67
commit
479ff127fc
@ -347,45 +347,46 @@ _register_state = 0
|
|||||||
async def msg_register(req: Msg) -> Cmd:
|
async def msg_register(req: Msg) -> Cmd:
|
||||||
global _register_state
|
global _register_state
|
||||||
|
|
||||||
if len(req.data) != 64:
|
|
||||||
return msg_error(req, _SW_WRONG_LENGTH)
|
|
||||||
|
|
||||||
chal = req.data[:32]
|
|
||||||
app_id = req.data[32:]
|
|
||||||
|
|
||||||
from apps.common import storage
|
from apps.common import storage
|
||||||
|
|
||||||
if not storage.is_initialized():
|
if not storage.is_initialized():
|
||||||
return msg_error(req, _SW_CONDITIONS_NOT_SATISFIED)
|
return msg_error(req, _SW_CONDITIONS_NOT_SATISFIED)
|
||||||
|
|
||||||
|
# check input data
|
||||||
|
if len(req.data) != 64:
|
||||||
|
return msg_error(req, _SW_WRONG_LENGTH)
|
||||||
|
|
||||||
|
# TODO: wait for a button press
|
||||||
if _register_state == 0:
|
if _register_state == 0:
|
||||||
_register_state = utime.ticks_ms()
|
_register_state = utime.ticks_ms()
|
||||||
if utime.ticks_ms() - _register_state < 2000:
|
if utime.ticks_ms() - _register_state < 500:
|
||||||
return msg_error(req, _SW_CONDITIONS_NOT_SATISFIED)
|
return msg_error(req, _SW_CONDITIONS_NOT_SATISFIED)
|
||||||
|
|
||||||
_register_state = 0
|
_register_state = 0
|
||||||
buf = msg_register_sign(chal, app_id, _U2F_ATT_CERT)
|
|
||||||
|
chal = req.data[:32]
|
||||||
|
app_id = req.data[32:]
|
||||||
|
buf = msg_register_sign(chal, app_id)
|
||||||
|
|
||||||
return Cmd(req.cid, _CMD_MSG, buf)
|
return Cmd(req.cid, _CMD_MSG, buf)
|
||||||
|
|
||||||
|
|
||||||
def msg_register_sign(challenge: bytes, app_id: bytes, cert: bytes) -> bytes:
|
def msg_register_sign(challenge: bytes, app_id: bytes) -> bytes:
|
||||||
|
|
||||||
from apps.common import seed
|
from apps.common import seed
|
||||||
|
|
||||||
# derivation path is m/U2F'/r'/r'/r'/r'/r'/r'/r'/r'
|
# derivation path is m/U2F'/r'/r'/r'/r'/r'/r'/r'/r'
|
||||||
key_path = [0x80000000 | random.uniform(0xf0000000) for _ in range(0, 8)]
|
keypath = [0x80000000 | random.uniform(0xf0000000) for _ in range(0, 8)]
|
||||||
node_path = [_U2F_KEY_PATH] + key_path
|
nodepath = [_U2F_KEY_PATH] + keypath
|
||||||
|
|
||||||
# prepare signing key from random path, compute decompressed public key
|
# prepare signing key from random path, compute decompressed public key
|
||||||
node = seed.get_root_without_passphrase('nist256p1')
|
node = seed.get_root_without_passphrase('nist256p1')
|
||||||
node.derive_path(node_path)
|
node.derive_path(nodepath)
|
||||||
pubkey = nist256p1.publickey(node.private_key(), False)
|
pubkey = nist256p1.publickey(node.private_key(), False)
|
||||||
|
|
||||||
# first half of keyhandle is key_path
|
# first half of keyhandle is keypath
|
||||||
keybuf = ustruct.pack('>8L', *key_path)
|
keybuf = ustruct.pack('>8L', *keypath)
|
||||||
|
|
||||||
# second half of keyhandle is a hmac of app_id and key_path
|
# second half of keyhandle is a hmac of app_id and keypath
|
||||||
keybase = hmac.Hmac(node.private_key(), app_id, hashlib.sha256)
|
keybase = hmac.Hmac(node.private_key(), app_id, hashlib.sha256)
|
||||||
keybase.update(keybuf)
|
keybase.update(keybuf)
|
||||||
keybase = keybase.digest()
|
keybase = keybase.digest()
|
||||||
@ -393,11 +394,11 @@ def msg_register_sign(challenge: bytes, app_id: bytes, cert: bytes) -> bytes:
|
|||||||
# hash the request data together with keyhandle and pubkey
|
# hash the request data together with keyhandle and pubkey
|
||||||
dig = hashlib.sha256()
|
dig = hashlib.sha256()
|
||||||
dig.update(b'\x00') # uint8_t reserved;
|
dig.update(b'\x00') # uint8_t reserved;
|
||||||
dig.update(app_id) # uint8_t appId[U2F_APPID_SIZE];
|
dig.update(app_id) # uint8_t appId[32];
|
||||||
dig.update(challenge) # uint8_t chal[U2F_CHAL_SIZE];
|
dig.update(challenge) # uint8_t chal[32];
|
||||||
dig.update(keybuf) # uint8_t keyHandle[KEY_HANDLE_LEN];
|
dig.update(keybuf) # uint8_t keyHandle[64];
|
||||||
dig.update(keybase)
|
dig.update(keybase)
|
||||||
dig.update(pubkey) # uint8_t pubKey[U2F_PUBKEY_LEN];
|
dig.update(pubkey) # uint8_t pubKey[65];
|
||||||
dig = dig.digest()
|
dig = dig.digest()
|
||||||
|
|
||||||
# sign the digest and convert to der
|
# sign the digest and convert to der
|
||||||
@ -405,15 +406,16 @@ def msg_register_sign(challenge: bytes, app_id: bytes, cert: bytes) -> bytes:
|
|||||||
sig = der.encode_seq((sig[1:33], sig[33:]))
|
sig = der.encode_seq((sig[1:33], sig[33:]))
|
||||||
|
|
||||||
# pack to a response
|
# pack to a response
|
||||||
buf, resp = make_struct(resp_cmd_register(len(keybuf) + len(keybase), len(cert), len(sig)))
|
buf, resp = make_struct(resp_cmd_register(
|
||||||
|
len(keybuf) + len(keybase), len(_U2F_ATT_CERT), len(sig)))
|
||||||
resp.registerId = _U2F_REGISTER_ID
|
resp.registerId = _U2F_REGISTER_ID
|
||||||
resp.status = _SW_NO_ERROR
|
|
||||||
resp.keyHandleLen = len(keybuf) + len(keybase)
|
|
||||||
utils.memcpy(resp.cert, 0, cert, 0, len(cert))
|
|
||||||
utils.memcpy(resp.pubKey, 0, pubkey, 0, len(pubkey))
|
utils.memcpy(resp.pubKey, 0, pubkey, 0, len(pubkey))
|
||||||
|
resp.keyHandleLen = len(keybuf) + len(keybase)
|
||||||
utils.memcpy(resp.keyHandle, 0, keybuf, 0, len(keybuf))
|
utils.memcpy(resp.keyHandle, 0, keybuf, 0, len(keybuf))
|
||||||
utils.memcpy(resp.keyHandle, len(keybuf), keybase, 0, len(keybase))
|
utils.memcpy(resp.keyHandle, len(keybuf), keybase, 0, len(keybase))
|
||||||
|
utils.memcpy(resp.cert, 0, _U2F_ATT_CERT, 0, len(_U2F_ATT_CERT))
|
||||||
utils.memcpy(resp.sig, 0, sig, 0, len(sig))
|
utils.memcpy(resp.sig, 0, sig, 0, len(sig))
|
||||||
|
resp.status = _SW_NO_ERROR
|
||||||
|
|
||||||
return buf
|
return buf
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user