diff --git a/src/game_interpreter.cpp b/src/game_interpreter.cpp index 1d18d6b843..053ae21758 100644 --- a/src/game_interpreter.cpp +++ b/src/game_interpreter.cpp @@ -5246,23 +5246,75 @@ bool Game_Interpreter::CommandManiacCallCommand(lcf::rpg::EventCommand const& co return true; } + enum class ProcessingMode { + Constant = 0, // 0 and 1: Parameters read from variables + Variable = 1, + //VariableIndirect = 2, // Somehow not implemented by ManiacPatch + Inline = 3, // Parameters are directly provided by the command + Expression = 4 // Like 3, but the parameters are calculated from expressions + }; + + std::vector values; + + // Create command with basic parameters lcf::rpg::EventCommand cmd; cmd.code = ValueOrVariableBitfield(com.parameters[0], 0, com.parameters[1]); cmd.string = lcf::DBString(CommandStringOrVariableBitfield(com, 0, 3, 4)); - int arr_begin = ValueOrVariableBitfield(com.parameters[0], 1, com.parameters[2]); - int arr_length = ValueOrVariableBitfield(com.parameters[0], 2, com.parameters[3]); + // Determine processing mode + auto processing_mode = static_cast((com.parameters[0] >> 4) & 0b1111); - std::vector output_args; - if (arr_length > 0) { - output_args.reserve(arr_length); - for (int i = 0; i < arr_length; ++i) { - output_args.push_back(Main_Data::game_variables->Get(arr_begin + i)); + switch (processing_mode) { + case ProcessingMode::Constant: + case ProcessingMode::Variable: { + int start_index = ValueOrVariable(static_cast(processing_mode), com.parameters[2]); + int length = ValueOrVariableBitfield(com.parameters[0], 2, com.parameters[3]); + + for (int i = 0; i < length; ++i) { + values.push_back(Main_Data::game_variables->Get(start_index + i)); } + break; } + case ProcessingMode::Inline: { + int value_index = 5; // Start of the values + int mode_index = com.parameters[2]; // Mode of the values + int length = com.parameters[3]; + + for (int i = 0; i < length; ++i) { + // The mode is the typical 4 bit packing + // Always 4 modes (16 bit) are packing into one parameter + // Then the mode_index increments + if (i != 0 && i % 4 == 0) { + ++mode_index; + } - cmd.parameters = lcf::DBArray(output_args.begin(), output_args.end()); + values.push_back(ValueOrVariableBitfield(com, mode_index, i % 4, value_index + i)); + } + break; + } + case ProcessingMode::Expression: { + values = ManiacPatch::ParseExpressions(MakeSpan(com.parameters).subspan(5), *this); + break; + } + default: + Output::Warning("Call Command: Unsupported Processing Mode: {}", static_cast(processing_mode)); + return true; + } + + // Finalize command parameters + cmd.parameters = lcf::DBArray(values.begin(), values.end()); + + // Debug output + /*Output::Warning("Processing mode: {}", static_cast(processing_mode)); + Output::Warning("Command code: {}", cmd.code); + Output::Warning("Command string: {}", cmd.string); + std::string params_str; + for (const auto& param : values) { + params_str += " " + std::to_string(param); + } + Output::Warning("Command parameters:{}", params_str); + Output::Info("--------------------\n");*/ // Our implementation pushes a new frame containing the command instead of invoking it directly. // This is incompatible to Maniacs but has a better compatibility with our code. diff --git a/src/maniac_patch.cpp b/src/maniac_patch.cpp index 0ed805adf9..0926849e3c 100644 --- a/src/maniac_patch.cpp +++ b/src/maniac_patch.cpp @@ -424,6 +424,7 @@ int Process(std::vector::iterator& it, std::vector::iterator e Output::Warning("Maniac: Expression actor args {} != 2", imm2); return 0; } + imm3 = Process(it, end, ip); return ControlVariables::Actor(Process(it, end, ip), imm3); case Fn::Party: if (imm2 != 2) { @@ -570,6 +571,35 @@ int32_t ManiacPatch::ParseExpression(Span op_codes, const Game_Ba return Process(beg, ops.end(), interpreter); } +std::vector ManiacPatch::ParseExpressions(Span op_codes, const Game_BaseInterpreterContext& interpreter) { + std::vector ops; + for (auto& o : op_codes) { + auto uo = static_cast(o); + ops.push_back(static_cast(uo & 0x000000FF)); + ops.push_back(static_cast((uo & 0x0000FF00) >> 8)); + ops.push_back(static_cast((uo & 0x00FF0000) >> 16)); + ops.push_back(static_cast((uo & 0xFF000000) >> 24)); + } + + if (ops.empty()) { + return {}; + } + + auto it = ops.begin(); + + std::vector results; + + while (true) { + results.push_back(Process(it, ops.end(), interpreter)); + + if (it == ops.end() || static_cast(*it) == Op::Null) { + break; + } + } + + return results; +} + std::array ManiacPatch::GetKeyRange() { std::array keys = { Input::Keys::A, diff --git a/src/maniac_patch.h b/src/maniac_patch.h index 789834fcf9..2b5c2c4edd 100644 --- a/src/maniac_patch.h +++ b/src/maniac_patch.h @@ -21,6 +21,7 @@ #include #include +#include #include "span.h" #include "game_strings.h" @@ -29,6 +30,8 @@ class Game_BaseInterpreterContext; namespace ManiacPatch { int32_t ParseExpression(Span op_codes, const Game_BaseInterpreterContext& interpreter); + std::vector ParseExpressions(Span op_codes, const Game_BaseInterpreterContext& interpreter); + std::array GetKeyRange();