mirror of
				https://github.com/gonum/gonum.git
				synced 2025-10-25 16:20:25 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			120 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			120 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright ©2015 The gonum Authors. All rights reserved.
 | |
| // Use of this source code is governed by a BSD-style
 | |
| // license that can be found in the LICENSE file.
 | |
| 
 | |
| package native
 | |
| 
 | |
| import (
 | |
| 	"math"
 | |
| 
 | |
| 	"github.com/gonum/lapack"
 | |
| )
 | |
| 
 | |
| // Implementation is the native Go implementation of LAPACK routines. It
 | |
| // is built on top of calls to the return of blas64.Implementation(), so while
 | |
| // this code is in pure Go, the underlying BLAS implementation may not be.
 | |
| type Implementation struct{}
 | |
| 
 | |
| var _ lapack.Float64 = Implementation{}
 | |
| 
 | |
| // This list is duplicated in lapack/cgo. Keep in sync.
 | |
| const (
 | |
| 	absIncNotOne    = "lapack: increment not one or negative one"
 | |
| 	badD            = "lapack: d has insufficient length"
 | |
| 	badDecompUpdate = "lapack: bad decomp update"
 | |
| 	badDiag         = "lapack: bad diag"
 | |
| 	badDims         = "lapack: bad input dimensions"
 | |
| 	badDirect       = "lapack: bad direct"
 | |
| 	badE            = "lapack: e has insufficient length"
 | |
| 	badEigComp      = "lapack: bad EigComp"
 | |
| 	badEVSide       = "lapack: bad EVSide"
 | |
| 	badHowMany      = "lapack: bad HowMany"
 | |
| 	badIlo          = "lapack: ilo out of range"
 | |
| 	badIhi          = "lapack: ihi out of range"
 | |
| 	badIpiv         = "lapack: insufficient permutation length"
 | |
| 	badJob          = "lapack: bad Job"
 | |
| 	badLdA          = "lapack: index of a out of range"
 | |
| 	badNorm         = "lapack: bad norm"
 | |
| 	badPivot        = "lapack: bad pivot"
 | |
| 	badS            = "lapack: s has insufficient length"
 | |
| 	badShifts       = "lapack: bad shifts"
 | |
| 	badSide         = "lapack: bad side"
 | |
| 	badSlice        = "lapack: bad input slice length"
 | |
| 	badStore        = "lapack: bad store"
 | |
| 	badTau          = "lapack: tau has insufficient length"
 | |
| 	badTauQ         = "lapack: tauQ has insufficient length"
 | |
| 	badTauP         = "lapack: tauP has insufficient length"
 | |
| 	badTrans        = "lapack: bad trans"
 | |
| 	badUplo         = "lapack: illegal triangle"
 | |
| 	badWork         = "lapack: insufficient working memory"
 | |
| 	badWorkStride   = "lapack: insufficient working array stride"
 | |
| 	badZ            = "lapack: insufficient z length"
 | |
| 	kGTM            = "lapack: k > m"
 | |
| 	kGTN            = "lapack: k > n"
 | |
| 	kLT0            = "lapack: k < 0"
 | |
| 	mLTN            = "lapack: m < n"
 | |
| 	negDimension    = "lapack: negative matrix dimension"
 | |
| 	negZ            = "lapack: negative z value"
 | |
| 	nLT0            = "lapack: n < 0"
 | |
| 	nLTM            = "lapack: n < m"
 | |
| 	shortWork       = "lapack: working array shorter than declared"
 | |
| )
 | |
| 
 | |
| // checkMatrix verifies the parameters of a matrix input.
 | |
| func checkMatrix(m, n int, a []float64, lda int) {
 | |
| 	if m < 0 {
 | |
| 		panic("lapack: has negative number of rows")
 | |
| 	}
 | |
| 	if n < 0 {
 | |
| 		panic("lapack: has negative number of columns")
 | |
| 	}
 | |
| 	if lda < n {
 | |
| 		panic("lapack: stride less than number of columns")
 | |
| 	}
 | |
| 	if len(a) < (m-1)*lda+n {
 | |
| 		panic("lapack: insufficient matrix slice length")
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func checkVector(n int, v []float64, inc int) {
 | |
| 	if n < 0 {
 | |
| 		panic("lapack: negative vector length")
 | |
| 	}
 | |
| 	if (inc > 0 && (n-1)*inc >= len(v)) || (inc < 0 && (1-n)*inc >= len(v)) {
 | |
| 		panic("lapack: insufficient vector slice length")
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func min(a, b int) int {
 | |
| 	if a < b {
 | |
| 		return a
 | |
| 	}
 | |
| 	return b
 | |
| }
 | |
| 
 | |
| func max(a, b int) int {
 | |
| 	if a > b {
 | |
| 		return a
 | |
| 	}
 | |
| 	return b
 | |
| }
 | |
| 
 | |
| var (
 | |
| 	// dlamchE is the machine epsilon. For IEEE this is 2^-53.
 | |
| 	dlamchE = 1.0 / (1 << 53)
 | |
| 
 | |
| 	// dlamchP is 2 * eps
 | |
| 	dlamchP = 2 * dlamchE
 | |
| 
 | |
| 	// dlamchS is the "safe min", that is, the lowest number such that 1/sfmin does
 | |
| 	// not overflow. The Netlib code for calculating this number is not correct --
 | |
| 	// it overflows. Found by comparison with the FORTRAN value.
 | |
| 	dlamchS = math.Nextafter((4 / math.MaxFloat64), 0)
 | |
| 
 | |
| 	smlnum = dlamchS / dlamchP
 | |
| 	bignum = 1 / smlnum
 | |
| 
 | |
| 	// dlamchB is the radix of the machine (the base of the number system).
 | |
| 	dlamchB = 2
 | |
| )
 | 
