Fix type of intrinsics

This commit is contained in:
Antoni Boucher 2024-06-28 12:51:18 -04:00
parent 63d308bd76
commit 59de4fa991
2 changed files with 18 additions and 19 deletions

View File

@ -154,9 +154,9 @@ fn array_as_slice(arr: &[u8; 3]) -> &[u8] {
} }
// FIXME: fix the intrinsic implementation to work with the new ->u32 signature // FIXME: fix the intrinsic implementation to work with the new ->u32 signature
// unsafe fn use_ctlz_nonzero(a: u16) -> u32 { unsafe fn use_ctlz_nonzero(a: u16) -> u32 {
// intrinsics::ctlz_nonzero(a) intrinsics::ctlz_nonzero(a)
// } }
fn ptr_as_usize(ptr: *const u8) -> usize { fn ptr_as_usize(ptr: *const u8) -> usize {
ptr as usize ptr as usize

View File

@ -220,12 +220,12 @@ impl<'a, 'gcc, 'tcx> IntrinsicCallMethods<'tcx> for Builder<'a, 'gcc, 'tcx> {
let after_block = func.new_block("after"); let after_block = func.new_block("after");
let arg = args[0].immediate(); let arg = args[0].immediate();
let result = func.new_local(None, arg.get_type(), "zeros"); let result = func.new_local(None, self.u32_type, "zeros");
let zero = self.cx.gcc_zero(arg.get_type()); let zero = self.cx.gcc_zero(arg.get_type());
let cond = self.gcc_icmp(IntPredicate::IntEQ, arg, zero); let cond = self.gcc_icmp(IntPredicate::IntEQ, arg, zero);
self.llbb().end_with_conditional(None, cond, then_block, else_block); self.llbb().end_with_conditional(None, cond, then_block, else_block);
let zero_result = self.cx.gcc_uint(arg.get_type(), width); let zero_result = self.cx.gcc_uint(self.u32_type, width);
then_block.add_assignment(None, result, zero_result); then_block.add_assignment(None, result, zero_result);
then_block.end_with_jump(None, after_block); then_block.end_with_jump(None, after_block);
@ -709,6 +709,7 @@ impl<'a, 'gcc, 'tcx> Builder<'a, 'gcc, 'tcx> {
fn count_leading_zeroes(&mut self, width: u64, arg: RValue<'gcc>) -> RValue<'gcc> { fn count_leading_zeroes(&mut self, width: u64, arg: RValue<'gcc>) -> RValue<'gcc> {
// TODO(antoyo): use width? // TODO(antoyo): use width?
let arg_type = arg.get_type(); let arg_type = arg.get_type();
let result_type = self.u32_type;
let count_leading_zeroes = let count_leading_zeroes =
// TODO(antoyo): write a new function Type::is_compatible_with(&Type) and use it here // TODO(antoyo): write a new function Type::is_compatible_with(&Type) and use it here
// instead of using is_uint(). // instead of using is_uint().
@ -766,7 +767,7 @@ impl<'a, 'gcc, 'tcx> Builder<'a, 'gcc, 'tcx> {
let res = self.context.new_array_access(self.location, result, index); let res = self.context.new_array_access(self.location, result, index);
return self.gcc_int_cast(res.to_rvalue(), arg_type); return self.gcc_int_cast(res.to_rvalue(), result_type);
} }
else { else {
let count_leading_zeroes = self.context.get_builtin_function("__builtin_clzll"); let count_leading_zeroes = self.context.get_builtin_function("__builtin_clzll");
@ -774,22 +775,22 @@ impl<'a, 'gcc, 'tcx> Builder<'a, 'gcc, 'tcx> {
let diff = self.ulonglong_type.get_size() as i64 - arg_type.get_size() as i64; let diff = self.ulonglong_type.get_size() as i64 - arg_type.get_size() as i64;
let diff = self.context.new_rvalue_from_long(self.int_type, diff * 8); let diff = self.context.new_rvalue_from_long(self.int_type, diff * 8);
let res = self.context.new_call(self.location, count_leading_zeroes, &[arg]) - diff; let res = self.context.new_call(self.location, count_leading_zeroes, &[arg]) - diff;
return self.context.new_cast(self.location, res, arg_type); return self.context.new_cast(self.location, res, result_type);
}; };
let count_leading_zeroes = self.context.get_builtin_function(count_leading_zeroes); let count_leading_zeroes = self.context.get_builtin_function(count_leading_zeroes);
let res = self.context.new_call(self.location, count_leading_zeroes, &[arg]); let res = self.context.new_call(self.location, count_leading_zeroes, &[arg]);
self.context.new_cast(self.location, res, arg_type) self.context.new_cast(self.location, res, result_type)
} }
fn count_trailing_zeroes(&mut self, _width: u64, arg: RValue<'gcc>) -> RValue<'gcc> { fn count_trailing_zeroes(&mut self, _width: u64, arg: RValue<'gcc>) -> RValue<'gcc> {
let result_type = arg.get_type(); let arg_type = arg.get_type();
let arg = if result_type.is_signed(self.cx) { let result_type = self.u32_type;
let new_type = result_type.to_unsigned(self.cx); let arg = if arg_type.is_signed(self.cx) {
let new_type = arg_type.to_unsigned(self.cx);
self.gcc_int_cast(arg, new_type) self.gcc_int_cast(arg, new_type)
} else { } else {
arg arg
}; };
let arg_type = arg.get_type();
let (count_trailing_zeroes, expected_type) = let (count_trailing_zeroes, expected_type) =
// TODO(antoyo): write a new function Type::is_compatible_with(&Type) and use it here // TODO(antoyo): write a new function Type::is_compatible_with(&Type) and use it here
// instead of using is_uint(). // instead of using is_uint().
@ -874,14 +875,12 @@ impl<'a, 'gcc, 'tcx> Builder<'a, 'gcc, 'tcx> {
fn pop_count(&mut self, value: RValue<'gcc>) -> RValue<'gcc> { fn pop_count(&mut self, value: RValue<'gcc>) -> RValue<'gcc> {
// TODO(antoyo): use the optimized version with fewer operations. // TODO(antoyo): use the optimized version with fewer operations.
let result_type = value.get_type(); let result_type = self.u32_type;
let value_type = result_type.to_unsigned(self.cx); let arg_type = value.get_type();
let value_type = arg_type.to_unsigned(self.cx);
let value = if result_type.is_signed(self.cx) { let value =
self.gcc_int_cast(value, value_type) if arg_type.is_signed(self.cx) { self.gcc_int_cast(value, value_type) } else { value };
} else {
value
};
// only break apart 128-bit ints if they're not natively supported // only break apart 128-bit ints if they're not natively supported
// TODO(antoyo): remove this if/when native 128-bit integers land in libgccjit // TODO(antoyo): remove this if/when native 128-bit integers land in libgccjit