forked from LeenkxTeam/LNXSDK
		
	
		
			
	
	
		
			555 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			555 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|  | /*
 | ||
|  | Bullet Continuous Collision Detection and Physics Library | ||
|  | Copyright (c) 2003-2013 Erwin Coumans  http://bulletphysics.org
 | ||
|  | 
 | ||
|  | This software is provided 'as-is', without any express or implied warranty. | ||
|  | In no event will the authors be held liable for any damages arising from the use of this software. | ||
|  | Permission is granted to anyone to use this software for any purpose,  | ||
|  | including commercial applications, and to alter it and redistribute it freely,  | ||
|  | subject to the following restrictions: | ||
|  | 
 | ||
|  | 1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required. | ||
|  | 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software. | ||
|  | 3. This notice may not be removed or altered from any source distribution. | ||
|  | */ | ||
|  | ///original version written by Erwin Coumans, October 2013
 | ||
|  | 
 | ||
|  | #ifndef BT_MATRIX_X_H
 | ||
|  | #define BT_MATRIX_X_H
 | ||
|  | 
 | ||
|  | #include "LinearMath/btQuickprof.h"
 | ||
|  | #include "LinearMath/btAlignedObjectArray.h"
 | ||
|  | #include <stdio.h>
 | ||
|  | 
 | ||
|  | //#define BT_DEBUG_OSTREAM
 | ||
|  | #ifdef BT_DEBUG_OSTREAM
 | ||
|  | #include <iostream>
 | ||
|  | #include <iomanip>      // std::setw
 | ||
|  | #endif //BT_DEBUG_OSTREAM
 | ||
|  | 
 | ||
|  | class btIntSortPredicate | ||
|  | { | ||
|  | 	public: | ||
|  | 		bool operator() ( const int& a, const int& b ) const | ||
|  | 		{ | ||
|  | 			 return a < b; | ||
|  | 		} | ||
|  | }; | ||
|  | 
 | ||
|  | 
 | ||
|  | template <typename T> | ||
|  | struct btVectorX | ||
|  | { | ||
|  | 	btAlignedObjectArray<T>	m_storage; | ||
|  | 	 | ||
|  | 	btVectorX() | ||
|  | 	{ | ||
|  | 	} | ||
|  | 	btVectorX(int numRows) | ||
|  | 	{ | ||
|  | 		m_storage.resize(numRows); | ||
|  | 	} | ||
|  | 	 | ||
|  | 	void resize(int rows) | ||
|  | 	{ | ||
|  | 		m_storage.resize(rows); | ||
|  | 	} | ||
|  | 	int cols() const | ||
|  | 	{ | ||
|  | 		return 1; | ||
|  | 	} | ||
|  | 	int rows() const | ||
|  | 	{ | ||
|  | 		return m_storage.size(); | ||
|  | 	} | ||
|  | 	int size() const | ||
|  | 	{ | ||
|  | 		return rows(); | ||
|  | 	} | ||
|  | 	 | ||
|  | 	T nrm2() const | ||
|  | 	{ | ||
|  | 		T norm = T(0); | ||
|  | 		 | ||
|  | 		int nn = rows(); | ||
|  | 		 | ||
|  | 		{ | ||
|  | 			if (nn == 1) | ||
|  | 			{ | ||
|  | 				norm = btFabs((*this)[0]); | ||
|  | 			} | ||
|  | 			else | ||
|  | 			{ | ||
|  | 				T scale = 0.0; | ||
|  | 				T ssq = 1.0; | ||
|  | 				 | ||
|  | 				/* The following loop is equivalent to this call to the LAPACK
 | ||
|  | 				 auxiliary routine:   CALL SLASSQ( N, X, INCX, SCALE, SSQ ) */ | ||
|  | 				 | ||
|  | 				for (int ix=0;ix<nn;ix++) | ||
|  | 				{ | ||
|  | 					if ((*this)[ix] != 0.0) | ||
|  | 					{ | ||
|  | 						T absxi = btFabs((*this)[ix]); | ||
|  | 						if (scale < absxi) | ||
|  | 						{ | ||
|  | 							T temp; | ||
|  | 							temp = scale / absxi; | ||
|  | 							ssq = ssq * (temp * temp) + BT_ONE; | ||
|  | 							scale = absxi; | ||
|  | 						} | ||
|  | 						else | ||
|  | 						{ | ||
|  | 							T temp; | ||
|  | 							temp = absxi / scale; | ||
|  | 							ssq += temp * temp; | ||
|  | 						} | ||
|  | 					} | ||
|  | 				} | ||
|  | 				norm = scale * sqrt(ssq); | ||
|  | 			} | ||
|  | 		} | ||
|  | 		return norm; | ||
|  | 		 | ||
|  | 	} | ||
|  | 	void	setZero() | ||
|  | 	{ | ||
|  | 		if (m_storage.size()) | ||
|  | 		{ | ||
|  | 			//	for (int i=0;i<m_storage.size();i++)
 | ||
|  | 			//		m_storage[i]=0;
 | ||
|  | 			//memset(&m_storage[0],0,sizeof(T)*m_storage.size());
 | ||
|  | 			btSetZero(&m_storage[0],m_storage.size()); | ||
|  | 		} | ||
|  | 	} | ||
|  | 	const T& operator[] (int index) const | ||
|  | 	{ | ||
|  | 		return m_storage[index]; | ||
|  | 	} | ||
|  | 	 | ||
|  | 	T& operator[] (int index) | ||
|  | 	{ | ||
|  | 		return m_storage[index]; | ||
|  | 	} | ||
|  | 	 | ||
|  | 	T* getBufferPointerWritable() | ||
|  | 	{ | ||
|  | 		return m_storage.size() ? &m_storage[0] : 0; | ||
|  | 	} | ||
|  | 	 | ||
|  | 	const T* getBufferPointer() const | ||
|  | 	{ | ||
|  | 		return m_storage.size() ? &m_storage[0] : 0; | ||
|  | 	} | ||
|  | 	 | ||
|  | }; | ||
|  | /*
 | ||
|  |  template <typename T> | ||
|  |  void setElem(btMatrixX<T>& mat, int row, int col, T val) | ||
|  |  { | ||
|  |  mat.setElem(row,col,val); | ||
|  |  } | ||
|  |  */ | ||
|  | 
 | ||
|  | 
 | ||
|  | template <typename T>  | ||
|  | struct btMatrixX | ||
|  | { | ||
|  | 	int m_rows; | ||
|  | 	int m_cols; | ||
|  | 	int m_operations; | ||
|  | 	int m_resizeOperations; | ||
|  | 	int m_setElemOperations; | ||
|  | 
 | ||
|  | 	btAlignedObjectArray<T>	m_storage; | ||
|  | 	mutable btAlignedObjectArray< btAlignedObjectArray<int> > m_rowNonZeroElements1; | ||
|  | 
 | ||
|  | 	T* getBufferPointerWritable()  | ||
|  | 	{ | ||
|  | 		return m_storage.size() ? &m_storage[0] : 0; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	const T* getBufferPointer() const | ||
|  | 	{ | ||
|  | 		return m_storage.size() ? &m_storage[0] : 0; | ||
|  | 	} | ||
|  | 	btMatrixX() | ||
|  | 		:m_rows(0), | ||
|  | 		m_cols(0), | ||
|  | 		m_operations(0), | ||
|  | 		m_resizeOperations(0), | ||
|  | 		m_setElemOperations(0) | ||
|  | 	{ | ||
|  | 	} | ||
|  | 	btMatrixX(int rows,int cols) | ||
|  | 		:m_rows(rows), | ||
|  | 		m_cols(cols), | ||
|  | 		m_operations(0), | ||
|  | 		m_resizeOperations(0), | ||
|  | 		m_setElemOperations(0) | ||
|  | 	{ | ||
|  | 		resize(rows,cols); | ||
|  | 	} | ||
|  | 	void resize(int rows, int cols) | ||
|  | 	{ | ||
|  | 		m_resizeOperations++; | ||
|  | 		m_rows = rows; | ||
|  | 		m_cols = cols; | ||
|  | 		{ | ||
|  | 			BT_PROFILE("m_storage.resize"); | ||
|  | 			m_storage.resize(rows*cols); | ||
|  | 		} | ||
|  | 	} | ||
|  | 	int cols() const | ||
|  | 	{ | ||
|  | 		return m_cols; | ||
|  | 	} | ||
|  | 	int rows() const | ||
|  | 	{ | ||
|  | 		return m_rows; | ||
|  | 	} | ||
|  | 	///we don't want this read/write operator(), because we cannot keep track of non-zero elements, use setElem instead
 | ||
|  | 	/*T& operator() (int row,int col)
 | ||
|  | 	{ | ||
|  | 		return m_storage[col*m_rows+row]; | ||
|  | 	} | ||
|  | 	*/ | ||
|  | 
 | ||
|  | 	void addElem(int row,int col, T val) | ||
|  | 	{ | ||
|  | 		if (val) | ||
|  | 		{ | ||
|  | 			if (m_storage[col+row*m_cols]==0.f) | ||
|  | 			{ | ||
|  | 				setElem(row,col,val); | ||
|  | 			} else | ||
|  | 			{ | ||
|  | 				m_storage[row*m_cols+col] += val; | ||
|  | 			} | ||
|  | 		} | ||
|  | 	} | ||
|  | 	 | ||
|  | 	 | ||
|  | 	void setElem(int row,int col, T val) | ||
|  | 	{ | ||
|  | 		m_setElemOperations++; | ||
|  | 		m_storage[row*m_cols+col] = val; | ||
|  | 	} | ||
|  | 	 | ||
|  | 	void mulElem(int row,int col, T val) | ||
|  | 	{ | ||
|  | 		m_setElemOperations++; | ||
|  | 		//mul doesn't change sparsity info
 | ||
|  | 
 | ||
|  | 		m_storage[row*m_cols+col] *= val; | ||
|  | 	} | ||
|  | 	 | ||
|  | 	 | ||
|  | 	 | ||
|  | 	 | ||
|  | 	void copyLowerToUpperTriangle() | ||
|  | 	{ | ||
|  | 		int count=0; | ||
|  | 		for (int row=0;row<rows();row++) | ||
|  | 		{ | ||
|  | 			for (int col=0;col<row;col++) | ||
|  | 			{ | ||
|  | 				setElem(col,row, (*this)(row,col)); | ||
|  | 				count++; | ||
|  | 				 | ||
|  | 			} | ||
|  | 		} | ||
|  | 		//printf("copyLowerToUpperTriangle copied %d elements out of %dx%d=%d\n", count,rows(),cols(),cols()*rows());
 | ||
|  | 	} | ||
|  | 	 | ||
|  | 	const T& operator() (int row,int col) const | ||
|  | 	{ | ||
|  | 		return m_storage[col+row*m_cols]; | ||
|  | 	} | ||
|  | 
 | ||
|  | 
 | ||
|  | 	void setZero() | ||
|  | 	{ | ||
|  | 		{ | ||
|  | 			BT_PROFILE("storage=0"); | ||
|  | 			btSetZero(&m_storage[0],m_storage.size()); | ||
|  | 			//memset(&m_storage[0],0,sizeof(T)*m_storage.size());
 | ||
|  | 			//for (int i=0;i<m_storage.size();i++)
 | ||
|  | 	//			m_storage[i]=0;
 | ||
|  | 		} | ||
|  | 	} | ||
|  | 	 | ||
|  | 	void setIdentity() | ||
|  | 	{ | ||
|  | 		btAssert(rows() == cols()); | ||
|  | 		 | ||
|  | 		setZero(); | ||
|  | 		for (int row=0;row<rows();row++) | ||
|  | 		{ | ||
|  | 			setElem(row,row,1); | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	 | ||
|  | 
 | ||
|  | 	void	printMatrix(const char* msg) | ||
|  | 	{ | ||
|  | 		printf("%s ---------------------\n",msg); | ||
|  | 		for (int i=0;i<rows();i++) | ||
|  | 		{ | ||
|  | 			printf("\n"); | ||
|  | 			for (int j=0;j<cols();j++) | ||
|  | 			{ | ||
|  | 				printf("%2.1f\t",(*this)(i,j)); | ||
|  | 			} | ||
|  | 		} | ||
|  | 		printf("\n---------------------\n"); | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 
 | ||
|  | 	void rowComputeNonZeroElements() const | ||
|  | 	{ | ||
|  | 		m_rowNonZeroElements1.resize(rows()); | ||
|  | 		for (int i=0;i<rows();i++) | ||
|  | 		{ | ||
|  | 			m_rowNonZeroElements1[i].resize(0); | ||
|  | 			for (int j=0;j<cols();j++) | ||
|  | 			{ | ||
|  | 				if ((*this)(i,j)!=0.f) | ||
|  | 				{ | ||
|  | 					m_rowNonZeroElements1[i].push_back(j); | ||
|  | 				} | ||
|  | 			} | ||
|  | 		} | ||
|  | 	} | ||
|  | 	btMatrixX transpose() const | ||
|  | 	{ | ||
|  | 		//transpose is optimized for sparse matrices
 | ||
|  | 		btMatrixX tr(m_cols,m_rows); | ||
|  | 		tr.setZero(); | ||
|  | 		for (int i=0;i<m_cols;i++) | ||
|  | 			for (int j=0;j<m_rows;j++) | ||
|  | 			{ | ||
|  | 				T v = (*this)(j,i); | ||
|  | 				if (v) | ||
|  | 				{ | ||
|  | 					tr.setElem(i,j,v); | ||
|  | 				} | ||
|  | 			} | ||
|  | 		return tr; | ||
|  | 	} | ||
|  | 
 | ||
|  | 
 | ||
|  | 	btMatrixX operator*(const btMatrixX& other) | ||
|  | 	{ | ||
|  | 		//btMatrixX*btMatrixX implementation, brute force
 | ||
|  | 		btAssert(cols() == other.rows()); | ||
|  | 
 | ||
|  | 		btMatrixX res(rows(),other.cols()); | ||
|  | 		res.setZero(); | ||
|  | //		BT_PROFILE("btMatrixX mul");
 | ||
|  | 		for (int j=0; j < res.cols(); ++j) | ||
|  | 		{ | ||
|  | 			{ | ||
|  | 				for (int i=0; i < res.rows(); ++i) | ||
|  | 				{ | ||
|  | 					T dotProd=0; | ||
|  | //					T dotProd2=0;
 | ||
|  | 					//int waste=0,waste2=0;
 | ||
|  | 
 | ||
|  | 					{ | ||
|  | //						bool useOtherCol = true;
 | ||
|  | 						{ | ||
|  | 							for (int v=0;v<rows();v++) | ||
|  | 							{ | ||
|  | 								T w = (*this)(i,v); | ||
|  | 								if (other(v,j)!=0.f) | ||
|  | 								{ | ||
|  | 									dotProd+=w*other(v,j);	 | ||
|  | 								} | ||
|  | 						 | ||
|  | 							} | ||
|  | 						} | ||
|  | 					} | ||
|  | 					if (dotProd) | ||
|  | 						res.setElem(i,j,dotProd); | ||
|  | 				} | ||
|  | 			} | ||
|  | 		} | ||
|  | 		return res; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	// this assumes the 4th and 8th rows of B and C are zero.
 | ||
|  | 	void multiplyAdd2_p8r (const btScalar *B, const btScalar *C,  int numRows,  int numRowsOther ,int row, int col) | ||
|  | 	{ | ||
|  | 		const btScalar *bb = B; | ||
|  | 		for ( int i = 0;i<numRows;i++) | ||
|  | 		{ | ||
|  | 			const btScalar *cc = C; | ||
|  | 			for ( int j = 0;j<numRowsOther;j++) | ||
|  | 			{ | ||
|  | 				btScalar sum; | ||
|  | 				sum  = bb[0]*cc[0]; | ||
|  | 				sum += bb[1]*cc[1]; | ||
|  | 				sum += bb[2]*cc[2]; | ||
|  | 				sum += bb[4]*cc[4]; | ||
|  | 				sum += bb[5]*cc[5]; | ||
|  | 				sum += bb[6]*cc[6]; | ||
|  | 				addElem(row+i,col+j,sum); | ||
|  | 				cc += 8; | ||
|  | 			} | ||
|  | 			bb += 8; | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	void multiply2_p8r (const btScalar *B, const btScalar *C,  int numRows,  int numRowsOther, int row, int col) | ||
|  | 	{ | ||
|  | 		btAssert (numRows>0 && numRowsOther>0 && B && C); | ||
|  | 		const btScalar *bb = B; | ||
|  | 		for ( int i = 0;i<numRows;i++) | ||
|  | 		{ | ||
|  | 			const btScalar *cc = C; | ||
|  | 			for ( int j = 0;j<numRowsOther;j++) | ||
|  | 			{ | ||
|  | 				btScalar sum; | ||
|  | 				sum  = bb[0]*cc[0]; | ||
|  | 				sum += bb[1]*cc[1]; | ||
|  | 				sum += bb[2]*cc[2]; | ||
|  | 				sum += bb[4]*cc[4]; | ||
|  | 				sum += bb[5]*cc[5]; | ||
|  | 				sum += bb[6]*cc[6]; | ||
|  | 				setElem(row+i,col+j,sum); | ||
|  | 				cc += 8; | ||
|  | 			} | ||
|  | 			bb += 8; | ||
|  | 		} | ||
|  | 	} | ||
|  | 	 | ||
|  | 	void setSubMatrix(int rowstart,int colstart,int rowend,int colend,const T value) | ||
|  | 	{ | ||
|  | 		int numRows = rowend+1-rowstart; | ||
|  | 		int numCols = colend+1-colstart; | ||
|  | 		 | ||
|  | 		for (int row=0;row<numRows;row++) | ||
|  | 		{ | ||
|  | 			for (int col=0;col<numCols;col++) | ||
|  | 			{ | ||
|  | 				setElem(rowstart+row,colstart+col,value); | ||
|  | 			} | ||
|  | 		} | ||
|  | 	} | ||
|  | 	 | ||
|  | 	void setSubMatrix(int rowstart,int colstart,int rowend,int colend,const btMatrixX& block) | ||
|  | 	{ | ||
|  | 		btAssert(rowend+1-rowstart == block.rows()); | ||
|  | 		btAssert(colend+1-colstart == block.cols()); | ||
|  | 		for (int row=0;row<block.rows();row++) | ||
|  | 		{ | ||
|  | 			for (int col=0;col<block.cols();col++) | ||
|  | 			{ | ||
|  | 				setElem(rowstart+row,colstart+col,block(row,col)); | ||
|  | 			} | ||
|  | 		} | ||
|  | 	} | ||
|  | 	void setSubMatrix(int rowstart,int colstart,int rowend,int colend,const btVectorX<T>& block) | ||
|  | 	{ | ||
|  | 		btAssert(rowend+1-rowstart == block.rows()); | ||
|  | 		btAssert(colend+1-colstart == block.cols()); | ||
|  | 		for (int row=0;row<block.rows();row++) | ||
|  | 		{ | ||
|  | 			for (int col=0;col<block.cols();col++) | ||
|  | 			{ | ||
|  | 				setElem(rowstart+row,colstart+col,block[row]); | ||
|  | 			} | ||
|  | 		} | ||
|  | 	} | ||
|  | 	 | ||
|  | 	 | ||
|  | 	btMatrixX negative() | ||
|  | 	{ | ||
|  | 		btMatrixX neg(rows(),cols()); | ||
|  | 		for (int i=0;i<rows();i++) | ||
|  | 			for (int j=0;j<cols();j++) | ||
|  | 			{ | ||
|  | 				T v = (*this)(i,j); | ||
|  | 				neg.setElem(i,j,-v); | ||
|  | 			} | ||
|  | 		return neg; | ||
|  | 	} | ||
|  | 
 | ||
|  | }; | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | typedef btMatrixX<float> btMatrixXf; | ||
|  | typedef btVectorX<float> btVectorXf; | ||
|  | 
 | ||
|  | typedef btMatrixX<double> btMatrixXd; | ||
|  | typedef btVectorX<double> btVectorXd; | ||
|  | 
 | ||
|  | 
 | ||
|  | #ifdef BT_DEBUG_OSTREAM
 | ||
|  | template <typename T>  | ||
|  | std::ostream& operator<< (std::ostream& os, const btMatrixX<T>& mat) | ||
|  | 	{ | ||
|  | 		 | ||
|  | 		os << " ["; | ||
|  | 		//printf("%s ---------------------\n",msg);
 | ||
|  | 		for (int i=0;i<mat.rows();i++) | ||
|  | 		{ | ||
|  | 			for (int j=0;j<mat.cols();j++) | ||
|  | 			{ | ||
|  | 				os << std::setw(12) << mat(i,j); | ||
|  | 			} | ||
|  | 			if (i!=mat.rows()-1) | ||
|  | 				os << std::endl << "  "; | ||
|  | 		} | ||
|  | 		os << " ]"; | ||
|  | 		//printf("\n---------------------\n");
 | ||
|  | 
 | ||
|  | 		return os; | ||
|  | 	} | ||
|  | template <typename T>  | ||
|  | std::ostream& operator<< (std::ostream& os, const btVectorX<T>& mat) | ||
|  | 	{ | ||
|  | 		 | ||
|  | 		os << " ["; | ||
|  | 		//printf("%s ---------------------\n",msg);
 | ||
|  | 		for (int i=0;i<mat.rows();i++) | ||
|  | 		{ | ||
|  | 				os << std::setw(12) << mat[i]; | ||
|  | 			if (i!=mat.rows()-1) | ||
|  | 				os << std::endl << "  "; | ||
|  | 		} | ||
|  | 		os << " ]"; | ||
|  | 		//printf("\n---------------------\n");
 | ||
|  | 
 | ||
|  | 		return os; | ||
|  | 	} | ||
|  | 
 | ||
|  | #endif //BT_DEBUG_OSTREAM
 | ||
|  | 
 | ||
|  | 
 | ||
|  | inline void setElem(btMatrixXd& mat, int row, int col, double val) | ||
|  | { | ||
|  | 	mat.setElem(row,col,val); | ||
|  | } | ||
|  | 
 | ||
|  | inline void setElem(btMatrixXf& mat, int row, int col, float val) | ||
|  | { | ||
|  | 	mat.setElem(row,col,val); | ||
|  | } | ||
|  | 
 | ||
|  | #ifdef BT_USE_DOUBLE_PRECISION
 | ||
|  | 	#define btVectorXu btVectorXd
 | ||
|  | 	#define btMatrixXu btMatrixXd
 | ||
|  | #else
 | ||
|  | 	#define btVectorXu btVectorXf
 | ||
|  | 	#define btMatrixXu btMatrixXf
 | ||
|  | #endif //BT_USE_DOUBLE_PRECISION
 | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | #endif//BT_MATRIX_H_H
 |