2022-12-06 16:12:02 +00:00
|
|
|
#include "script_function.hpp"
|
2023-03-01 21:27:15 +00:00
|
|
|
|
2022-12-06 16:12:02 +00:00
|
|
|
#include "gta_util.hpp"
|
2023-03-01 21:27:15 +00:00
|
|
|
#include "pointers.hpp"
|
2022-12-06 16:12:02 +00:00
|
|
|
#include "util/scripts.hpp"
|
2023-03-01 21:27:15 +00:00
|
|
|
|
2022-12-06 16:12:02 +00:00
|
|
|
#include <script/scrProgram.hpp>
|
|
|
|
#include <script/scrProgramTable.hpp>
|
|
|
|
|
|
|
|
namespace big
|
|
|
|
{
|
|
|
|
script_function::script_function(const std::string& name, const rage::joaat_t script, const std::string& pattern, int32_t offset) :
|
2023-03-01 21:27:15 +00:00
|
|
|
m_name(name),
|
|
|
|
m_script(script),
|
|
|
|
m_pattern(pattern),
|
|
|
|
m_offset(offset),
|
|
|
|
m_ip(0)
|
2022-12-06 16:12:02 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void script_function::populate_ip()
|
|
|
|
{
|
|
|
|
if (m_ip == 0)
|
|
|
|
{
|
|
|
|
auto program = gta_util::find_script_program(m_script);
|
|
|
|
|
|
|
|
if (!program)
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto location = scripts::get_code_location_by_pattern(program, m_pattern);
|
2023-03-01 21:27:15 +00:00
|
|
|
|
2022-12-06 16:12:02 +00:00
|
|
|
if (!location)
|
|
|
|
LOG(FATAL) << "Failed to find pattern " << m_name << " in script " << program->m_name;
|
|
|
|
else
|
2023-02-08 23:36:55 +01:00
|
|
|
LOG(VERBOSE) << "Found pattern " << m_name << " in script " << program->m_name;
|
2022-12-06 16:12:02 +00:00
|
|
|
|
|
|
|
m_ip = location.value() + m_offset;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-20 22:46:32 +02:00
|
|
|
void script_function::call(rage::scrThread* thread, rage::scrProgram* program, std::initializer_list<uint64_t> args)
|
2022-12-06 16:12:02 +00:00
|
|
|
{
|
2023-03-01 21:27:15 +00:00
|
|
|
auto tls_ctx = rage::tlsContext::get();
|
|
|
|
auto stack = (uint64_t*)thread->m_stack;
|
2022-12-06 16:12:02 +00:00
|
|
|
auto og_thread = tls_ctx->m_script_thread;
|
|
|
|
|
2023-03-01 21:27:15 +00:00
|
|
|
tls_ctx->m_script_thread = thread;
|
2022-12-06 16:12:02 +00:00
|
|
|
tls_ctx->m_is_script_thread_active = true;
|
|
|
|
|
|
|
|
rage::scrThreadContext ctx = thread->m_context;
|
|
|
|
|
|
|
|
for (auto& arg : args)
|
|
|
|
stack[ctx.m_stack_pointer++] = arg;
|
|
|
|
|
|
|
|
stack[ctx.m_stack_pointer++] = 0;
|
2023-03-01 21:27:15 +00:00
|
|
|
ctx.m_instruction_pointer = m_ip;
|
|
|
|
ctx.m_state = rage::eThreadState::idle;
|
2022-12-06 16:12:02 +00:00
|
|
|
|
2023-04-14 18:54:07 +02:00
|
|
|
g_pointers->m_gta.m_script_vm(stack, g_pointers->m_gta.m_script_globals, program, &ctx);
|
2022-12-06 16:12:02 +00:00
|
|
|
|
2023-03-01 21:27:15 +00:00
|
|
|
tls_ctx->m_script_thread = og_thread;
|
2022-12-06 16:12:02 +00:00
|
|
|
tls_ctx->m_is_script_thread_active = og_thread != nullptr;
|
|
|
|
}
|
|
|
|
|
2023-07-20 22:46:32 +02:00
|
|
|
void script_function::call_latent(rage::scrThread* thread, rage::scrProgram* program, std::initializer_list<uint64_t> args, bool& done)
|
2023-03-09 12:23:01 +00:00
|
|
|
{
|
|
|
|
g_fiber_pool->queue_job([this, thread, program, args, &done] {
|
|
|
|
auto stack = (uint64_t*)thread->m_stack;
|
|
|
|
|
|
|
|
rage::eThreadState result = rage::eThreadState::idle;
|
|
|
|
|
|
|
|
rage::scrThreadContext ctx = thread->m_context;
|
|
|
|
|
|
|
|
for (auto& arg : args)
|
|
|
|
stack[ctx.m_stack_pointer++] = arg;
|
|
|
|
|
|
|
|
stack[ctx.m_stack_pointer++] = 0;
|
|
|
|
ctx.m_instruction_pointer = m_ip;
|
|
|
|
ctx.m_state = rage::eThreadState::idle;
|
|
|
|
|
|
|
|
while (result != rage::eThreadState::killed)
|
|
|
|
{
|
|
|
|
auto tls_ctx = rage::tlsContext::get();
|
|
|
|
auto og_thread = tls_ctx->m_script_thread;
|
|
|
|
|
|
|
|
tls_ctx->m_script_thread = thread;
|
|
|
|
tls_ctx->m_is_script_thread_active = true;
|
|
|
|
|
|
|
|
auto old_ctx = thread->m_context;
|
|
|
|
thread->m_context = ctx;
|
2023-04-14 18:54:07 +02:00
|
|
|
result = g_pointers->m_gta.m_script_vm(stack, g_pointers->m_gta.m_script_globals, program, &thread->m_context);
|
2023-03-09 12:23:01 +00:00
|
|
|
thread->m_context = old_ctx;
|
|
|
|
|
|
|
|
tls_ctx->m_script_thread = og_thread;
|
|
|
|
tls_ctx->m_is_script_thread_active = og_thread != nullptr;
|
|
|
|
|
|
|
|
script::get_current()->yield();
|
|
|
|
}
|
|
|
|
|
|
|
|
done = true;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-07-20 22:46:32 +02:00
|
|
|
void script_function::static_call(std::initializer_list<uint64_t> args)
|
2022-12-17 14:47:01 +00:00
|
|
|
{
|
|
|
|
populate_ip();
|
|
|
|
|
|
|
|
rage::scrThread* thread = (rage::scrThread*)new uint8_t[sizeof(rage::scrThread)];
|
2023-07-16 12:24:40 -04:00
|
|
|
memcpy(thread, rage::tlsContext::get()->m_script_thread, sizeof(rage::scrThread));
|
2022-12-17 14:47:01 +00:00
|
|
|
|
2023-03-01 21:27:15 +00:00
|
|
|
void* stack = new uint64_t[25000];
|
2023-07-16 12:24:40 -04:00
|
|
|
thread->m_stack = (rage::scrValue*)stack;
|
2023-03-01 21:27:15 +00:00
|
|
|
thread->m_context.m_stack_size = 25000;
|
2022-12-17 14:47:01 +00:00
|
|
|
thread->m_context.m_stack_pointer = 1;
|
|
|
|
|
|
|
|
call(thread, gta_util::find_script_program(m_script), args);
|
|
|
|
|
|
|
|
delete[] stack;
|
2023-07-02 00:52:36 +02:00
|
|
|
delete[] (uint8_t*)thread; // without the cast it ends up calling the destructor which leads to some pretty funny crashes
|
2022-12-17 14:47:01 +00:00
|
|
|
}
|
|
|
|
|
2023-07-20 22:46:32 +02:00
|
|
|
void script_function::operator()(std::initializer_list<uint64_t> args)
|
2022-12-06 16:12:02 +00:00
|
|
|
{
|
|
|
|
populate_ip();
|
|
|
|
|
|
|
|
if (m_ip == 0)
|
|
|
|
return;
|
|
|
|
|
2023-03-01 21:27:15 +00:00
|
|
|
auto thread = gta_util::find_script_thread(m_script);
|
2022-12-06 16:12:02 +00:00
|
|
|
auto program = gta_util::find_script_program(m_script);
|
|
|
|
|
|
|
|
if (thread && program)
|
|
|
|
{
|
|
|
|
call(thread, program, args);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|