#include #include #include #include #include #include extern "C" { #include } #include namespace Poincare { Expression::Type MatrixInverse::type() const { return Type::MatrixInverse; } Expression * MatrixInverse::clone() const { MatrixInverse * a = new MatrixInverse(m_operands, true); return a; } Expression * MatrixInverse::shallowReduce(Context& context, AngleUnit angleUnit) { Expression * e = Expression::shallowReduce(context, angleUnit); if (e != this) { return e; } Expression * op = editableOperand(0); #if MATRIX_EXACT_REDUCING if (!op->recursivelyMatches(Expression::IsMatrix)) { detachOperand(op); return replaceWith(new Power(op, new Rational(-1), false), true)->shallowReduce(context, angleUnit); } if (op->type() == Type::Matrix) { Matrix * mat = static_cast(op); if (mat->numberOfRows() != mat->numberOfColumns()) { return replaceWith(new Undefined(), true); } } return this; #else detachOperand(op); return replaceWith(new Power(op, new Rational(-1), false), true)->shallowReduce(context, angleUnit); #endif } // TODO: handle this exactly in shallowReduce for small dimensions. template Expression * MatrixInverse::templatedApproximate(Context& context, AngleUnit angleUnit) const { Expression * input = operand(0)->approximate(context, angleUnit); Expression * result = nullptr; if (input->type() == Type::Complex) { Complex * c = static_cast *>(input); result = new Complex(Division::compute(Complex::Cartesian(1, 0), *c)); } else { assert(input->type() == Type::Matrix); result = static_cast(input)->createInverse(); } if (result == nullptr) { result = new Complex(Complex::Float(NAN)); } delete input; return result; } }