intpool.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  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. // putOne 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. // As opposed to 'put' with variadic args, this method becomes inlined by the
  50. // go compiler
  51. func (p *intPool) putOne(i *big.Int) {
  52. if len(p.pool.data) > poolLimit {
  53. return
  54. }
  55. p.pool.push(i)
  56. }
  57. // put returns an allocated big int to the pool to be later reused by get calls.
  58. // Note, the values as saved as is; neither put nor get zeroes the ints out!
  59. func (p *intPool) put(is ...*big.Int) {
  60. if len(p.pool.data) > poolLimit {
  61. return
  62. }
  63. for _, i := range is {
  64. // verifyPool is a build flag. Pool verification makes sure the integrity
  65. // of the integer pool by comparing values to a default value.
  66. if verifyPool {
  67. i.Set(checkVal)
  68. }
  69. p.pool.push(i)
  70. }
  71. }
  72. // The intPool pool's default capacity
  73. const poolDefaultCap = 25
  74. // intPoolPool manages a pool of intPools.
  75. type intPoolPool struct {
  76. pools []*intPool
  77. lock sync.Mutex
  78. }
  79. var poolOfIntPools = &intPoolPool{
  80. pools: make([]*intPool, 0, poolDefaultCap),
  81. }
  82. // get is looking for an available pool to return.
  83. func (ipp *intPoolPool) get() *intPool {
  84. ipp.lock.Lock()
  85. defer ipp.lock.Unlock()
  86. if len(poolOfIntPools.pools) > 0 {
  87. ip := ipp.pools[len(ipp.pools)-1]
  88. ipp.pools = ipp.pools[:len(ipp.pools)-1]
  89. return ip
  90. }
  91. return newIntPool()
  92. }
  93. // put a pool that has been allocated with get.
  94. func (ipp *intPoolPool) put(ip *intPool) {
  95. ipp.lock.Lock()
  96. defer ipp.lock.Unlock()
  97. if len(ipp.pools) < cap(ipp.pools) {
  98. ipp.pools = append(ipp.pools, ip)
  99. }
  100. }