use crate::enums::builtin_impl_type::BuiltinImplType;
use crate::enums::ir_block_kind::IrBlockKind;
use crate::enums::ir_cmd::IrCmd;
use crate::enums::ir_condition::IrCondition;
use crate::enums::ir_op_kind::IrOpKind;
use crate::records::builtin_impl_result::BuiltinImplResult;
use crate::records::ir_builder::IrBuilder;
use crate::records::ir_op::IrOp;
pub fn translate_builtin_vector_clamp(
build: &mut IrBuilder,
nparams: i32,
ra: i32,
arg: i32,
args: IrOp,
arg3: IrOp,
nresults: i32,
fallback: IrOp,
pcpos: i32,
) -> BuiltinImplResult {
let arg1 = build.vm_reg(arg as u8);
if nparams != 3
|| nresults > 1
|| arg1.kind() == IrOpKind::Constant
|| args.kind() == IrOpKind::Constant
|| arg3.kind() == IrOpKind::Constant
{
return BuiltinImplResult {
r#type: BuiltinImplType::None,
actual_result_count: -1,
};
}
let lua_tvector = luaur_vm::enums::lua_type::lua_Type::LUA_TVECTOR as u8;
let exit = build.vm_exit(pcpos as u32);
build.load_and_check_tag(arg1, lua_tvector, exit);
let exit = build.vm_exit(pcpos as u32);
build.load_and_check_tag(args, lua_tvector, exit);
let exit = build.vm_exit(pcpos as u32);
build.load_and_check_tag(arg3, lua_tvector, exit);
let block1 = build.block(IrBlockKind::Internal);
let block2 = build.block(IrBlockKind::Internal);
let block3 = build.block(IrBlockKind::Internal);
let zero = build.const_int(0);
let x = build.inst_ir_cmd_ir_op_ir_op(IrCmd::LOAD_FLOAT, arg1, zero);
let zero = build.const_int(0);
let xmin = build.inst_ir_cmd_ir_op_ir_op(IrCmd::LOAD_FLOAT, args, zero);
let zero = build.const_int(0);
let xmax = build.inst_ir_cmd_ir_op_ir_op(IrCmd::LOAD_FLOAT, arg3, zero);
let cond = build.cond(IrCondition::NotLessEqual);
build.inst_ir_cmd_ir_op_ir_op_ir_op_ir_op_ir_op(
IrCmd::JUMP_CMP_FLOAT,
xmin,
xmax,
cond,
fallback,
block1,
);
build.begin_block(block1);
let four = build.const_int(4);
let y = build.inst_ir_cmd_ir_op_ir_op(IrCmd::LOAD_FLOAT, arg1, four);
let four = build.const_int(4);
let ymin = build.inst_ir_cmd_ir_op_ir_op(IrCmd::LOAD_FLOAT, args, four);
let four = build.const_int(4);
let ymax = build.inst_ir_cmd_ir_op_ir_op(IrCmd::LOAD_FLOAT, arg3, four);
let cond = build.cond(IrCondition::NotLessEqual);
build.inst_ir_cmd_ir_op_ir_op_ir_op_ir_op_ir_op(
IrCmd::JUMP_CMP_FLOAT,
ymin,
ymax,
cond,
fallback,
block2,
);
build.begin_block(block2);
let eight = build.const_int(8);
let z = build.inst_ir_cmd_ir_op_ir_op(IrCmd::LOAD_FLOAT, arg1, eight);
let eight = build.const_int(8);
let zmin = build.inst_ir_cmd_ir_op_ir_op(IrCmd::LOAD_FLOAT, args, eight);
let eight = build.const_int(8);
let zmax = build.inst_ir_cmd_ir_op_ir_op(IrCmd::LOAD_FLOAT, arg3, eight);
let cond = build.cond(IrCondition::NotLessEqual);
build.inst_ir_cmd_ir_op_ir_op_ir_op_ir_op_ir_op(
IrCmd::JUMP_CMP_FLOAT,
zmin,
zmax,
cond,
fallback,
block3,
);
build.begin_block(block3);
let xtemp = build.inst_ir_cmd_ir_op_ir_op(IrCmd::MAX_FLOAT, xmin, x);
let xclamped = build.inst_ir_cmd_ir_op_ir_op(IrCmd::MIN_FLOAT, xmax, xtemp);
let ytemp = build.inst_ir_cmd_ir_op_ir_op(IrCmd::MAX_FLOAT, ymin, y);
let yclamped = build.inst_ir_cmd_ir_op_ir_op(IrCmd::MIN_FLOAT, ymax, ytemp);
let ztemp = build.inst_ir_cmd_ir_op_ir_op(IrCmd::MAX_FLOAT, zmin, z);
let zclamped = build.inst_ir_cmd_ir_op_ir_op(IrCmd::MIN_FLOAT, zmax, ztemp);
let ra_reg = build.vm_reg(ra as u8);
build.inst_ir_cmd_ir_op_ir_op_ir_op_ir_op(
IrCmd::STORE_VECTOR,
ra_reg,
xclamped,
yclamped,
zclamped,
);
let tag = build.const_tag(lua_tvector);
build.inst_ir_cmd_ir_op_ir_op(IrCmd::STORE_TAG, ra_reg, tag);
BuiltinImplResult {
r#type: BuiltinImplType::UsesFallback,
actual_result_count: 1,
}
}