From ffd21d08f4c4af6e55b6b958a44eea8d336fd236 Mon Sep 17 00:00:00 2001 From: yowl00 Date: Sun, 27 Oct 2024 14:41:41 -0500 Subject: [PATCH 1/3] minor improvement to reduce unused using statements --- crates/csharp/src/lib.rs | 77 ++++++++++++++++++++++++++++------------ 1 file changed, 54 insertions(+), 23 deletions(-) diff --git a/crates/csharp/src/lib.rs b/crates/csharp/src/lib.rs index 6f11ef47b..003ba13bf 100644 --- a/crates/csharp/src/lib.rs +++ b/crates/csharp/src/lib.rs @@ -25,18 +25,6 @@ use wit_component::{StringEncoding, WitPrinter}; mod csproj; pub use csproj::CSProject; -//TODO remove unused -const CSHARP_IMPORTS: &str = "\ -using System; -using System.Runtime.CompilerServices; -using System.Collections; -using System.Runtime.InteropServices; -using System.Text; -using System.Collections.Generic; -using System.Diagnostics; -using System.Diagnostics.CodeAnalysis; -"; - #[derive(Default, Debug, Clone)] #[cfg_attr(feature = "clap", derive(clap::Args))] pub struct Opts { @@ -133,6 +121,8 @@ pub enum FunctionLevel { pub struct CSharp { opts: Opts, name: String, + //TODO remove unused + usings: HashSet, return_area_size: usize, return_area_align: usize, tuple_counts: HashSet, @@ -196,6 +186,13 @@ impl CSharp { (String::new(), String::new()) } } + + fn require_using(&mut self, using_ns: &str) { + if !self.usings.contains(using_ns) { + let using_ns_string = using_ns.to_string(); + self.usings.insert(using_ns_string); + } + } } impl WorldGenerator for CSharp { @@ -405,9 +402,11 @@ impl WorldGenerator for CSharp { let access = self.access_modifier(); + let using_pos = src.len(); + uwrite!( src, - "{CSHARP_IMPORTS} + " namespace {world_namespace} {{ @@ -434,6 +433,7 @@ impl WorldGenerator for CSharp { src.push_str("}\n"); if self.needs_result { + self.require_using("System.Runtime.InteropServices"); uwrite!( src, r#" @@ -495,6 +495,7 @@ impl WorldGenerator for CSharp { } if self.needs_option { + self.require_using("System.Diagnostics.CodeAnalysis"); uwrite!( src, r#" @@ -525,6 +526,7 @@ impl WorldGenerator for CSharp { } if self.needs_interop_string { + self.require_using("System.Text"); uwrite!( src, r#" @@ -568,6 +570,8 @@ impl WorldGenerator for CSharp { let (array_size, element_type) = dotnet_aligned_array(self.return_area_size, self.return_area_align); + + self.require_using("System.Runtime.CompilerServices"); uwrite!( ret_area_str, " @@ -623,6 +627,16 @@ impl WorldGenerator for CSharp { src.push_str("}\n"); + src.insert_str( + using_pos, + &self + .usings + .iter() + .map(|s| "using ".to_owned() + s + ";") + .collect::>() + .join("\n"), + ); + files.push(&format!("{name}.cs"), indent(&src).as_bytes()); let generate_stub = |name: String, files: &mut Files, stubs: Stubs| { @@ -668,8 +682,6 @@ impl WorldGenerator for CSharp { let body = format!( "{header} - {CSHARP_IMPORTS} - namespace {fully_qualified_namespace}; {access} partial class {stub_class_name} : {interface_or_class_name} {{ @@ -789,14 +801,20 @@ impl WorldGenerator for CSharp { if body.len() > 0 { let body = format!( "{header} - {CSHARP_IMPORTS} + {0} namespace {namespace}; {access} interface {interface_name} {{ {body} }} - " + ", + &self + .usings + .iter() + .map(|s| "using ".to_owned() + s + ";") + .collect::>() + .join("\n"), ); files.push(&format!("{full_name}.cs"), indent(&body).as_bytes()); @@ -812,7 +830,7 @@ impl WorldGenerator for CSharp { let class_name = interface_name.strip_prefix("I").unwrap(); let body = format!( "{header} - {CSHARP_IMPORTS} + {0} namespace {namespace} {{ @@ -820,7 +838,13 @@ impl WorldGenerator for CSharp { {body} }} }} - " + ", + &self + .usings + .iter() + .map(|s| "using ".to_owned() + s + ";\n") + .collect::>() + .join("\n"), ); files.push( @@ -1082,6 +1106,8 @@ impl InterfaceGenerator<'_> { let import_name = &func.name; + self.gen.require_using("System.Runtime.InteropServices"); + let target = if let FunctionKind::Freestanding = &func.kind { &mut self.csharp_interop_src } else { @@ -1229,6 +1255,7 @@ impl InterfaceGenerator<'_> { let export_name = func.legacy_core_export_name(core_module_name.as_deref()); let access = self.gen.access_modifier(); + self.gen.require_using("System.Runtime.InteropServices"); uwrite!( self.csharp_interop_src, r#" @@ -1482,6 +1509,7 @@ impl InterfaceGenerator<'_> { .map(|s| format!("{}#", self.resolve.name_world_key(s))) .unwrap_or_else(String::new); + self.gen.require_using("System.Runtime.InteropServices"); uwrite!( self.csharp_interop_src, r#" @@ -2584,10 +2612,13 @@ impl Bindgen for FunctionBindgen<'_, '_> { self.gen.gen.needs_interop_string = true; } - Instruction::StringLift { .. } => results.push(format!( - "Encoding.UTF8.GetString((byte*){}, {})", - operands[0], operands[1] - )), + Instruction::StringLift { .. } => { + self.gen.gen.require_using("System.Text"); + results.push(format!( + "Encoding.UTF8.GetString((byte*){}, {})", + operands[0], operands[1] + )); + } Instruction::ListLower { element, realloc } => { let Block { From 162371bc19333e4124730ff6269dbfa66e5420c2 Mon Sep 17 00:00:00 2001 From: yowl00 Date: Tue, 12 Nov 2024 11:57:21 -0500 Subject: [PATCH 2/3] tidy usings, update to net10 --- crates/csharp/src/csproj.rs | 4 +- crates/csharp/src/lib.rs | 74 ++++++++++++++++++++++++++++++------- 2 files changed, 63 insertions(+), 15 deletions(-) diff --git a/crates/csharp/src/csproj.rs b/crates/csharp/src/csproj.rs index 626e89df2..fe5aa24b1 100644 --- a/crates/csharp/src/csproj.rs +++ b/crates/csharp/src/csproj.rs @@ -92,8 +92,8 @@ impl CSProjectLLVMBuilder { csproj.push_str( r#" - - + + "#, ); diff --git a/crates/csharp/src/lib.rs b/crates/csharp/src/lib.rs index 003ba13bf..6eb65f499 100644 --- a/crates/csharp/src/lib.rs +++ b/crates/csharp/src/lib.rs @@ -94,6 +94,8 @@ struct InterfaceFragment { csharp_src: String, csharp_interop_src: String, stub: String, + usings: HashSet, + interop_usings: HashSet, } pub struct InterfaceTypeAndFragments { @@ -121,8 +123,8 @@ pub enum FunctionLevel { pub struct CSharp { opts: Opts, name: String, - //TODO remove unused usings: HashSet, + interop_usings: HashSet, return_area_size: usize, return_area_align: usize, tuple_counts: HashSet, @@ -170,6 +172,8 @@ impl CSharp { resolve, name, direction, + usings: HashSet::::new(), + interop_usings: HashSet::::new(), } } @@ -193,6 +197,13 @@ impl CSharp { self.usings.insert(using_ns_string); } } + + fn require_interop_using(&mut self, using_ns: &str) { + if !self.interop_usings.contains(using_ns) { + let using_ns_string = using_ns.to_string(); + self.interop_usings.insert(using_ns_string); + } + } } impl WorldGenerator for CSharp { @@ -407,7 +418,6 @@ impl WorldGenerator for CSharp { uwrite!( src, " - namespace {world_namespace} {{ {access} interface I{name}World {{ @@ -423,6 +433,9 @@ impl WorldGenerator for CSharp { .join("\n"), ); + let usings: Vec<_> = self.world_fragments .iter() .flat_map(|f| &f.usings) .cloned() .collect(); + usings.iter().for_each(|u| { self.require_using(u); }); + let mut producers = wasm_metadata::Producers::empty(); producers.add( "processed-by", @@ -527,6 +540,7 @@ impl WorldGenerator for CSharp { if self.needs_interop_string { self.require_using("System.Text"); + self.require_using("System.Runtime.InteropServices"); uwrite!( src, r#" @@ -611,6 +625,14 @@ impl WorldGenerator for CSharp { src.push_str("\n"); src.push_str("namespace exports {\n"); + + src.push_str(&self.world_fragments + .iter() + .flat_map(|f| &f.interop_usings) + .map(|s| "using ".to_owned() + s + ";") + .collect::>() + .join("\n")); + src.push_str(&format!("{access} static class {name}World\n")); src.push_str("{"); @@ -809,9 +831,8 @@ impl WorldGenerator for CSharp { {body} }} ", - &self - .usings - .iter() + fragments.iter() + .flat_map(|f| &f.usings) .map(|s| "using ".to_owned() + s + ";") .collect::>() .join("\n"), @@ -839,12 +860,12 @@ impl WorldGenerator for CSharp { }} }} ", - &self - .usings + fragments .iter() + .flat_map(|f| &f.interop_usings) .map(|s| "using ".to_owned() + s + ";\n") .collect::>() - .join("\n"), + .join(""), ); files.push( @@ -869,6 +890,8 @@ struct InterfaceGenerator<'a> { resolve: &'a Resolve, name: &'a str, direction: Direction, + usings: HashSet, + interop_usings: HashSet, } impl InterfaceGenerator<'_> { @@ -980,6 +1003,8 @@ impl InterfaceGenerator<'_> { csharp_src: self.src, csharp_interop_src: self.csharp_interop_src, stub: self.stub, + usings: self.usings, + interop_usings: self.interop_usings, }); } @@ -988,6 +1013,8 @@ impl InterfaceGenerator<'_> { csharp_src: self.src, csharp_interop_src: self.csharp_interop_src, stub: self.stub, + usings: self.usings, + interop_usings: self.interop_usings, }); } @@ -1106,11 +1133,11 @@ impl InterfaceGenerator<'_> { let import_name = &func.name; - self.gen.require_using("System.Runtime.InteropServices"); - let target = if let FunctionKind::Freestanding = &func.kind { + self.require_interop_using("System.Runtime.InteropServices"); &mut self.csharp_interop_src } else { + self.require_using("System.Runtime.InteropServices"); &mut self.src }; @@ -1255,7 +1282,7 @@ impl InterfaceGenerator<'_> { let export_name = func.legacy_core_export_name(core_module_name.as_deref()); let access = self.gen.access_modifier(); - self.gen.require_using("System.Runtime.InteropServices"); + self.require_interop_using("System.Runtime.InteropServices"); uwrite!( self.csharp_interop_src, r#" @@ -1456,6 +1483,20 @@ impl InterfaceGenerator<'_> { } } + fn require_using(&mut self, using_ns: &str) { + if !self.usings.contains(using_ns) { + let using_ns_string = using_ns.to_string(); + self.usings.insert(using_ns_string); + } + } + + fn require_interop_using(&mut self, using_ns: &str) { + if !self.interop_usings.contains(using_ns) { + let using_ns_string = using_ns.to_string(); + self.interop_usings.insert(using_ns_string); + } + } + fn start_resource(&mut self, id: TypeId, key: Option<&WorldKey>) { let access = self.gen.access_modifier(); let qualified = self.type_name_with_qualifier(&Type::Id(id), true); @@ -1471,6 +1512,7 @@ impl InterfaceGenerator<'_> { .map(|key| self.resolve.name_world_key(key)) .unwrap_or_else(|| "$root".into()); + self.require_using("System.Runtime.InteropServices"); // As of this writing, we cannot safely drop a handle to an imported resource from a .NET finalizer // because it may still have one or more open child resources. Once WIT has explicit syntax for // indicating parent/child relationships, we should be able to use that information to keep track @@ -1509,7 +1551,7 @@ impl InterfaceGenerator<'_> { .map(|s| format!("{}#", self.resolve.name_world_key(s))) .unwrap_or_else(String::new); - self.gen.require_using("System.Runtime.InteropServices"); + self.require_interop_using("System.Runtime.InteropServices"); uwrite!( self.csharp_interop_src, r#" @@ -1528,6 +1570,7 @@ impl InterfaceGenerator<'_> { .map(|key| format!("[export]{}", self.resolve.name_world_key(key))) .unwrap_or_else(|| "[export]$root".into()); + self.require_using("System.Runtime.InteropServices"); // The ergonomics of exported resources are not ideal, currently. Implementing such a resource // requires both extending a class and implementing an interface. The reason for the class is to // allow implementers to inherit code which tracks and disposes of the resource handle; the reason @@ -2613,7 +2656,12 @@ impl Bindgen for FunctionBindgen<'_, '_> { } Instruction::StringLift { .. } => { - self.gen.gen.require_using("System.Text"); + if FunctionKind::Freestanding == *self.kind || self.gen.direction == Direction::Export { + self.gen.require_interop_using("System.Text"); + } else { + self.gen.require_using("System.Text"); + }; + results.push(format!( "Encoding.UTF8.GetString((byte*){}, {})", operands[0], operands[1] From 542e0951b7f5952ed8762d3abba4c448653c37d7 Mon Sep 17 00:00:00 2001 From: yowl00 Date: Tue, 12 Nov 2024 12:02:08 -0500 Subject: [PATCH 3/3] cargo fmt --- crates/csharp/src/lib.rs | 33 ++++++++++++++++++++++----------- 1 file changed, 22 insertions(+), 11 deletions(-) diff --git a/crates/csharp/src/lib.rs b/crates/csharp/src/lib.rs index 6eb65f499..ff4c767c0 100644 --- a/crates/csharp/src/lib.rs +++ b/crates/csharp/src/lib.rs @@ -433,8 +433,15 @@ impl WorldGenerator for CSharp { .join("\n"), ); - let usings: Vec<_> = self.world_fragments .iter() .flat_map(|f| &f.usings) .cloned() .collect(); - usings.iter().for_each(|u| { self.require_using(u); }); + let usings: Vec<_> = self + .world_fragments + .iter() + .flat_map(|f| &f.usings) + .cloned() + .collect(); + usings.iter().for_each(|u| { + self.require_using(u); + }); let mut producers = wasm_metadata::Producers::empty(); producers.add( @@ -626,12 +633,15 @@ impl WorldGenerator for CSharp { src.push_str("namespace exports {\n"); - src.push_str(&self.world_fragments - .iter() - .flat_map(|f| &f.interop_usings) - .map(|s| "using ".to_owned() + s + ";") - .collect::>() - .join("\n")); + src.push_str( + &self + .world_fragments + .iter() + .flat_map(|f| &f.interop_usings) + .map(|s| "using ".to_owned() + s + ";") + .collect::>() + .join("\n"), + ); src.push_str(&format!("{access} static class {name}World\n")); src.push_str("{"); @@ -831,7 +841,8 @@ impl WorldGenerator for CSharp { {body} }} ", - fragments.iter() + fragments + .iter() .flat_map(|f| &f.usings) .map(|s| "using ".to_owned() + s + ";") .collect::>() @@ -2660,8 +2671,8 @@ impl Bindgen for FunctionBindgen<'_, '_> { self.gen.require_interop_using("System.Text"); } else { self.gen.require_using("System.Text"); - }; - + } + results.push(format!( "Encoding.UTF8.GetString((byte*){}, {})", operands[0], operands[1]