-
Notifications
You must be signed in to change notification settings - Fork 245
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 6 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 |
---|---|---|
|
@@ -5,13 +5,16 @@ use crate::builder_spirv::SpirvValue; | |
use crate::spirv_type::SpirvType; | ||
use rspirv::dr::Operand; | ||
use rspirv::spirv::{Decoration, ExecutionModel, FunctionControl, StorageClass, Word}; | ||
use rustc_codegen_ssa::traits::BaseTypeMethods; | ||
use rustc_hir as hir; | ||
use rustc_middle::ty::layout::TyAndLayout; | ||
use rustc_middle::ty::layout::{HasParamEnv, TyAndLayout}; | ||
use rustc_middle::ty::{Instance, Ty, TyKind}; | ||
use rustc_span::Span; | ||
use rustc_target::abi::call::{FnAbi, PassMode}; | ||
use rustc_target::abi::LayoutOf; | ||
use std::collections::HashMap; | ||
use rustc_target::abi::{ | ||
call::{ArgAbi, ArgAttribute, ArgAttributes, FnAbi, PassMode}, | ||
LayoutOf, Size, | ||
}; | ||
use std::{collections::HashMap, iter}; | ||
|
||
impl<'tcx> CodegenCx<'tcx> { | ||
// Entry points declare their "interface" (all uniforms, inputs, outputs, etc.) as parameters. | ||
|
@@ -37,9 +40,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) { | ||
match abi.mode { | ||
PassMode::Direct(_) | PassMode::Indirect { .. } => {} | ||
PassMode::Direct(_) | ||
| PassMode::Indirect { .. } | ||
// plain DST/RTA/VLA | ||
| PassMode::Pair( | ||
ArgAttributes { | ||
pointee_size: Size::ZERO, | ||
.. | ||
}, | ||
ArgAttributes { regular: EMPTY, .. }, | ||
) | ||
// DST struct with fields before the DST member | ||
| PassMode::Pair( | ||
ArgAttributes { .. }, | ||
ArgAttributes { | ||
pointee_size: Size::ZERO, | ||
.. | ||
}, | ||
) => {} | ||
_ => self.tcx.sess.span_err( | ||
arg.span, | ||
&format!("PassMode {:?} invalid for entry point parameter", abi.mode), | ||
|
@@ -63,7 +84,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, | ||
|
@@ -82,7 +103,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, | ||
|
@@ -94,25 +115,22 @@ impl<'tcx> CodegenCx<'tcx> { | |
} | ||
.def(span, self); | ||
let entry_func_return_type = match self.lookup_type(entry_func.ty) { | ||
SpirvType::Function { | ||
return_type, | ||
arguments: _, | ||
} => return_type, | ||
SpirvType::Function { return_type, .. } => return_type, | ||
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 declared_params = entry_fn_abi | ||
.args | ||
let declared_params = arg_abis | ||
.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 len_t = self.type_isize(); | ||
let mut emit = self.emit_global(); | ||
let fn_id = emit | ||
.begin_function(void, None, FunctionControl::NONE, fn_void_void) | ||
|
@@ -121,14 +139,41 @@ impl<'tcx> CodegenCx<'tcx> { | |
// Adjust any global `OpVariable`s as needed (e.g. loading from `Input`s). | ||
let arguments: Vec<_> = declared_params | ||
.iter() | ||
.zip(&entry_fn_abi.args) | ||
.zip(arg_abis) | ||
.zip(hir_params) | ||
.map(|((&(var, storage_class), entry_fn_arg), hir_param)| { | ||
.flat_map(|((&(var, storage_class), entry_fn_arg), hir_param)| { | ||
match entry_fn_arg.layout.ty.kind() { | ||
TyKind::Ref(..) => var, | ||
|
||
TyKind::Ref(_, ty, _) if ty.is_sized(self.tcx.at(span), self.param_env()) => iter::once(var).chain(None), | ||
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. Why are you doing Also, could we extract this into a function? 10 levels of indentation is giving me a headache |
||
TyKind::Ref(_, ty, _) => { | ||
match ty.kind() { | ||
TyKind::Adt(adt_def, substs) => { | ||
let (member_idx, field_def) = adt_def.all_fields().enumerate().last().unwrap(); | ||
let field_ty = field_def.ty(self.tcx, substs); | ||
if !matches!(field_ty.kind(), TyKind::Slice(..)) { | ||
self.tcx.sess.span_fatal( | ||
hir_param.ty_span, | ||
"DST parameters are currently restricted to a reference to a struct whose last field is a slice.", | ||
) | ||
} | ||
let len = emit | ||
.array_length(len_t, None, var, member_idx as u32) | ||
.unwrap(); | ||
iter::once(var).chain(Some(len)) | ||
} | ||
TyKind::Slice(..) | TyKind::Str => self.tcx.sess.span_fatal( | ||
hir_param.ty_span, | ||
"Straight slices are not yet supported, wrap the slice in a newtype.", | ||
), | ||
// TODO: Is this needed? | ||
TyKind::Dynamic(..) => self.tcx.sess.span_fatal( | ||
hir_param.ty_span, | ||
"Trait objects are not supported.", | ||
), | ||
_ => unreachable!(), | ||
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 don't think this is unreachable. For example, here's a compiletest that hits it: // build-pass
#![feature(extern_types)]
use spirv_std as _;
extern "C" {
pub type Blah;
}
#[spirv(fragment)]
pub fn main(#[spirv(uniform)] x: &mut Blah) {} |
||
} | ||
} | ||
_ => match entry_fn_arg.mode { | ||
PassMode::Indirect { .. } => var, | ||
PassMode::Indirect { .. } => iter::once(var).chain(None), | ||
PassMode::Direct(_) => { | ||
assert_eq!(storage_class, StorageClass::Input); | ||
|
||
|
@@ -137,8 +182,10 @@ impl<'tcx> CodegenCx<'tcx> { | |
let value_spirv_type = | ||
entry_fn_arg.layout.spirv_type(hir_param.span, self); | ||
|
||
emit.load(value_spirv_type, None, var, None, std::iter::empty()) | ||
.unwrap() | ||
let loaded_var = emit | ||
.load(value_spirv_type, None, var, None, iter::empty()) | ||
.unwrap(); | ||
iter::once(loaded_var).chain(None) | ||
} | ||
_ => unreachable!(), | ||
}, | ||
|
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.
The second element in a pair for
&SomeUnsizedType
is always the "metadata" likeusize
for anything based on slices. I assumepointee_size
is just zero for non-pointers (and I'm not sure checkingpointee_size
even makes sense?).You could probably just check that the type of the
PassMode::Pair
argument isTyKind::Ref
, that should limit it to&SomeUnsizedType
.