intpool.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  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. package vm
  17. import (
  18. "math/big"
  19. "sync"
  20. )
  21. var checkVal = big.NewInt(-42)
  22. const poolLimit = 256
  23. // intPool is a pool of big integers that
  24. // can be reused for all big.Int operations.
  25. type intPool struct {
  26. pool *Stack
  27. }
  28. func newIntPool() *intPool {
  29. return &intPool{pool: newstack()}
  30. }
  31. // get retrieves a big int from the pool, allocating one if the pool is empty.
  32. // Note, the returned int's value is arbitrary and will not be zeroed!
  33. func (p *intPool) get() *big.Int {
  34. if p.pool.len() > 0 {
  35. return p.pool.pop()
  36. }
  37. return new(big.Int)
  38. }
  39. // getZero retrieves a big int from the pool, setting it to zero or allocating
  40. // a new one if the pool is empty.
  41. func (p *intPool) getZero() *big.Int {
  42. if p.pool.len() > 0 {
  43. return p.pool.pop().SetUint64(0)
  44. }
  45. return new(big.Int)
  46. }
  47. // put returns an allocated big int to the pool to be later reused by get calls.
  48. // Note, the values as saved as is; neither put nor get zeroes the ints out!
  49. func (p *intPool) put(is ...*big.Int) {
  50. if len(p.pool.data) > poolLimit {
  51. return
  52. }
  53. for _, i := range is {
  54. // verifyPool is a build flag. Pool verification makes sure the integrity
  55. // of the integer pool by comparing values to a default value.
  56. if verifyPool {
  57. i.Set(checkVal)
  58. }
  59. p.pool.push(i)
  60. }
  61. }
  62. // The intPool pool's default capacity
  63. const poolDefaultCap = 25
  64. // intPoolPool manages a pool of intPools.
  65. type intPoolPool struct {
  66. pools []*intPool
  67. lock sync.Mutex
  68. }
  69. var poolOfIntPools = &intPoolPool{
  70. pools: make([]*intPool, 0, poolDefaultCap),
  71. }
  72. // get is looking for an available pool to return.
  73. func (ipp *intPoolPool) get() *intPool {
  74. ipp.lock.Lock()
  75. defer ipp.lock.Unlock()
  76. if len(poolOfIntPools.pools) > 0 {
  77. ip := ipp.pools[len(ipp.pools)-1]
  78. ipp.pools = ipp.pools[:len(ipp.pools)-1]
  79. return ip
  80. }
  81. return newIntPool()
  82. }
  83. // put a pool that has been allocated with get.
  84. func (ipp *intPoolPool) put(ip *intPool) {
  85. ipp.lock.Lock()
  86. defer ipp.lock.Unlock()
  87. if len(ipp.pools) < cap(ipp.pools) {
  88. ipp.pools = append(ipp.pools, ip)
  89. }
  90. }