| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // Copyright 2014 The go-ethereum Authors | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // This file is part of the go-ethereum library. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-23 18:35:11 +02:00
										 |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // 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. | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-31 12:37:43 +01:00
										 |  |  | package crypto | 
					
						
							| 
									
										
										
										
											2014-06-29 15:57:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2014-12-10 00:03:21 +01:00
										 |  |  | 	"crypto/ecdsa" | 
					
						
							|  |  |  | 	"crypto/elliptic" | 
					
						
							|  |  |  | 	"crypto/rand" | 
					
						
							| 
									
										
										
										
											2017-02-18 09:24:12 +01:00
										 |  |  | 	"encoding/hex" | 
					
						
							|  |  |  | 	"errors" | 
					
						
							| 
									
										
										
										
											2015-02-10 12:29:50 +01:00
										 |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2015-03-25 14:58:52 +00:00
										 |  |  | 	"io/ioutil" | 
					
						
							| 
									
										
										
										
											2015-06-01 20:27:20 +02:00
										 |  |  | 	"math/big" | 
					
						
							| 
									
										
										
										
											2015-02-10 12:29:50 +01:00
										 |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2014-10-08 12:00:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-16 17:27:24 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2015-01-22 00:25:00 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/crypto/sha3" | 
					
						
							| 
									
										
										
										
											2015-03-17 11:19:23 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/rlp" | 
					
						
							| 
									
										
										
										
											2017-02-18 09:24:12 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ( | 
					
						
							|  |  |  | 	secp256k1_N, _  = new(big.Int).SetString("fffffffffffffffffffffffffffffffebaaedce6af48a03bbfd25e8cd0364141", 16) | 
					
						
							|  |  |  | 	secp256k1_halfN = new(big.Int).Div(secp256k1_N, big.NewInt(2)) | 
					
						
							| 
									
										
										
										
											2014-06-29 15:57:12 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-21 18:40:27 +00:00
										 |  |  | func Keccak256(data ...[]byte) []byte { | 
					
						
							| 
									
										
										
										
											2014-06-29 15:57:12 +01:00
										 |  |  | 	d := sha3.NewKeccak256() | 
					
						
							| 
									
										
										
										
											2015-01-27 14:29:33 +01:00
										 |  |  | 	for _, b := range data { | 
					
						
							|  |  |  | 		d.Write(b) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-06-29 15:57:12 +01:00
										 |  |  | 	return d.Sum(nil) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-06-29 16:08:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-21 18:40:27 +00:00
										 |  |  | func Keccak256Hash(data ...[]byte) (h common.Hash) { | 
					
						
							| 
									
										
										
										
											2015-03-16 17:27:24 +01:00
										 |  |  | 	d := sha3.NewKeccak256() | 
					
						
							|  |  |  | 	for _, b := range data { | 
					
						
							|  |  |  | 		d.Write(b) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-17 01:32:35 +01:00
										 |  |  | 	d.Sum(h[:0]) | 
					
						
							| 
									
										
										
										
											2015-03-16 17:27:24 +01:00
										 |  |  | 	return h | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-21 22:45:08 +00:00
										 |  |  | // Deprecated: For backward compatibility as other packages depend on these | 
					
						
							|  |  |  | func Sha3Hash(data ...[]byte) common.Hash { return Keccak256Hash(data...) } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-29 16:08:33 +01:00
										 |  |  | // Creates an ethereum address given the bytes and the nonce | 
					
						
							| 
									
										
										
										
											2015-03-17 11:19:23 +01:00
										 |  |  | func CreateAddress(b common.Address, nonce uint64) common.Address { | 
					
						
							|  |  |  | 	data, _ := rlp.EncodeToBytes([]interface{}{b, nonce}) | 
					
						
							| 
									
										
										
										
											2016-02-21 18:40:27 +00:00
										 |  |  | 	return common.BytesToAddress(Keccak256(data)[12:]) | 
					
						
							| 
									
										
										
										
											2014-06-29 16:08:33 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-10-08 12:00:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-18 09:24:12 +01:00
										 |  |  | // ToECDSA creates a private key with the given D value. | 
					
						
							| 
									
										
										
										
											2014-12-10 14:17:10 +01:00
										 |  |  | func ToECDSA(prv []byte) *ecdsa.PrivateKey { | 
					
						
							| 
									
										
										
										
											2014-12-12 22:24:04 +01:00
										 |  |  | 	if len(prv) == 0 { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-10 14:17:10 +01:00
										 |  |  | 	priv := new(ecdsa.PrivateKey) | 
					
						
							| 
									
										
										
										
											2017-02-18 09:24:12 +01:00
										 |  |  | 	priv.PublicKey.Curve = S256() | 
					
						
							| 
									
										
										
										
											2015-03-16 11:27:38 +01:00
										 |  |  | 	priv.D = common.BigD(prv) | 
					
						
							| 
									
										
										
										
											2017-02-18 09:24:12 +01:00
										 |  |  | 	priv.PublicKey.X, priv.PublicKey.Y = priv.PublicKey.Curve.ScalarBaseMult(prv) | 
					
						
							| 
									
										
										
										
											2014-12-10 14:17:10 +01:00
										 |  |  | 	return priv | 
					
						
							| 
									
										
										
										
											2014-12-10 00:03:21 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-10 14:17:10 +01:00
										 |  |  | func FromECDSA(prv *ecdsa.PrivateKey) []byte { | 
					
						
							| 
									
										
										
										
											2014-12-12 22:24:04 +01:00
										 |  |  | 	if prv == nil { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-12-10 14:17:10 +01:00
										 |  |  | 	return prv.D.Bytes() | 
					
						
							| 
									
										
										
										
											2014-12-10 00:03:21 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-12 22:24:04 +01:00
										 |  |  | func ToECDSAPub(pub []byte) *ecdsa.PublicKey { | 
					
						
							|  |  |  | 	if len(pub) == 0 { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-18 09:24:12 +01:00
										 |  |  | 	x, y := elliptic.Unmarshal(S256(), pub) | 
					
						
							|  |  |  | 	return &ecdsa.PublicKey{Curve: S256(), X: x, Y: y} | 
					
						
							| 
									
										
										
										
											2014-12-10 14:17:10 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-12 22:24:04 +01:00
										 |  |  | func FromECDSAPub(pub *ecdsa.PublicKey) []byte { | 
					
						
							| 
									
										
										
										
											2015-01-30 13:24:20 +01:00
										 |  |  | 	if pub == nil || pub.X == nil || pub.Y == nil { | 
					
						
							| 
									
										
										
										
											2014-12-12 22:24:04 +01:00
										 |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-18 09:24:12 +01:00
										 |  |  | 	return elliptic.Marshal(S256(), pub.X, pub.Y) | 
					
						
							| 
									
										
										
										
											2014-12-12 22:24:04 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-10 12:29:50 +01:00
										 |  |  | // HexToECDSA parses a secp256k1 private key. | 
					
						
							|  |  |  | func HexToECDSA(hexkey string) (*ecdsa.PrivateKey, error) { | 
					
						
							|  |  |  | 	b, err := hex.DecodeString(hexkey) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, errors.New("invalid hex string") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if len(b) != 32 { | 
					
						
							|  |  |  | 		return nil, errors.New("invalid length, need 256 bits") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return ToECDSA(b), nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // LoadECDSA loads a secp256k1 private key from the given file. | 
					
						
							| 
									
										
										
										
											2015-04-19 01:33:00 +02:00
										 |  |  | // The key data is expected to be hex-encoded. | 
					
						
							| 
									
										
										
										
											2015-02-10 12:29:50 +01:00
										 |  |  | func LoadECDSA(file string) (*ecdsa.PrivateKey, error) { | 
					
						
							| 
									
										
										
										
											2015-04-08 23:03:47 +02:00
										 |  |  | 	buf := make([]byte, 64) | 
					
						
							| 
									
										
										
										
											2015-02-10 12:29:50 +01:00
										 |  |  | 	fd, err := os.Open(file) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer fd.Close() | 
					
						
							|  |  |  | 	if _, err := io.ReadFull(fd, buf); err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-04-09 10:59:37 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	key, err := hex.DecodeString(string(buf)) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ToECDSA(key), nil | 
					
						
							| 
									
										
										
										
											2015-02-10 12:29:50 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-19 01:33:00 +02:00
										 |  |  | // SaveECDSA saves a secp256k1 private key to the given file with | 
					
						
							|  |  |  | // restrictive permissions. The key data is saved hex-encoded. | 
					
						
							| 
									
										
										
										
											2015-03-23 13:00:06 +00:00
										 |  |  | func SaveECDSA(file string, key *ecdsa.PrivateKey) error { | 
					
						
							| 
									
										
										
										
											2015-04-09 10:59:37 +02:00
										 |  |  | 	k := hex.EncodeToString(FromECDSA(key)) | 
					
						
							|  |  |  | 	return ioutil.WriteFile(file, []byte(k), 0600) | 
					
						
							| 
									
										
										
										
											2015-03-23 13:00:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-10 14:17:10 +01:00
										 |  |  | func GenerateKey() (*ecdsa.PrivateKey, error) { | 
					
						
							| 
									
										
										
										
											2017-02-18 09:24:12 +01:00
										 |  |  | 	return ecdsa.GenerateKey(S256(), rand.Reader) | 
					
						
							| 
									
										
										
										
											2014-12-10 14:17:10 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-05 12:35:23 +02:00
										 |  |  | // ValidateSignatureValues verifies whether the signature values are valid with | 
					
						
							|  |  |  | // the given chain rules. The v value is assumed to be either 0 or 1. | 
					
						
							| 
									
										
										
										
											2015-11-27 15:40:29 +01:00
										 |  |  | func ValidateSignatureValues(v byte, r, s *big.Int, homestead bool) bool { | 
					
						
							| 
									
										
										
										
											2015-09-21 15:48:15 +02:00
										 |  |  | 	if r.Cmp(common.Big1) < 0 || s.Cmp(common.Big1) < 0 { | 
					
						
							| 
									
										
										
										
											2015-06-01 20:27:20 +02:00
										 |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-11-27 15:40:29 +01:00
										 |  |  | 	// reject upper range of s values (ECDSA malleability) | 
					
						
							|  |  |  | 	// see discussion in secp256k1/libsecp256k1/include/secp256k1.h | 
					
						
							| 
									
										
										
										
											2017-02-18 09:24:12 +01:00
										 |  |  | 	if homestead && s.Cmp(secp256k1_halfN) > 0 { | 
					
						
							| 
									
										
										
										
											2015-11-27 15:40:29 +01:00
										 |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Frontier: allow s to be in full N range | 
					
						
							| 
									
										
										
										
											2017-02-18 09:24:12 +01:00
										 |  |  | 	return r.Cmp(secp256k1_N) < 0 && s.Cmp(secp256k1_N) < 0 && (v == 0 || v == 1) | 
					
						
							| 
									
										
										
										
											2014-12-10 00:03:21 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-01-20 23:55:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-04 16:52:23 +02:00
										 |  |  | func PubkeyToAddress(p ecdsa.PublicKey) common.Address { | 
					
						
							| 
									
										
										
										
											2015-01-25 02:07:20 +01:00
										 |  |  | 	pubBytes := FromECDSAPub(&p) | 
					
						
							| 
									
										
										
										
											2016-02-21 18:40:27 +00:00
										 |  |  | 	return common.BytesToAddress(Keccak256(pubBytes[1:])[12:]) | 
					
						
							| 
									
										
										
										
											2015-01-25 02:07:20 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-09-28 11:19:23 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | func zeroBytes(bytes []byte) { | 
					
						
							|  |  |  | 	for i := range bytes { | 
					
						
							|  |  |  | 		bytes[i] = 0 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |