mirror of
https://github.com/UpsilonNumworks/Upsilon.git
synced 2026-03-18 21:30:38 +01:00
96 lines
3.5 KiB
C++
96 lines
3.5 KiB
C++
#include <poincare/division_quotient.h>
|
|
#include <poincare/infinity.h>
|
|
#include <poincare/layout_helper.h>
|
|
#include <poincare/rational.h>
|
|
#include <poincare/serialization_helper.h>
|
|
#include <poincare/undefined.h>
|
|
#include <cmath>
|
|
|
|
namespace Poincare {
|
|
|
|
constexpr Expression::FunctionHelper DivisionQuotient::s_functionHelper;
|
|
|
|
int DivisionQuotientNode::numberOfChildren() const { return DivisionQuotient::s_functionHelper.numberOfChildren(); }
|
|
|
|
ExpressionNode::Sign DivisionQuotientNode::sign(Context * context) const {
|
|
ExpressionNode::Sign numeratorSign = childAtIndex(0)->sign(context);
|
|
ExpressionNode::Sign denominatorSign = childAtIndex(1)->sign(context);
|
|
if (numeratorSign == ExpressionNode::Sign::Unknown || denominatorSign == ExpressionNode::Sign::Unknown) {
|
|
return ExpressionNode::Sign::Unknown;
|
|
}
|
|
return numeratorSign == denominatorSign ? ExpressionNode::Sign::Positive : ExpressionNode::Sign::Negative;
|
|
}
|
|
|
|
Expression DivisionQuotientNode::shallowReduce(ReductionContext reductionContext) {
|
|
return DivisionQuotient(this).shallowReduce(reductionContext.context());
|
|
}
|
|
|
|
Layout DivisionQuotientNode::createLayout(Preferences::PrintFloatMode floatDisplayMode, int numberOfSignificantDigits) const {
|
|
return LayoutHelper::Prefix(DivisionQuotient(this), floatDisplayMode, numberOfSignificantDigits, DivisionQuotient::s_functionHelper.name());
|
|
}
|
|
int DivisionQuotientNode::serialize(char * buffer, int bufferSize, Preferences::PrintFloatMode floatDisplayMode, int numberOfSignificantDigits) const {
|
|
return SerializationHelper::Prefix(this, buffer, bufferSize, floatDisplayMode, numberOfSignificantDigits, DivisionQuotient::s_functionHelper.name());
|
|
}
|
|
|
|
template<typename T>
|
|
Evaluation<T> DivisionQuotientNode::templatedApproximate(ApproximationContext approximationContext) const {
|
|
Evaluation<T> f1Input = childAtIndex(0)->approximate(T(), approximationContext);
|
|
Evaluation<T> f2Input = childAtIndex(1)->approximate(T(), approximationContext);
|
|
T f1 = f1Input.toScalar();
|
|
T f2 = f2Input.toScalar();
|
|
if (std::isnan(f1) || std::isnan(f2) || f1 != (int)f1 || f2 != (int)f2) {
|
|
return Complex<T>::RealUndefined();
|
|
}
|
|
return Complex<T>::Builder(std::floor(f1/f2));
|
|
}
|
|
|
|
|
|
Expression DivisionQuotient::shallowReduce(Context * context) {
|
|
{
|
|
Expression e = Expression::defaultShallowReduce();
|
|
e = e.defaultHandleUnitsInChildren();
|
|
if (e.isUndefined()) {
|
|
return e;
|
|
}
|
|
}
|
|
Expression c0 = childAtIndex(0);
|
|
Expression c1 = childAtIndex(1);
|
|
if (c0.deepIsMatrix(context) || c1.deepIsMatrix(context)) {
|
|
return replaceWithUndefinedInPlace();
|
|
}
|
|
if (c0.type() == ExpressionNode::Type::Rational) {
|
|
Rational r0 = static_cast<Rational &>(c0);
|
|
if (!r0.isInteger()) {
|
|
return replaceWithUndefinedInPlace();
|
|
}
|
|
}
|
|
if (c1.type() == ExpressionNode::Type::Rational) {
|
|
Rational r1 = static_cast<Rational &>(c1);
|
|
if (!r1.isInteger()) {
|
|
return replaceWithUndefinedInPlace();
|
|
}
|
|
}
|
|
if (c0.type() != ExpressionNode::Type::Rational || c1.type() != ExpressionNode::Type::Rational) {
|
|
return *this;
|
|
}
|
|
Rational r0 = static_cast<Rational &>(c0);
|
|
Rational r1 = static_cast<Rational &>(c1);
|
|
|
|
Integer a = r0.signedIntegerNumerator();
|
|
Integer b = r1.signedIntegerNumerator();
|
|
Expression result = Reduce(a, b);
|
|
replaceWithInPlace(result);
|
|
return result;
|
|
}
|
|
|
|
Expression DivisionQuotient::Reduce(const Integer & a, const Integer & b) {
|
|
if (b.isZero()) {
|
|
return Infinity::Builder(a.isNegative());
|
|
}
|
|
Integer result = Integer::Division(a, b).quotient;
|
|
assert(!result.isOverflow());
|
|
return Rational::Builder(result);
|
|
}
|
|
|
|
}
|