| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782 |
- #include "JsonExpression.h"
- #include "Chunk.h"
- #include "Game.h"
- JExpressionMemory::JExpressionMemory()
- : ReferenceCounter(),
- currentChunk(0)
- {}
- JExpressionMemory::~JExpressionMemory()
- {
- if (currentChunk) currentChunk->release();
- }
- void JExpressionMemory::lock()
- {
- cs.lock();
- }
- void JExpressionMemory::unlock()
- {
- cs.unlock();
- }
- Noise* JExpressionMemory::zNoiseP(Framework::Text name)
- {
- return noises.z(name, name.getLength());
- }
- void JExpressionMemory::setNoise(Framework::Text name, Noise* noise)
- {
- noises.set(name, name.getLength(), noise);
- }
- void JExpressionMemory::setCurrentChunk(Chunk* chunk)
- {
- if (currentChunk) currentChunk->release();
- currentChunk = chunk;
- }
- float* JExpressionMemory::getFloatVariableP(const Framework::Text& name)
- {
- if (!floatVariables.contains(name, name.getLength()))
- {
- floatVariables.set(name, name.getLength(), 0.f);
- }
- return floatVariables.getP(name, name.getLength());
- }
- bool* JExpressionMemory::getBoolVariableP(const Framework::Text& name)
- {
- if (!boolVariables.contains(name, name.getLength()))
- {
- boolVariables.set(name, name.getLength(), 0);
- }
- return boolVariables.getP(name, name.getLength());
- }
- Chunk** JExpressionMemory::zzCurrentChunk()
- {
- return ¤tChunk;
- }
- JFloatExpression::JFloatExpression()
- : ReferenceCounter(),
- compiled(0)
- {}
- float JFloatExpression::getValue()
- {
- return compiled();
- }
- FloatFunc JFloatExpression::compile(JExpressionMemory* zMemory)
- {
- if (compiled)
- {
- if (zMemory != memory)
- {
- throw "Cannot compile the same expression for different memories";
- }
- return compiled;
- }
- memory = zMemory;
- return compiled = buildAssembly(zMemory).compileToFunction<FloatFunc>();
- }
- JBoolExpression::JBoolExpression()
- : ReferenceCounter(),
- compiled(0)
- {}
- bool JBoolExpression::getValue()
- {
- return compiled();
- }
- BoolFunc JBoolExpression::compile(JExpressionMemory* zMemory)
- {
- if (compiled)
- {
- if (zMemory != memory)
- {
- throw "Cannot compile the same expression for different "
- "memories";
- }
- return compiled;
- }
- memory = zMemory;
- return compiled = buildAssembly(zMemory).compileToFunction<BoolFunc>();
- }
- JVariableFloatExpression::JVariableFloatExpression()
- : JFloatExpression()
- {}
- Framework::Assembly::AssemblyBlock& JVariableFloatExpression::buildAssembly(
- JExpressionMemory* zMemory)
- {
- codeBlock.addLoadValue(
- zMemory->getFloatVariableP(name), Framework::Assembly::MM0);
- return codeBlock;
- }
- void JVariableFloatExpression::setName(Framework::Text name)
- {
- this->name = name;
- }
- Framework::Text JVariableFloatExpression::getName() const
- {
- return name;
- }
- JVariableFloatExpressionFactory::JVariableFloatExpressionFactory()
- : SubTypeFactory()
- {}
- JVariableFloatExpression* JVariableFloatExpressionFactory::fromJson(
- Framework::JSON::JSONObject* zJson) const
- {
- JVariableFloatExpression* result = new JVariableFloatExpression();
- result->setName(zJson->zValue("name")->asString()->getString());
- return result;
- }
- Framework::JSON::JSONObject* JVariableFloatExpressionFactory::toJsonObject(
- JVariableFloatExpression* zObject) const
- {
- Framework::JSON::JSONObject* result = new Framework::JSON::JSONObject();
- result->addValue(
- "name", new Framework::JSON::JSONString(zObject->getName()));
- return result;
- }
- JSONObjectValidationBuilder* JVariableFloatExpressionFactory::addToValidator(
- JSONObjectValidationBuilder* builder) const
- {
- return builder->withRequiredString("name")->finishString();
- }
- const char* JVariableFloatExpressionFactory::getTypeToken() const
- {
- return "variable";
- }
- JVariableBoolExpression::JVariableBoolExpression()
- : JBoolExpression()
- {}
- Framework::Assembly::AssemblyBlock& JVariableBoolExpression::buildAssembly(
- JExpressionMemory* zMemory)
- {
- codeBlock.addLoadValue(
- (char*)zMemory->getBoolVariableP(name), Framework::Assembly::RAX);
- return codeBlock;
- }
- void JVariableBoolExpression::setName(Framework::Text name)
- {
- this->name = name;
- }
- Framework::Text JVariableBoolExpression::getName() const
- {
- return name;
- }
- JVariableBoolExpressionFactory::JVariableBoolExpressionFactory()
- : SubTypeFactory()
- {}
- JVariableBoolExpression* JVariableBoolExpressionFactory::fromJson(
- Framework::JSON::JSONObject* zJson) const
- {
- JVariableBoolExpression* result = new JVariableBoolExpression();
- result->setName(zJson->zValue("name")->asString()->getString());
- return result;
- }
- Framework::JSON::JSONObject* JVariableBoolExpressionFactory::toJsonObject(
- JVariableBoolExpression* zObject) const
- {
- Framework::JSON::JSONObject* result = new Framework::JSON::JSONObject();
- result->addValue(
- "name", new Framework::JSON::JSONString(zObject->getName()));
- return result;
- }
- JSONObjectValidationBuilder* JVariableBoolExpressionFactory::addToValidator(
- JSONObjectValidationBuilder* builder) const
- {
- return builder->withRequiredString("name")->finishString();
- }
- const char* JVariableBoolExpressionFactory::getTypeToken() const
- {
- return "variable";
- }
- JConstantFloatExpression::JConstantFloatExpression()
- : JFloatExpression(),
- value(0)
- {}
- Framework::Assembly::AssemblyBlock& JConstantFloatExpression::buildAssembly(
- JExpressionMemory* zMemory)
- {
- codeBlock.addLoadValue(&value, Framework::Assembly::MM0);
- return codeBlock;
- }
- void JConstantFloatExpression::setValue(float value)
- {
- this->value = value;
- }
- float JConstantFloatExpression::getValue() const
- {
- return value;
- }
- JConstantFloatExpressionFactory::JConstantFloatExpressionFactory()
- : SubTypeFactory()
- {}
- JConstantFloatExpression* JConstantFloatExpressionFactory::fromJson(
- Framework::JSON::JSONObject* zJson) const
- {
- JConstantFloatExpression* result = new JConstantFloatExpression();
- result->setValue((float)zJson->zValue("value")->asNumber()->getNumber());
- return result;
- }
- Framework::JSON::JSONObject* JConstantFloatExpressionFactory::toJsonObject(
- JConstantFloatExpression* zObject) const
- {
- Framework::JSON::JSONObject* result = new Framework::JSON::JSONObject();
- result->addValue(
- "value", new Framework::JSON::JSONNumber(zObject->getValue()));
- return result;
- }
- JSONObjectValidationBuilder* JConstantFloatExpressionFactory::addToValidator(
- JSONObjectValidationBuilder* builder) const
- {
- return builder->withRequiredNumber("value")->finishNumber();
- }
- const char* JConstantFloatExpressionFactory::getTypeToken() const
- {
- return "constant";
- }
- JConstantBoolExpression::JConstantBoolExpression()
- : JBoolExpression()
- {}
- Framework::Assembly::AssemblyBlock& JConstantBoolExpression::buildAssembly(
- JExpressionMemory* zMemory)
- {
- codeBlock.addMoveValue(Framework::Assembly::RAX, (char)(value ? 1 : 0));
- return codeBlock;
- }
- void JConstantBoolExpression::setValue(bool value)
- {
- this->value = value;
- }
- bool JConstantBoolExpression::getValue() const
- {
- return value;
- }
- JConstantBoolExpressionFactory::JConstantBoolExpressionFactory()
- : SubTypeFactory()
- {}
- JConstantBoolExpression* JConstantBoolExpressionFactory::fromJson(
- Framework::JSON::JSONObject* zJson) const
- {
- JConstantBoolExpression* result = new JConstantBoolExpression();
- result->setValue(zJson->zValue("value")->asBool()->getBool());
- return result;
- }
- Framework::JSON::JSONObject* JConstantBoolExpressionFactory::toJsonObject(
- JConstantBoolExpression* zObject) const
- {
- Framework::JSON::JSONObject* zResult = new Framework::JSON::JSONObject();
- zResult->addValue(
- "value", new Framework::JSON::JSONBool(zObject->getValue()));
- return zResult;
- }
- JSONObjectValidationBuilder* JConstantBoolExpressionFactory::addToValidator(
- JSONObjectValidationBuilder* builder) const
- {
- return builder->withRequiredBool("value")->finishBool();
- }
- const char* JConstantBoolExpressionFactory::getTypeToken() const
- {
- return "constant";
- }
- JNoiseFloatExpression::JNoiseFloatExpression()
- : JFloatExpression(),
- x(0),
- y(0),
- z(0)
- {}
- JNoiseFloatExpression::~JNoiseFloatExpression()
- {
- if (x) x->release();
- if (y) y->release();
- if (z) z->release();
- }
- Framework::Assembly::AssemblyBlock& JNoiseFloatExpression::buildAssembly(
- JExpressionMemory* zMemory)
- {
- Noise* noise = zMemory->zNoiseP(name);
- if (!noise)
- {
- Framework::Logging::error() << "no noise with name '" << name.getText()
- << "' found, behavior is undefined\n";
- return codeBlock;
- }
- Framework::Assembly::AssemblyBlock& xBlock = x->buildAssembly(zMemory);
- Framework::Assembly::AssemblyBlock& yBlock = y->buildAssembly(zMemory);
- Framework::Assembly::AssemblyBlock& zBlock = z->buildAssembly(zMemory);
- Framework::Assembly::FPRegister xTarget = Framework::Assembly::MM0;
- if (xBlock.isReplacementPossible(
- Framework::Assembly::MM0, Framework::Assembly::MM1))
- {
- xBlock.replaceRegister(
- Framework::Assembly::MM0, Framework::Assembly::MM1);
- xTarget = Framework::Assembly::MM1;
- }
- Framework::Assembly::FPRegister yTarget = Framework::Assembly::MM0;
- if (yBlock.isReplacementPossible(
- Framework::Assembly::MM0, Framework::Assembly::MM2))
- {
- yBlock.replaceRegister(
- Framework::Assembly::MM0, Framework::Assembly::MM2);
- yTarget = Framework::Assembly::MM2;
- }
- Framework::Assembly::FPRegister zTarget = Framework::Assembly::MM0;
- if (zBlock.isReplacementPossible(
- Framework::Assembly::MM0, Framework::Assembly::MM3))
- {
- zBlock.replaceRegister(
- Framework::Assembly::MM0, Framework::Assembly::MM3);
- zTarget = Framework::Assembly::MM3;
- }
- codeBlock.addBlock(&zBlock, {}, {}, {}, 0, &zTarget);
- if (zTarget != Framework::Assembly::MM3)
- {
- codeBlock.addMoveValue(Framework::Assembly::MM3,
- zTarget,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::X);
- }
- codeBlock.addBlock(&yBlock,
- {},
- {Framework::Assembly::MM3},
- {Framework::Assembly::SINGLE_FLOAT},
- 0,
- &yTarget);
- if (yTarget != Framework::Assembly::MM2)
- {
- codeBlock.addMoveValue(Framework::Assembly::MM2,
- yTarget,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::X);
- }
- codeBlock.addBlock(&xBlock,
- {},
- {Framework::Assembly::MM2, Framework::Assembly::MM3},
- {Framework::Assembly::SINGLE_FLOAT, Framework::Assembly::SINGLE_FLOAT},
- 0,
- &xTarget);
- if (xTarget != Framework::Assembly::MM1)
- {
- codeBlock.addMoveValue(Framework::Assembly::MM1,
- xTarget,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::X);
- }
- codeBlock.addLoadAddress(noise, Framework::Assembly::RCX);
- codeBlock.addMemberCall<float (Noise::*)(float, float, float)>(
- &Noise::getNoise,
- Framework::Assembly::FLOAT_VALUE,
- {Framework::Assembly::RCX},
- {Framework::Assembly::MM1,
- Framework::Assembly::MM2,
- Framework::Assembly::MM3});
- return codeBlock;
- }
- void JNoiseFloatExpression::setName(Framework::Text name)
- {
- this->name = name;
- }
- Framework::Text JNoiseFloatExpression::getName() const
- {
- return name;
- }
- void JNoiseFloatExpression::setX(JFloatExpression* x)
- {
- if (this->x) this->x->release();
- this->x = x;
- }
- JFloatExpression* JNoiseFloatExpression::zX() const
- {
- return x;
- }
- void JNoiseFloatExpression::setY(JFloatExpression* y)
- {
- if (this->y) this->y->release();
- this->y = y;
- }
- JFloatExpression* JNoiseFloatExpression::zY() const
- {
- return y;
- }
- void JNoiseFloatExpression::setZ(JFloatExpression* z)
- {
- if (this->z) this->z->release();
- this->z = z;
- }
- JFloatExpression* JNoiseFloatExpression::zZ() const
- {
- return z;
- }
- JNoiseFloatExpressionFactory::JNoiseFloatExpressionFactory()
- : SubTypeFactory()
- {}
- JNoiseFloatExpression* JNoiseFloatExpressionFactory::fromJson(
- Framework::JSON::JSONObject* zJson) const
- {
- JNoiseFloatExpression* result = new JNoiseFloatExpression();
- result->setName(zJson->zValue("name")->asString()->getString());
- result->setX(Game::INSTANCE->zTypeRegistry()->fromJson<JFloatExpression>(
- zJson->zValue("x")));
- result->setY(Game::INSTANCE->zTypeRegistry()->fromJson<JFloatExpression>(
- zJson->zValue("y")));
- result->setZ(Game::INSTANCE->zTypeRegistry()->fromJson<JFloatExpression>(
- zJson->zValue("z")));
- return result;
- }
- Framework::JSON::JSONObject* JNoiseFloatExpressionFactory::toJsonObject(
- JNoiseFloatExpression* zObject) const
- {
- Framework::JSON::JSONObject* result = new Framework::JSON::JSONObject();
- result->addValue(
- "name", new Framework::JSON::JSONString(zObject->getName()));
- result->addValue(
- "x", Game::INSTANCE->zTypeRegistry()->toJson(zObject->zX()));
- result->addValue(
- "y", Game::INSTANCE->zTypeRegistry()->toJson(zObject->zY()));
- result->addValue(
- "z", Game::INSTANCE->zTypeRegistry()->toJson(zObject->zZ()));
- return result;
- }
- JSONObjectValidationBuilder* JNoiseFloatExpressionFactory::addToValidator(
- JSONObjectValidationBuilder* builder) const
- {
- return builder->withRequiredString("name")
- ->finishString()
- ->withRequiredAttribute("x",
- Game::INSTANCE->zTypeRegistry()->getValidator<JFloatExpression>())
- ->withRequiredAttribute("y",
- Game::INSTANCE->zTypeRegistry()->getValidator<JFloatExpression>())
- ->withRequiredAttribute("z",
- Game::INSTANCE->zTypeRegistry()->getValidator<JFloatExpression>());
- }
- const char* JNoiseFloatExpressionFactory::getTypeToken() const
- {
- return "noise";
- }
- JOperatorFloatExpression::JOperatorFloatExpression()
- : JFloatExpression()
- {}
- Framework::Assembly::AssemblyBlock& JOperatorFloatExpression::buildAssembly(
- JExpressionMemory* zMemory)
- {
- bool first = 1;
- if (!values.getEntryCount())
- {
- codeBlock.addMoveValue(Framework::Assembly::MM0, 0.f);
- }
- for (JFloatExpression* expression : values)
- {
- if (first)
- {
- first = 0;
- codeBlock.addBlock(
- &expression->buildAssembly(zMemory), {}, {}, {}, 0, 0);
- }
- else
- {
- Framework::Assembly::AssemblyBlock& exprBlock
- = expression->buildAssembly(zMemory);
- if (exprBlock.isReplacementPossible(
- Framework::Assembly::MM0, Framework::Assembly::MM1))
- {
- exprBlock.replaceRegister(
- Framework::Assembly::MM0, Framework::Assembly::MM1);
- }
- else
- {
- exprBlock.addMoveValue(Framework::Assembly::MM1,
- Framework::Assembly::MM0,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::X);
- }
- codeBlock.addBlock(&exprBlock,
- {},
- {Framework::Assembly::MM0},
- {Framework::Assembly::SINGLE_FLOAT},
- 0,
- 0);
- if (op.isEqual("+"))
- {
- codeBlock.addAddition(Framework::Assembly::MM0,
- Framework::Assembly::MM1,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::X);
- }
- else if (op.isEqual("-"))
- {
- codeBlock.addSubtraction(Framework::Assembly::MM0,
- Framework::Assembly::MM1,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::X);
- }
- else if (op.isEqual("*"))
- {
- codeBlock.addMultiplication(Framework::Assembly::MM0,
- Framework::Assembly::MM1,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::X);
- }
- else if (op.isEqual("/"))
- {
- codeBlock.addDivision(Framework::Assembly::MM0,
- Framework::Assembly::MM1,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::X);
- }
- }
- }
- return codeBlock;
- }
- void JOperatorFloatExpression::setOperator(Framework::Text op)
- {
- this->op = op;
- }
- Framework::Text JOperatorFloatExpression::getOperator()
- {
- return op;
- }
- void JOperatorFloatExpression::addValue(JFloatExpression* value)
- {
- values.add(value);
- }
- const Framework::RCArray<JFloatExpression>&
- JOperatorFloatExpression::getValues() const
- {
- return values;
- }
- JOperatorFloatExpressionFactory::JOperatorFloatExpressionFactory()
- : SubTypeFactory()
- {}
- JOperatorFloatExpression* JOperatorFloatExpressionFactory::fromJson(
- Framework::JSON::JSONObject* zJson) const
- {
- JOperatorFloatExpression* result = new JOperatorFloatExpression();
- result->setOperator(zJson->zValue("operator")->asString()->getString());
- for (Framework::JSON::JSONValue* value :
- *zJson->zValue("values")->asArray())
- {
- result->addValue(
- Game::INSTANCE->zTypeRegistry()->fromJson<JFloatExpression>(value));
- }
- return result;
- }
- Framework::JSON::JSONObject* JOperatorFloatExpressionFactory::toJsonObject(
- JOperatorFloatExpression* zObject) const
- {
- Framework::JSON::JSONObject* result = new Framework::JSON::JSONObject();
- result->addValue(
- "operator", new Framework::JSON::JSONString(zObject->getOperator()));
- Framework::JSON::JSONArray* values = new Framework::JSON::JSONArray();
- for (JFloatExpression* expression : zObject->getValues())
- {
- values->addValue(
- Game::INSTANCE->zTypeRegistry()->toJson<JFloatExpression>(
- expression));
- }
- result->addValue("values", values);
- return result;
- }
- JSONObjectValidationBuilder* JOperatorFloatExpressionFactory::addToValidator(
- JSONObjectValidationBuilder* builder) const
- {
- return builder->withRequiredString("operator")
- ->whichIsOneOf({"+", "-", "*", "/"})
- ->finishString()
- ->withRequiredArray("values")
- ->addAcceptedTypeInArray(
- Game::INSTANCE->zTypeRegistry()->getValidator<JFloatExpression>())
- ->finishArray();
- }
- const char* JOperatorFloatExpressionFactory::getTypeToken() const
- {
- return "operator";
- }
- JBoolOperatorBoolExpression::JBoolOperatorBoolExpression()
- : JBoolExpression()
- {}
- Framework::Assembly::AssemblyBlock& JBoolOperatorBoolExpression::buildAssembly(
- JExpressionMemory* zMemory)
- {
- bool first = 1;
- if (!values.getEntryCount())
- {
- codeBlock.addMoveValue(Framework::Assembly::RAX, (char)0);
- }
- for (JBoolExpression* expression : values)
- {
- if (first)
- {
- first = 0;
- codeBlock.addBlock(
- &expression->buildAssembly(zMemory), {}, {}, {}, 0, 0);
- }
- else
- {
- Framework::Assembly::AssemblyBlock& exprBlock
- = expression->buildAssembly(zMemory);
- if (exprBlock.isReplacementPossible(
- Framework::Assembly::RAX, Framework::Assembly::RCX))
- {
- exprBlock.replaceRegister(
- Framework::Assembly::RAX, Framework::Assembly::RCX);
- }
- else
- {
- exprBlock.addMoveValue(Framework::Assembly::RCX,
- Framework::Assembly::RAX,
- Framework::Assembly::LOWER8);
- }
- codeBlock.addBlock(
- &exprBlock, {Framework::Assembly::RAX}, {}, {}, 0, 0);
- if (op.isEqual("&&"))
- {
- codeBlock.addAnd(Framework::Assembly::RAX,
- Framework::Assembly::RCX,
- Framework::Assembly::LOWER8);
- }
- else if (op.isEqual("||"))
- {
- codeBlock.addOr(Framework::Assembly::RAX,
- Framework::Assembly::RCX,
- Framework::Assembly::LOWER8);
- }
- }
- }
- return codeBlock;
- }
- void JBoolOperatorBoolExpression::setOperator(Framework::Text op)
- {
- this->op = op;
- }
- Framework::Text JBoolOperatorBoolExpression::getOperator()
- {
- return op;
- }
- void JBoolOperatorBoolExpression::addValue(JBoolExpression* value)
- {
- values.add(value);
- }
- const Framework::RCArray<JBoolExpression>&
- JBoolOperatorBoolExpression::getValues() const
- {
- return values;
- }
- JBoolOperatorBoolExpressionFactory::JBoolOperatorBoolExpressionFactory()
- : SubTypeFactory()
- {}
- JBoolOperatorBoolExpression* JBoolOperatorBoolExpressionFactory::fromJson(
- Framework::JSON::JSONObject* zJson) const
- {
- JBoolOperatorBoolExpression* result = new JBoolOperatorBoolExpression();
- for (Framework::JSON::JSONValue* value :
- *zJson->zValue("values")->asArray())
- {
- result->addValue(
- Game::INSTANCE->zTypeRegistry()->fromJson<JBoolExpression>(value));
- }
- result->setOperator(zJson->zValue("operator")->asString()->getString());
- return result;
- }
- Framework::JSON::JSONObject* JBoolOperatorBoolExpressionFactory::toJsonObject(
- JBoolOperatorBoolExpression* zObject) const
- {
- Framework::JSON::JSONObject* zResult = new Framework::JSON::JSONObject();
- Framework::JSON::JSONArray* values = new Framework::JSON::JSONArray();
- for (JBoolExpression* expression : zObject->getValues())
- {
- values->addValue(
- Game::INSTANCE->zTypeRegistry()->toJson<JBoolExpression>(
- expression));
- }
- zResult->addValue("values", values);
- zResult->addValue(
- "operator", new Framework::JSON::JSONString(zObject->getOperator()));
- return zResult;
- }
- JSONObjectValidationBuilder* JBoolOperatorBoolExpressionFactory::addToValidator(
- JSONObjectValidationBuilder* builder) const
- {
- return builder->withRequiredString("operator")
- ->whichIsOneOf({"&&", "||"})
- ->finishString()
- ->withRequiredArray("values")
- ->addAcceptedTypeInArray(
- Game::INSTANCE->zTypeRegistry()->getValidator<JBoolExpression>())
- ->finishArray();
- }
- const char* JBoolOperatorBoolExpressionFactory::getTypeToken() const
- {
- return "operator";
- }
- JFloatOperatorBoolExpression::JFloatOperatorBoolExpression()
- : JBoolExpression()
- {}
- Framework::Assembly::AssemblyBlock& JFloatOperatorBoolExpression::buildAssembly(
- JExpressionMemory* zMemory)
- {
- bool first = 1;
- Framework::Assembly::FPRegister lastResultSorage = Framework::Assembly::MM0;
- for (JFloatExpression* expression : values)
- {
- if (first)
- {
- first = 0;
- codeBlock.addBlock(
- &expression->buildAssembly(zMemory), {}, {}, {}, 0, 0);
- }
- else
- {
- Framework::Assembly::FPRegister currentResultSorage
- = lastResultSorage == Framework::Assembly::MM0
- ? Framework::Assembly::MM1
- : Framework::Assembly::MM0;
- Framework::Assembly::AssemblyBlock& exprBlock
- = expression->buildAssembly(zMemory);
- if (currentResultSorage != Framework::Assembly::MM0)
- {
- if (exprBlock.isReplacementPossible(
- Framework::Assembly::MM0, currentResultSorage))
- {
- exprBlock.replaceRegister(
- Framework::Assembly::MM0, currentResultSorage);
- }
- else
- {
- exprBlock.addMoveValue(currentResultSorage,
- Framework::Assembly::MM0,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::X);
- }
- }
- codeBlock.addBlock(&exprBlock,
- {},
- {lastResultSorage},
- {Framework::Assembly::SINGLE_FLOAT},
- 0,
- 0);
- Framework::Assembly::Operation jumpOp = Framework::Assembly::NOP;
- bool needConversion = false;
- if (op.isEqual(">"))
- {
- jumpOp = Framework::Assembly::JBE; // jump if below or equal
- }
- else if (op.isEqual("<"))
- {
- jumpOp = Framework::Assembly::JNB; // jump if not below
- }
- else if (op.isEqual(">="))
- {
- jumpOp = Framework::Assembly::JB; // jump if below
- }
- else if (op.isEqual("<="))
- {
- jumpOp = Framework::Assembly::JA; // jump if above
- }
- else if (op.isEqual("=="))
- {
- jumpOp = Framework::Assembly::JNE; // jump if not equal
- }
- else if (op.isEqual("!="))
- {
- jumpOp = Framework::Assembly::JE; // jump if equal
- }
- else
- {
- needConversion = true;
- if (op.isEqual(">i"))
- {
- jumpOp = Framework::Assembly::JLE; // jump if less or equal
- }
- else if (op.isEqual("<i"))
- {
- jumpOp = Framework::Assembly::JGE; // jump if greater or
- // equal
- }
- else if (op.isEqual(">=i"))
- {
- jumpOp = Framework::Assembly::JL; // jump if less
- }
- else if (op.isEqual("<=i"))
- {
- jumpOp = Framework::Assembly::JG; // jump if greater
- }
- else if (op.isEqual("==i"))
- {
- jumpOp = Framework::Assembly::JNE; // jump if not equal
- }
- else if (op.isEqual("!=i"))
- {
- jumpOp = Framework::Assembly::JE; // jump if equal
- }
- }
- if (needConversion)
- {
- codeBlock.addConversion(Framework::Assembly::RAX,
- lastResultSorage,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::LOWER32);
- codeBlock.addConversion(Framework::Assembly::RCX,
- currentResultSorage,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::LOWER32);
- codeBlock.addCompare(Framework::Assembly::RAX,
- Framework::Assembly::RCX,
- Framework::Assembly::LOWER32);
- }
- else
- {
- codeBlock.addCompare(lastResultSorage,
- currentResultSorage,
- Framework::Assembly::SINGLE_FLOAT);
- }
- codeBlock.addJump(jumpOp, "end_false");
- lastResultSorage = currentResultSorage;
- }
- }
- codeBlock.addMoveValue(Framework::Assembly::RAX, (char)1);
- codeBlock.addJump(Framework::Assembly::JMP, "end");
- codeBlock.defineJumpTarget("end_false");
- codeBlock.addMoveValue(Framework::Assembly::RAX, (char)0);
- codeBlock.defineJumpTarget("end");
- return codeBlock;
- }
- void JFloatOperatorBoolExpression::setOperator(Framework::Text op)
- {
- this->op = op;
- }
- Framework::Text JFloatOperatorBoolExpression::getOperator()
- {
- return op;
- }
- void JFloatOperatorBoolExpression::addValue(JFloatExpression* value)
- {
- values.add(value);
- }
- const Framework::RCArray<JFloatExpression>&
- JFloatOperatorBoolExpression::getValues() const
- {
- return values;
- }
- JFloatOperatorBoolExpressionFactory::JFloatOperatorBoolExpressionFactory()
- : SubTypeFactory()
- {}
- JFloatOperatorBoolExpression* JFloatOperatorBoolExpressionFactory::fromJson(
- Framework::JSON::JSONObject* zJson) const
- {
- JFloatOperatorBoolExpression* result = new JFloatOperatorBoolExpression();
- result->setOperator(zJson->zValue("operator")->asString()->getString());
- for (Framework::JSON::JSONValue* value :
- *zJson->zValue("values")->asArray())
- {
- result->addValue(
- Game::INSTANCE->zTypeRegistry()->fromJson<JFloatExpression>(value));
- }
- return result;
- }
- Framework::JSON::JSONObject* JFloatOperatorBoolExpressionFactory::toJsonObject(
- JFloatOperatorBoolExpression* zObject) const
- {
- Framework::JSON::JSONObject* result = new Framework::JSON::JSONObject();
- result->addValue(
- "operator", new Framework::JSON::JSONString(zObject->getOperator()));
- Framework::JSON::JSONArray* values = new Framework::JSON::JSONArray();
- for (JFloatExpression* expression : zObject->getValues())
- {
- values->addValue(
- Game::INSTANCE->zTypeRegistry()->toJson<JFloatExpression>(
- expression));
- }
- result->addValue("values", values);
- return result;
- }
- JSONObjectValidationBuilder*
- JFloatOperatorBoolExpressionFactory::addToValidator(
- JSONObjectValidationBuilder* builder) const
- {
- return builder->withRequiredString("operator")
- ->whichIsOneOf({">",
- "<",
- ">=",
- "<=",
- "==",
- "!=",
- "<i",
- ">i",
- ">=i",
- "<=i",
- "==i",
- "!=i"})
- ->finishString()
- ->withRequiredArray("values")
- ->addAcceptedTypeInArray(
- Game::INSTANCE->zTypeRegistry()->getValidator<JFloatExpression>())
- ->finishArray();
- }
- const char* JFloatOperatorBoolExpressionFactory::getTypeToken() const
- {
- return "comparsion";
- }
- JSpecificBlockBoolExpression::JSpecificBlockBoolExpression()
- : JBoolExpression(),
- filter(0),
- x(0),
- y(0),
- z(0)
- {}
- JSpecificBlockBoolExpression::~JSpecificBlockBoolExpression()
- {
- if (filter) filter->release();
- if (x) x->release();
- if (y) y->release();
- if (z) z->release();
- }
- bool JSpecificBlockBoolExpression::isValidPosition(
- int x, int y, Chunk* currentChunk)
- {
- return currentChunk
- && Game::getChunkCenter(x, y) == currentChunk->getCenter();
- }
- Framework::Assembly::AssemblyBlock& JSpecificBlockBoolExpression::buildAssembly(
- JExpressionMemory* zMemory)
- {
- Framework::Assembly::AssemblyBlock& xBlock = x->buildAssembly(zMemory);
- Framework::Assembly::AssemblyBlock& yBlock = y->buildAssembly(zMemory);
- Framework::Assembly::AssemblyBlock& zBlock = z->buildAssembly(zMemory);
- Framework::Assembly::FPRegister xTarget = Framework::Assembly::MM0;
- if (xBlock.isReplacementPossible(
- Framework::Assembly::MM0, Framework::Assembly::MM1))
- {
- xBlock.replaceRegister(
- Framework::Assembly::MM0, Framework::Assembly::MM1);
- xTarget = Framework::Assembly::MM1;
- }
- Framework::Assembly::FPRegister yTarget = Framework::Assembly::MM0;
- if (yBlock.isReplacementPossible(
- Framework::Assembly::MM0, Framework::Assembly::MM2))
- {
- yBlock.replaceRegister(
- Framework::Assembly::MM0, Framework::Assembly::MM2);
- yTarget = Framework::Assembly::MM2;
- }
- Framework::Assembly::FPRegister zTarget = Framework::Assembly::MM0;
- if (zBlock.isReplacementPossible(
- Framework::Assembly::MM0, Framework::Assembly::MM3))
- {
- zBlock.replaceRegister(
- Framework::Assembly::MM0, Framework::Assembly::MM3);
- zTarget = Framework::Assembly::MM3;
- }
- codeBlock.addBlock(&zBlock, {}, {}, {}, 0, &zTarget);
- codeBlock.addConversion(Framework::Assembly::R9,
- zTarget,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::LOWER32,
- 1);
- codeBlock.addTest(Framework::Assembly::R9,
- Framework::Assembly::R9,
- Framework::Assembly::LOWER32);
- codeBlock.addJump(Framework::Assembly::JL, "end_false");
- codeBlock.addCompare(Framework::Assembly::R9, WORLD_HEIGHT);
- codeBlock.addJump(Framework::Assembly::JGE, "end_false");
- codeBlock.addBlock(&yBlock, {Framework::Assembly::R9}, {}, {}, 0, &yTarget);
- codeBlock.addConversion(Framework::Assembly::R8,
- yTarget,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::LOWER32,
- 1);
- codeBlock.addBlock(&xBlock,
- {Framework::Assembly::R8, Framework::Assembly::R9},
- {},
- {},
- 0,
- &xTarget);
- codeBlock.addConversion(Framework::Assembly::RDX,
- xTarget,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::LOWER32,
- 1);
- codeBlock.addLoadAddress(this, Framework::Assembly::RCX);
- codeBlock.addPush(Framework::Assembly::RDX, Framework::Assembly::LOWER32);
- codeBlock.addPush(Framework::Assembly::R8, Framework::Assembly::LOWER32);
- codeBlock.addPush(Framework::Assembly::R9, Framework::Assembly::LOWER32);
- codeBlock.addLoadValue(
- (__int64*)zMemory->zzCurrentChunk(), Framework::Assembly::R9);
- codeBlock.addMemberCall<bool (JSpecificBlockBoolExpression::*)(
- int, int, Chunk*)>(&JSpecificBlockBoolExpression::isValidPosition,
- Framework::Assembly::INT_VALUE,
- {Framework::Assembly::R9},
- {});
- codeBlock.addPop(Framework::Assembly::R9, Framework::Assembly::LOWER32);
- codeBlock.addPop(Framework::Assembly::R8, Framework::Assembly::LOWER32);
- codeBlock.addPop(Framework::Assembly::RDX, Framework::Assembly::LOWER32);
- codeBlock.addTest(Framework::Assembly::RAX,
- Framework::Assembly::RAX,
- Framework::Assembly::LOWER8);
- codeBlock.addJump(Framework::Assembly::JZ, "end");
- codeBlock.addLoadValue(
- (__int64*)zMemory->zzCurrentChunk(), Framework::Assembly::RCX);
- codeBlock.addMemberCall<const Block* (Chunk::*)(int, int, int) const>(
- &Chunk::zBlockConstWC,
- Framework::Assembly::INT_VALUE,
- {Framework::Assembly::RCX,
- Framework::Assembly::RDX,
- Framework::Assembly::R8,
- Framework::Assembly::R9},
- {});
- codeBlock.addMoveValue(Framework::Assembly::RDX, Framework::Assembly::RAX);
- codeBlock.addLoadValue((__int64*)&filter, Framework::Assembly::RCX);
- codeBlock.addMemberCall<bool (BlockFilter::*)(const Block*) const>(
- &BlockFilter::test,
- Framework::Assembly::INT_VALUE,
- {Framework::Assembly::RCX, Framework::Assembly::RDX},
- {});
- codeBlock.addTest(Framework::Assembly::RAX,
- Framework::Assembly::RAX,
- Framework::Assembly::LOWER8);
- codeBlock.addJump(Framework::Assembly::JZ, "end");
- codeBlock.addMoveValue(Framework::Assembly::RAX, (char)1);
- codeBlock.addJump(Framework::Assembly::JMP, "end");
- codeBlock.defineJumpTarget("end_false");
- codeBlock.addMoveValue(Framework::Assembly::RAX, (char)0);
- codeBlock.defineJumpTarget("end");
- return codeBlock;
- }
- void JSpecificBlockBoolExpression::setFilter(BlockFilter* filter)
- {
- if (this->filter)
- {
- this->filter->release();
- }
- this->filter = filter;
- }
- BlockFilter* JSpecificBlockBoolExpression::zFilter() const
- {
- return filter;
- }
- void JSpecificBlockBoolExpression::setX(JFloatExpression* x)
- {
- if (this->x) this->x->release();
- this->x = x;
- }
- JFloatExpression* JSpecificBlockBoolExpression::zX() const
- {
- return x;
- }
- void JSpecificBlockBoolExpression::setY(JFloatExpression* y)
- {
- if (this->y) this->y->release();
- this->y = y;
- }
- JFloatExpression* JSpecificBlockBoolExpression::zY() const
- {
- return y;
- }
- void JSpecificBlockBoolExpression::setZ(JFloatExpression* z)
- {
- if (this->z) this->z->release();
- this->z = z;
- }
- JFloatExpression* JSpecificBlockBoolExpression::zZ() const
- {
- return z;
- }
- JSpecificBlockBoolExpressionFactory::JSpecificBlockBoolExpressionFactory()
- : SubTypeFactory()
- {}
- JSpecificBlockBoolExpression* JSpecificBlockBoolExpressionFactory::fromJson(
- Framework::JSON::JSONObject* zJson) const
- {
- JSpecificBlockBoolExpression* result = new JSpecificBlockBoolExpression();
- result->setFilter(Game::INSTANCE->zTypeRegistry()->fromJson<BlockFilter>(
- zJson->zValue("condition")));
- result->setX(Game::INSTANCE->zTypeRegistry()->fromJson<JFloatExpression>(
- zJson->zValue("x")));
- result->setY(Game::INSTANCE->zTypeRegistry()->fromJson<JFloatExpression>(
- zJson->zValue("y")));
- result->setZ(Game::INSTANCE->zTypeRegistry()->fromJson<JFloatExpression>(
- zJson->zValue("z")));
- return result;
- }
- Framework::JSON::JSONObject* JSpecificBlockBoolExpressionFactory::toJsonObject(
- JSpecificBlockBoolExpression* zObject) const
- {
- Framework::JSON::JSONObject* result = new Framework::JSON::JSONObject();
- result->addValue("condition",
- Game::INSTANCE->zTypeRegistry()->toJson(zObject->zFilter()));
- result->addValue(
- "x", Game::INSTANCE->zTypeRegistry()->toJson(zObject->zX()));
- result->addValue(
- "y", Game::INSTANCE->zTypeRegistry()->toJson(zObject->zY()));
- result->addValue(
- "z", Game::INSTANCE->zTypeRegistry()->toJson(zObject->zZ()));
- return result;
- }
- JSONObjectValidationBuilder*
- JSpecificBlockBoolExpressionFactory::addToValidator(
- JSONObjectValidationBuilder* builder) const
- {
- return builder
- ->withRequiredAttribute("condition",
- Game::INSTANCE->zTypeRegistry()->getValidator<BlockFilter>())
- ->withRequiredAttribute("x",
- Game::INSTANCE->zTypeRegistry()->getValidator<JFloatExpression>())
- ->withRequiredAttribute("y",
- Game::INSTANCE->zTypeRegistry()->getValidator<JFloatExpression>())
- ->withRequiredAttribute("z",
- Game::INSTANCE->zTypeRegistry()->getValidator<JFloatExpression>());
- }
- const char* JSpecificBlockBoolExpressionFactory::getTypeToken() const
- {
- return "specificBlockMatches";
- }
- JFirstBlockAboveBoolExpression::JFirstBlockAboveBoolExpression()
- : JBoolExpression(),
- filter(0),
- maxDistance(0)
- {}
- JFirstBlockAboveBoolExpression::~JFirstBlockAboveBoolExpression()
- {
- if (filter) filter->release();
- }
- Framework::Assembly::AssemblyBlock&
- JFirstBlockAboveBoolExpression::buildAssembly(JExpressionMemory* zMemory)
- {
- // load x into R12
- codeBlock.addLoadValue(
- zMemory->getFloatVariableP("x"), Framework::Assembly::MM0);
- codeBlock.addConversion(Framework::Assembly::R12,
- Framework::Assembly::MM0,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::LOWER32,
- 1);
- // load y into R13
- codeBlock.addLoadValue(
- zMemory->getFloatVariableP("y"), Framework::Assembly::MM0);
- codeBlock.addConversion(Framework::Assembly::R13,
- Framework::Assembly::MM0,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::LOWER32,
- 1);
- // load z into R14
- codeBlock.addLoadValue(
- zMemory->getFloatVariableP("z"), Framework::Assembly::MM0);
- codeBlock.addConversion(Framework::Assembly::R14,
- Framework::Assembly::MM0,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::LOWER32,
- 1);
- // load current chunk into R15
- codeBlock.addLoadValue(
- (__int64*)zMemory->zzCurrentChunk(), Framework::Assembly::R15);
- if (maxDistance)
- {
- // calculate loop end
- codeBlock.addMoveValue(
- Framework::Assembly::RBX, Framework::Assembly::R14);
- codeBlock.addAddition(Framework::Assembly::RBX, maxDistance + 1);
- codeBlock.addCompare(Framework::Assembly::RBX, WORLD_HEIGHT);
- codeBlock.addJump(Framework::Assembly::JL, "continue");
- codeBlock.addMoveValue(Framework::Assembly::RBX, WORLD_HEIGHT);
- codeBlock.defineJumpTarget("continue");
- }
- // begin loop to check above blocks
- codeBlock.defineJumpTarget("loop_start");
- // increment height
- codeBlock.addAddition(
- Framework::Assembly::R14, (char)1, Framework::Assembly::LOWER32);
- // check if height is above loop end
- if (maxDistance)
- {
- codeBlock.addCompare(Framework::Assembly::R14,
- Framework::Assembly::RBX,
- Framework::Assembly::LOWER32);
- }
- else
- {
- codeBlock.addCompare(Framework::Assembly::R14, WORLD_HEIGHT);
- }
- codeBlock.addJump(Framework::Assembly::JGE, "end_false");
- // load block type at current position into RAX
- codeBlock.addMoveValue(Framework::Assembly::RCX, Framework::Assembly::R15);
- codeBlock.addMoveValue(Framework::Assembly::RDX, Framework::Assembly::R12);
- codeBlock.addMoveValue(Framework::Assembly::R8, Framework::Assembly::R13);
- codeBlock.addMoveValue(Framework::Assembly::R9, Framework::Assembly::R14);
- if (maxDistance)
- { // save loop end in RBX for later comparison
- codeBlock.addPush(
- Framework::Assembly::RBX, Framework::Assembly::LOWER32);
- }
- codeBlock.addMemberCall<int (Chunk::*)(int, int, int) const>(
- &Chunk::getBlockTypeAtWC,
- Framework::Assembly::INT_VALUE,
- {Framework::Assembly::RCX,
- Framework::Assembly::RDX,
- Framework::Assembly::R8,
- Framework::Assembly::R9},
- {});
- if (maxDistance)
- {
- // restore loop end from stack
- codeBlock.addPop(
- Framework::Assembly::RBX, Framework::Assembly::LOWER32);
- }
- // check if block type is 0 (NO_BLOCK) or 1 (AIR)
- codeBlock.addCompare(Framework::Assembly::RAX, 1);
- codeBlock.addJump(Framework::Assembly::JLE, "loop_start");
- // end of loop
- // load current block into RDX
- codeBlock.addMoveValue(Framework::Assembly::RCX, Framework::Assembly::R15);
- codeBlock.addMoveValue(Framework::Assembly::RDX, Framework::Assembly::R12);
- codeBlock.addMoveValue(Framework::Assembly::R8, Framework::Assembly::R13);
- codeBlock.addMoveValue(Framework::Assembly::R9, Framework::Assembly::R14);
- codeBlock.addMemberCall<const Block* (Chunk::*)(int, int, int) const>(
- &Chunk::zBlockConstWC,
- Framework::Assembly::INT_VALUE,
- {Framework::Assembly::RCX,
- Framework::Assembly::RDX,
- Framework::Assembly::R8,
- Framework::Assembly::R9},
- {});
- codeBlock.addMoveValue(Framework::Assembly::RDX, Framework::Assembly::RAX);
- // load filter into RCX
- codeBlock.addLoadValue((__int64*)&filter, Framework::Assembly::RCX);
- // check filter condition
- codeBlock.addMemberCall<bool (BlockFilter::*)(const Block*) const>(
- &BlockFilter::test,
- Framework::Assembly::INT_VALUE,
- {Framework::Assembly::RCX, Framework::Assembly::RDX},
- {});
- codeBlock.addJump(Framework::Assembly::JMP, "end");
- codeBlock.defineJumpTarget("end_false");
- codeBlock.addMoveValue(Framework::Assembly::RAX, (char)0);
- codeBlock.defineJumpTarget("end");
- return codeBlock;
- }
- void JFirstBlockAboveBoolExpression::setFilter(BlockFilter* filter)
- {
- if (this->filter)
- {
- this->filter->release();
- }
- this->filter = filter;
- }
- BlockFilter* JFirstBlockAboveBoolExpression::zFilter() const
- {
- return filter;
- }
- void JFirstBlockAboveBoolExpression::setMaxDistance(short maxDistance)
- {
- this->maxDistance = maxDistance;
- }
- short JFirstBlockAboveBoolExpression::getMaxDistance() const
- {
- return maxDistance;
- }
- JFirstBlockAboveBoolExpressionFactory::JFirstBlockAboveBoolExpressionFactory()
- : SubTypeFactory()
- {}
- JFirstBlockAboveBoolExpression* JFirstBlockAboveBoolExpressionFactory::fromJson(
- Framework::JSON::JSONObject* zJson) const
- {
- JFirstBlockAboveBoolExpression* result
- = new JFirstBlockAboveBoolExpression();
- result->setFilter(Game::INSTANCE->zTypeRegistry()->fromJson<BlockFilter>(
- zJson->zValue("condition")));
- if (zJson->hasValue("maxDistance"))
- {
- result->setMaxDistance(
- (short)zJson->zValue("maxDistance")->asNumber()->getNumber());
- }
- return result;
- }
- Framework::JSON::JSONObject*
- JFirstBlockAboveBoolExpressionFactory::toJsonObject(
- JFirstBlockAboveBoolExpression* zObject) const
- {
- Framework::JSON::JSONObject* result = new Framework::JSON::JSONObject();
- result->addValue("condition",
- Game::INSTANCE->zTypeRegistry()->toJson(zObject->zFilter()));
- if (zObject->getMaxDistance())
- {
- result->addValue("maxDistance",
- new Framework::JSON::JSONNumber(zObject->getMaxDistance()));
- }
- return result;
- }
- JSONObjectValidationBuilder*
- JFirstBlockAboveBoolExpressionFactory::addToValidator(
- JSONObjectValidationBuilder* builder) const
- {
- return builder
- ->withRequiredAttribute("condition",
- Game::INSTANCE->zTypeRegistry()->getValidator<BlockFilter>())
- ->withOptionalNumber("maxDistance")
- ->whichIsGreaterThen(0)
- ->whichIsLessThen(WORLD_HEIGHT)
- ->finishNumber();
- }
- const char* JFirstBlockAboveBoolExpressionFactory::getTypeToken() const
- {
- return "firstBlockAboveMatches";
- }
- JNaighborBlockBoolExpression::JNaighborBlockBoolExpression()
- : JBoolExpression(),
- filter(0),
- validDirections(Direction::NO_DIRECTION)
- {}
- JNaighborBlockBoolExpression::~JNaighborBlockBoolExpression()
- {
- if (filter)
- {
- filter->release();
- }
- }
- bool JNaighborBlockBoolExpression::isValidPosition(
- int x, int y, Chunk* currentChunk)
- {
- return currentChunk
- && Game::getChunkCenter(x, y) == currentChunk->getCenter();
- }
- void JNaighborBlockBoolExpression::addEvaluation(
- Framework::Assembly::AssemblyBlock& codeBlock, JExpressionMemory* zMemory)
- {
- // call currentChunk->zBlockConstWC(x, y, z)
- codeBlock.addLoadValue(
- (__int64*)zMemory->zzCurrentChunk(), Framework::Assembly::RCX);
- codeBlock.addMemberCall<const Block* (Chunk::*)(int, int, int) const>(
- &Chunk::zBlockConstWC,
- Framework::Assembly::INT_VALUE,
- {Framework::Assembly::RCX,
- Framework::Assembly::RDX,
- Framework::Assembly::R8,
- Framework::Assembly::R9},
- {});
- // call filter->test(block)
- codeBlock.addLoadValue((__int64*)&filter, Framework::Assembly::RCX);
- codeBlock.addMoveValue(Framework::Assembly::RDX, Framework::Assembly::RAX);
- codeBlock.addMemberCall<bool (BlockFilter::*)(const Block*) const>(
- &BlockFilter::test,
- Framework::Assembly::INT_VALUE,
- {Framework::Assembly::RCX, Framework::Assembly::RDX},
- {});
- // if filter returns true, jump to end
- codeBlock.addTest(Framework::Assembly::RAX,
- Framework::Assembly::RAX,
- Framework::Assembly::LOWER8);
- codeBlock.addJump(Framework::Assembly::JNZ, "end_true");
- }
- Framework::Assembly::AssemblyBlock& JNaighborBlockBoolExpression::buildAssembly(
- JExpressionMemory* zMemory)
- {
- if (validDirections == Direction::NO_DIRECTION)
- {
- // no directions to check, return false
- codeBlock.addMoveValue(Framework::Assembly::RAX, (char)0);
- }
- else
- {
- // load x into R12
- codeBlock.addLoadValue(
- zMemory->getFloatVariableP("x"), Framework::Assembly::MM0);
- codeBlock.addConversion(Framework::Assembly::R12,
- Framework::Assembly::MM0,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::LOWER32,
- 1);
- // load y into R13
- codeBlock.addLoadValue(
- zMemory->getFloatVariableP("y"), Framework::Assembly::MM0);
- codeBlock.addConversion(Framework::Assembly::R13,
- Framework::Assembly::MM0,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::LOWER32,
- 1);
- // load z into R14
- codeBlock.addLoadValue(
- zMemory->getFloatVariableP("z"), Framework::Assembly::MM0);
- codeBlock.addConversion(Framework::Assembly::R14,
- Framework::Assembly::MM0,
- Framework::Assembly::SINGLE_FLOAT,
- Framework::Assembly::LOWER32,
- 1);
- if (validDirections & Direction::TOP)
- {
- // add 1 to z and check if it's above world height
- codeBlock.addMoveValue(
- Framework::Assembly::R9, Framework::Assembly::R14);
- codeBlock.addAddition(
- Framework::Assembly::R9, (char)1, Framework::Assembly::LOWER16);
- codeBlock.addCompare(Framework::Assembly::R9, WORLD_HEIGHT);
- codeBlock.addJump(Framework::Assembly::JGE, "skip_top");
- // evaluates filter on block at current position
- codeBlock.addMoveValue(
- Framework::Assembly::RDX, Framework::Assembly::R12);
- codeBlock.addMoveValue(
- Framework::Assembly::R8, Framework::Assembly::R13);
- addEvaluation(codeBlock, zMemory);
- codeBlock.defineJumpTarget("skip_top");
- }
- if (validDirections & Direction::BOTTOM)
- {
- // subtract 1 from z and check if it's >= 0
- codeBlock.addMoveValue(
- Framework::Assembly::R9, Framework::Assembly::R14);
- codeBlock.addSubtraction(
- Framework::Assembly::R9, (char)1, Framework::Assembly::LOWER16);
- codeBlock.addTest(Framework::Assembly::R9,
- Framework::Assembly::R9,
- Framework::Assembly::LOWER32);
- codeBlock.addJump(Framework::Assembly::JL, "skip_bottom");
- // evaluates filter on block at current position
- codeBlock.addMoveValue(
- Framework::Assembly::RDX, Framework::Assembly::R12);
- codeBlock.addMoveValue(
- Framework::Assembly::R8, Framework::Assembly::R13);
- addEvaluation(codeBlock, zMemory);
- codeBlock.defineJumpTarget("skip_bottom");
- }
- for (int i = 0; i < 4; i++)
- { // check horizontal directions
- Direction dir = getDirectionFromIndex(i);
- if (validDirections & dir)
- {
- Framework::Vec3<int> offset = getDirection(dir);
- codeBlock.addMoveValue(
- Framework::Assembly::RDX, Framework::Assembly::R12);
- codeBlock.addMoveValue(
- Framework::Assembly::R8, Framework::Assembly::R13);
- if (offset.x > 0)
- {
- codeBlock.addAddition(Framework::Assembly::RDX,
- (char)1,
- Framework::Assembly::LOWER32);
- }
- else if (offset.x < 0)
- {
- codeBlock.addSubtraction(Framework::Assembly::RDX,
- (char)1,
- Framework::Assembly::LOWER32);
- }
- if (offset.y > 0)
- {
- codeBlock.addAddition(Framework::Assembly::R8,
- (char)1,
- Framework::Assembly::LOWER32);
- }
- else if (offset.y < 0)
- {
- codeBlock.addSubtraction(Framework::Assembly::R8,
- (char)1,
- Framework::Assembly::LOWER32);
- }
- // check if new position is in the same chunk
- codeBlock.addLoadValue((__int64*)zMemory->zzCurrentChunk(),
- Framework::Assembly::R9);
- codeBlock.addLoadAddress(this, Framework::Assembly::RCX);
- codeBlock.addPush(
- Framework::Assembly::RDX, Framework::Assembly::LOWER32);
- codeBlock.addPush(
- Framework::Assembly::R8, Framework::Assembly::LOWER32);
- codeBlock.addMemberCall<bool (JNaighborBlockBoolExpression::*)(
- int, int, Chunk*)>(
- &JNaighborBlockBoolExpression::isValidPosition,
- Framework::Assembly::INT_VALUE,
- {Framework::Assembly::RCX,
- Framework::Assembly::RDX,
- Framework::Assembly::R8,
- Framework::Assembly::R9},
- {});
- codeBlock.addPop(
- Framework::Assembly::R8, Framework::Assembly::LOWER32);
- codeBlock.addPop(
- Framework::Assembly::RDX, Framework::Assembly::LOWER32);
- // abort if position is not valid
- codeBlock.addTest(Framework::Assembly::RAX,
- Framework::Assembly::RAX,
- Framework::Assembly::LOWER8);
- Framework::Text jumpLabel = "skip_dir_";
- jumpLabel.append(i);
- codeBlock.addJump(Framework::Assembly::JZ, jumpLabel);
- // restore parameters for block access
- codeBlock.addMoveValue(
- Framework::Assembly::R9, Framework::Assembly::R14);
- addEvaluation(codeBlock, zMemory);
- codeBlock.defineJumpTarget(jumpLabel);
- }
- }
- // set return value to false if no direction matched
- codeBlock.addMoveValue(Framework::Assembly::RAX, (char)0);
- codeBlock.addJump(Framework::Assembly::JMP, "end");
- codeBlock.defineJumpTarget("end_true");
- // set return value to true if any direction matched
- codeBlock.addMoveValue(Framework::Assembly::RAX, (char)1);
- codeBlock.defineJumpTarget("end");
- }
- return codeBlock;
- }
- void JNaighborBlockBoolExpression::setFilter(BlockFilter* filter)
- {
- this->filter = filter;
- }
- BlockFilter* JNaighborBlockBoolExpression::zFilter() const
- {
- return filter;
- }
- void JNaighborBlockBoolExpression::setValidDirections(
- Directions validDirections)
- {
- this->validDirections = validDirections;
- }
- Directions JNaighborBlockBoolExpression::getValidDirections() const
- {
- return validDirections;
- }
- JNaighborBlockBoolExpressionFactory::JNaighborBlockBoolExpressionFactory()
- : SubTypeFactory()
- {}
- JNaighborBlockBoolExpression* JNaighborBlockBoolExpressionFactory::fromJson(
- Framework::JSON::JSONObject* zJson) const
- {
- JNaighborBlockBoolExpression* result = new JNaighborBlockBoolExpression();
- result->setFilter(Game::INSTANCE->zTypeRegistry()->fromJson<BlockFilter>(
- zJson->zValue("condition")));
- for (Framework::JSON::JSONValue* direction :
- *zJson->zValue("directions")->asArray())
- {
- Framework::Text dirStr = direction->asString()->getString();
- if (dirStr.isEqual("TOP"))
- {
- result->setValidDirections(
- result->getValidDirections() | Direction::TOP);
- }
- else if (dirStr.isEqual("BOTTOM"))
- {
- result->setValidDirections(
- result->getValidDirections() | Direction::BOTTOM);
- }
- else if (dirStr.isEqual("EAST"))
- {
- result->setValidDirections(
- result->getValidDirections() | Direction::EAST);
- }
- else if (dirStr.isEqual("NORTH"))
- {
- result->setValidDirections(
- result->getValidDirections() | Direction::NORTH);
- }
- else if (dirStr.isEqual("WEST"))
- {
- result->setValidDirections(
- result->getValidDirections() | Direction::WEST);
- }
- else if (dirStr.isEqual("SOUTH"))
- {
- result->setValidDirections(
- result->getValidDirections() | Direction::SOUTH);
- }
- }
- return result;
- }
- Framework::JSON::JSONObject* JNaighborBlockBoolExpressionFactory::toJsonObject(
- JNaighborBlockBoolExpression* zObject) const
- {
- Framework::JSON::JSONObject* result = new Framework::JSON::JSONObject();
- result->addValue("condition",
- Game::INSTANCE->zTypeRegistry()->toJson(zObject->zFilter()));
- Framework::JSON::JSONArray* directionsArray
- = new Framework::JSON::JSONArray();
- if (zObject->getValidDirections() & Direction::TOP)
- {
- directionsArray->addValue(new Framework::JSON::JSONString("TOP"));
- }
- if (zObject->getValidDirections() & Direction::BOTTOM)
- {
- directionsArray->addValue(new Framework::JSON::JSONString("BOTTOM"));
- }
- if (zObject->getValidDirections() & Direction::EAST)
- {
- directionsArray->addValue(new Framework::JSON::JSONString("EAST"));
- }
- if (zObject->getValidDirections() & Direction::NORTH)
- {
- directionsArray->addValue(new Framework::JSON::JSONString("NORTH"));
- }
- if (zObject->getValidDirections() & Direction::WEST)
- {
- directionsArray->addValue(new Framework::JSON::JSONString("WEST"));
- }
- if (zObject->getValidDirections() & Direction::SOUTH)
- {
- directionsArray->addValue(new Framework::JSON::JSONString("SOUTH"));
- }
- result->addValue("directions", directionsArray);
- return result;
- }
- JSONObjectValidationBuilder*
- JNaighborBlockBoolExpressionFactory::addToValidator(
- JSONObjectValidationBuilder* builder) const
- {
- Framework::JSON::JSONArray* defaultDirectionsArray
- = new Framework::JSON::JSONArray();
- defaultDirectionsArray->addValue(new Framework::JSON::JSONString("EAST"));
- defaultDirectionsArray->addValue(new Framework::JSON::JSONString("NORTH"));
- defaultDirectionsArray->addValue(new Framework::JSON::JSONString("WEST"));
- defaultDirectionsArray->addValue(new Framework::JSON::JSONString("SOUTH"));
- return builder
- ->withRequiredAttribute("condition",
- Game::INSTANCE->zTypeRegistry()->getValidator<BlockFilter>())
- ->withRequiredArray("directions")
- ->addAcceptedStringInArray()
- ->whichIsOneOf({"TOP", "BOTTOM", "EAST", "NORTH", "WEST", "SOUTH"})
- ->finishString()
- ->withDefault(defaultDirectionsArray)
- ->finishArray();
- }
- const char* JNaighborBlockBoolExpressionFactory::getTypeToken() const
- {
- return "anyNeighborBlockMatches";
- }
|