-
Notifications
You must be signed in to change notification settings - Fork 244
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add basic support for struct DSTs #504
Changes from 4 commits
1947777
4c06864
7e9728d
57abe13
401fcc3
22cc775
367ac4d
7b9f73f
2704ef1
e8df0ac
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,5 +1,4 @@ | ||
use super::CodegenCx; | ||
use crate::abi::ConvSpirvType; | ||
use crate::builder_spirv::SpirvValue; | ||
use crate::spirv_type::SpirvType; | ||
use crate::symbols::{parse_attrs, Entry, SpirvAttribute}; | ||
|
@@ -9,7 +8,10 @@ use rustc_hir as hir; | |
use rustc_middle::ty::layout::TyAndLayout; | ||
use rustc_middle::ty::{Instance, Ty}; | ||
use rustc_span::Span; | ||
use rustc_target::abi::call::{FnAbi, PassMode}; | ||
use rustc_target::abi::{ | ||
call::{ArgAbi, ArgAttribute, ArgAttributes, FnAbi, PassMode}, | ||
Size, | ||
}; | ||
use std::collections::HashMap; | ||
|
||
impl<'tcx> CodegenCx<'tcx> { | ||
|
@@ -36,8 +38,27 @@ impl<'tcx> CodegenCx<'tcx> { | |
}; | ||
let fn_hir_id = self.tcx.hir().local_def_id_to_hir_id(local_id); | ||
let body = self.tcx.hir().body(self.tcx.hir().body_owned_by(fn_hir_id)); | ||
const EMPTY: ArgAttribute = ArgAttribute::empty(); | ||
for (abi, arg) in fn_abi.args.iter().zip(body.params) { | ||
if let PassMode::Direct(_) = abi.mode { | ||
} else if let PassMode::Pair( | ||
// plain DST/RTA/VLA | ||
ArgAttributes { | ||
pointee_size: Size::ZERO, | ||
.. | ||
}, | ||
ArgAttributes { regular: EMPTY, .. }, | ||
) = abi.mode | ||
{ | ||
} else if let PassMode::Pair( | ||
// DST struct with fields before the DST member | ||
ArgAttributes { .. }, | ||
ArgAttributes { | ||
pointee_size: Size::ZERO, | ||
.. | ||
}, | ||
) = abi.mode | ||
{ | ||
} else { | ||
self.tcx.sess.span_err( | ||
arg.span, | ||
|
@@ -62,7 +83,7 @@ impl<'tcx> CodegenCx<'tcx> { | |
self.shader_entry_stub( | ||
self.tcx.def_span(instance.def_id()), | ||
entry_func, | ||
fn_abi, | ||
&fn_abi.args, | ||
body.params, | ||
name, | ||
execution_model, | ||
|
@@ -81,7 +102,7 @@ impl<'tcx> CodegenCx<'tcx> { | |
&self, | ||
span: Span, | ||
entry_func: SpirvValue, | ||
entry_fn_abi: &FnAbi<'tcx, Ty<'tcx>>, | ||
arg_abis: &[ArgAbi<'tcx, Ty<'tcx>>], | ||
Comment on lines
-85
to
+106
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'm not sure this is a good idea, especially if we want to do something with the return type at some point. |
||
hir_params: &[hir::Param<'tcx>], | ||
name: String, | ||
execution_model: ExecutionModel, | ||
|
@@ -92,52 +113,85 @@ impl<'tcx> CodegenCx<'tcx> { | |
arguments: vec![], | ||
} | ||
.def(span, self); | ||
let entry_func_return_type = match self.lookup_type(entry_func.ty) { | ||
let (entry_func_return_type, entry_func_arg_types) = match self.lookup_type(entry_func.ty) { | ||
SpirvType::Function { | ||
return_type, | ||
arguments: _, | ||
} => return_type, | ||
arguments, | ||
} => (return_type, arguments), | ||
other => self.tcx.sess.fatal(&format!( | ||
"Invalid entry_stub type: {}", | ||
other.debug(entry_func.ty, self) | ||
)), | ||
}; | ||
let mut decoration_locations = HashMap::new(); | ||
// Create OpVariables before OpFunction so they're global instead of local vars. | ||
let arguments = entry_fn_abi | ||
.args | ||
.iter() | ||
.zip(hir_params) | ||
.map(|(entry_fn_arg, hir_param)| { | ||
self.declare_parameter(entry_fn_arg.layout, hir_param, &mut decoration_locations) | ||
}) | ||
.collect::<Vec<_>>(); | ||
let new_spirv = self.emit_global().version().unwrap() > (1, 3); | ||
let arg_len = arg_abis.len(); | ||
let mut arguments = Vec::with_capacity(arg_len); | ||
let mut interface = Vec::with_capacity(arg_len); | ||
let mut rta_lens = Vec::with_capacity(arg_len / 2); | ||
let mut arg_types = entry_func_arg_types.iter(); | ||
for (hir_param, arg_abi) in hir_params.iter().zip(arg_abis) { | ||
// explicit next because there are two args for scalar pairs, but only one param & abi | ||
let arg_t = *arg_types.next().unwrap_or_else(|| { | ||
self.tcx.sess.span_fatal( | ||
hir_param.span, | ||
&format!( | ||
"Invalid function arguments: Param {:?} Abi {:?} missing type", | ||
hir_param, arg_abi.layout.ty | ||
), | ||
) | ||
}); | ||
let (argument, storage_class) = | ||
self.declare_parameter(arg_abi.layout, hir_param, arg_t, &mut decoration_locations); | ||
// SPIR-V <= v1.3 only includes Input and Output in the interface. | ||
if new_spirv | ||
|| storage_class == StorageClass::Input | ||
|| storage_class == StorageClass::Output | ||
{ | ||
interface.push(argument); | ||
} | ||
arguments.push(argument); | ||
if let SpirvType::Pointer { pointee } = self.lookup_type(arg_t) { | ||
if let SpirvType::Adt { | ||
size: None, | ||
field_types, | ||
.. | ||
} = self.lookup_type(pointee) | ||
{ | ||
let len_t = *arg_types.next().unwrap_or_else(|| { | ||
self.tcx.sess.span_fatal( | ||
hir_param.span, | ||
&format!( | ||
"Invalid function arguments: Param {:?} Abi {:?} fat pointer missing length", | ||
hir_param, arg_abi.layout.ty | ||
), | ||
) | ||
}); | ||
rta_lens.push((arguments.len() as u32, len_t, field_types.len() as u32 - 1)); | ||
arguments.push(u32::MAX); | ||
} | ||
} | ||
} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This seems pretty ad-hoc and confusing to follow (splitting it up into functions would probably help), and has some issues with it. For example, this compiletest passes, but it shouldn't (no way to get the length of the slice):
|
||
let mut emit = self.emit_global(); | ||
let fn_id = emit | ||
.begin_function(void, None, FunctionControl::NONE, fn_void_void) | ||
.unwrap(); | ||
emit.begin_block(None).unwrap(); | ||
rta_lens.iter().for_each(|&(len_idx, len_t, member_idx)| { | ||
arguments[len_idx as usize] = emit | ||
.array_length(len_t, None, arguments[len_idx as usize - 1], member_idx) | ||
.unwrap() | ||
}); | ||
emit.function_call( | ||
entry_func_return_type, | ||
None, | ||
entry_func.def_cx(self), | ||
arguments.iter().map(|&(a, _)| a), | ||
arguments, | ||
) | ||
.unwrap(); | ||
emit.ret().unwrap(); | ||
emit.end_function().unwrap(); | ||
|
||
let interface: Vec<_> = if emit.version().unwrap() > (1, 3) { | ||
// SPIR-V >= v1.4 includes all OpVariables in the interface. | ||
arguments.into_iter().map(|(a, _)| a).collect() | ||
} else { | ||
// SPIR-V <= v1.3 only includes Input and Output in the interface. | ||
arguments | ||
.into_iter() | ||
.filter(|&(_, s)| s == StorageClass::Input || s == StorageClass::Output) | ||
.map(|(a, _)| a) | ||
.collect() | ||
}; | ||
emit.entry_point(execution_model, fn_id, name, interface); | ||
fn_id | ||
} | ||
|
@@ -146,6 +200,7 @@ impl<'tcx> CodegenCx<'tcx> { | |
&self, | ||
layout: TyAndLayout<'tcx>, | ||
hir_param: &hir::Param<'tcx>, | ||
arg_t: Word, | ||
decoration_locations: &mut HashMap<StorageClass, u32>, | ||
) -> (Word, StorageClass) { | ||
let storage_class = crate::abi::get_storage_class(self, layout).unwrap_or_else(|| { | ||
|
@@ -159,10 +214,9 @@ impl<'tcx> CodegenCx<'tcx> { | |
StorageClass::Input | StorageClass::Output | StorageClass::UniformConstant | ||
); | ||
// Note: this *declares* the variable too. | ||
let spirv_type = layout.spirv_type(hir_param.span, self); | ||
let variable = self | ||
.emit_global() | ||
.variable(spirv_type, None, storage_class, None); | ||
.variable(arg_t, None, storage_class, None); | ||
if let hir::PatKind::Binding(_, _, ident, _) = &hir_param.pat.kind { | ||
self.emit_global().name(variable, ident.to_string()); | ||
} | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -159,6 +159,33 @@ fn dis_fn(src: &str, func: &str, expect: &str) { | |
assert_str_eq(expect, &func.disassemble()) | ||
} | ||
|
||
fn dis_entry_fn(src: &str, func: &str, expect: &str) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I was confused by this so maybe |
||
let _lock = global_lock(); | ||
let module = read_module(&build(src)).unwrap(); | ||
let id = module | ||
.entry_points | ||
.iter() | ||
.find(|inst| inst.operands.last().unwrap().unwrap_literal_string() == func) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Isn't it always |
||
.unwrap_or_else(|| { | ||
panic!( | ||
"no entry point with the name `{}` found in:\n{}\n", | ||
func, | ||
module.disassemble() | ||
) | ||
}) | ||
.operands[1] | ||
.unwrap_id_ref(); | ||
let mut func = module | ||
.functions | ||
.into_iter() | ||
.find(|f| f.def_id().unwrap() == id) | ||
.unwrap(); | ||
// Compact to make IDs more stable | ||
compact_ids(&mut func); | ||
use rspirv::binary::Disassemble; | ||
assert_str_eq(expect, &func.disassemble()) | ||
Comment on lines
+178
to
+186
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This part could maybe be shared between the two functions (so only the picking the function ID would be different). |
||
} | ||
|
||
fn dis_globals(src: &str, expect: &str) { | ||
let _lock = global_lock(); | ||
let module = read_module(&build(src)).unwrap(); | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: this should probably be turned into a
match
, the reason it's anif let
is because there's only one case