mirror of
https://github.com/UpsilonNumworks/Upsilon.git
synced 2026-01-18 16:27:34 +01:00
220 lines
8.6 KiB
C++
220 lines
8.6 KiB
C++
#include "expression_model.h"
|
|
#include "global_context.h"
|
|
#include "poincare_helpers.h"
|
|
#include <apps/apps_container.h>
|
|
#include <poincare/horizontal_layout.h>
|
|
#include <poincare/undefined.h>
|
|
#include <string.h>
|
|
#include <cmath>
|
|
#include <assert.h>
|
|
#include <algorithm>
|
|
|
|
using namespace Ion;
|
|
using namespace Poincare;
|
|
|
|
namespace Shared {
|
|
|
|
ExpressionModel::ExpressionModel() :
|
|
m_expression(),
|
|
m_layout(),
|
|
m_circular(-1)
|
|
{
|
|
}
|
|
|
|
void ExpressionModel::text(const Storage::Record * record, char * buffer, size_t bufferSize, CodePoint symbol) const {
|
|
Expression e = expressionClone(record);
|
|
if (e.isUninitialized()) {
|
|
if (bufferSize > 0) {
|
|
buffer[0] = 0;
|
|
}
|
|
return;
|
|
}
|
|
if (symbol != 0) {
|
|
e = e.replaceSymbolWithExpression(Symbol::Builder(UCodePointUnknown), Symbol::Builder(symbol));
|
|
}
|
|
int serializedSize = e.serialize(buffer, bufferSize);
|
|
if (serializedSize >= (int)bufferSize - 1) {
|
|
// It is very likely that the buffer is overflowed
|
|
buffer[0] = 0;
|
|
}
|
|
}
|
|
|
|
bool ExpressionModel::isCircularlyDefined(const Storage::Record * record, Poincare::Context * context) const {
|
|
if (m_circular == -1) {
|
|
m_circular = Expression::ExpressionWithoutSymbols(expressionClone(record), context).isUninitialized();
|
|
}
|
|
return m_circular;
|
|
}
|
|
|
|
Expression ExpressionModel::expressionReduced(const Storage::Record * record, Poincare::Context * context) const {
|
|
/* TODO
|
|
* By calling isCircularlyDefined and then Simplify, the expression tree is
|
|
* browsed twice. Note that Simplify does ALMOST the job of
|
|
* isCircularlyDefined, since Simplify reduces the expression, replaces the
|
|
* symbols it encounters (see SymbolAbstract::Expand). isCircularlyDefined
|
|
* should probably be removed. The difficulty relies in the ambiguous
|
|
* conventions about the values returned or set when a symbol has no proper
|
|
* expression: for example,
|
|
* - GlobalContext::expressionForSymbolAbstract returns an uninitialized
|
|
* expression,
|
|
* - so do Expression::ExpressionWithoutSymbols and SymbolAbstract::Expand,
|
|
* - Expression::deepReplaceReplaceableSymbols leaves unchanged the symbols,
|
|
* whose expression is uninitialized, but returns Undefined if the
|
|
* expression for a symbol contains the symbol itself,
|
|
* - Symbol::shallowReduce and Function::shallowReduce return Undefined or
|
|
* the expression unaltered according to symbolic-computation setting,
|
|
* - expressionReduced returns Undefined if the expression
|
|
* isCircularlyDefined but leaves the expression unchanged if Simplify
|
|
* returns an uninitialized expression...
|
|
*/
|
|
if (m_expression.isUninitialized()) {
|
|
assert(record->fullName() != nullptr);
|
|
if (isCircularlyDefined(record, context)) {
|
|
m_expression = Undefined::Builder();
|
|
} else {
|
|
m_expression = Expression::ExpressionFromAddress(expressionAddress(record), expressionSize(record), record);
|
|
/* 'Simplify' routine might need to call expressionReduced on the very
|
|
* same function. So we need to keep a valid m_expression while executing
|
|
* 'Simplify'. Thus, we use a temporary expression. */
|
|
Expression tempExpression = m_expression.clone();
|
|
PoincareHelpers::Simplify(&tempExpression, context, ExpressionNode::ReductionTarget::SystemForApproximation);
|
|
// simplify might return an uninitialized Expression if interrupted
|
|
if (!tempExpression.isUninitialized()) {
|
|
m_expression = tempExpression;
|
|
}
|
|
}
|
|
}
|
|
return m_expression;
|
|
}
|
|
|
|
Expression ExpressionModel::expressionClone(const Storage::Record * record) const {
|
|
assert(record->fullName() != nullptr);
|
|
/* A new Expression has to be created at each call (because it might be tempered with after calling) */
|
|
return Expression::ExpressionFromAddress(expressionAddress(record), expressionSize(record), record);
|
|
/* TODO
|
|
* The substitution of UCodePointUnknown back and forth is done in the
|
|
* methods text, setContent (through BuildExpressionFromText), layout and
|
|
* also in GlobalContext::expressionForSymbolAbstract and
|
|
* GlobalContext::setExpressionForSymbolAbstract. When getting the
|
|
* expression, the substitutions may probably be gathered here.
|
|
*/
|
|
}
|
|
|
|
Layout ExpressionModel::layout(const Storage::Record * record, CodePoint symbol) const {
|
|
if (m_layout.isUninitialized()) {
|
|
Expression clone = expressionClone(record);
|
|
if (!clone.isUninitialized() && symbol != 0) {
|
|
clone = clone.replaceSymbolWithExpression(Symbol::Builder(UCodePointUnknown), Symbol::Builder(symbol));
|
|
}
|
|
m_layout = PoincareHelpers::CreateLayout(clone);
|
|
if (m_layout.isUninitialized()) {
|
|
m_layout = HorizontalLayout::Builder();
|
|
}
|
|
}
|
|
return m_layout;
|
|
}
|
|
|
|
Ion::Storage::Record::ErrorStatus ExpressionModel::setContent(Ion::Storage::Record * record, const char * c, Context * context, CodePoint symbol) {
|
|
Expression e = ExpressionModel::BuildExpressionFromText(c, symbol, context);
|
|
return setExpressionContent(record, e);
|
|
}
|
|
|
|
Ion::Storage::Record::ErrorStatus ExpressionModel::setExpressionContent(Ion::Storage::Record * record, const Expression & newExpression) {
|
|
assert(record->fullName() != nullptr);
|
|
// Prepare the new data to be stored
|
|
Ion::Storage::Record::Data newData = record->value();
|
|
size_t previousExpressionSize = expressionSize(record);
|
|
size_t newExpressionSize = newExpression.isUninitialized() ? 0 : newExpression.size();
|
|
#ifdef STRING_STORAGE
|
|
size_t stringsize = 0;
|
|
char buf[1024] = {0};
|
|
char repl = 0;
|
|
buf[0] = '"';
|
|
if (!newExpression.isUninitialized()) {
|
|
size_t l = newExpression.serialize(buf+1,sizeof(buf)-2);
|
|
if (l >= sizeof(buf) - 3) {
|
|
newExpressionSize = 0;
|
|
} else {
|
|
buf[l + 1] = '"';
|
|
// replace 0x1 by x for func or n for seq
|
|
const char * name = record->fullName();
|
|
int namel = strlen(name);
|
|
if (namel > 4 && strncmp(name + namel - 4, ".seq", 4) == 0) {
|
|
repl = 'n';
|
|
} else if (namel > 5 && strncmp(name + namel - 5, ".func", 5) == 0) {
|
|
repl = 'x';
|
|
}
|
|
if (repl) {
|
|
for (char * ptr = buf; *ptr; ++ptr) {
|
|
if (*ptr == 1) {
|
|
*ptr = repl;
|
|
}
|
|
}
|
|
}
|
|
stringsize= l + 3; // 2 quotes and 0 at end
|
|
if (newExpressionSize < stringsize) {
|
|
newExpressionSize = stringsize;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
size_t previousDataSize = newData.size;
|
|
size_t newDataSize = previousDataSize - previousExpressionSize + newExpressionSize;
|
|
void * expAddress = expressionAddress(record);
|
|
// Update size of record to maximal size between previous and new data
|
|
newData.size = std::max(previousDataSize, newDataSize);
|
|
Ion::Storage::Record::ErrorStatus error = record->setValue(newData);
|
|
if (error != Ion::Storage::Record::ErrorStatus::None) {
|
|
assert(error == Ion::Storage::Record::ErrorStatus::NotEnoughSpaceAvailable);
|
|
return error;
|
|
}
|
|
// Prepare the new data content
|
|
/* WARNING: expressionAddress() cannot be used while the metadata is invalid
|
|
* (as it is sometimes computed from metadata). Thus, the expression address
|
|
* is given as a parameter to updateNewDataWithExpression. */
|
|
updateNewDataWithExpression(record, newExpression, expAddress, newExpressionSize, previousExpressionSize);
|
|
#ifdef STRING_STORAGE
|
|
if (stringsize && stringsize<newExpressionSize) {
|
|
// print size is smaller than expression size
|
|
strncpy((char *)expAddress, buf, stringsize);
|
|
}
|
|
#endif
|
|
// Set the data with the right size
|
|
newData.size = newDataSize;
|
|
error = record->setValue(newData);
|
|
// Any error would have occurred at the first call to setValue
|
|
assert(error == Ion::Storage::Record::ErrorStatus::None);
|
|
|
|
/* Here we delete only the elements relative to the expression model kept in
|
|
* this handle. */
|
|
tidy();
|
|
return error;
|
|
}
|
|
|
|
void ExpressionModel::updateNewDataWithExpression(Ion::Storage::Record * record, const Expression & expressionToStore, void * expressionAddress, size_t expressionToStoreSize, size_t previousExpressionSize) {
|
|
if (!expressionToStore.isUninitialized()) {
|
|
memmove(expressionAddress, expressionToStore.addressInPool(), expressionToStoreSize);
|
|
}
|
|
}
|
|
|
|
void ExpressionModel::tidy() const {
|
|
m_layout = Layout();
|
|
m_expression = Expression();
|
|
m_circular = -1;
|
|
}
|
|
|
|
Poincare::Expression ExpressionModel::BuildExpressionFromText(const char * c, CodePoint symbol, Poincare::Context * context) {
|
|
Expression expressionToStore;
|
|
// if c = "", we want to reinit the Expression
|
|
if (c && *c != 0) {
|
|
// Compute the expression to store, without replacing symbols
|
|
expressionToStore = Expression::Parse(c, context);
|
|
if (!expressionToStore.isUninitialized() && symbol != 0) {
|
|
expressionToStore = expressionToStore.replaceSymbolWithExpression(Symbol::Builder(symbol), Symbol::Builder(UCodePointUnknown));
|
|
}
|
|
}
|
|
return expressionToStore;
|
|
}
|
|
|
|
}
|