| 
									
										
										
										
											2017-08-09 12:51:16 +03:00
										 |  |  | // Copyright 2017 The go-ethereum Authors | 
					
						
							|  |  |  | // This file is part of the go-ethereum library. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | // it under the terms of the GNU Lesser General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							|  |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							|  |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // This file contains the implementation for interacting with the Trezor hardware | 
					
						
							|  |  |  | // wallets. The wire protocol spec can be found on the SatoshiLabs website: | 
					
						
							|  |  |  | // https://doc.satoshilabs.com/trezor-tech/api-protobuf.html | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package usbwallet | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"encoding/binary" | 
					
						
							|  |  |  | 	"errors" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io" | 
					
						
							|  |  |  | 	"math/big" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/accounts" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/accounts/usbwallet/internal/trezor" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/common/hexutil" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/core/types" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/log" | 
					
						
							|  |  |  | 	"github.com/golang/protobuf/proto" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ErrTrezorPINNeeded is returned if opening the trezor requires a PIN code. In | 
					
						
							|  |  |  | // this case, the calling application should display a pinpad and send back the | 
					
						
							|  |  |  | // encoded passphrase. | 
					
						
							|  |  |  | var ErrTrezorPINNeeded = errors.New("trezor: pin needed") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // errTrezorReplyInvalidHeader is the error message returned by a Trezor data exchange | 
					
						
							|  |  |  | // if the device replies with a mismatching header. This usually means the device | 
					
						
							|  |  |  | // is in browser mode. | 
					
						
							|  |  |  | var errTrezorReplyInvalidHeader = errors.New("trezor: invalid reply header") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // trezorDriver implements the communication with a Trezor hardware wallet. | 
					
						
							|  |  |  | type trezorDriver struct { | 
					
						
							|  |  |  | 	device  io.ReadWriter // USB device connection to communicate through | 
					
						
							|  |  |  | 	version [3]uint32     // Current version of the Trezor firmware | 
					
						
							|  |  |  | 	label   string        // Current textual label of the Trezor device | 
					
						
							|  |  |  | 	pinwait bool          // Flags whether the device is waiting for PIN entry | 
					
						
							|  |  |  | 	failure error         // Any failure that would make the device unusable | 
					
						
							|  |  |  | 	log     log.Logger    // Contextual logger to tag the trezor with its id | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // newTrezorDriver creates a new instance of a Trezor USB protocol driver. | 
					
						
							|  |  |  | func newTrezorDriver(logger log.Logger) driver { | 
					
						
							|  |  |  | 	return &trezorDriver{ | 
					
						
							|  |  |  | 		log: logger, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Status implements accounts.Wallet, always whether the Trezor is opened, closed | 
					
						
							|  |  |  | // or whether the Ethereum app was not started on it. | 
					
						
							|  |  |  | func (w *trezorDriver) Status() (string, error) { | 
					
						
							|  |  |  | 	if w.failure != nil { | 
					
						
							|  |  |  | 		return fmt.Sprintf("Failed: %v", w.failure), w.failure | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if w.device == nil { | 
					
						
							|  |  |  | 		return "Closed", w.failure | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if w.pinwait { | 
					
						
							|  |  |  | 		return fmt.Sprintf("Trezor v%d.%d.%d '%s' waiting for PIN", w.version[0], w.version[1], w.version[2], w.label), w.failure | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return fmt.Sprintf("Trezor v%d.%d.%d '%s' online", w.version[0], w.version[1], w.version[2], w.label), w.failure | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Open implements usbwallet.driver, attempting to initialize the connection to | 
					
						
							|  |  |  | // the Trezor hardware wallet. Initializing the Trezor is a two phase operation: | 
					
						
							|  |  |  | //  * The first phase is to initialize the connection and read the wallet's | 
					
						
							|  |  |  | //    features. This phase is invoked is the provided passphrase is empty. The | 
					
						
							|  |  |  | //    device will display the pinpad as a result and will return an appropriate | 
					
						
							|  |  |  | //    error to notify the user that a second open phase is needed. | 
					
						
							|  |  |  | //  * The second phase is to unlock access to the Trezor, which is done by the | 
					
						
							|  |  |  | //    user actually providing a passphrase mapping a keyboard keypad to the pin | 
					
						
							|  |  |  | //    number of the user (shuffled according to the pinpad displayed). | 
					
						
							|  |  |  | func (w *trezorDriver) Open(device io.ReadWriter, passphrase string) error { | 
					
						
							|  |  |  | 	w.device, w.failure = device, nil | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// If phase 1 is requested, init the connection and wait for user callback | 
					
						
							|  |  |  | 	if passphrase == "" { | 
					
						
							|  |  |  | 		// If we're already waiting for a PIN entry, insta-return | 
					
						
							|  |  |  | 		if w.pinwait { | 
					
						
							|  |  |  | 			return ErrTrezorPINNeeded | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// Initialize a connection to the device | 
					
						
							|  |  |  | 		features := new(trezor.Features) | 
					
						
							|  |  |  | 		if _, err := w.trezorExchange(&trezor.Initialize{}, features); err != nil { | 
					
						
							|  |  |  | 			return err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		w.version = [3]uint32{features.GetMajorVersion(), features.GetMinorVersion(), features.GetPatchVersion()} | 
					
						
							|  |  |  | 		w.label = features.GetLabel() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Do a manual ping, forcing the device to ask for its PIN | 
					
						
							|  |  |  | 		askPin := true | 
					
						
							|  |  |  | 		res, err := w.trezorExchange(&trezor.Ping{PinProtection: &askPin}, new(trezor.PinMatrixRequest), new(trezor.Success)) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// Only return the PIN request if the device wasn't unlocked until now | 
					
						
							|  |  |  | 		if res == 1 { | 
					
						
							|  |  |  | 			return nil // Device responded with trezor.Success | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		w.pinwait = true | 
					
						
							|  |  |  | 		return ErrTrezorPINNeeded | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Phase 2 requested with actual PIN entry | 
					
						
							|  |  |  | 	w.pinwait = false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if _, err := w.trezorExchange(&trezor.PinMatrixAck{Pin: &passphrase}, new(trezor.Success)); err != nil { | 
					
						
							|  |  |  | 		w.failure = err | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Close implements usbwallet.driver, cleaning up and metadata maintained within | 
					
						
							|  |  |  | // the Trezor driver. | 
					
						
							|  |  |  | func (w *trezorDriver) Close() error { | 
					
						
							|  |  |  | 	w.version, w.label, w.pinwait = [3]uint32{}, "", false | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Heartbeat implements usbwallet.driver, performing a sanity check against the | 
					
						
							|  |  |  | // Trezor to see if it's still online. | 
					
						
							|  |  |  | func (w *trezorDriver) Heartbeat() error { | 
					
						
							|  |  |  | 	if _, err := w.trezorExchange(&trezor.Ping{}, new(trezor.Success)); err != nil { | 
					
						
							|  |  |  | 		w.failure = err | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Derive implements usbwallet.driver, sending a derivation request to the Trezor | 
					
						
							|  |  |  | // and returning the Ethereum address located on that derivation path. | 
					
						
							|  |  |  | func (w *trezorDriver) Derive(path accounts.DerivationPath) (common.Address, error) { | 
					
						
							|  |  |  | 	return w.trezorDerive(path) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // SignTx implements usbwallet.driver, sending the transaction to the Trezor and | 
					
						
							|  |  |  | // waiting for the user to confirm or deny the transaction. | 
					
						
							|  |  |  | func (w *trezorDriver) SignTx(path accounts.DerivationPath, tx *types.Transaction, chainID *big.Int) (common.Address, *types.Transaction, error) { | 
					
						
							|  |  |  | 	if w.device == nil { | 
					
						
							|  |  |  | 		return common.Address{}, nil, accounts.ErrWalletClosed | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return w.trezorSign(path, tx, chainID) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // trezorDerive sends a derivation request to the Trezor device and returns the | 
					
						
							|  |  |  | // Ethereum address located on that path. | 
					
						
							|  |  |  | func (w *trezorDriver) trezorDerive(derivationPath []uint32) (common.Address, error) { | 
					
						
							|  |  |  | 	address := new(trezor.EthereumAddress) | 
					
						
							|  |  |  | 	if _, err := w.trezorExchange(&trezor.EthereumGetAddress{AddressN: derivationPath}, address); err != nil { | 
					
						
							|  |  |  | 		return common.Address{}, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return common.BytesToAddress(address.GetAddress()), nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // trezorSign sends the transaction to the Trezor wallet, and waits for the user | 
					
						
							|  |  |  | // to confirm or deny the transaction. | 
					
						
							|  |  |  | func (w *trezorDriver) trezorSign(derivationPath []uint32, tx *types.Transaction, chainID *big.Int) (common.Address, *types.Transaction, error) { | 
					
						
							|  |  |  | 	// Create the transaction initiation message | 
					
						
							|  |  |  | 	data := tx.Data() | 
					
						
							|  |  |  | 	length := uint32(len(data)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	request := &trezor.EthereumSignTx{ | 
					
						
							|  |  |  | 		AddressN:   derivationPath, | 
					
						
							|  |  |  | 		Nonce:      new(big.Int).SetUint64(tx.Nonce()).Bytes(), | 
					
						
							|  |  |  | 		GasPrice:   tx.GasPrice().Bytes(), | 
					
						
							| 
									
										
										
										
											2017-11-13 13:47:27 +02:00
										 |  |  | 		GasLimit:   new(big.Int).SetUint64(tx.Gas()).Bytes(), | 
					
						
							| 
									
										
										
										
											2017-08-09 12:51:16 +03:00
										 |  |  | 		Value:      tx.Value().Bytes(), | 
					
						
							|  |  |  | 		DataLength: &length, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if to := tx.To(); to != nil { | 
					
						
							|  |  |  | 		request.To = (*to)[:] // Non contract deploy, set recipient explicitly | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if length > 1024 { // Send the data chunked if that was requested | 
					
						
							|  |  |  | 		request.DataInitialChunk, data = data[:1024], data[1024:] | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		request.DataInitialChunk, data = data, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if chainID != nil { // EIP-155 transaction, set chain ID explicitly (only 32 bit is supported!?) | 
					
						
							|  |  |  | 		id := uint32(chainID.Int64()) | 
					
						
							|  |  |  | 		request.ChainId = &id | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Send the initiation message and stream content until a signature is returned | 
					
						
							|  |  |  | 	response := new(trezor.EthereumTxRequest) | 
					
						
							|  |  |  | 	if _, err := w.trezorExchange(request, response); err != nil { | 
					
						
							|  |  |  | 		return common.Address{}, nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for response.DataLength != nil && int(*response.DataLength) <= len(data) { | 
					
						
							|  |  |  | 		chunk := data[:*response.DataLength] | 
					
						
							|  |  |  | 		data = data[*response.DataLength:] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if _, err := w.trezorExchange(&trezor.EthereumTxAck{DataChunk: chunk}, response); err != nil { | 
					
						
							|  |  |  | 			return common.Address{}, nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Extract the Ethereum signature and do a sanity validation | 
					
						
							|  |  |  | 	if len(response.GetSignatureR()) == 0 || len(response.GetSignatureS()) == 0 || response.GetSignatureV() == 0 { | 
					
						
							|  |  |  | 		return common.Address{}, nil, errors.New("reply lacks signature") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	signature := append(append(response.GetSignatureR(), response.GetSignatureS()...), byte(response.GetSignatureV())) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Create the correct signer and signature transform based on the chain ID | 
					
						
							|  |  |  | 	var signer types.Signer | 
					
						
							|  |  |  | 	if chainID == nil { | 
					
						
							|  |  |  | 		signer = new(types.HomesteadSigner) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		signer = types.NewEIP155Signer(chainID) | 
					
						
							|  |  |  | 		signature[64] = signature[64] - byte(chainID.Uint64()*2+35) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Inject the final signature into the transaction and sanity check the sender | 
					
						
							|  |  |  | 	signed, err := tx.WithSignature(signer, signature) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return common.Address{}, nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	sender, err := types.Sender(signer, signed) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return common.Address{}, nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return sender, signed, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // trezorExchange performs a data exchange with the Trezor wallet, sending it a | 
					
						
							|  |  |  | // message and retrieving the response. If multiple responses are possible, the | 
					
						
							|  |  |  | // method will also return the index of the destination object used. | 
					
						
							|  |  |  | func (w *trezorDriver) trezorExchange(req proto.Message, results ...proto.Message) (int, error) { | 
					
						
							|  |  |  | 	// Construct the original message payload to chunk up | 
					
						
							|  |  |  | 	data, err := proto.Marshal(req) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return 0, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	payload := make([]byte, 8+len(data)) | 
					
						
							|  |  |  | 	copy(payload, []byte{0x23, 0x23}) | 
					
						
							|  |  |  | 	binary.BigEndian.PutUint16(payload[2:], trezor.Type(req)) | 
					
						
							|  |  |  | 	binary.BigEndian.PutUint32(payload[4:], uint32(len(data))) | 
					
						
							|  |  |  | 	copy(payload[8:], data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Stream all the chunks to the device | 
					
						
							|  |  |  | 	chunk := make([]byte, 64) | 
					
						
							|  |  |  | 	chunk[0] = 0x3f // Report ID magic number | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for len(payload) > 0 { | 
					
						
							|  |  |  | 		// Construct the new message to stream, padding with zeroes if needed | 
					
						
							|  |  |  | 		if len(payload) > 63 { | 
					
						
							|  |  |  | 			copy(chunk[1:], payload[:63]) | 
					
						
							|  |  |  | 			payload = payload[63:] | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			copy(chunk[1:], payload) | 
					
						
							|  |  |  | 			copy(chunk[1+len(payload):], make([]byte, 63-len(payload))) | 
					
						
							|  |  |  | 			payload = nil | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// Send over to the device | 
					
						
							|  |  |  | 		w.log.Trace("Data chunk sent to the Trezor", "chunk", hexutil.Bytes(chunk)) | 
					
						
							|  |  |  | 		if _, err := w.device.Write(chunk); err != nil { | 
					
						
							|  |  |  | 			return 0, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Stream the reply back from the wallet in 64 byte chunks | 
					
						
							|  |  |  | 	var ( | 
					
						
							|  |  |  | 		kind  uint16 | 
					
						
							|  |  |  | 		reply []byte | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 	for { | 
					
						
							|  |  |  | 		// Read the next chunk from the Trezor wallet | 
					
						
							|  |  |  | 		if _, err := io.ReadFull(w.device, chunk); err != nil { | 
					
						
							|  |  |  | 			return 0, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		w.log.Trace("Data chunk received from the Trezor", "chunk", hexutil.Bytes(chunk)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Make sure the transport header matches | 
					
						
							|  |  |  | 		if chunk[0] != 0x3f || (len(reply) == 0 && (chunk[1] != 0x23 || chunk[2] != 0x23)) { | 
					
						
							|  |  |  | 			return 0, errTrezorReplyInvalidHeader | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// If it's the first chunk, retrieve the reply message type and total message length | 
					
						
							|  |  |  | 		var payload []byte | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if len(reply) == 0 { | 
					
						
							|  |  |  | 			kind = binary.BigEndian.Uint16(chunk[3:5]) | 
					
						
							|  |  |  | 			reply = make([]byte, 0, int(binary.BigEndian.Uint32(chunk[5:9]))) | 
					
						
							|  |  |  | 			payload = chunk[9:] | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			payload = chunk[1:] | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// Append to the reply and stop when filled up | 
					
						
							|  |  |  | 		if left := cap(reply) - len(reply); left > len(payload) { | 
					
						
							|  |  |  | 			reply = append(reply, payload...) | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			reply = append(reply, payload[:left]...) | 
					
						
							|  |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Try to parse the reply into the requested reply message | 
					
						
							|  |  |  | 	if kind == uint16(trezor.MessageType_MessageType_Failure) { | 
					
						
							|  |  |  | 		// Trezor returned a failure, extract and return the message | 
					
						
							|  |  |  | 		failure := new(trezor.Failure) | 
					
						
							|  |  |  | 		if err := proto.Unmarshal(reply, failure); err != nil { | 
					
						
							|  |  |  | 			return 0, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return 0, errors.New("trezor: " + failure.GetMessage()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if kind == uint16(trezor.MessageType_MessageType_ButtonRequest) { | 
					
						
							|  |  |  | 		// Trezor is waiting for user confirmation, ack and wait for the next message | 
					
						
							|  |  |  | 		return w.trezorExchange(&trezor.ButtonAck{}, results...) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for i, res := range results { | 
					
						
							|  |  |  | 		if trezor.Type(res) == kind { | 
					
						
							|  |  |  | 			return i, proto.Unmarshal(reply, res) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	expected := make([]string, len(results)) | 
					
						
							|  |  |  | 	for i, res := range results { | 
					
						
							|  |  |  | 		expected[i] = trezor.Name(trezor.Type(res)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0, fmt.Errorf("trezor: expected reply types %s, got %s", expected, trezor.Name(kind)) | 
					
						
							|  |  |  | } |