gonum.org/v1/gonum@v0.14.0/mat/pool.go (about)

     1  // Copyright ©2014 The Gonum Authors. All rights reserved.
     2  // Use of this source code is governed by a BSD-style
     3  // license that can be found in the LICENSE file.
     4  
     5  package mat
     6  
     7  import (
     8  	"math/bits"
     9  	"sync"
    10  
    11  	"gonum.org/v1/gonum/blas"
    12  	"gonum.org/v1/gonum/blas/blas64"
    13  	"gonum.org/v1/gonum/blas/cblas128"
    14  )
    15  
    16  // poolFor returns the ceiling of base 2 log of size. It provides an index
    17  // into a pool array to a sync.Pool that will return values able to hold
    18  // size elements.
    19  func poolFor(size uint) int {
    20  	if size == 0 {
    21  		return 0
    22  	}
    23  	return bits.Len(size - 1)
    24  }
    25  
    26  var (
    27  	// poolDense contains size stratified workspace Dense pools.
    28  	// Each poolDense element i returns sized matrices with a data
    29  	// slice capped at 1<<i.
    30  	poolDense [63]sync.Pool
    31  
    32  	// poolSymDense is the SymDense equivalent of poolDense.
    33  	poolSymDense [63]sync.Pool
    34  
    35  	// poolTriDense is the TriDense equivalent of poolDense.
    36  	poolTriDense [63]sync.Pool
    37  
    38  	// poolVecDense is the VecDense equivalent of poolDense.
    39  	poolVecDense [63]sync.Pool
    40  
    41  	// poolCDense is the CDense equivalent of poolDense.
    42  	poolCDense [63]sync.Pool
    43  
    44  	// poolFloat64s is the []float64 equivalent of poolDense.
    45  	poolFloat64s [63]sync.Pool
    46  
    47  	// poolInts is the []int equivalent of poolDense.
    48  	poolInts [63]sync.Pool
    49  )
    50  
    51  func init() {
    52  	for i := range poolDense {
    53  		l := 1 << uint(i)
    54  		// Real matrix pools.
    55  		poolDense[i].New = func() interface{} {
    56  			return &Dense{mat: blas64.General{
    57  				Data: make([]float64, l),
    58  			}}
    59  		}
    60  		poolSymDense[i].New = func() interface{} {
    61  			return &SymDense{mat: blas64.Symmetric{
    62  				Uplo: blas.Upper,
    63  				Data: make([]float64, l),
    64  			}}
    65  		}
    66  		poolTriDense[i].New = func() interface{} {
    67  			return &TriDense{mat: blas64.Triangular{
    68  				Data: make([]float64, l),
    69  			}}
    70  		}
    71  		poolVecDense[i].New = func() interface{} {
    72  			return &VecDense{mat: blas64.Vector{
    73  				Inc:  1,
    74  				Data: make([]float64, l),
    75  			}}
    76  		}
    77  
    78  		// Complex matrix pools.
    79  		poolCDense[i].New = func() interface{} {
    80  			return &CDense{mat: cblas128.General{
    81  				Data: make([]complex128, l),
    82  			}}
    83  		}
    84  
    85  		// Helper pools.
    86  		poolFloat64s[i].New = func() interface{} {
    87  			s := make([]float64, l)
    88  			return &s
    89  		}
    90  		poolInts[i].New = func() interface{} {
    91  			s := make([]int, l)
    92  			return &s
    93  		}
    94  	}
    95  }
    96  
    97  // getDenseWorkspace returns a *Dense of size r×c and a data slice
    98  // with a cap that is less than 2*r*c. If clear is true, the
    99  // data slice visible through the Matrix interface is zeroed.
   100  func getDenseWorkspace(r, c int, clear bool) *Dense {
   101  	l := uint(r * c)
   102  	w := poolDense[poolFor(l)].Get().(*Dense)
   103  	w.mat.Data = w.mat.Data[:l]
   104  	if clear {
   105  		zero(w.mat.Data)
   106  	}
   107  	w.mat.Rows = r
   108  	w.mat.Cols = c
   109  	w.mat.Stride = c
   110  	w.capRows = r
   111  	w.capCols = c
   112  	return w
   113  }
   114  
   115  // putDenseWorkspace replaces a used *Dense into the appropriate size
   116  // workspace pool. putDenseWorkspace must not be called with a matrix
   117  // where references to the underlying data slice have been kept.
   118  func putDenseWorkspace(w *Dense) {
   119  	poolDense[poolFor(uint(cap(w.mat.Data)))].Put(w)
   120  }
   121  
   122  // getSymDenseWorkspace returns a *SymDense of size n and a cap that
   123  // is less than 2*n. If clear is true, the data slice visible
   124  // through the Matrix interface is zeroed.
   125  func getSymDenseWorkspace(n int, clear bool) *SymDense {
   126  	l := uint(n)
   127  	l *= l
   128  	s := poolSymDense[poolFor(l)].Get().(*SymDense)
   129  	s.mat.Data = s.mat.Data[:l]
   130  	if clear {
   131  		zero(s.mat.Data)
   132  	}
   133  	s.mat.N = n
   134  	s.mat.Stride = n
   135  	s.cap = n
   136  	return s
   137  }
   138  
   139  // putSymDenseWorkspace replaces a used *SymDense into the appropriate size
   140  // workspace pool. putSymDenseWorkspace must not be called with a matrix
   141  // where references to the underlying data slice have been kept.
   142  func putSymDenseWorkspace(s *SymDense) {
   143  	poolSymDense[poolFor(uint(cap(s.mat.Data)))].Put(s)
   144  }
   145  
   146  // getTriDenseWorkspace returns a *TriDense of size n and a cap that
   147  // is less than 2*n. If clear is true, the data slice visible
   148  // through the Matrix interface is zeroed.
   149  func getTriDenseWorkspace(n int, kind TriKind, clear bool) *TriDense {
   150  	l := uint(n)
   151  	l *= l
   152  	t := poolTriDense[poolFor(l)].Get().(*TriDense)
   153  	t.mat.Data = t.mat.Data[:l]
   154  	if clear {
   155  		zero(t.mat.Data)
   156  	}
   157  	t.mat.N = n
   158  	t.mat.Stride = n
   159  	if kind == Upper {
   160  		t.mat.Uplo = blas.Upper
   161  	} else if kind == Lower {
   162  		t.mat.Uplo = blas.Lower
   163  	} else {
   164  		panic(ErrTriangle)
   165  	}
   166  	t.mat.Diag = blas.NonUnit
   167  	t.cap = n
   168  	return t
   169  }
   170  
   171  // putTriWorkspace replaces a used *TriDense into the appropriate size
   172  // workspace pool. putTriWorkspace must not be called with a matrix
   173  // where references to the underlying data slice have been kept.
   174  func putTriWorkspace(t *TriDense) {
   175  	poolTriDense[poolFor(uint(cap(t.mat.Data)))].Put(t)
   176  }
   177  
   178  // getVecDenseWorkspace returns a *VecDense of length n and a cap that
   179  // is less than 2*n. If clear is true, the data slice visible
   180  // through the Matrix interface is zeroed.
   181  func getVecDenseWorkspace(n int, clear bool) *VecDense {
   182  	l := uint(n)
   183  	v := poolVecDense[poolFor(l)].Get().(*VecDense)
   184  	v.mat.Data = v.mat.Data[:l]
   185  	if clear {
   186  		zero(v.mat.Data)
   187  	}
   188  	v.mat.N = n
   189  	return v
   190  }
   191  
   192  // putVecDenseWorkspace replaces a used *VecDense into the appropriate size
   193  // workspace pool. putVecDenseWorkspace must not be called with a matrix
   194  // where references to the underlying data slice have been kept.
   195  func putVecDenseWorkspace(v *VecDense) {
   196  	poolVecDense[poolFor(uint(cap(v.mat.Data)))].Put(v)
   197  }
   198  
   199  // getCDenseWorkspace returns a *CDense of size r×c and a data slice
   200  // with a cap that is less than 2*r*c. If clear is true, the
   201  // data slice visible through the CMatrix interface is zeroed.
   202  func getCDenseWorkspace(r, c int, clear bool) *CDense {
   203  	l := uint(r * c)
   204  	w := poolCDense[poolFor(l)].Get().(*CDense)
   205  	w.mat.Data = w.mat.Data[:l]
   206  	if clear {
   207  		zeroC(w.mat.Data)
   208  	}
   209  	w.mat.Rows = r
   210  	w.mat.Cols = c
   211  	w.mat.Stride = c
   212  	w.capRows = r
   213  	w.capCols = c
   214  	return w
   215  }
   216  
   217  // putCDenseWorkspace replaces a used *CDense into the appropriate size
   218  // workspace pool. putWorkspace must not be called with a matrix
   219  // where references to the underlying data slice have been kept.
   220  func putCDenseWorkspace(w *CDense) {
   221  	poolCDense[poolFor(uint(cap(w.mat.Data)))].Put(w)
   222  }
   223  
   224  // getFloat64s returns a []float64 of length l and a cap that is
   225  // less than 2*l. If clear is true, the slice visible is zeroed.
   226  func getFloat64s(l int, clear bool) []float64 {
   227  	w := *poolFloat64s[poolFor(uint(l))].Get().(*[]float64)
   228  	w = w[:l]
   229  	if clear {
   230  		zero(w)
   231  	}
   232  	return w
   233  }
   234  
   235  // putFloat64s replaces a used []float64 into the appropriate size
   236  // workspace pool. putFloat64s must not be called with a slice
   237  // where references to the underlying data have been kept.
   238  func putFloat64s(w []float64) {
   239  	poolFloat64s[poolFor(uint(cap(w)))].Put(&w)
   240  }
   241  
   242  // getInts returns a []int of length l and a cap that is
   243  // less than 2*l. If clear is true, the slice visible is zeroed.
   244  func getInts(l int, clear bool) []int {
   245  	w := *poolInts[poolFor(uint(l))].Get().(*[]int)
   246  	w = w[:l]
   247  	if clear {
   248  		for i := range w {
   249  			w[i] = 0
   250  		}
   251  	}
   252  	return w
   253  }
   254  
   255  // putInts replaces a used []int into the appropriate size
   256  // workspace pool. putInts must not be called with a slice
   257  // where references to the underlying data have been kept.
   258  func putInts(w []int) {
   259  	poolInts[poolFor(uint(cap(w)))].Put(&w)
   260  }