use crate::enums::ir_block_kind::IrBlockKind;
use crate::enums::ir_cmd::IrCmd;
use crate::records::ir_builder::IrBuilder;
use crate::records::ir_op::IrOp;
use crate::type_aliases::instruction_ir_builder::Instruction;
use luaur_common::macros::luau_insn_a::LUAU_INSN_A;
use luaur_vm::enums::lua_type::lua_Type;
use crate::functions::get_jump_target::get_jump_target;
use crate::functions::get_op_length::get_op_length;
use crate::macros::codegen_assert::CODEGEN_ASSERT;
use luaur_common::enums::luau_opcode::LuauOpcode;
use luaur_common::macros::luau_insn_op::LUAU_INSN_OP;
pub fn translate_inst_for_g_loop_ipairs(build: &mut IrBuilder, pc: *const Instruction, pcpos: i32) {
let ra = LUAU_INSN_A(unsafe { *pc }) as u8;
CODEGEN_ASSERT!((unsafe { *pc.add(1) } as i32) < 0);
let loop_repeat = build.block_at_inst(get_jump_target(unsafe { *pc }, pcpos as u32) as u32);
let op = LuauOpcode::from(LUAU_INSN_OP(unsafe { *pc }) as u8);
let loop_exit = build.block_at_inst((pcpos + get_op_length(op)) as u32);
let fallback = build.fallback_block(pcpos as u32);
let has_elem = build.block(IrBlockKind::Internal);
let pcpos_op = build.const_uint(pcpos as u32);
build.inst_ir_cmd_ir_op(IrCmd::INTERRUPT, pcpos_op);
let reg_ra = build.vm_reg(ra);
let tag_a = build.inst_ir_cmd_ir_op(IrCmd::LOAD_TAG, reg_ra);
let nil_tag = build.const_tag(lua_Type::LUA_TNIL as u8);
build.inst_ir_cmd_ir_op_ir_op_ir_op(IrCmd::CHECK_TAG, tag_a, nil_tag, fallback);
let reg_table = build.vm_reg(ra + 1);
let table = build.inst_ir_cmd_ir_op(IrCmd::LOAD_POINTER, reg_table);
let reg_index = build.vm_reg(ra + 2);
let index = build.inst_ir_cmd_ir_op(IrCmd::LOAD_INT, reg_index);
let elem_ptr = build.inst_ir_cmd_ir_op_ir_op(IrCmd::GET_ARR_ADDR, table, index);
build.inst_ir_cmd_ir_op_ir_op_ir_op(IrCmd::CHECK_ARRAY_SIZE, table, index, loop_exit);
let elem_tag = build.inst_ir_cmd_ir_op(IrCmd::LOAD_TAG, elem_ptr);
let nil_tag = build.const_tag(lua_Type::LUA_TNIL as u8);
build.inst_ir_cmd_ir_op_ir_op_ir_op_ir_op(
IrCmd::JUMP_EQ_TAG,
elem_tag,
nil_tag,
loop_exit,
has_elem,
);
build.begin_block(has_elem);
let one = build.const_int(1);
let next_index = build.inst_ir_cmd_ir_op_ir_op(IrCmd::ADD_INT, index, one);
let reg_iter = build.vm_reg(ra + 2);
build.inst_ir_cmd_ir_op_ir_op(IrCmd::STORE_INT, reg_iter, next_index);
let next_index_num = build.inst_ir_cmd_ir_op(IrCmd::INT_TO_NUM, next_index);
let reg_value = build.vm_reg(ra + 3);
build.inst_ir_cmd_ir_op_ir_op(IrCmd::STORE_DOUBLE, reg_value, next_index_num);
let reg_value = build.vm_reg(ra + 3);
let number_tag = build.const_tag(lua_Type::LUA_TNUMBER as u8);
build.inst_ir_cmd_ir_op_ir_op(IrCmd::STORE_TAG, reg_value, number_tag);
let elem_tv = build.inst_ir_cmd_ir_op(IrCmd::LOAD_TVALUE, elem_ptr);
let reg_elem = build.vm_reg(ra + 4);
build.inst_ir_cmd_ir_op_ir_op(IrCmd::STORE_TVALUE, reg_elem, elem_tv);
build.inst_ir_cmd_ir_op(IrCmd::JUMP, loop_repeat);
build.begin_block(fallback);
let savedpc = build.const_uint((pcpos + 1) as u32);
build.inst_ir_cmd_ir_op(IrCmd::SET_SAVEDPC, savedpc);
let reg_ra = build.vm_reg(ra);
let aux = build.const_int(unsafe { *pc.add(1) } as i32);
build.inst_ir_cmd_ir_op_ir_op_ir_op_ir_op(
IrCmd::FORGLOOP_FALLBACK,
reg_ra,
aux,
loop_repeat,
loop_exit,
);
if build.is_internal_block(loop_exit) {
build.begin_block(loop_exit);
}
}