| 
									
										
										
										
											2017-04-05 01:16:29 +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/>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package ethash | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2018-01-23 11:05:30 +01:00
										 |  |  | 	"io/ioutil" | 
					
						
							| 
									
										
										
										
											2017-04-05 01:16:29 +03:00
										 |  |  | 	"math/big" | 
					
						
							| 
									
										
										
										
											2018-01-23 11:05:30 +01:00
										 |  |  | 	"math/rand" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2017-04-05 01:16:29 +03:00
										 |  |  | 	"testing" | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2017-04-05 01:16:29 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/common/hexutil" | 
					
						
							| 
									
										
										
										
											2017-04-05 01:16:29 +03:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/types" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Tests that ethash works correctly in test mode. | 
					
						
							|  |  |  | func TestTestMode(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	header := &types.Header{Number: big.NewInt(1), Difficulty: big.NewInt(100)} | 
					
						
							| 
									
										
										
										
											2017-04-05 01:16:29 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 	ethash := NewTester(nil, false) | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	defer ethash.Close() | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 	results := make(chan *types.Block) | 
					
						
							|  |  |  | 	err := ethash.Seal(nil, types.NewBlockWithHeader(header), results, nil) | 
					
						
							| 
									
										
										
										
											2017-04-05 01:16:29 +03:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("failed to seal block: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 	select { | 
					
						
							|  |  |  | 	case block := <-results: | 
					
						
							|  |  |  | 		header.Nonce = types.EncodeNonce(block.Nonce()) | 
					
						
							|  |  |  | 		header.MixDigest = block.MixDigest() | 
					
						
							| 
									
										
										
										
											2021-02-05 19:44:34 +01:00
										 |  |  | 		if err := ethash.verifySeal(nil, header, false); err != nil { | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 			t.Fatalf("unexpected verification error: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-01-13 11:44:20 +01:00
										 |  |  | 	case <-time.NewTimer(4 * time.Second).C: | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 		t.Error("sealing result timeout") | 
					
						
							| 
									
										
										
										
											2017-04-05 01:16:29 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-01-23 11:05:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | // This test checks that cache lru logic doesn't crash under load. | 
					
						
							|  |  |  | // It reproduces https://github.com/ethereum/go-ethereum/issues/14943 | 
					
						
							|  |  |  | func TestCacheFileEvict(t *testing.T) { | 
					
						
							|  |  |  | 	tmpdir, err := ioutil.TempDir("", "ethash-test") | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer os.RemoveAll(tmpdir) | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 	e := New(Config{CachesInMem: 3, CachesOnDisk: 10, CacheDir: tmpdir, PowMode: ModeTest}, nil, false) | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	defer e.Close() | 
					
						
							| 
									
										
										
										
											2018-01-23 11:05:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	workers := 8 | 
					
						
							|  |  |  | 	epochs := 100 | 
					
						
							|  |  |  | 	var wg sync.WaitGroup | 
					
						
							|  |  |  | 	wg.Add(workers) | 
					
						
							|  |  |  | 	for i := 0; i < workers; i++ { | 
					
						
							|  |  |  | 		go verifyTest(&wg, e, i, epochs) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	wg.Wait() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func verifyTest(wg *sync.WaitGroup, e *Ethash, workerIndex, epochs int) { | 
					
						
							|  |  |  | 	defer wg.Done() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const wiggle = 4 * epochLength | 
					
						
							|  |  |  | 	r := rand.New(rand.NewSource(int64(workerIndex))) | 
					
						
							|  |  |  | 	for epoch := 0; epoch < epochs; epoch++ { | 
					
						
							|  |  |  | 		block := int64(epoch)*epochLength - wiggle/2 + r.Int63n(wiggle) | 
					
						
							|  |  |  | 		if block < 0 { | 
					
						
							|  |  |  | 			block = 0 | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 		header := &types.Header{Number: big.NewInt(block), Difficulty: big.NewInt(100)} | 
					
						
							| 
									
										
										
										
											2021-02-05 19:44:34 +01:00
										 |  |  | 		e.verifySeal(nil, header, false) | 
					
						
							| 
									
										
										
										
											2018-01-23 11:05:30 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func TestRemoteSealer(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 	ethash := NewTester(nil, false) | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	defer ethash.Close() | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	api := &API{ethash} | 
					
						
							|  |  |  | 	if _, err := api.GetWork(); err != errNoMiningWork { | 
					
						
							|  |  |  | 		t.Error("expect to return an error indicate there is no mining work") | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	header := &types.Header{Number: big.NewInt(1), Difficulty: big.NewInt(100)} | 
					
						
							|  |  |  | 	block := types.NewBlockWithHeader(header) | 
					
						
							| 
									
										
										
										
											2018-08-23 21:02:57 +08:00
										 |  |  | 	sealhash := ethash.SealHash(header) | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Push new work. | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 	results := make(chan *types.Block) | 
					
						
							|  |  |  | 	ethash.Seal(nil, block, results, nil) | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	var ( | 
					
						
							| 
									
										
										
										
											2018-11-08 16:08:57 +01:00
										 |  |  | 		work [4]string | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 		err  error | 
					
						
							|  |  |  | 	) | 
					
						
							| 
									
										
										
										
											2018-08-23 21:02:57 +08:00
										 |  |  | 	if work, err = api.GetWork(); err != nil || work[0] != sealhash.Hex() { | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 		t.Error("expect to return a mining work has same hash") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-23 21:02:57 +08:00
										 |  |  | 	if res := api.SubmitWork(types.BlockNonce{}, sealhash, common.Hash{}); res { | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 		t.Error("expect to return false when submit a fake solution") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Push new block with same block number to replace the original one. | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	header = &types.Header{Number: big.NewInt(1), Difficulty: big.NewInt(1000)} | 
					
						
							|  |  |  | 	block = types.NewBlockWithHeader(header) | 
					
						
							| 
									
										
										
										
											2018-08-23 21:02:57 +08:00
										 |  |  | 	sealhash = ethash.SealHash(header) | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 	ethash.Seal(nil, block, results, nil) | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-23 21:02:57 +08:00
										 |  |  | 	if work, err = api.GetWork(); err != nil || work[0] != sealhash.Hex() { | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 		t.Error("expect to return the latest pushed work") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestHashRate(t *testing.T) { | 
					
						
							|  |  |  | 	var ( | 
					
						
							|  |  |  | 		hashrate = []hexutil.Uint64{100, 200, 300} | 
					
						
							|  |  |  | 		expect   uint64 | 
					
						
							|  |  |  | 		ids      = []common.Hash{common.HexToHash("a"), common.HexToHash("b"), common.HexToHash("c")} | 
					
						
							|  |  |  | 	) | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 	ethash := NewTester(nil, false) | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	defer ethash.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if tot := ethash.Hashrate(); tot != 0 { | 
					
						
							|  |  |  | 		t.Error("expect the result should be zero") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	api := &API{ethash} | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	for i := 0; i < len(hashrate); i += 1 { | 
					
						
							|  |  |  | 		if res := api.SubmitHashRate(hashrate[i], ids[i]); !res { | 
					
						
							|  |  |  | 			t.Error("remote miner submit hashrate failed") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		expect += uint64(hashrate[i]) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if tot := ethash.Hashrate(); tot != float64(expect) { | 
					
						
							|  |  |  | 		t.Error("expect total hashrate should be same") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestClosedRemoteSealer(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 	ethash := NewTester(nil, false) | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	time.Sleep(1 * time.Second) // ensure exit channel is listening | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	ethash.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	api := &API{ethash} | 
					
						
							|  |  |  | 	if _, err := api.GetWork(); err != errEthashStopped { | 
					
						
							|  |  |  | 		t.Error("expect to return an error to indicate ethash is stopped") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if res := api.SubmitHashRate(hexutil.Uint64(100), common.HexToHash("a")); res { | 
					
						
							|  |  |  | 		t.Error("expect to return false when submit hashrate to a stopped ethash") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |