366 lines
		
	
	
		
			9.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			366 lines
		
	
	
		
			9.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright (c) 2013 Kyle Isom <kyle@tyrfingr.is>
 | |
| // Copyright (c) 2012 The Go Authors. All rights reserved.
 | |
| //
 | |
| // Redistribution and use in source and binary forms, with or without
 | |
| // modification, are permitted provided that the following conditions are
 | |
| // met:
 | |
| //
 | |
| //    * Redistributions of source code must retain the above copyright
 | |
| // notice, this list of conditions and the following disclaimer.
 | |
| //    * Redistributions in binary form must reproduce the above
 | |
| // copyright notice, this list of conditions and the following disclaimer
 | |
| // in the documentation and/or other materials provided with the
 | |
| // distribution.
 | |
| //    * Neither the name of Google Inc. nor the names of its
 | |
| // contributors may be used to endorse or promote products derived from
 | |
| // this software without specific prior written permission.
 | |
| //
 | |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 | |
| // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 | |
| // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 | |
| // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 | |
| // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 | |
| // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 | |
| // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 | |
| // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 | |
| // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 | |
| // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 | |
| // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | |
| 
 | |
| package ecies
 | |
| 
 | |
| import (
 | |
| 	"crypto/cipher"
 | |
| 	"crypto/ecdsa"
 | |
| 	"crypto/elliptic"
 | |
| 	"crypto/hmac"
 | |
| 	"crypto/subtle"
 | |
| 	"fmt"
 | |
| 	"hash"
 | |
| 	"io"
 | |
| 	"math/big"
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	ErrImport                     = fmt.Errorf("ecies: failed to import key")
 | |
| 	ErrInvalidCurve               = fmt.Errorf("ecies: invalid elliptic curve")
 | |
| 	ErrInvalidParams              = fmt.Errorf("ecies: invalid ECIES parameters")
 | |
| 	ErrInvalidPublicKey           = fmt.Errorf("ecies: invalid public key")
 | |
| 	ErrSharedKeyIsPointAtInfinity = fmt.Errorf("ecies: shared key is point at infinity")
 | |
| 	ErrSharedKeyTooBig            = fmt.Errorf("ecies: shared key params are too big")
 | |
| )
 | |
| 
 | |
| // PublicKey is a representation of an elliptic curve public key.
 | |
| type PublicKey struct {
 | |
| 	X *big.Int
 | |
| 	Y *big.Int
 | |
| 	elliptic.Curve
 | |
| 	Params *ECIESParams
 | |
| }
 | |
| 
 | |
| // Export an ECIES public key as an ECDSA public key.
 | |
| func (pub *PublicKey) ExportECDSA() *ecdsa.PublicKey {
 | |
| 	return &ecdsa.PublicKey{pub.Curve, pub.X, pub.Y}
 | |
| }
 | |
| 
 | |
| // Import an ECDSA public key as an ECIES public key.
 | |
| func ImportECDSAPublic(pub *ecdsa.PublicKey) *PublicKey {
 | |
| 	return &PublicKey{
 | |
| 		X:      pub.X,
 | |
| 		Y:      pub.Y,
 | |
| 		Curve:  pub.Curve,
 | |
| 		Params: ParamsFromCurve(pub.Curve),
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // PrivateKey is a representation of an elliptic curve private key.
 | |
| type PrivateKey struct {
 | |
| 	PublicKey
 | |
| 	D *big.Int
 | |
| }
 | |
| 
 | |
| // Export an ECIES private key as an ECDSA private key.
 | |
| func (prv *PrivateKey) ExportECDSA() *ecdsa.PrivateKey {
 | |
| 	pub := &prv.PublicKey
 | |
| 	pubECDSA := pub.ExportECDSA()
 | |
| 	return &ecdsa.PrivateKey{*pubECDSA, prv.D}
 | |
| }
 | |
| 
 | |
| // Import an ECDSA private key as an ECIES private key.
 | |
| func ImportECDSA(prv *ecdsa.PrivateKey) *PrivateKey {
 | |
| 	pub := ImportECDSAPublic(&prv.PublicKey)
 | |
| 	return &PrivateKey{*pub, prv.D}
 | |
| }
 | |
| 
 | |
| // Generate an elliptic curve public / private keypair. If params is nil,
 | |
| // the recommended default paramters for the key will be chosen.
 | |
| func GenerateKey(rand io.Reader, curve elliptic.Curve, params *ECIESParams) (prv *PrivateKey, err error) {
 | |
| 	pb, x, y, err := elliptic.GenerateKey(curve, rand)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 	prv = new(PrivateKey)
 | |
| 	prv.PublicKey.X = x
 | |
| 	prv.PublicKey.Y = y
 | |
| 	prv.PublicKey.Curve = curve
 | |
| 	prv.D = new(big.Int).SetBytes(pb)
 | |
| 	if params == nil {
 | |
| 		params = ParamsFromCurve(curve)
 | |
| 	}
 | |
| 	prv.PublicKey.Params = params
 | |
| 	return
 | |
| }
 | |
| 
 | |
| // MaxSharedKeyLength returns the maximum length of the shared key the
 | |
| // public key can produce.
 | |
| func MaxSharedKeyLength(pub *PublicKey) int {
 | |
| 	return (pub.Curve.Params().BitSize + 7) / 8
 | |
| }
 | |
| 
 | |
| // ECDH key agreement method used to establish secret keys for encryption.
 | |
| func (prv *PrivateKey) GenerateShared(pub *PublicKey, skLen, macLen int) (sk []byte, err error) {
 | |
| 	if prv.PublicKey.Curve != pub.Curve {
 | |
| 		return nil, ErrInvalidCurve
 | |
| 	}
 | |
| 	if skLen+macLen > MaxSharedKeyLength(pub) {
 | |
| 		return nil, ErrSharedKeyTooBig
 | |
| 	}
 | |
| 
 | |
| 	x, _ := pub.Curve.ScalarMult(pub.X, pub.Y, prv.D.Bytes())
 | |
| 	if x == nil {
 | |
| 		return nil, ErrSharedKeyIsPointAtInfinity
 | |
| 	}
 | |
| 
 | |
| 	sk = make([]byte, skLen+macLen)
 | |
| 	skBytes := x.Bytes()
 | |
| 	copy(sk[len(sk)-len(skBytes):], skBytes)
 | |
| 	return sk, nil
 | |
| }
 | |
| 
 | |
| var (
 | |
| 	ErrKeyDataTooLong = fmt.Errorf("ecies: can't supply requested key data")
 | |
| 	ErrSharedTooLong  = fmt.Errorf("ecies: shared secret is too long")
 | |
| 	ErrInvalidMessage = fmt.Errorf("ecies: invalid message")
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	big2To32   = new(big.Int).Exp(big.NewInt(2), big.NewInt(32), nil)
 | |
| 	big2To32M1 = new(big.Int).Sub(big2To32, big.NewInt(1))
 | |
| )
 | |
| 
 | |
| func incCounter(ctr []byte) {
 | |
| 	if ctr[3]++; ctr[3] != 0 {
 | |
| 		return
 | |
| 	} else if ctr[2]++; ctr[2] != 0 {
 | |
| 		return
 | |
| 	} else if ctr[1]++; ctr[1] != 0 {
 | |
| 		return
 | |
| 	} else if ctr[0]++; ctr[0] != 0 {
 | |
| 		return
 | |
| 	}
 | |
| 	return
 | |
| }
 | |
| 
 | |
| // NIST SP 800-56 Concatenation Key Derivation Function (see section 5.8.1).
 | |
| func concatKDF(hash hash.Hash, z, s1 []byte, kdLen int) (k []byte, err error) {
 | |
| 	if s1 == nil {
 | |
| 		s1 = make([]byte, 0)
 | |
| 	}
 | |
| 
 | |
| 	reps := ((kdLen + 7) * 8) / (hash.BlockSize() * 8)
 | |
| 	if big.NewInt(int64(reps)).Cmp(big2To32M1) > 0 {
 | |
| 		fmt.Println(big2To32M1)
 | |
| 		return nil, ErrKeyDataTooLong
 | |
| 	}
 | |
| 
 | |
| 	counter := []byte{0, 0, 0, 1}
 | |
| 	k = make([]byte, 0)
 | |
| 
 | |
| 	for i := 0; i <= reps; i++ {
 | |
| 		hash.Write(counter)
 | |
| 		hash.Write(z)
 | |
| 		hash.Write(s1)
 | |
| 		k = append(k, hash.Sum(nil)...)
 | |
| 		hash.Reset()
 | |
| 		incCounter(counter)
 | |
| 	}
 | |
| 
 | |
| 	k = k[:kdLen]
 | |
| 	return
 | |
| }
 | |
| 
 | |
| // messageTag computes the MAC of a message (called the tag) as per
 | |
| // SEC 1, 3.5.
 | |
| func messageTag(hash func() hash.Hash, km, msg, shared []byte) []byte {
 | |
| 	mac := hmac.New(hash, km)
 | |
| 	mac.Write(msg)
 | |
| 	mac.Write(shared)
 | |
| 	tag := mac.Sum(nil)
 | |
| 	return tag
 | |
| }
 | |
| 
 | |
| // Generate an initialisation vector for CTR mode.
 | |
| func generateIV(params *ECIESParams, rand io.Reader) (iv []byte, err error) {
 | |
| 	iv = make([]byte, params.BlockSize)
 | |
| 	_, err = io.ReadFull(rand, iv)
 | |
| 	return
 | |
| }
 | |
| 
 | |
| // symEncrypt carries out CTR encryption using the block cipher specified in the
 | |
| // parameters.
 | |
| func symEncrypt(rand io.Reader, params *ECIESParams, key, m []byte) (ct []byte, err error) {
 | |
| 	c, err := params.Cipher(key)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	iv, err := generateIV(params, rand)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 	ctr := cipher.NewCTR(c, iv)
 | |
| 
 | |
| 	ct = make([]byte, len(m)+params.BlockSize)
 | |
| 	copy(ct, iv)
 | |
| 	ctr.XORKeyStream(ct[params.BlockSize:], m)
 | |
| 	return
 | |
| }
 | |
| 
 | |
| // symDecrypt carries out CTR decryption using the block cipher specified in
 | |
| // the parameters
 | |
| func symDecrypt(rand io.Reader, params *ECIESParams, key, ct []byte) (m []byte, err error) {
 | |
| 	c, err := params.Cipher(key)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	ctr := cipher.NewCTR(c, ct[:params.BlockSize])
 | |
| 
 | |
| 	m = make([]byte, len(ct)-params.BlockSize)
 | |
| 	ctr.XORKeyStream(m, ct[params.BlockSize:])
 | |
| 	return
 | |
| }
 | |
| 
 | |
| // Encrypt encrypts a message using ECIES as specified in SEC 1, 5.1.
 | |
| //
 | |
| // s1 and s2 contain shared information that is not part of the resulting
 | |
| // ciphertext. s1 is fed into key derivation, s2 is fed into the MAC. If the
 | |
| // shared information parameters aren't being used, they should be nil.
 | |
| func Encrypt(rand io.Reader, pub *PublicKey, m, s1, s2 []byte) (ct []byte, err error) {
 | |
| 	params := pub.Params
 | |
| 	if params == nil {
 | |
| 		if params = ParamsFromCurve(pub.Curve); params == nil {
 | |
| 			err = ErrUnsupportedECIESParameters
 | |
| 			return
 | |
| 		}
 | |
| 	}
 | |
| 	R, err := GenerateKey(rand, pub.Curve, params)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	hash := params.Hash()
 | |
| 	z, err := R.GenerateShared(pub, params.KeyLen, params.KeyLen)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 	K, err := concatKDF(hash, z, s1, params.KeyLen+params.KeyLen)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 	Ke := K[:params.KeyLen]
 | |
| 	Km := K[params.KeyLen:]
 | |
| 	hash.Write(Km)
 | |
| 	Km = hash.Sum(nil)
 | |
| 	hash.Reset()
 | |
| 
 | |
| 	em, err := symEncrypt(rand, params, Ke, m)
 | |
| 	if err != nil || len(em) <= params.BlockSize {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	d := messageTag(params.Hash, Km, em, s2)
 | |
| 
 | |
| 	Rb := elliptic.Marshal(pub.Curve, R.PublicKey.X, R.PublicKey.Y)
 | |
| 	ct = make([]byte, len(Rb)+len(em)+len(d))
 | |
| 	copy(ct, Rb)
 | |
| 	copy(ct[len(Rb):], em)
 | |
| 	copy(ct[len(Rb)+len(em):], d)
 | |
| 	return
 | |
| }
 | |
| 
 | |
| // Decrypt decrypts an ECIES ciphertext.
 | |
| func (prv *PrivateKey) Decrypt(rand io.Reader, c, s1, s2 []byte) (m []byte, err error) {
 | |
| 	if c == nil || len(c) == 0 {
 | |
| 		err = ErrInvalidMessage
 | |
| 		return
 | |
| 	}
 | |
| 	params := prv.PublicKey.Params
 | |
| 	if params == nil {
 | |
| 		if params = ParamsFromCurve(prv.PublicKey.Curve); params == nil {
 | |
| 			err = ErrUnsupportedECIESParameters
 | |
| 			return
 | |
| 		}
 | |
| 	}
 | |
| 	hash := params.Hash()
 | |
| 
 | |
| 	var (
 | |
| 		rLen   int
 | |
| 		hLen   int = hash.Size()
 | |
| 		mStart int
 | |
| 		mEnd   int
 | |
| 	)
 | |
| 
 | |
| 	switch c[0] {
 | |
| 	case 2, 3, 4:
 | |
| 		rLen = ((prv.PublicKey.Curve.Params().BitSize + 7) / 4)
 | |
| 		if len(c) < (rLen + hLen + 1) {
 | |
| 			err = ErrInvalidMessage
 | |
| 			return
 | |
| 		}
 | |
| 	default:
 | |
| 		err = ErrInvalidPublicKey
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	mStart = rLen
 | |
| 	mEnd = len(c) - hLen
 | |
| 
 | |
| 	R := new(PublicKey)
 | |
| 	R.Curve = prv.PublicKey.Curve
 | |
| 	R.X, R.Y = elliptic.Unmarshal(R.Curve, c[:rLen])
 | |
| 	if R.X == nil {
 | |
| 		err = ErrInvalidPublicKey
 | |
| 		return
 | |
| 	}
 | |
| 	if !R.Curve.IsOnCurve(R.X, R.Y) {
 | |
| 		err = ErrInvalidCurve
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	z, err := prv.GenerateShared(R, params.KeyLen, params.KeyLen)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	K, err := concatKDF(hash, z, s1, params.KeyLen+params.KeyLen)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	Ke := K[:params.KeyLen]
 | |
| 	Km := K[params.KeyLen:]
 | |
| 	hash.Write(Km)
 | |
| 	Km = hash.Sum(nil)
 | |
| 	hash.Reset()
 | |
| 
 | |
| 	d := messageTag(params.Hash, Km, c[mStart:mEnd], s2)
 | |
| 	if subtle.ConstantTimeCompare(c[mEnd:], d) != 1 {
 | |
| 		err = ErrInvalidMessage
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	m, err = symDecrypt(rand, params, Ke, c[mStart:mEnd])
 | |
| 	return
 | |
| }
 |