mirror of
https://github.com/UpsilonNumworks/Upsilon.git
synced 2026-03-18 21:30:38 +01:00
345 lines
11 KiB
C++
345 lines
11 KiB
C++
#include <poincare/matrix.h>
|
|
#include <poincare/rational.h>
|
|
#include <poincare/division.h>
|
|
#include <poincare/subtraction.h>
|
|
#include <poincare/matrix_complex.h>
|
|
#include <poincare/matrix_layout.h>
|
|
#include <cmath>
|
|
#include <float.h>
|
|
#include <string.h>
|
|
#include <assert.h>
|
|
#include <stdlib.h>
|
|
|
|
namespace Poincare {
|
|
|
|
void MatrixNode::didAddChildAtIndex(int newNumberOfChildren) {
|
|
setNumberOfRows(1);
|
|
setNumberOfColumns(newNumberOfChildren);
|
|
}
|
|
|
|
int MatrixNode::polynomialDegree(Context & context, char symbolName) const {
|
|
return -1;
|
|
}
|
|
|
|
Layout MatrixNode::createLayout(Preferences::PrintFloatMode floatDisplayMode, int numberOfSignificantDigits) const {
|
|
assert(numberOfChildren() > 0);
|
|
MatrixLayout layout;
|
|
for (ExpressionNode * c : children()) {
|
|
layout.addChildAtIndex(c->createLayout(floatDisplayMode, numberOfSignificantDigits), layout.numberOfChildren(), layout.numberOfChildren(), nullptr);
|
|
}
|
|
layout.setDimensions(m_numberOfRows, m_numberOfColumns);
|
|
return layout;
|
|
}
|
|
|
|
int MatrixNode::serialize(char * buffer, int bufferSize, Preferences::PrintFloatMode floatDisplayMode, int numberOfSignificantDigits) const {
|
|
if (bufferSize == 0) {
|
|
return -1;
|
|
}
|
|
buffer[bufferSize-1] = 0;
|
|
int currentChar = 0;
|
|
if (currentChar >= bufferSize-1) {
|
|
return 0;
|
|
}
|
|
buffer[currentChar++] = '[';
|
|
if (currentChar >= bufferSize-1) {
|
|
return currentChar;
|
|
}
|
|
for (int i = 0; i < m_numberOfRows; i++) {
|
|
buffer[currentChar++] = '[';
|
|
if (currentChar >= bufferSize-1) {
|
|
return currentChar;
|
|
}
|
|
currentChar += childAtIndex(i*m_numberOfColumns)->serialize(buffer+currentChar, bufferSize-currentChar, floatDisplayMode, numberOfSignificantDigits);
|
|
if (currentChar >= bufferSize-1) {
|
|
return currentChar;
|
|
}
|
|
for (int j = 1; j < m_numberOfColumns; j++) {
|
|
buffer[currentChar++] = ',';
|
|
if (currentChar >= bufferSize-1) {
|
|
return currentChar;
|
|
}
|
|
currentChar += childAtIndex(i*m_numberOfColumns+j)->serialize(buffer+currentChar, bufferSize-currentChar, floatDisplayMode, numberOfSignificantDigits);
|
|
if (currentChar >= bufferSize-1) {
|
|
return currentChar;
|
|
}
|
|
}
|
|
currentChar = strlen(buffer);
|
|
if (currentChar >= bufferSize-1) {
|
|
return currentChar;
|
|
}
|
|
buffer[currentChar++] = ']';
|
|
if (currentChar >= bufferSize-1) {
|
|
return currentChar;
|
|
}
|
|
}
|
|
buffer[currentChar++] = ']';
|
|
buffer[currentChar] = 0;
|
|
return currentChar;
|
|
}
|
|
|
|
template<typename T>
|
|
Evaluation<T> MatrixNode::templatedApproximate(Context& context, Preferences::AngleUnit angleUnit) const {
|
|
MatrixComplex<T> matrix;
|
|
for (ExpressionNode * c : children()) {
|
|
matrix.addChildAtIndexInPlace(c->approximate(T(), context, angleUnit), matrix.numberOfChildren(), matrix.numberOfChildren());
|
|
}
|
|
matrix.setDimensions(numberOfRows(), numberOfColumns());
|
|
return matrix;
|
|
}
|
|
|
|
// MATRIX
|
|
|
|
void Matrix::setDimensions(int rows, int columns) {
|
|
assert(rows * columns == numberOfChildren());
|
|
setNumberOfRows(rows);
|
|
setNumberOfColumns(columns);
|
|
}
|
|
|
|
void Matrix::addChildrenAsRowInPlace(TreeHandle t, int i) {
|
|
int previousNumberOfColumns = numberOfColumns();
|
|
if (previousNumberOfColumns > 0) {
|
|
assert(t.numberOfChildren() == numberOfColumns());
|
|
}
|
|
int previousNumberOfRows = numberOfRows();
|
|
for (int i = 0; i < t.numberOfChildren(); i++) {
|
|
addChildAtIndexInPlace(t.childAtIndex(i), numberOfChildren(), numberOfChildren());
|
|
}
|
|
setDimensions(previousNumberOfRows + 1, previousNumberOfColumns == 0 ? t.numberOfChildren() : previousNumberOfColumns);
|
|
}
|
|
|
|
int Matrix::rank(Context & context, Preferences::AngleUnit angleUnit, bool inPlace) {
|
|
Matrix m = inPlace ? *this : clone().convert<Matrix>();
|
|
m = m.rowCanonize(context, angleUnit);
|
|
int rank = m.numberOfRows();
|
|
int i = rank-1;
|
|
while (i >= 0) {
|
|
int j = m.numberOfColumns()-1;
|
|
while (j >= i && matrixChild(i,j).isRationalZero()) {
|
|
j--;
|
|
}
|
|
if (j == i-1) {
|
|
rank--;
|
|
} else {
|
|
break;
|
|
}
|
|
i--;
|
|
}
|
|
return rank;
|
|
}
|
|
|
|
template<typename T>
|
|
int Matrix::ArrayInverse(T * array, int numberOfRows, int numberOfColumns) {
|
|
if (numberOfRows != numberOfColumns) {
|
|
return -1;
|
|
}
|
|
assert(numberOfRows*numberOfColumns <= k_maxNumberOfCoefficients);
|
|
int dim = numberOfRows;
|
|
/* Create the matrix inv = (A|I) with A the input matrix and I the dim identity matrix */
|
|
T operands[k_maxNumberOfCoefficients*(3/2)]; // inv dimensions: dim x (2*dim)
|
|
for (int i = 0; i < dim; i++) {
|
|
for (int j = 0; j < dim; j++) {
|
|
operands[i*2*dim+j] = array[i*numberOfColumns+j];
|
|
}
|
|
for (int j = dim; j < 2*dim; j++) {
|
|
operands[i*2*dim+j] = j-dim == i ? 1.0 : 0.0;
|
|
}
|
|
}
|
|
ArrayRowCanonize(operands, dim, 2*dim);
|
|
// Check inversibility
|
|
for (int i = 0; i < dim; i++) {
|
|
if (std::abs(operands[i*2*dim+i] - (T)1.0) > Expression::epsilon<float>()) {
|
|
return -2;
|
|
}
|
|
}
|
|
for (int i = 0; i < dim; i++) {
|
|
for (int j = 0; j < dim; j++) {
|
|
array[i*numberOfColumns+j] = operands[i*2*dim+j+dim];
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
Matrix Matrix::rowCanonize(Context & context, Preferences::AngleUnit angleUnit, Multiplication determinant) {
|
|
// The matrix children have to be reduced to be able to spot 0
|
|
reduceChildren(context, angleUnit);
|
|
|
|
int m = numberOfRows();
|
|
int n = numberOfColumns();
|
|
|
|
int h = 0; // row pivot
|
|
int k = 0; // column pivot
|
|
|
|
while (h < m && k < n) {
|
|
// Find the first non-null pivot
|
|
int iPivot = h;
|
|
while (iPivot < m && matrixChild(iPivot, k).isRationalZero()) {
|
|
iPivot++;
|
|
}
|
|
if (iPivot == m) {
|
|
// No non-null coefficient in this column, skip
|
|
k++;
|
|
// Update determinant: det *= 0
|
|
if (!determinant.isUninitialized()) { determinant.addChildAtIndexInPlace(Rational(0), 0, determinant.numberOfChildren()); }
|
|
} else {
|
|
// Swap row h and iPivot
|
|
if (iPivot != h) {
|
|
for (int col = h; col < n; col++) {
|
|
swapChildrenInPlace(iPivot*n+col, h*n+col);
|
|
}
|
|
// Update determinant: det *= -1
|
|
if (!determinant.isUninitialized()) { determinant.addChildAtIndexInPlace(Rational(-1), 0, determinant.numberOfChildren()); }
|
|
}
|
|
/* Set to 1 M[h][k] by linear combination */
|
|
Expression divisor = matrixChild(h, k);
|
|
// Update determinant: det *= divisor
|
|
if (!determinant.isUninitialized()) { determinant.addChildAtIndexInPlace(divisor.clone(), 0, determinant.numberOfChildren()); }
|
|
for (int j = k+1; j < n; j++) {
|
|
Expression opHJ = matrixChild(h, j);
|
|
Expression newOpHJ = Division(opHJ, divisor.clone());
|
|
replaceChildAtIndexInPlace(h*n+j, newOpHJ);
|
|
newOpHJ = newOpHJ.shallowReduce(context, angleUnit);
|
|
}
|
|
replaceChildInPlace(divisor, Rational(1));
|
|
|
|
/* Set to 0 all M[i][j] i != h, j > k by linear combination */
|
|
for (int i = 0; i < m; i++) {
|
|
if (i == h) { continue; }
|
|
Expression factor = matrixChild(i, k);
|
|
for (int j = k+1; j < n; j++) {
|
|
Expression opIJ = matrixChild(i, j);
|
|
Expression newOpIJ = Subtraction(opIJ, Multiplication(matrixChild(h, j).clone(), factor.clone()));
|
|
replaceChildAtIndexInPlace(i*n+j, newOpIJ);
|
|
newOpIJ.childAtIndex(1).shallowReduce(context, angleUnit);
|
|
newOpIJ = newOpIJ.shallowReduce(context, angleUnit);
|
|
}
|
|
replaceChildAtIndexInPlace(i*n+k, Rational(0));
|
|
}
|
|
h++;
|
|
k++;
|
|
}
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
template<typename T>
|
|
void Matrix::ArrayRowCanonize(T * array, int numberOfRows, int numberOfColumns, T * determinant) {
|
|
int h = 0; // row pivot
|
|
int k = 0; // column pivot
|
|
|
|
while (h < numberOfRows && k < numberOfColumns) {
|
|
// Find the first non-null pivot
|
|
int iPivot = h;
|
|
while (iPivot < numberOfRows && std::abs(array[iPivot*numberOfColumns+k]) < Expression::epsilon<double>()) {
|
|
iPivot++;
|
|
}
|
|
if (iPivot == numberOfRows) {
|
|
// No non-null coefficient in this column, skip
|
|
k++;
|
|
// Update determinant: det *= 0
|
|
if (determinant) { *determinant *= 0.0; }
|
|
} else {
|
|
// Swap row h and iPivot
|
|
if (iPivot != h) {
|
|
for (int col = h; col < numberOfColumns; col++) {
|
|
// Swap array[iPivot, col] and array[h, col]
|
|
T temp = array[iPivot*numberOfColumns+col];
|
|
array[iPivot*numberOfColumns+col] = array[h*numberOfColumns+col];
|
|
array[h*numberOfColumns+col] = temp;
|
|
}
|
|
// Update determinant: det *= -1
|
|
if (determinant) { *determinant *= -1.0; }
|
|
}
|
|
/* Set to 1 array[h][k] by linear combination */
|
|
T divisor = array[h*numberOfColumns+k];
|
|
// Update determinant: det *= divisor
|
|
if (determinant) { *determinant *= divisor; }
|
|
for (int j = k+1; j < numberOfColumns; j++) {
|
|
array[h*numberOfColumns+j] /= divisor;
|
|
}
|
|
array[h*numberOfColumns+k] = 1;
|
|
|
|
/* Set to 0 all M[i][j] i != h, j > k by linear combination */
|
|
for (int i = 0; i < numberOfRows; i++) {
|
|
if (i == h) { continue; }
|
|
T factor = array[i*numberOfColumns+k];
|
|
for (int j = k+1; j < numberOfColumns; j++) {
|
|
array[i*numberOfColumns+j] -= array[h*numberOfColumns+j]*factor;
|
|
}
|
|
array[i*numberOfColumns+k] = 0;
|
|
}
|
|
h++;
|
|
k++;
|
|
}
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
#if MATRIX_EXACT_REDUCING
|
|
Matrix Matrix::transpose() const {
|
|
Matrix matrix();
|
|
for (int i = 0; i < m_numberOfRows; i++) {
|
|
for (int j = 0; j < m_numberOfColumns; j++) {
|
|
matrix.addChildAtIndexInPlace(childAtIndex(i*m_numberOfRows+j), j*m_numberOfRows+i, j*m_numberOfRows+i);
|
|
}
|
|
}
|
|
// Intentionally swapping dimensions for transpose
|
|
matrix.setDimensions(m_numberOfColumns, m_numberOfRows);
|
|
return matrix;
|
|
}
|
|
|
|
Matrix Matrix::createIdentity(int dim) {
|
|
Matrix matrix();
|
|
for (int i = 0; i < dim; i++) {
|
|
for (int j = 0; j < dim; j++) {
|
|
matrix.addChildAtIndexInPlace(i == j ? Rational(1) : Rational(0), i*dim+j, i*dim+j);
|
|
}
|
|
}
|
|
matrix.setDimensions(dim, dim);
|
|
return matrix;
|
|
}
|
|
|
|
Expression Matrix::inverse(Context & context, Preferences::AngleUnit angleUnit) const {
|
|
if (m_numberOfRows != m_numberOfColumns) {
|
|
return Undefined();
|
|
}
|
|
int dim = m_numberOfRows;
|
|
/* Create the matrix inv = (A|I) with A the input matrix and I the dim identity matrix */
|
|
Matrix AI;
|
|
for (int i = 0; i < dim; i++) {
|
|
for (int j = 0; j < dim; j++) {
|
|
AI.addChildAtIndexInPlace(matrixChild(i, j), i*2*dim+j, i*2*dim+j);
|
|
}
|
|
for (int j = dim; j < 2*dim; j++) {
|
|
AI.addChildAtIndexInPlace(j-dim == i ? Rational(1) : Rational(0), i*2*dim+j, i*2*dim+j);
|
|
}
|
|
}
|
|
AI.setDimensions(dim, 2*dim);
|
|
AI = AI.rowCanonize(context, angleUnit);
|
|
// Check inversibility
|
|
for (int i = 0; i < dim; i++) {
|
|
if (AI.matrixChild(i, i)->type() != ExpressionNode::Type::Rational || !static_cast<RationalNode *>(AI.matrixChild(i, i)->node())->isOne()) {
|
|
return Undefined();
|
|
}
|
|
}
|
|
Matrix inverse();
|
|
for (int i = 0; i < dim; i++) {
|
|
for (int j = 0; j < dim; j++) {
|
|
addChildAtIndexInPlace(AI.matrixChild(i, j+dim), i*dim+j, i*dim+j);
|
|
}
|
|
}
|
|
inverse.setDimensions(dim, dim);
|
|
return inverse;
|
|
}
|
|
|
|
#endif
|
|
#endif
|
|
|
|
template int Matrix::ArrayInverse<float>(float *, int, int);
|
|
template int Matrix::ArrayInverse<double>(double *, int, int);
|
|
template int Matrix::ArrayInverse<std::complex<float>>(std::complex<float> *, int, int);
|
|
template int Matrix::ArrayInverse<std::complex<double>>(std::complex<double> *, int, int);
|
|
template void Matrix::ArrayRowCanonize<std::complex<float> >(std::complex<float>*, int, int, std::complex<float>*);
|
|
template void Matrix::ArrayRowCanonize<std::complex<double> >(std::complex<double>*, int, int, std::complex<double>*);
|
|
|
|
}
|