diff --git a/.github/workflows/test.sh b/.github/workflows/test.sh index 8e904efc63..668ff88be2 100755 --- a/.github/workflows/test.sh +++ b/.github/workflows/test.sh @@ -52,4 +52,4 @@ cargo_test_no_features examples/runners/cpu cargo_test_no_features examples/shaders/sky-shader cargo_test_no_features examples/shaders/simplest-shader -cargo compiletest +cargo compiletest --target-env unknown,vulkan1.1,spv1.3 diff --git a/crates/rustc_codegen_spirv/src/lib.rs b/crates/rustc_codegen_spirv/src/lib.rs index b0afddc050..2d1d0c951c 100644 --- a/crates/rustc_codegen_spirv/src/lib.rs +++ b/crates/rustc_codegen_spirv/src/lib.rs @@ -205,7 +205,7 @@ fn is_blocklisted_fn<'tcx>( false } -fn target_options() -> Target { +fn target_options(env: Option) -> Target { Target { llvm_target: "no-llvm".to_string(), pointer_width: 32, @@ -222,6 +222,10 @@ fn target_options() -> Target { linker_flavor: LinkerFlavor::Ld, panic_strategy: PanicStrategy::Abort, os: "unknown".to_string(), + env: env + .as_ref() + .map(ToString::to_string) + .unwrap_or_else(|| "unknown".to_string()), // TODO: Investigate if main_needs_argc_argv is useful (for building exes) main_needs_argc_argv: false, ..Default::default() @@ -284,12 +288,20 @@ impl CodegenBackend for SpirvCodegenBackend { fn target_override(&self, opts: &config::Options) -> Option { match opts.target_triple { - TargetTriple::TargetTriple(ref target_triple) => match &**target_triple { - // TODO: Do we want to match *everything* here, since, well, we only support one thing? that will allow - // folks to not specify --target spirv-unknown-unknown on the commandline. - "spirv-unknown-unknown" => Some(target_options()), - _ => None, - }, + TargetTriple::TargetTriple(ref target_triple) => { + const ARCH_VENDOR: &str = "spirv-unknown-"; + if !target_triple.starts_with(ARCH_VENDOR) { + return None; + } + + let env = &target_triple[ARCH_VENDOR.len()..]; + + match env.parse() { + Ok(env) => Some(target_options(Some(env))), + _ if env == "unknown" => Some(target_options(None)), + _ => None, + } + } TargetTriple::TargetPath(_) => None, } } diff --git a/crates/rustc_codegen_spirv/src/link.rs b/crates/rustc_codegen_spirv/src/link.rs index e4336a9466..b22feab4aa 100644 --- a/crates/rustc_codegen_spirv/src/link.rs +++ b/crates/rustc_codegen_spirv/src/link.rs @@ -218,7 +218,7 @@ fn do_spirv_opt(sess: &Session, spv_binary: Vec, filename: &Path) -> Vec, + + /// The environment to compile to the SPIR-V tests. + #[structopt(long)] + target_env: Option, +} + +impl Opt { + pub fn environments(&self) -> Vec { + match &self.target_env { + Some(env) => env.split(",").map(String::from).collect(), + None => vec!["unknown".into()], + } + } } -const TARGET: &str = "spirv-unknown-unknown"; +const TARGET_PREFIX: &str = "spirv-unknown-"; #[derive(Copy, Clone)] enum DepKind { @@ -38,10 +51,10 @@ impl DepKind { } } - fn target_dir_suffix(self) -> &'static str { + fn target_dir_suffix(self, target: &str) -> String { match self { - Self::SpirvLib => "spirv-unknown-unknown/debug/deps", - Self::ProcMacro => "debug/deps", + Self::SpirvLib => format!("{}/debug/deps", target), + Self::ProcMacro => "debug/deps".into(), } } } @@ -49,7 +62,7 @@ impl DepKind { fn main() { let opt = Opt::from_args(); - let tests_dir = Path::new(env!("CARGO_MANIFEST_DIR")); + let tests_dir = PathBuf::from(env!("CARGO_MANIFEST_DIR")); let workspace_root = tests_dir.parent().unwrap(); let original_target_dir = workspace_root.join("target"); let deps_target_dir = original_target_dir.join("compiletest-deps"); @@ -58,90 +71,99 @@ fn main() { // Pull in rustc_codegen_spirv as a dynamic library in the same way // spirv-builder does. let codegen_backend_path = find_rustc_codegen_spirv(); - let libs = build_deps(&deps_target_dir, &codegen_backend_path); - run_mode( - "ui", + let runner = Runner { opt, tests_dir, compiletest_build_dir, - &deps_target_dir, - &codegen_backend_path, - &libs, - ); + deps_target_dir, + codegen_backend_path, + }; + + runner.run_mode("ui"); } // FIXME(eddyb) a bunch of these functions could be nicer if they were methods. -/// Runs the given `mode` on the directory that matches that name, using the -/// backend provided by `codegen_backend_path`. -fn run_mode( - mode: &'static str, +struct Runner { opt: Opt, - tests_dir: &Path, + tests_dir: PathBuf, compiletest_build_dir: PathBuf, - deps_target_dir: &Path, - codegen_backend_path: &Path, - libs: &TestDeps, -) { - let mut config = compiletest::Config::default(); - - /// RUSTFLAGS passed to all test files. - fn test_rustc_flags( - codegen_backend_path: &Path, - deps: &TestDeps, - indirect_deps_dirs: &[&Path], - ) -> String { - [ - &*rust_flags(codegen_backend_path), - &*indirect_deps_dirs - .iter() - .map(|dir| format!("-L dependency={}", dir.display())) - .fold(String::new(), |a, b| b + " " + &a), - "--edition 2018", - &*format!("--extern noprelude:core={}", deps.core.display()), - &*format!( - "--extern noprelude:compiler_builtins={}", - deps.compiler_builtins.display() - ), - &*format!( - "--extern spirv_std_macros={}", - deps.spirv_std_macros.display() - ), - &*format!("--extern spirv_std={}", deps.spirv_std.display()), - &*format!("--extern glam={}", deps.glam.display()), - "--crate-type dylib", - "-Zunstable-options", - "-Zcrate-attr=no_std", - "-Zcrate-attr=feature(register_attr,asm)", - "-Zcrate-attr=register_attr(spirv)", - ] - .join(" ") - } - - let flags = test_rustc_flags( - codegen_backend_path, - libs, - &[ - &deps_target_dir.join(DepKind::SpirvLib.target_dir_suffix()), - &deps_target_dir.join(DepKind::ProcMacro.target_dir_suffix()), - ], - ); + deps_target_dir: PathBuf, + codegen_backend_path: PathBuf, +} - config.target_rustcflags = Some(flags); - config.mode = mode.parse().expect("Invalid mode"); - config.target = String::from(TARGET); - config.src_base = tests_dir.join(mode); - config.build_base = compiletest_build_dir; - config.bless = opt.bless; - config.filters = opt.filters; - config.clean_rmeta(); +impl Runner { + /// Runs the given `mode` on the directory that matches that name, using the + /// backend provided by `codegen_backend_path`. + fn run_mode(&self, mode: &'static str) { + /// RUSTFLAGS passed to all test files. + fn test_rustc_flags( + codegen_backend_path: &Path, + deps: &TestDeps, + indirect_deps_dirs: &[&Path], + ) -> String { + [ + &*rust_flags(codegen_backend_path), + &*indirect_deps_dirs + .iter() + .map(|dir| format!("-L dependency={}", dir.display())) + .fold(String::new(), |a, b| b + " " + &a), + "--edition 2018", + &*format!("--extern noprelude:core={}", deps.core.display()), + &*format!( + "--extern noprelude:compiler_builtins={}", + deps.compiler_builtins.display() + ), + &*format!( + "--extern spirv_std_macros={}", + deps.spirv_std_macros.display() + ), + &*format!("--extern spirv_std={}", deps.spirv_std.display()), + &*format!("--extern glam={}", deps.glam.display()), + "--crate-type dylib", + "-Zunstable-options", + "-Zcrate-attr=no_std", + "-Zcrate-attr=feature(register_attr,asm)", + "-Zcrate-attr=register_attr(spirv)", + ] + .join(" ") + } - compiletest::run_tests(&config); + for env in self.opt.environments() { + let target = format!("{}{}", TARGET_PREFIX, env); + let mut config = compiletest::Config::default(); + let libs = build_deps(&self.deps_target_dir, &self.codegen_backend_path, &target); + + let flags = test_rustc_flags( + &self.codegen_backend_path, + &libs, + &[ + &self + .deps_target_dir + .join(DepKind::SpirvLib.target_dir_suffix(&target)), + &self + .deps_target_dir + .join(DepKind::ProcMacro.target_dir_suffix(&target)), + ], + ); + + config.target_rustcflags = Some(flags); + config.mode = mode.parse().expect("Invalid mode"); + config.target = target; + config.src_base = self.tests_dir.join(mode); + config.build_base = self.compiletest_build_dir.clone(); + config.bless = self.opt.bless; + config.filters = self.opt.filters.clone(); + config.clean_rmeta(); + + compiletest::run_tests(&config); + } + } } /// Runs the processes needed to build `spirv-std` & other deps. -fn build_deps(deps_target_dir: &Path, codegen_backend_path: &Path) -> TestDeps { +fn build_deps(deps_target_dir: &Path, codegen_backend_path: &Path, target: &str) -> TestDeps { // HACK(eddyb) this is only needed until we enable `resolver = "2"`, as the // old ("1") resolver has a bug where it picks up extra features based on the // current directory (and so we always set the working dir as a workaround). @@ -154,7 +176,7 @@ fn build_deps(deps_target_dir: &Path, codegen_backend_path: &Path) -> TestDeps { "-p", "compiletests-deps-helper", "-Zbuild-std=core", - &*format!("--target={}", TARGET), + &*format!("--target={}", target), ]) .arg("--target-dir") .arg(deps_target_dir) @@ -166,13 +188,23 @@ fn build_deps(deps_target_dir: &Path, codegen_backend_path: &Path) -> TestDeps { .and_then(map_status_to_result) .unwrap(); - let compiler_builtins = - find_lib(deps_target_dir, "compiler_builtins", DepKind::SpirvLib).unwrap(); - let core = find_lib(deps_target_dir, "core", DepKind::SpirvLib).unwrap(); - let spirv_std = find_lib(deps_target_dir, "spirv_std", DepKind::SpirvLib).unwrap(); - let glam = find_lib(deps_target_dir, "glam", DepKind::SpirvLib).unwrap(); - let spirv_std_macros = - find_lib(deps_target_dir, "spirv_std_macros", DepKind::ProcMacro).unwrap(); + let compiler_builtins = find_lib( + deps_target_dir, + "compiler_builtins", + DepKind::SpirvLib, + target, + ) + .unwrap(); + let core = find_lib(deps_target_dir, "core", DepKind::SpirvLib, target).unwrap(); + let spirv_std = find_lib(deps_target_dir, "spirv_std", DepKind::SpirvLib, target).unwrap(); + let glam = find_lib(deps_target_dir, "glam", DepKind::SpirvLib, target).unwrap(); + let spirv_std_macros = find_lib( + deps_target_dir, + "spirv_std_macros", + DepKind::ProcMacro, + target, + ) + .unwrap(); if [ &compiler_builtins, @@ -185,7 +217,7 @@ fn build_deps(deps_target_dir: &Path, codegen_backend_path: &Path) -> TestDeps { .any(|o| o.is_none()) { clean_deps(deps_target_dir); - build_deps(deps_target_dir, codegen_backend_path) + build_deps(deps_target_dir, codegen_backend_path, target) } else { TestDeps { core: core.unwrap(), @@ -215,12 +247,13 @@ fn find_lib( deps_target_dir: &Path, base: impl AsRef, dep_kind: DepKind, + target: &str, ) -> Result> { let base = base.as_ref(); let (expected_prefix, expected_extension) = dep_kind.prefix_and_extension(); let expected_name = format!("{}{}", expected_prefix, base.display()); - let dir = deps_target_dir.join(dep_kind.target_dir_suffix()); + let dir = deps_target_dir.join(dep_kind.target_dir_suffix(target)); let paths = std::fs::read_dir(dir)? .filter_map(Result::ok) diff --git a/tests/ui/image/fetch.rs b/tests/ui/image/fetch.rs index cd83fa85ba..ba6d176293 100644 --- a/tests/ui/image/fetch.rs +++ b/tests/ui/image/fetch.rs @@ -3,7 +3,7 @@ use spirv_std::{arch, Image2d}; #[spirv(fragment)] -pub fn main(#[spirv(uniform_constant)] image: &Image2d, output: &mut glam::Vec4) { +pub fn main(#[spirv(uniform_constant, descriptor_set = 0, binding = 0)] image: &Image2d, output: &mut glam::Vec4) { let texel = image.fetch(glam::IVec2::new(0, 1)); *output = texel; } diff --git a/tests/ui/image/issue_527.rs b/tests/ui/image/issue_527.rs new file mode 100644 index 0000000000..a46ba8e716 --- /dev/null +++ b/tests/ui/image/issue_527.rs @@ -0,0 +1,29 @@ +use glam::*; + +#[spirv(block)] +pub struct PointsBuffer { + points: [UVec2; 100], +} + +#[spirv(compute(threads(1)))] +pub fn main_cs( + #[spirv(global_invocation_id)] + id: UVec3, + #[spirv(storage_buffer, descriptor_set = 0, binding = 0)] + points_buffer: &mut PointsBuffer, + #[spirv(uniform_constant, descriptor_set = 1, binding = 1)] + image: &spirv_std::StorageImage2d, +) { + unsafe { asm!("OpCapability StorageImageWriteWithoutFormat") }; + let position = id.xy(); + for i in 0..100usize { + let p0 = &points_buffer.points[i]; + let p1 = &points_buffer.points[i + 1]; + if p0.x == position.x && p1.y == position.y { + unsafe { + image.write(position, vec2(1.0, 0.0)); + }; + } + } +} + diff --git a/tests/ui/image/read.rs b/tests/ui/image/read.rs index 849f5fd2f1..d98fb92246 100644 --- a/tests/ui/image/read.rs +++ b/tests/ui/image/read.rs @@ -4,7 +4,8 @@ use spirv_std::{arch, StorageImage2d}; #[spirv(fragment)] -pub fn main(#[spirv(uniform_constant)] image: &StorageImage2d, output: &mut glam::Vec2) { +pub fn main(#[spirv(uniform_constant, descriptor_set = 0, binding = 0)] image: &StorageImage2d, output: &mut glam::Vec4) { + unsafe { asm!("OpCapability StorageImageReadWithoutFormat") }; let coords = image.read(glam::IVec2::new(0, 1)); *output = coords; } diff --git a/tests/ui/image/sample.rs b/tests/ui/image/sample.rs index 176ec08492..4b2f7298b4 100644 --- a/tests/ui/image/sample.rs +++ b/tests/ui/image/sample.rs @@ -5,10 +5,10 @@ use spirv_std::{arch, Cubemap, Image2d, Image2dArray, Sampler}; #[spirv(fragment)] pub fn main( - #[spirv(uniform_constant)] image2d: &Image2d, - #[spirv(uniform_constant)] image2d_array: &Image2dArray, - #[spirv(uniform_constant)] cubemap: &Cubemap, - #[spirv(uniform_constant)] sampler: &Sampler, + #[spirv(uniform_constant, descriptor_set = 0, binding = 0)] image2d: &Image2d, + #[spirv(uniform_constant, descriptor_set = 1, binding = 1)] image2d_array: &Image2dArray, + #[spirv(uniform_constant, descriptor_set = 2, binding = 2)] cubemap: &Cubemap, + #[spirv(uniform_constant, descriptor_set = 3, binding = 3)] sampler: &Sampler, output: &mut glam::Vec4, ) { let v2 = glam::Vec2::new(0.0, 1.0); diff --git a/tests/ui/image/sample_gradient.rs b/tests/ui/image/sample_gradient.rs index 24e20a2bec..2d4aa63420 100644 --- a/tests/ui/image/sample_gradient.rs +++ b/tests/ui/image/sample_gradient.rs @@ -5,10 +5,10 @@ use spirv_std::{arch, Cubemap, Image2d, Image2dArray, Sampler}; #[spirv(fragment)] pub fn main( - #[spirv(uniform_constant)] image2d: &Image2d, - #[spirv(uniform_constant)] image2d_array: &Image2dArray, - #[spirv(uniform_constant)] cubemap: &Cubemap, - #[spirv(uniform_constant)] sampler: &Sampler, + #[spirv(uniform_constant, descriptor_set = 0, binding = 0)] image2d: &Image2d, + #[spirv(uniform_constant, descriptor_set = 1, binding = 1)] image2d_array: &Image2dArray, + #[spirv(uniform_constant, descriptor_set = 2, binding = 2)] cubemap: &Cubemap, + #[spirv(uniform_constant, descriptor_set = 3, binding = 3)] sampler: &Sampler, output: &mut glam::Vec4, ) { let v2 = glam::Vec2::new(0.0, 1.0); diff --git a/tests/ui/image/sample_lod.rs b/tests/ui/image/sample_lod.rs index a547d5e370..129d540c7a 100644 --- a/tests/ui/image/sample_lod.rs +++ b/tests/ui/image/sample_lod.rs @@ -5,10 +5,10 @@ use spirv_std::{arch, Cubemap, Image2d, Image2dArray, Sampler}; #[spirv(fragment)] pub fn main( - #[spirv(uniform_constant)] image2d: &Image2d, - #[spirv(uniform_constant)] image2d_array: &Image2dArray, - #[spirv(uniform_constant)] cubemap: &Cubemap, - #[spirv(uniform_constant)] sampler: &Sampler, + #[spirv(uniform_constant, descriptor_set = 0, binding = 0)] image2d: &Image2d, + #[spirv(uniform_constant, descriptor_set = 1, binding = 1)] image2d_array: &Image2dArray, + #[spirv(uniform_constant, descriptor_set = 2, binding = 2)] cubemap: &Cubemap, + #[spirv(uniform_constant, descriptor_set = 3, binding = 3)] sampler: &Sampler, output: &mut glam::Vec4, ) { let v2 = glam::Vec2::new(0.0, 1.0); diff --git a/tests/ui/image/write.rs b/tests/ui/image/write.rs index e2cf172446..a6045c4d10 100644 --- a/tests/ui/image/write.rs +++ b/tests/ui/image/write.rs @@ -4,8 +4,9 @@ use spirv_std::{arch, StorageImage2d}; #[spirv(fragment)] -pub fn main(texels: glam::Vec2, #[spirv(uniform_constant)] image: &StorageImage2d) { +pub fn main(texels: glam::Vec2, #[spirv(uniform_constant, descriptor_set = 0, binding = 0)] image: &StorageImage2d) { unsafe { + asm!("OpCapability StorageImageWriteWithoutFormat"); image.write(glam::UVec2::new(0, 1), texels); } } diff --git a/tests/ui/storage_class/push_constant.rs b/tests/ui/storage_class/push_constant.rs index 3248d14b74..2e64a66bde 100644 --- a/tests/ui/storage_class/push_constant.rs +++ b/tests/ui/storage_class/push_constant.rs @@ -7,6 +7,7 @@ use spirv_std as _; #[derive(Copy, Clone)] +#[spirv(block)] pub struct ShaderConstants { pub width: u32, pub height: u32,