github.com/myafeier/go-ethereum@v1.6.8-0.20170719123245-3e0dbe0eaa72/les/handler.go (about)

     1  // Copyright 2016 The go-ethereum Authors
     2  // This file is part of the go-ethereum library.
     3  //
     4  // The go-ethereum library is free software: you can redistribute it and/or modify
     5  // it under the terms of the GNU Lesser General Public License as published by
     6  // the Free Software Foundation, either version 3 of the License, or
     7  // (at your option) any later version.
     8  //
     9  // The go-ethereum library is distributed in the hope that it will be useful,
    10  // but WITHOUT ANY WARRANTY; without even the implied warranty of
    11  // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
    12  // GNU Lesser General Public License for more details.
    13  //
    14  // You should have received a copy of the GNU Lesser General Public License
    15  // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
    16  
    17  // Package les implements the Light Ethereum Subprotocol.
    18  package les
    19  
    20  import (
    21  	"encoding/binary"
    22  	"errors"
    23  	"fmt"
    24  	"math/big"
    25  	"net"
    26  	"sync"
    27  	"time"
    28  
    29  	"github.com/ethereum/go-ethereum/common"
    30  	"github.com/ethereum/go-ethereum/consensus"
    31  	"github.com/ethereum/go-ethereum/core"
    32  	"github.com/ethereum/go-ethereum/core/state"
    33  	"github.com/ethereum/go-ethereum/core/types"
    34  	"github.com/ethereum/go-ethereum/eth"
    35  	"github.com/ethereum/go-ethereum/eth/downloader"
    36  	"github.com/ethereum/go-ethereum/ethdb"
    37  	"github.com/ethereum/go-ethereum/event"
    38  	"github.com/ethereum/go-ethereum/log"
    39  	"github.com/ethereum/go-ethereum/p2p"
    40  	"github.com/ethereum/go-ethereum/p2p/discover"
    41  	"github.com/ethereum/go-ethereum/p2p/discv5"
    42  	"github.com/ethereum/go-ethereum/params"
    43  	"github.com/ethereum/go-ethereum/rlp"
    44  	"github.com/ethereum/go-ethereum/trie"
    45  )
    46  
    47  const (
    48  	softResponseLimit = 2 * 1024 * 1024 // Target maximum size of returned blocks, headers or node data.
    49  	estHeaderRlpSize  = 500             // Approximate size of an RLP encoded block header
    50  
    51  	ethVersion = 63 // equivalent eth version for the downloader
    52  
    53  	MaxHeaderFetch       = 192 // Amount of block headers to be fetched per retrieval request
    54  	MaxBodyFetch         = 32  // Amount of block bodies to be fetched per retrieval request
    55  	MaxReceiptFetch      = 128 // Amount of transaction receipts to allow fetching per request
    56  	MaxCodeFetch         = 64  // Amount of contract codes to allow fetching per request
    57  	MaxProofsFetch       = 64  // Amount of merkle proofs to be fetched per retrieval request
    58  	MaxHeaderProofsFetch = 64  // Amount of merkle proofs to be fetched per retrieval request
    59  	MaxTxSend            = 64  // Amount of transactions to be send per request
    60  
    61  	disableClientRemovePeer = false
    62  )
    63  
    64  // errIncompatibleConfig is returned if the requested protocols and configs are
    65  // not compatible (low protocol version restrictions and high requirements).
    66  var errIncompatibleConfig = errors.New("incompatible configuration")
    67  
    68  func errResp(code errCode, format string, v ...interface{}) error {
    69  	return fmt.Errorf("%v - %v", code, fmt.Sprintf(format, v...))
    70  }
    71  
    72  type hashFetcherFn func(common.Hash) error
    73  
    74  type BlockChain interface {
    75  	HasHeader(hash common.Hash) bool
    76  	GetHeader(hash common.Hash, number uint64) *types.Header
    77  	GetHeaderByHash(hash common.Hash) *types.Header
    78  	CurrentHeader() *types.Header
    79  	GetTdByHash(hash common.Hash) *big.Int
    80  	InsertHeaderChain(chain []*types.Header, checkFreq int) (int, error)
    81  	Rollback(chain []common.Hash)
    82  	Status() (td *big.Int, currentBlock common.Hash, genesisBlock common.Hash)
    83  	GetHeaderByNumber(number uint64) *types.Header
    84  	GetBlockHashesFromHash(hash common.Hash, max uint64) []common.Hash
    85  	LastBlockHash() common.Hash
    86  	Genesis() *types.Block
    87  }
    88  
    89  type txPool interface {
    90  	// AddRemotes should add the given transactions to the pool.
    91  	AddRemotes([]*types.Transaction) error
    92  }
    93  
    94  type ProtocolManager struct {
    95  	lightSync   bool
    96  	txpool      txPool
    97  	txrelay     *LesTxRelay
    98  	networkId   uint64
    99  	chainConfig *params.ChainConfig
   100  	blockchain  BlockChain
   101  	chainDb     ethdb.Database
   102  	odr         *LesOdr
   103  	server      *LesServer
   104  	serverPool  *serverPool
   105  	lesTopic    discv5.Topic
   106  	reqDist     *requestDistributor
   107  	retriever   *retrieveManager
   108  
   109  	downloader *downloader.Downloader
   110  	fetcher    *lightFetcher
   111  	peers      *peerSet
   112  
   113  	SubProtocols []p2p.Protocol
   114  
   115  	eventMux *event.TypeMux
   116  
   117  	// channels for fetcher, syncer, txsyncLoop
   118  	newPeerCh   chan *peer
   119  	quitSync    chan struct{}
   120  	noMorePeers chan struct{}
   121  
   122  	syncMu   sync.Mutex
   123  	syncing  bool
   124  	syncDone chan struct{}
   125  
   126  	// wait group is used for graceful shutdowns during downloading
   127  	// and processing
   128  	wg *sync.WaitGroup
   129  }
   130  
   131  // NewProtocolManager returns a new ethereum sub protocol manager. The Ethereum sub protocol manages peers capable
   132  // with the ethereum network.
   133  func NewProtocolManager(chainConfig *params.ChainConfig, lightSync bool, networkId uint64, mux *event.TypeMux, engine consensus.Engine, peers *peerSet, blockchain BlockChain, txpool txPool, chainDb ethdb.Database, odr *LesOdr, txrelay *LesTxRelay, quitSync chan struct{}, wg *sync.WaitGroup) (*ProtocolManager, error) {
   134  	// Create the protocol manager with the base fields
   135  	manager := &ProtocolManager{
   136  		lightSync:   lightSync,
   137  		eventMux:    mux,
   138  		blockchain:  blockchain,
   139  		chainConfig: chainConfig,
   140  		chainDb:     chainDb,
   141  		odr:         odr,
   142  		networkId:   networkId,
   143  		txpool:      txpool,
   144  		txrelay:     txrelay,
   145  		peers:       peers,
   146  		newPeerCh:   make(chan *peer),
   147  		quitSync:    quitSync,
   148  		wg:          wg,
   149  		noMorePeers: make(chan struct{}),
   150  	}
   151  	if odr != nil {
   152  		manager.retriever = odr.retriever
   153  		manager.reqDist = odr.retriever.dist
   154  	}
   155  	// Initiate a sub-protocol for every implemented version we can handle
   156  	manager.SubProtocols = make([]p2p.Protocol, 0, len(ProtocolVersions))
   157  	for i, version := range ProtocolVersions {
   158  		// Compatible, initialize the sub-protocol
   159  		version := version // Closure for the run
   160  		manager.SubProtocols = append(manager.SubProtocols, p2p.Protocol{
   161  			Name:    "les",
   162  			Version: version,
   163  			Length:  ProtocolLengths[i],
   164  			Run: func(p *p2p.Peer, rw p2p.MsgReadWriter) error {
   165  				var entry *poolEntry
   166  				peer := manager.newPeer(int(version), networkId, p, rw)
   167  				if manager.serverPool != nil {
   168  					addr := p.RemoteAddr().(*net.TCPAddr)
   169  					entry = manager.serverPool.connect(peer, addr.IP, uint16(addr.Port))
   170  				}
   171  				peer.poolEntry = entry
   172  				select {
   173  				case manager.newPeerCh <- peer:
   174  					manager.wg.Add(1)
   175  					defer manager.wg.Done()
   176  					err := manager.handle(peer)
   177  					if entry != nil {
   178  						manager.serverPool.disconnect(entry)
   179  					}
   180  					return err
   181  				case <-manager.quitSync:
   182  					if entry != nil {
   183  						manager.serverPool.disconnect(entry)
   184  					}
   185  					return p2p.DiscQuitting
   186  				}
   187  			},
   188  			NodeInfo: func() interface{} {
   189  				return manager.NodeInfo()
   190  			},
   191  			PeerInfo: func(id discover.NodeID) interface{} {
   192  				if p := manager.peers.Peer(fmt.Sprintf("%x", id[:8])); p != nil {
   193  					return p.Info()
   194  				}
   195  				return nil
   196  			},
   197  		})
   198  	}
   199  	if len(manager.SubProtocols) == 0 {
   200  		return nil, errIncompatibleConfig
   201  	}
   202  
   203  	removePeer := manager.removePeer
   204  	if disableClientRemovePeer {
   205  		removePeer = func(id string) {}
   206  	}
   207  
   208  	if lightSync {
   209  		manager.downloader = downloader.New(downloader.LightSync, chainDb, manager.eventMux, nil, blockchain, removePeer)
   210  		manager.peers.notify((*downloaderPeerNotify)(manager))
   211  		manager.fetcher = newLightFetcher(manager)
   212  	}
   213  
   214  	return manager, nil
   215  }
   216  
   217  // removePeer initiates disconnection from a peer by removing it from the peer set
   218  func (pm *ProtocolManager) removePeer(id string) {
   219  	pm.peers.Unregister(id)
   220  }
   221  
   222  func (pm *ProtocolManager) Start() {
   223  	if pm.lightSync {
   224  		go pm.syncer()
   225  	} else {
   226  		go func() {
   227  			for range pm.newPeerCh {
   228  			}
   229  		}()
   230  	}
   231  }
   232  
   233  func (pm *ProtocolManager) Stop() {
   234  	// Showing a log message. During download / process this could actually
   235  	// take between 5 to 10 seconds and therefor feedback is required.
   236  	log.Info("Stopping light Ethereum protocol")
   237  
   238  	// Quit the sync loop.
   239  	// After this send has completed, no new peers will be accepted.
   240  	pm.noMorePeers <- struct{}{}
   241  
   242  	close(pm.quitSync) // quits syncer, fetcher
   243  
   244  	// Disconnect existing sessions.
   245  	// This also closes the gate for any new registrations on the peer set.
   246  	// sessions which are already established but not added to pm.peers yet
   247  	// will exit when they try to register.
   248  	pm.peers.Close()
   249  
   250  	// Wait for any process action
   251  	pm.wg.Wait()
   252  
   253  	log.Info("Light Ethereum protocol stopped")
   254  }
   255  
   256  func (pm *ProtocolManager) newPeer(pv int, nv uint64, p *p2p.Peer, rw p2p.MsgReadWriter) *peer {
   257  	return newPeer(pv, nv, p, newMeteredMsgWriter(rw))
   258  }
   259  
   260  // handle is the callback invoked to manage the life cycle of a les peer. When
   261  // this function terminates, the peer is disconnected.
   262  func (pm *ProtocolManager) handle(p *peer) error {
   263  	p.Log().Debug("Light Ethereum peer connected", "name", p.Name())
   264  
   265  	// Execute the LES handshake
   266  	td, head, genesis := pm.blockchain.Status()
   267  	headNum := core.GetBlockNumber(pm.chainDb, head)
   268  	if err := p.Handshake(td, head, headNum, genesis, pm.server); err != nil {
   269  		p.Log().Debug("Light Ethereum handshake failed", "err", err)
   270  		return err
   271  	}
   272  	if rw, ok := p.rw.(*meteredMsgReadWriter); ok {
   273  		rw.Init(p.version)
   274  	}
   275  	// Register the peer locally
   276  	if err := pm.peers.Register(p); err != nil {
   277  		p.Log().Error("Light Ethereum peer registration failed", "err", err)
   278  		return err
   279  	}
   280  	defer func() {
   281  		if pm.server != nil && pm.server.fcManager != nil && p.fcClient != nil {
   282  			p.fcClient.Remove(pm.server.fcManager)
   283  		}
   284  		pm.removePeer(p.id)
   285  	}()
   286  	// Register the peer in the downloader. If the downloader considers it banned, we disconnect
   287  	if pm.lightSync {
   288  		p.lock.Lock()
   289  		head := p.headInfo
   290  		p.lock.Unlock()
   291  		if pm.fetcher != nil {
   292  			pm.fetcher.announce(p, head)
   293  		}
   294  
   295  		if p.poolEntry != nil {
   296  			pm.serverPool.registered(p.poolEntry)
   297  		}
   298  	}
   299  
   300  	stop := make(chan struct{})
   301  	defer close(stop)
   302  	go func() {
   303  		// new block announce loop
   304  		for {
   305  			select {
   306  			case announce := <-p.announceChn:
   307  				p.SendAnnounce(announce)
   308  			case <-stop:
   309  				return
   310  			}
   311  		}
   312  	}()
   313  
   314  	// main loop. handle incoming messages.
   315  	for {
   316  		if err := pm.handleMsg(p); err != nil {
   317  			p.Log().Debug("Light Ethereum message handling failed", "err", err)
   318  			return err
   319  		}
   320  	}
   321  }
   322  
   323  var reqList = []uint64{GetBlockHeadersMsg, GetBlockBodiesMsg, GetCodeMsg, GetReceiptsMsg, GetProofsMsg, SendTxMsg, GetHeaderProofsMsg}
   324  
   325  // handleMsg is invoked whenever an inbound message is received from a remote
   326  // peer. The remote connection is torn down upon returning any error.
   327  func (pm *ProtocolManager) handleMsg(p *peer) error {
   328  	// Read the next message from the remote peer, and ensure it's fully consumed
   329  	msg, err := p.rw.ReadMsg()
   330  	if err != nil {
   331  		return err
   332  	}
   333  	p.Log().Trace("Light Ethereum message arrived", "code", msg.Code, "bytes", msg.Size)
   334  
   335  	costs := p.fcCosts[msg.Code]
   336  	reject := func(reqCnt, maxCnt uint64) bool {
   337  		if p.fcClient == nil || reqCnt > maxCnt {
   338  			return true
   339  		}
   340  		bufValue, _ := p.fcClient.AcceptRequest()
   341  		cost := costs.baseCost + reqCnt*costs.reqCost
   342  		if cost > pm.server.defParams.BufLimit {
   343  			cost = pm.server.defParams.BufLimit
   344  		}
   345  		if cost > bufValue {
   346  			recharge := time.Duration((cost - bufValue) * 1000000 / pm.server.defParams.MinRecharge)
   347  			p.Log().Error("Request came too early", "recharge", common.PrettyDuration(recharge))
   348  			return true
   349  		}
   350  		return false
   351  	}
   352  
   353  	if msg.Size > ProtocolMaxMsgSize {
   354  		return errResp(ErrMsgTooLarge, "%v > %v", msg.Size, ProtocolMaxMsgSize)
   355  	}
   356  	defer msg.Discard()
   357  
   358  	var deliverMsg *Msg
   359  
   360  	// Handle the message depending on its contents
   361  	switch msg.Code {
   362  	case StatusMsg:
   363  		p.Log().Trace("Received status message")
   364  		// Status messages should never arrive after the handshake
   365  		return errResp(ErrExtraStatusMsg, "uncontrolled status message")
   366  
   367  	// Block header query, collect the requested headers and reply
   368  	case AnnounceMsg:
   369  		p.Log().Trace("Received announce message")
   370  
   371  		var req announceData
   372  		if err := msg.Decode(&req); err != nil {
   373  			return errResp(ErrDecode, "%v: %v", msg, err)
   374  		}
   375  		p.Log().Trace("Announce message content", "number", req.Number, "hash", req.Hash, "td", req.Td, "reorg", req.ReorgDepth)
   376  		if pm.fetcher != nil {
   377  			pm.fetcher.announce(p, &req)
   378  		}
   379  
   380  	case GetBlockHeadersMsg:
   381  		p.Log().Trace("Received block header request")
   382  		// Decode the complex header query
   383  		var req struct {
   384  			ReqID uint64
   385  			Query getBlockHeadersData
   386  		}
   387  		if err := msg.Decode(&req); err != nil {
   388  			return errResp(ErrDecode, "%v: %v", msg, err)
   389  		}
   390  
   391  		query := req.Query
   392  		if reject(query.Amount, MaxHeaderFetch) {
   393  			return errResp(ErrRequestRejected, "")
   394  		}
   395  
   396  		hashMode := query.Origin.Hash != (common.Hash{})
   397  
   398  		// Gather headers until the fetch or network limits is reached
   399  		var (
   400  			bytes   common.StorageSize
   401  			headers []*types.Header
   402  			unknown bool
   403  		)
   404  		for !unknown && len(headers) < int(query.Amount) && bytes < softResponseLimit {
   405  			// Retrieve the next header satisfying the query
   406  			var origin *types.Header
   407  			if hashMode {
   408  				origin = pm.blockchain.GetHeaderByHash(query.Origin.Hash)
   409  			} else {
   410  				origin = pm.blockchain.GetHeaderByNumber(query.Origin.Number)
   411  			}
   412  			if origin == nil {
   413  				break
   414  			}
   415  			number := origin.Number.Uint64()
   416  			headers = append(headers, origin)
   417  			bytes += estHeaderRlpSize
   418  
   419  			// Advance to the next header of the query
   420  			switch {
   421  			case query.Origin.Hash != (common.Hash{}) && query.Reverse:
   422  				// Hash based traversal towards the genesis block
   423  				for i := 0; i < int(query.Skip)+1; i++ {
   424  					if header := pm.blockchain.GetHeader(query.Origin.Hash, number); header != nil {
   425  						query.Origin.Hash = header.ParentHash
   426  						number--
   427  					} else {
   428  						unknown = true
   429  						break
   430  					}
   431  				}
   432  			case query.Origin.Hash != (common.Hash{}) && !query.Reverse:
   433  				// Hash based traversal towards the leaf block
   434  				if header := pm.blockchain.GetHeaderByNumber(origin.Number.Uint64() + query.Skip + 1); header != nil {
   435  					if pm.blockchain.GetBlockHashesFromHash(header.Hash(), query.Skip+1)[query.Skip] == query.Origin.Hash {
   436  						query.Origin.Hash = header.Hash()
   437  					} else {
   438  						unknown = true
   439  					}
   440  				} else {
   441  					unknown = true
   442  				}
   443  			case query.Reverse:
   444  				// Number based traversal towards the genesis block
   445  				if query.Origin.Number >= query.Skip+1 {
   446  					query.Origin.Number -= (query.Skip + 1)
   447  				} else {
   448  					unknown = true
   449  				}
   450  
   451  			case !query.Reverse:
   452  				// Number based traversal towards the leaf block
   453  				query.Origin.Number += (query.Skip + 1)
   454  			}
   455  		}
   456  
   457  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + query.Amount*costs.reqCost)
   458  		pm.server.fcCostStats.update(msg.Code, query.Amount, rcost)
   459  		return p.SendBlockHeaders(req.ReqID, bv, headers)
   460  
   461  	case BlockHeadersMsg:
   462  		if pm.downloader == nil {
   463  			return errResp(ErrUnexpectedResponse, "")
   464  		}
   465  
   466  		p.Log().Trace("Received block header response message")
   467  		// A batch of headers arrived to one of our previous requests
   468  		var resp struct {
   469  			ReqID, BV uint64
   470  			Headers   []*types.Header
   471  		}
   472  		if err := msg.Decode(&resp); err != nil {
   473  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   474  		}
   475  		p.fcServer.GotReply(resp.ReqID, resp.BV)
   476  		if pm.fetcher != nil && pm.fetcher.requestedID(resp.ReqID) {
   477  			pm.fetcher.deliverHeaders(p, resp.ReqID, resp.Headers)
   478  		} else {
   479  			err := pm.downloader.DeliverHeaders(p.id, resp.Headers)
   480  			if err != nil {
   481  				log.Debug(fmt.Sprint(err))
   482  			}
   483  		}
   484  
   485  	case GetBlockBodiesMsg:
   486  		p.Log().Trace("Received block bodies request")
   487  		// Decode the retrieval message
   488  		var req struct {
   489  			ReqID  uint64
   490  			Hashes []common.Hash
   491  		}
   492  		if err := msg.Decode(&req); err != nil {
   493  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   494  		}
   495  		// Gather blocks until the fetch or network limits is reached
   496  		var (
   497  			bytes  int
   498  			bodies []rlp.RawValue
   499  		)
   500  		reqCnt := len(req.Hashes)
   501  		if reject(uint64(reqCnt), MaxBodyFetch) {
   502  			return errResp(ErrRequestRejected, "")
   503  		}
   504  		for _, hash := range req.Hashes {
   505  			if bytes >= softResponseLimit {
   506  				break
   507  			}
   508  			// Retrieve the requested block body, stopping if enough was found
   509  			if data := core.GetBodyRLP(pm.chainDb, hash, core.GetBlockNumber(pm.chainDb, hash)); len(data) != 0 {
   510  				bodies = append(bodies, data)
   511  				bytes += len(data)
   512  			}
   513  		}
   514  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
   515  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
   516  		return p.SendBlockBodiesRLP(req.ReqID, bv, bodies)
   517  
   518  	case BlockBodiesMsg:
   519  		if pm.odr == nil {
   520  			return errResp(ErrUnexpectedResponse, "")
   521  		}
   522  
   523  		p.Log().Trace("Received block bodies response")
   524  		// A batch of block bodies arrived to one of our previous requests
   525  		var resp struct {
   526  			ReqID, BV uint64
   527  			Data      []*types.Body
   528  		}
   529  		if err := msg.Decode(&resp); err != nil {
   530  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   531  		}
   532  		p.fcServer.GotReply(resp.ReqID, resp.BV)
   533  		deliverMsg = &Msg{
   534  			MsgType: MsgBlockBodies,
   535  			ReqID:   resp.ReqID,
   536  			Obj:     resp.Data,
   537  		}
   538  
   539  	case GetCodeMsg:
   540  		p.Log().Trace("Received code request")
   541  		// Decode the retrieval message
   542  		var req struct {
   543  			ReqID uint64
   544  			Reqs  []CodeReq
   545  		}
   546  		if err := msg.Decode(&req); err != nil {
   547  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   548  		}
   549  		// Gather state data until the fetch or network limits is reached
   550  		var (
   551  			bytes int
   552  			data  [][]byte
   553  		)
   554  		reqCnt := len(req.Reqs)
   555  		if reject(uint64(reqCnt), MaxCodeFetch) {
   556  			return errResp(ErrRequestRejected, "")
   557  		}
   558  		for _, req := range req.Reqs {
   559  			// Retrieve the requested state entry, stopping if enough was found
   560  			if header := core.GetHeader(pm.chainDb, req.BHash, core.GetBlockNumber(pm.chainDb, req.BHash)); header != nil {
   561  				if trie, _ := trie.New(header.Root, pm.chainDb); trie != nil {
   562  					sdata := trie.Get(req.AccKey)
   563  					var acc state.Account
   564  					if err := rlp.DecodeBytes(sdata, &acc); err == nil {
   565  						entry, _ := pm.chainDb.Get(acc.CodeHash)
   566  						if bytes+len(entry) >= softResponseLimit {
   567  							break
   568  						}
   569  						data = append(data, entry)
   570  						bytes += len(entry)
   571  					}
   572  				}
   573  			}
   574  		}
   575  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
   576  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
   577  		return p.SendCode(req.ReqID, bv, data)
   578  
   579  	case CodeMsg:
   580  		if pm.odr == nil {
   581  			return errResp(ErrUnexpectedResponse, "")
   582  		}
   583  
   584  		p.Log().Trace("Received code response")
   585  		// A batch of node state data arrived to one of our previous requests
   586  		var resp struct {
   587  			ReqID, BV uint64
   588  			Data      [][]byte
   589  		}
   590  		if err := msg.Decode(&resp); err != nil {
   591  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   592  		}
   593  		p.fcServer.GotReply(resp.ReqID, resp.BV)
   594  		deliverMsg = &Msg{
   595  			MsgType: MsgCode,
   596  			ReqID:   resp.ReqID,
   597  			Obj:     resp.Data,
   598  		}
   599  
   600  	case GetReceiptsMsg:
   601  		p.Log().Trace("Received receipts request")
   602  		// Decode the retrieval message
   603  		var req struct {
   604  			ReqID  uint64
   605  			Hashes []common.Hash
   606  		}
   607  		if err := msg.Decode(&req); err != nil {
   608  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   609  		}
   610  		// Gather state data until the fetch or network limits is reached
   611  		var (
   612  			bytes    int
   613  			receipts []rlp.RawValue
   614  		)
   615  		reqCnt := len(req.Hashes)
   616  		if reject(uint64(reqCnt), MaxReceiptFetch) {
   617  			return errResp(ErrRequestRejected, "")
   618  		}
   619  		for _, hash := range req.Hashes {
   620  			if bytes >= softResponseLimit {
   621  				break
   622  			}
   623  			// Retrieve the requested block's receipts, skipping if unknown to us
   624  			results := core.GetBlockReceipts(pm.chainDb, hash, core.GetBlockNumber(pm.chainDb, hash))
   625  			if results == nil {
   626  				if header := pm.blockchain.GetHeaderByHash(hash); header == nil || header.ReceiptHash != types.EmptyRootHash {
   627  					continue
   628  				}
   629  			}
   630  			// If known, encode and queue for response packet
   631  			if encoded, err := rlp.EncodeToBytes(results); err != nil {
   632  				log.Error("Failed to encode receipt", "err", err)
   633  			} else {
   634  				receipts = append(receipts, encoded)
   635  				bytes += len(encoded)
   636  			}
   637  		}
   638  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
   639  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
   640  		return p.SendReceiptsRLP(req.ReqID, bv, receipts)
   641  
   642  	case ReceiptsMsg:
   643  		if pm.odr == nil {
   644  			return errResp(ErrUnexpectedResponse, "")
   645  		}
   646  
   647  		p.Log().Trace("Received receipts response")
   648  		// A batch of receipts arrived to one of our previous requests
   649  		var resp struct {
   650  			ReqID, BV uint64
   651  			Receipts  []types.Receipts
   652  		}
   653  		if err := msg.Decode(&resp); err != nil {
   654  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   655  		}
   656  		p.fcServer.GotReply(resp.ReqID, resp.BV)
   657  		deliverMsg = &Msg{
   658  			MsgType: MsgReceipts,
   659  			ReqID:   resp.ReqID,
   660  			Obj:     resp.Receipts,
   661  		}
   662  
   663  	case GetProofsMsg:
   664  		p.Log().Trace("Received proofs request")
   665  		// Decode the retrieval message
   666  		var req struct {
   667  			ReqID uint64
   668  			Reqs  []ProofReq
   669  		}
   670  		if err := msg.Decode(&req); err != nil {
   671  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   672  		}
   673  		// Gather state data until the fetch or network limits is reached
   674  		var (
   675  			bytes  int
   676  			proofs proofsData
   677  		)
   678  		reqCnt := len(req.Reqs)
   679  		if reject(uint64(reqCnt), MaxProofsFetch) {
   680  			return errResp(ErrRequestRejected, "")
   681  		}
   682  		for _, req := range req.Reqs {
   683  			if bytes >= softResponseLimit {
   684  				break
   685  			}
   686  			// Retrieve the requested state entry, stopping if enough was found
   687  			if header := core.GetHeader(pm.chainDb, req.BHash, core.GetBlockNumber(pm.chainDb, req.BHash)); header != nil {
   688  				if tr, _ := trie.New(header.Root, pm.chainDb); tr != nil {
   689  					if len(req.AccKey) > 0 {
   690  						sdata := tr.Get(req.AccKey)
   691  						tr = nil
   692  						var acc state.Account
   693  						if err := rlp.DecodeBytes(sdata, &acc); err == nil {
   694  							tr, _ = trie.New(acc.Root, pm.chainDb)
   695  						}
   696  					}
   697  					if tr != nil {
   698  						proof := tr.Prove(req.Key)
   699  						proofs = append(proofs, proof)
   700  						bytes += len(proof)
   701  					}
   702  				}
   703  			}
   704  		}
   705  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
   706  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
   707  		return p.SendProofs(req.ReqID, bv, proofs)
   708  
   709  	case ProofsMsg:
   710  		if pm.odr == nil {
   711  			return errResp(ErrUnexpectedResponse, "")
   712  		}
   713  
   714  		p.Log().Trace("Received proofs response")
   715  		// A batch of merkle proofs arrived to one of our previous requests
   716  		var resp struct {
   717  			ReqID, BV uint64
   718  			Data      [][]rlp.RawValue
   719  		}
   720  		if err := msg.Decode(&resp); err != nil {
   721  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   722  		}
   723  		p.fcServer.GotReply(resp.ReqID, resp.BV)
   724  		deliverMsg = &Msg{
   725  			MsgType: MsgProofs,
   726  			ReqID:   resp.ReqID,
   727  			Obj:     resp.Data,
   728  		}
   729  
   730  	case GetHeaderProofsMsg:
   731  		p.Log().Trace("Received headers proof request")
   732  		// Decode the retrieval message
   733  		var req struct {
   734  			ReqID uint64
   735  			Reqs  []ChtReq
   736  		}
   737  		if err := msg.Decode(&req); err != nil {
   738  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   739  		}
   740  		// Gather state data until the fetch or network limits is reached
   741  		var (
   742  			bytes  int
   743  			proofs []ChtResp
   744  		)
   745  		reqCnt := len(req.Reqs)
   746  		if reject(uint64(reqCnt), MaxHeaderProofsFetch) {
   747  			return errResp(ErrRequestRejected, "")
   748  		}
   749  		for _, req := range req.Reqs {
   750  			if bytes >= softResponseLimit {
   751  				break
   752  			}
   753  
   754  			if header := pm.blockchain.GetHeaderByNumber(req.BlockNum); header != nil {
   755  				if root := getChtRoot(pm.chainDb, req.ChtNum); root != (common.Hash{}) {
   756  					if tr, _ := trie.New(root, pm.chainDb); tr != nil {
   757  						var encNumber [8]byte
   758  						binary.BigEndian.PutUint64(encNumber[:], req.BlockNum)
   759  						proof := tr.Prove(encNumber[:])
   760  						proofs = append(proofs, ChtResp{Header: header, Proof: proof})
   761  						bytes += len(proof) + estHeaderRlpSize
   762  					}
   763  				}
   764  			}
   765  		}
   766  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
   767  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
   768  		return p.SendHeaderProofs(req.ReqID, bv, proofs)
   769  
   770  	case HeaderProofsMsg:
   771  		if pm.odr == nil {
   772  			return errResp(ErrUnexpectedResponse, "")
   773  		}
   774  
   775  		p.Log().Trace("Received headers proof response")
   776  		var resp struct {
   777  			ReqID, BV uint64
   778  			Data      []ChtResp
   779  		}
   780  		if err := msg.Decode(&resp); err != nil {
   781  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   782  		}
   783  		p.fcServer.GotReply(resp.ReqID, resp.BV)
   784  		deliverMsg = &Msg{
   785  			MsgType: MsgHeaderProofs,
   786  			ReqID:   resp.ReqID,
   787  			Obj:     resp.Data,
   788  		}
   789  
   790  	case SendTxMsg:
   791  		if pm.txpool == nil {
   792  			return errResp(ErrUnexpectedResponse, "")
   793  		}
   794  		// Transactions arrived, parse all of them and deliver to the pool
   795  		var txs []*types.Transaction
   796  		if err := msg.Decode(&txs); err != nil {
   797  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   798  		}
   799  		reqCnt := len(txs)
   800  		if reject(uint64(reqCnt), MaxTxSend) {
   801  			return errResp(ErrRequestRejected, "")
   802  		}
   803  
   804  		if err := pm.txpool.AddRemotes(txs); err != nil {
   805  			return errResp(ErrUnexpectedResponse, "msg: %v", err)
   806  		}
   807  
   808  		_, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
   809  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
   810  
   811  	default:
   812  		p.Log().Trace("Received unknown message", "code", msg.Code)
   813  		return errResp(ErrInvalidMsgCode, "%v", msg.Code)
   814  	}
   815  
   816  	if deliverMsg != nil {
   817  		err := pm.retriever.deliver(p, deliverMsg)
   818  		if err != nil {
   819  			p.responseErrors++
   820  			if p.responseErrors > maxResponseErrors {
   821  				return err
   822  			}
   823  		}
   824  	}
   825  	return nil
   826  }
   827  
   828  // NodeInfo retrieves some protocol metadata about the running host node.
   829  func (self *ProtocolManager) NodeInfo() *eth.EthNodeInfo {
   830  	return &eth.EthNodeInfo{
   831  		Network:    self.networkId,
   832  		Difficulty: self.blockchain.GetTdByHash(self.blockchain.LastBlockHash()),
   833  		Genesis:    self.blockchain.Genesis().Hash(),
   834  		Head:       self.blockchain.LastBlockHash(),
   835  	}
   836  }
   837  
   838  // downloaderPeerNotify implements peerSetNotify
   839  type downloaderPeerNotify ProtocolManager
   840  
   841  type peerConnection struct {
   842  	manager *ProtocolManager
   843  	peer    *peer
   844  }
   845  
   846  func (pc *peerConnection) Head() (common.Hash, *big.Int) {
   847  	return pc.peer.HeadAndTd()
   848  }
   849  
   850  func (pc *peerConnection) RequestHeadersByHash(origin common.Hash, amount int, skip int, reverse bool) error {
   851  	reqID := genReqID()
   852  	rq := &distReq{
   853  		getCost: func(dp distPeer) uint64 {
   854  			peer := dp.(*peer)
   855  			return peer.GetRequestCost(GetBlockHeadersMsg, amount)
   856  		},
   857  		canSend: func(dp distPeer) bool {
   858  			return dp.(*peer) == pc.peer
   859  		},
   860  		request: func(dp distPeer) func() {
   861  			peer := dp.(*peer)
   862  			cost := peer.GetRequestCost(GetBlockHeadersMsg, amount)
   863  			peer.fcServer.QueueRequest(reqID, cost)
   864  			return func() { peer.RequestHeadersByHash(reqID, cost, origin, amount, skip, reverse) }
   865  		},
   866  	}
   867  	_, ok := <-pc.manager.reqDist.queue(rq)
   868  	if !ok {
   869  		return ErrNoPeers
   870  	}
   871  	return nil
   872  }
   873  
   874  func (pc *peerConnection) RequestHeadersByNumber(origin uint64, amount int, skip int, reverse bool) error {
   875  	reqID := genReqID()
   876  	rq := &distReq{
   877  		getCost: func(dp distPeer) uint64 {
   878  			peer := dp.(*peer)
   879  			return peer.GetRequestCost(GetBlockHeadersMsg, amount)
   880  		},
   881  		canSend: func(dp distPeer) bool {
   882  			return dp.(*peer) == pc.peer
   883  		},
   884  		request: func(dp distPeer) func() {
   885  			peer := dp.(*peer)
   886  			cost := peer.GetRequestCost(GetBlockHeadersMsg, amount)
   887  			peer.fcServer.QueueRequest(reqID, cost)
   888  			return func() { peer.RequestHeadersByNumber(reqID, cost, origin, amount, skip, reverse) }
   889  		},
   890  	}
   891  	_, ok := <-pc.manager.reqDist.queue(rq)
   892  	if !ok {
   893  		return ErrNoPeers
   894  	}
   895  	return nil
   896  }
   897  
   898  func (d *downloaderPeerNotify) registerPeer(p *peer) {
   899  	pm := (*ProtocolManager)(d)
   900  	pc := &peerConnection{
   901  		manager: pm,
   902  		peer:    p,
   903  	}
   904  	pm.downloader.RegisterLightPeer(p.id, ethVersion, pc)
   905  }
   906  
   907  func (d *downloaderPeerNotify) unregisterPeer(p *peer) {
   908  	pm := (*ProtocolManager)(d)
   909  	pm.downloader.UnregisterPeer(p.id)
   910  }