algorithm_go1.8.go 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263
  1. // Copyright 2017 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. // +build go1.8
  17. package ethash
  18. import "math/big"
  19. // cacheSize returns the size of the ethash verification cache that belongs to a certain
  20. // block number.
  21. func cacheSize(block uint64) uint64 {
  22. epoch := int(block / epochLength)
  23. if epoch < maxEpoch {
  24. return cacheSizes[epoch]
  25. }
  26. return calcCacheSize(epoch)
  27. }
  28. // calcCacheSize calculates the cache size for epoch. The cache size grows linearly,
  29. // however, we always take the highest prime below the linearly growing threshold in order
  30. // to reduce the risk of accidental regularities leading to cyclic behavior.
  31. func calcCacheSize(epoch int) uint64 {
  32. size := cacheInitBytes + cacheGrowthBytes*uint64(epoch) - hashBytes
  33. for !new(big.Int).SetUint64(size / hashBytes).ProbablyPrime(1) { // Always accurate for n < 2^64
  34. size -= 2 * hashBytes
  35. }
  36. return size
  37. }
  38. // datasetSize returns the size of the ethash mining dataset that belongs to a certain
  39. // block number.
  40. func datasetSize(block uint64) uint64 {
  41. epoch := int(block / epochLength)
  42. if epoch < maxEpoch {
  43. return datasetSizes[epoch]
  44. }
  45. return calcDatasetSize(epoch)
  46. }
  47. // calcDatasetSize calculates the dataset size for epoch. The dataset size grows linearly,
  48. // however, we always take the highest prime below the linearly growing threshold in order
  49. // to reduce the risk of accidental regularities leading to cyclic behavior.
  50. func calcDatasetSize(epoch int) uint64 {
  51. size := datasetInitBytes + datasetGrowthBytes*uint64(epoch) - mixBytes
  52. for !new(big.Int).SetUint64(size / mixBytes).ProbablyPrime(1) { // Always accurate for n < 2^64
  53. size -= 2 * mixBytes
  54. }
  55. return size
  56. }