1 //! Defines `Module` and related types. 2 3 // TODO: Should `ir::Function` really have a `name`? 4 5 // TODO: Factor out `ir::Function`'s `ext_funcs` and `global_values` into a struct 6 // shared with `DataDescription`? 7 8 use super::HashMap; 9 use crate::data_context::DataDescription; 10 use core::fmt::Display; 11 use cranelift_codegen::binemit::{CodeOffset, Reloc}; 12 use cranelift_codegen::entity::{PrimaryMap, entity_impl}; 13 use cranelift_codegen::ir::ExternalName; 14 use cranelift_codegen::ir::function::{Function, VersionMarker}; 15 use cranelift_codegen::settings::SetError; 16 use cranelift_codegen::{ 17 CodegenError, CompileError, Context, FinalizedMachReloc, FinalizedRelocTarget, ir, isa, 18 }; 19 use cranelift_control::ControlPlane; 20 use std::borrow::{Cow, ToOwned}; 21 use std::boxed::Box; 22 use std::string::String; 23 24 /// A module relocation. 25 #[derive(Clone)] 26 pub struct ModuleReloc { 27 /// The offset at which the relocation applies, *relative to the 28 /// containing section*. 29 pub offset: CodeOffset, 30 /// The kind of relocation. 31 pub kind: Reloc, 32 /// The external symbol / name to which this relocation refers. 33 pub name: ModuleRelocTarget, 34 /// The addend to add to the symbol value. 35 pub addend: i64, 36 } 37 38 impl ModuleReloc { 39 /// Converts a `FinalizedMachReloc` produced from a `Function` into a `ModuleReloc`. 40 pub fn from_mach_reloc( 41 mach_reloc: &FinalizedMachReloc, 42 func: &Function, 43 func_id: FuncId, 44 ) -> Self { 45 let name = match mach_reloc.target { 46 FinalizedRelocTarget::ExternalName(ExternalName::User(reff)) => { 47 let name = &func.params.user_named_funcs()[reff]; 48 ModuleRelocTarget::user(name.namespace, name.index) 49 } 50 FinalizedRelocTarget::ExternalName(ExternalName::TestCase(_)) => unimplemented!(), 51 FinalizedRelocTarget::ExternalName(ExternalName::LibCall(libcall)) => { 52 ModuleRelocTarget::LibCall(libcall) 53 } 54 FinalizedRelocTarget::ExternalName(ExternalName::KnownSymbol(ks)) => { 55 ModuleRelocTarget::KnownSymbol(ks) 56 } 57 FinalizedRelocTarget::Func(offset) => { 58 ModuleRelocTarget::FunctionOffset(func_id, offset) 59 } 60 }; 61 Self { 62 offset: mach_reloc.offset, 63 kind: mach_reloc.kind, 64 name, 65 addend: mach_reloc.addend, 66 } 67 } 68 } 69 70 /// A function identifier for use in the `Module` interface. 71 #[derive(Copy, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)] 72 #[cfg_attr( 73 feature = "enable-serde", 74 derive(serde_derive::Serialize, serde_derive::Deserialize) 75 )] 76 pub struct FuncId(u32); 77 entity_impl!(FuncId, "funcid"); 78 79 /// Function identifiers are namespace 0 in `ir::ExternalName` 80 impl From<FuncId> for ModuleRelocTarget { 81 fn from(id: FuncId) -> Self { 82 Self::User { 83 namespace: 0, 84 index: id.0, 85 } 86 } 87 } 88 89 impl FuncId { 90 /// Get the `FuncId` for the function named by `name`. 91 pub fn from_name(name: &ModuleRelocTarget) -> FuncId { 92 if let ModuleRelocTarget::User { namespace, index } = name { 93 debug_assert_eq!(*namespace, 0); 94 FuncId::from_u32(*index) 95 } else { 96 panic!("unexpected name in DataId::from_name") 97 } 98 } 99 } 100 101 /// A data object identifier for use in the `Module` interface. 102 #[derive(Copy, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)] 103 #[cfg_attr( 104 feature = "enable-serde", 105 derive(serde_derive::Serialize, serde_derive::Deserialize) 106 )] 107 pub struct DataId(u32); 108 entity_impl!(DataId, "dataid"); 109 110 /// Data identifiers are namespace 1 in `ir::ExternalName` 111 impl From<DataId> for ModuleRelocTarget { 112 fn from(id: DataId) -> Self { 113 Self::User { 114 namespace: 1, 115 index: id.0, 116 } 117 } 118 } 119 120 impl DataId { 121 /// Get the `DataId` for the data object named by `name`. 122 pub fn from_name(name: &ModuleRelocTarget) -> DataId { 123 if let ModuleRelocTarget::User { namespace, index } = name { 124 debug_assert_eq!(*namespace, 1); 125 DataId::from_u32(*index) 126 } else { 127 panic!("unexpected name in DataId::from_name") 128 } 129 } 130 } 131 132 /// Linkage refers to where an entity is defined and who can see it. 133 #[derive(Copy, Clone, Debug, PartialEq, Eq)] 134 #[cfg_attr( 135 feature = "enable-serde", 136 derive(serde_derive::Serialize, serde_derive::Deserialize) 137 )] 138 pub enum Linkage { 139 /// Defined outside of a module. 140 Import, 141 /// Defined inside the module, but not visible outside it. 142 Local, 143 /// Defined inside the module, visible outside it, and may be preempted. 144 Preemptible, 145 /// Defined inside the module, visible inside the current static linkage unit, but not outside. 146 /// 147 /// A static linkage unit is the combination of all object files passed to a linker to create 148 /// an executable or dynamic library. 149 Hidden, 150 /// Defined inside the module, and visible outside it. 151 Export, 152 } 153 154 impl Linkage { 155 fn merge(a: Self, b: Self) -> Self { 156 match a { 157 Self::Export => Self::Export, 158 Self::Hidden => match b { 159 Self::Export => Self::Export, 160 Self::Preemptible => Self::Preemptible, 161 _ => Self::Hidden, 162 }, 163 Self::Preemptible => match b { 164 Self::Export => Self::Export, 165 _ => Self::Preemptible, 166 }, 167 Self::Local => match b { 168 Self::Export => Self::Export, 169 Self::Hidden => Self::Hidden, 170 Self::Preemptible => Self::Preemptible, 171 Self::Local | Self::Import => Self::Local, 172 }, 173 Self::Import => b, 174 } 175 } 176 177 /// Test whether this linkage can have a definition. 178 pub fn is_definable(self) -> bool { 179 match self { 180 Self::Import => false, 181 Self::Local | Self::Preemptible | Self::Hidden | Self::Export => true, 182 } 183 } 184 185 /// Test whether this linkage must have a definition. 186 pub fn requires_definition(self) -> bool { 187 match self { 188 Self::Import | Self::Preemptible => false, 189 Self::Local | Self::Hidden | Self::Export => true, 190 } 191 } 192 193 /// Test whether this linkage will have a definition that cannot be preempted. 194 pub fn is_final(self) -> bool { 195 match self { 196 Self::Import | Self::Preemptible => false, 197 Self::Local | Self::Hidden | Self::Export => true, 198 } 199 } 200 } 201 202 /// A declared name may refer to either a function or data declaration 203 #[derive(Copy, Clone, PartialEq, Eq, Hash, PartialOrd, Ord, Debug)] 204 #[cfg_attr( 205 feature = "enable-serde", 206 derive(serde_derive::Serialize, serde_derive::Deserialize) 207 )] 208 pub enum FuncOrDataId { 209 /// When it's a FuncId 210 Func(FuncId), 211 /// When it's a DataId 212 Data(DataId), 213 } 214 215 /// Mapping to `ModuleExtName` is trivial based on the `FuncId` and `DataId` mapping. 216 impl From<FuncOrDataId> for ModuleRelocTarget { 217 fn from(id: FuncOrDataId) -> Self { 218 match id { 219 FuncOrDataId::Func(funcid) => Self::from(funcid), 220 FuncOrDataId::Data(dataid) => Self::from(dataid), 221 } 222 } 223 } 224 225 /// Information about a function which can be called. 226 #[derive(Debug)] 227 #[cfg_attr( 228 feature = "enable-serde", 229 derive(serde_derive::Serialize, serde_derive::Deserialize) 230 )] 231 #[expect(missing_docs, reason = "self-describing fields")] 232 pub struct FunctionDeclaration { 233 pub name: Option<String>, 234 pub linkage: Linkage, 235 pub signature: ir::Signature, 236 } 237 238 impl FunctionDeclaration { 239 /// The linkage name of the function. 240 /// 241 /// Synthesized from the given function id if it is an anonymous function. 242 pub fn linkage_name(&self, id: FuncId) -> Cow<'_, str> { 243 match &self.name { 244 Some(name) => Cow::Borrowed(name), 245 // Symbols starting with .L are completely omitted from the symbol table after linking. 246 // Using hexadecimal instead of decimal for slightly smaller symbol names and often 247 // slightly faster linking. 248 None => Cow::Owned(format!(".Lfn{:x}", id.as_u32())), 249 } 250 } 251 252 fn merge( 253 &mut self, 254 id: FuncId, 255 linkage: Linkage, 256 sig: &ir::Signature, 257 ) -> Result<(), ModuleError> { 258 self.linkage = Linkage::merge(self.linkage, linkage); 259 if &self.signature != sig { 260 return Err(ModuleError::IncompatibleSignature( 261 self.linkage_name(id).into_owned(), 262 self.signature.clone(), 263 sig.clone(), 264 )); 265 } 266 Ok(()) 267 } 268 } 269 270 /// Error messages for all `Module` methods 271 #[derive(Debug)] 272 pub enum ModuleError { 273 /// Indicates an identifier was used before it was declared 274 Undeclared(String), 275 276 /// Indicates an identifier was used as data/function first, but then used as the other 277 IncompatibleDeclaration(String), 278 279 /// Indicates a function identifier was declared with a 280 /// different signature than declared previously 281 IncompatibleSignature(String, ir::Signature, ir::Signature), 282 283 /// Indicates an identifier was defined more than once 284 DuplicateDefinition(String), 285 286 /// Indicates an identifier was defined, but was declared as an import 287 InvalidImportDefinition(String), 288 289 /// Wraps a `cranelift-codegen` error 290 Compilation(CodegenError), 291 292 /// Memory allocation failure from a backend 293 Allocation { 294 /// Tell where the allocation came from 295 message: &'static str, 296 /// Io error the allocation failed with 297 err: std::io::Error, 298 }, 299 300 /// Wraps a generic error from a backend 301 Backend(anyhow::Error), 302 303 /// Wraps an error from a flag definition. 304 Flag(SetError), 305 } 306 307 impl<'a> From<CompileError<'a>> for ModuleError { 308 fn from(err: CompileError<'a>) -> Self { 309 Self::Compilation(err.inner) 310 } 311 } 312 313 // This is manually implementing Error and Display instead of using thiserror to reduce the amount 314 // of dependencies used by Cranelift. 315 impl std::error::Error for ModuleError { 316 fn source(&self) -> Option<&(dyn std::error::Error + 'static)> { 317 match self { 318 Self::Undeclared { .. } 319 | Self::IncompatibleDeclaration { .. } 320 | Self::IncompatibleSignature { .. } 321 | Self::DuplicateDefinition { .. } 322 | Self::InvalidImportDefinition { .. } => None, 323 Self::Compilation(source) => Some(source), 324 Self::Allocation { err: source, .. } => Some(source), 325 Self::Backend(source) => Some(&**source), 326 Self::Flag(source) => Some(source), 327 } 328 } 329 } 330 331 impl std::fmt::Display for ModuleError { 332 fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result { 333 match self { 334 Self::Undeclared(name) => { 335 write!(f, "Undeclared identifier: {name}") 336 } 337 Self::IncompatibleDeclaration(name) => { 338 write!(f, "Incompatible declaration of identifier: {name}",) 339 } 340 Self::IncompatibleSignature(name, prev_sig, new_sig) => { 341 write!( 342 f, 343 "Function {name} signature {new_sig:?} is incompatible with previous declaration {prev_sig:?}", 344 ) 345 } 346 Self::DuplicateDefinition(name) => { 347 write!(f, "Duplicate definition of identifier: {name}") 348 } 349 Self::InvalidImportDefinition(name) => { 350 write!( 351 f, 352 "Invalid to define identifier declared as an import: {name}", 353 ) 354 } 355 Self::Compilation(err) => { 356 write!(f, "Compilation error: {err}") 357 } 358 Self::Allocation { message, err } => { 359 write!(f, "Allocation error: {message}: {err}") 360 } 361 Self::Backend(err) => write!(f, "Backend error: {err}"), 362 Self::Flag(err) => write!(f, "Flag error: {err}"), 363 } 364 } 365 } 366 367 impl std::convert::From<CodegenError> for ModuleError { 368 fn from(source: CodegenError) -> Self { 369 Self::Compilation { 0: source } 370 } 371 } 372 373 impl std::convert::From<SetError> for ModuleError { 374 fn from(source: SetError) -> Self { 375 Self::Flag { 0: source } 376 } 377 } 378 379 /// A convenient alias for a `Result` that uses `ModuleError` as the error type. 380 pub type ModuleResult<T> = Result<T, ModuleError>; 381 382 /// Information about a data object which can be accessed. 383 #[derive(Debug)] 384 #[cfg_attr( 385 feature = "enable-serde", 386 derive(serde_derive::Serialize, serde_derive::Deserialize) 387 )] 388 #[expect(missing_docs, reason = "self-describing fields")] 389 pub struct DataDeclaration { 390 pub name: Option<String>, 391 pub linkage: Linkage, 392 pub writable: bool, 393 pub tls: bool, 394 } 395 396 impl DataDeclaration { 397 /// The linkage name of the data object. 398 /// 399 /// Synthesized from the given data id if it is an anonymous function. 400 pub fn linkage_name(&self, id: DataId) -> Cow<'_, str> { 401 match &self.name { 402 Some(name) => Cow::Borrowed(name), 403 // Symbols starting with .L are completely omitted from the symbol table after linking. 404 // Using hexadecimal instead of decimal for slightly smaller symbol names and often 405 // slightly faster linking. 406 None => Cow::Owned(format!(".Ldata{:x}", id.as_u32())), 407 } 408 } 409 410 fn merge(&mut self, linkage: Linkage, writable: bool, tls: bool) { 411 self.linkage = Linkage::merge(self.linkage, linkage); 412 self.writable = self.writable || writable; 413 assert_eq!( 414 self.tls, tls, 415 "Can't change TLS data object to normal or in the opposite way", 416 ); 417 } 418 } 419 420 /// A translated `ExternalName` into something global we can handle. 421 #[derive(Clone, Debug)] 422 #[cfg_attr( 423 feature = "enable-serde", 424 derive(serde_derive::Serialize, serde_derive::Deserialize) 425 )] 426 pub enum ModuleRelocTarget { 427 /// User defined function, converted from `ExternalName::User`. 428 User { 429 /// Arbitrary. 430 namespace: u32, 431 /// Arbitrary. 432 index: u32, 433 }, 434 /// Call into a library function. 435 LibCall(ir::LibCall), 436 /// Symbols known to the linker. 437 KnownSymbol(ir::KnownSymbol), 438 /// A offset inside a function 439 FunctionOffset(FuncId, CodeOffset), 440 } 441 442 impl ModuleRelocTarget { 443 /// Creates a user-defined external name. 444 pub fn user(namespace: u32, index: u32) -> Self { 445 Self::User { namespace, index } 446 } 447 } 448 449 impl Display for ModuleRelocTarget { 450 fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { 451 match self { 452 Self::User { namespace, index } => write!(f, "u{namespace}:{index}"), 453 Self::LibCall(lc) => write!(f, "%{lc}"), 454 Self::KnownSymbol(ks) => write!(f, "{ks}"), 455 Self::FunctionOffset(fname, offset) => write!(f, "{fname}+{offset}"), 456 } 457 } 458 } 459 460 /// This provides a view to the state of a module which allows `ir::ExternalName`s to be translated 461 /// into `FunctionDeclaration`s and `DataDeclaration`s. 462 #[derive(Debug, Default)] 463 pub struct ModuleDeclarations { 464 /// A version marker used to ensure that serialized clif ir is never deserialized with a 465 /// different version of Cranelift. 466 // Note: This must be the first field to ensure that Serde will deserialize it before 467 // attempting to deserialize other fields that are potentially changed between versions. 468 _version_marker: VersionMarker, 469 470 names: HashMap<String, FuncOrDataId>, 471 functions: PrimaryMap<FuncId, FunctionDeclaration>, 472 data_objects: PrimaryMap<DataId, DataDeclaration>, 473 } 474 475 #[cfg(feature = "enable-serde")] 476 mod serialize { 477 // This is manually implementing Serialize and Deserialize to avoid serializing the names field, 478 // which can be entirely reconstructed from the functions and data_objects fields, saving space. 479 480 use super::*; 481 482 use serde::de::{Deserialize, Deserializer, Error, MapAccess, SeqAccess, Unexpected, Visitor}; 483 use serde::ser::{Serialize, SerializeStruct, Serializer}; 484 use std::fmt; 485 486 fn get_names<E: Error>( 487 functions: &PrimaryMap<FuncId, FunctionDeclaration>, 488 data_objects: &PrimaryMap<DataId, DataDeclaration>, 489 ) -> Result<HashMap<String, FuncOrDataId>, E> { 490 let mut names = HashMap::new(); 491 for (func_id, decl) in functions.iter() { 492 if let Some(name) = &decl.name { 493 let old = names.insert(name.clone(), FuncOrDataId::Func(func_id)); 494 if old.is_some() { 495 return Err(E::invalid_value( 496 Unexpected::Other("duplicate name"), 497 &"FunctionDeclaration's with no duplicate names", 498 )); 499 } 500 } 501 } 502 for (data_id, decl) in data_objects.iter() { 503 if let Some(name) = &decl.name { 504 let old = names.insert(name.clone(), FuncOrDataId::Data(data_id)); 505 if old.is_some() { 506 return Err(E::invalid_value( 507 Unexpected::Other("duplicate name"), 508 &"DataDeclaration's with no duplicate names", 509 )); 510 } 511 } 512 } 513 Ok(names) 514 } 515 516 impl Serialize for ModuleDeclarations { 517 fn serialize<S: Serializer>(&self, s: S) -> Result<S::Ok, S::Error> { 518 let ModuleDeclarations { 519 _version_marker, 520 functions, 521 data_objects, 522 names: _, 523 } = self; 524 525 let mut state = s.serialize_struct("ModuleDeclarations", 4)?; 526 state.serialize_field("_version_marker", _version_marker)?; 527 state.serialize_field("functions", functions)?; 528 state.serialize_field("data_objects", data_objects)?; 529 state.end() 530 } 531 } 532 533 enum ModuleDeclarationsField { 534 VersionMarker, 535 Functions, 536 DataObjects, 537 Ignore, 538 } 539 540 struct ModuleDeclarationsFieldVisitor; 541 542 impl<'de> serde::de::Visitor<'de> for ModuleDeclarationsFieldVisitor { 543 type Value = ModuleDeclarationsField; 544 545 fn expecting(&self, f: &mut fmt::Formatter) -> fmt::Result { 546 f.write_str("field identifier") 547 } 548 549 fn visit_u64<E: Error>(self, val: u64) -> Result<Self::Value, E> { 550 match val { 551 0u64 => Ok(ModuleDeclarationsField::VersionMarker), 552 1u64 => Ok(ModuleDeclarationsField::Functions), 553 2u64 => Ok(ModuleDeclarationsField::DataObjects), 554 _ => Ok(ModuleDeclarationsField::Ignore), 555 } 556 } 557 558 fn visit_str<E: Error>(self, val: &str) -> Result<Self::Value, E> { 559 match val { 560 "_version_marker" => Ok(ModuleDeclarationsField::VersionMarker), 561 "functions" => Ok(ModuleDeclarationsField::Functions), 562 "data_objects" => Ok(ModuleDeclarationsField::DataObjects), 563 _ => Ok(ModuleDeclarationsField::Ignore), 564 } 565 } 566 567 fn visit_bytes<E: Error>(self, val: &[u8]) -> Result<Self::Value, E> { 568 match val { 569 b"_version_marker" => Ok(ModuleDeclarationsField::VersionMarker), 570 b"functions" => Ok(ModuleDeclarationsField::Functions), 571 b"data_objects" => Ok(ModuleDeclarationsField::DataObjects), 572 _ => Ok(ModuleDeclarationsField::Ignore), 573 } 574 } 575 } 576 577 impl<'de> Deserialize<'de> for ModuleDeclarationsField { 578 #[inline] 579 fn deserialize<D: Deserializer<'de>>(d: D) -> Result<Self, D::Error> { 580 d.deserialize_identifier(ModuleDeclarationsFieldVisitor) 581 } 582 } 583 584 struct ModuleDeclarationsVisitor; 585 586 impl<'de> Visitor<'de> for ModuleDeclarationsVisitor { 587 type Value = ModuleDeclarations; 588 589 fn expecting(&self, f: &mut fmt::Formatter) -> fmt::Result { 590 f.write_str("struct ModuleDeclarations") 591 } 592 593 #[inline] 594 fn visit_seq<A: SeqAccess<'de>>(self, mut seq: A) -> Result<Self::Value, A::Error> { 595 let _version_marker = match seq.next_element()? { 596 Some(val) => val, 597 None => { 598 return Err(Error::invalid_length( 599 0usize, 600 &"struct ModuleDeclarations with 4 elements", 601 )); 602 } 603 }; 604 let functions = match seq.next_element()? { 605 Some(val) => val, 606 None => { 607 return Err(Error::invalid_length( 608 2usize, 609 &"struct ModuleDeclarations with 4 elements", 610 )); 611 } 612 }; 613 let data_objects = match seq.next_element()? { 614 Some(val) => val, 615 None => { 616 return Err(Error::invalid_length( 617 3usize, 618 &"struct ModuleDeclarations with 4 elements", 619 )); 620 } 621 }; 622 let names = get_names(&functions, &data_objects)?; 623 Ok(ModuleDeclarations { 624 _version_marker, 625 names, 626 functions, 627 data_objects, 628 }) 629 } 630 631 #[inline] 632 fn visit_map<A: MapAccess<'de>>(self, mut map: A) -> Result<Self::Value, A::Error> { 633 let mut _version_marker: Option<VersionMarker> = None; 634 let mut functions: Option<PrimaryMap<FuncId, FunctionDeclaration>> = None; 635 let mut data_objects: Option<PrimaryMap<DataId, DataDeclaration>> = None; 636 while let Some(key) = map.next_key::<ModuleDeclarationsField>()? { 637 match key { 638 ModuleDeclarationsField::VersionMarker => { 639 if _version_marker.is_some() { 640 return Err(Error::duplicate_field("_version_marker")); 641 } 642 _version_marker = Some(map.next_value()?); 643 } 644 ModuleDeclarationsField::Functions => { 645 if functions.is_some() { 646 return Err(Error::duplicate_field("functions")); 647 } 648 functions = Some(map.next_value()?); 649 } 650 ModuleDeclarationsField::DataObjects => { 651 if data_objects.is_some() { 652 return Err(Error::duplicate_field("data_objects")); 653 } 654 data_objects = Some(map.next_value()?); 655 } 656 _ => { 657 map.next_value::<serde::de::IgnoredAny>()?; 658 } 659 } 660 } 661 let _version_marker = match _version_marker { 662 Some(_version_marker) => _version_marker, 663 None => return Err(Error::missing_field("_version_marker")), 664 }; 665 let functions = match functions { 666 Some(functions) => functions, 667 None => return Err(Error::missing_field("functions")), 668 }; 669 let data_objects = match data_objects { 670 Some(data_objects) => data_objects, 671 None => return Err(Error::missing_field("data_objects")), 672 }; 673 let names = get_names(&functions, &data_objects)?; 674 Ok(ModuleDeclarations { 675 _version_marker, 676 names, 677 functions, 678 data_objects, 679 }) 680 } 681 } 682 683 impl<'de> Deserialize<'de> for ModuleDeclarations { 684 fn deserialize<D: Deserializer<'de>>(d: D) -> Result<Self, D::Error> { 685 d.deserialize_struct( 686 "ModuleDeclarations", 687 &["_version_marker", "functions", "data_objects"], 688 ModuleDeclarationsVisitor, 689 ) 690 } 691 } 692 } 693 694 impl ModuleDeclarations { 695 /// Get the module identifier for a given name, if that name 696 /// has been declared. 697 pub fn get_name(&self, name: &str) -> Option<FuncOrDataId> { 698 self.names.get(name).copied() 699 } 700 701 /// Get an iterator of all function declarations 702 pub fn get_functions(&self) -> impl Iterator<Item = (FuncId, &FunctionDeclaration)> { 703 self.functions.iter() 704 } 705 706 /// Return whether `name` names a function, rather than a data object. 707 pub fn is_function(name: &ModuleRelocTarget) -> bool { 708 match name { 709 ModuleRelocTarget::User { namespace, .. } => *namespace == 0, 710 ModuleRelocTarget::LibCall(_) 711 | ModuleRelocTarget::KnownSymbol(_) 712 | ModuleRelocTarget::FunctionOffset(..) => { 713 panic!("unexpected module ext name") 714 } 715 } 716 } 717 718 /// Get the `FunctionDeclaration` for the function named by `name`. 719 pub fn get_function_decl(&self, func_id: FuncId) -> &FunctionDeclaration { 720 &self.functions[func_id] 721 } 722 723 /// Get an iterator of all data declarations 724 pub fn get_data_objects(&self) -> impl Iterator<Item = (DataId, &DataDeclaration)> { 725 self.data_objects.iter() 726 } 727 728 /// Get the `DataDeclaration` for the data object named by `name`. 729 pub fn get_data_decl(&self, data_id: DataId) -> &DataDeclaration { 730 &self.data_objects[data_id] 731 } 732 733 /// Declare a function in this module. 734 pub fn declare_function( 735 &mut self, 736 name: &str, 737 linkage: Linkage, 738 signature: &ir::Signature, 739 ) -> ModuleResult<(FuncId, Linkage)> { 740 // TODO: Can we avoid allocating names so often? 741 use super::hash_map::Entry::*; 742 match self.names.entry(name.to_owned()) { 743 Occupied(entry) => match *entry.get() { 744 FuncOrDataId::Func(id) => { 745 let existing = &mut self.functions[id]; 746 existing.merge(id, linkage, signature)?; 747 Ok((id, existing.linkage)) 748 } 749 FuncOrDataId::Data(..) => { 750 Err(ModuleError::IncompatibleDeclaration(name.to_owned())) 751 } 752 }, 753 Vacant(entry) => { 754 let id = self.functions.push(FunctionDeclaration { 755 name: Some(name.to_owned()), 756 linkage, 757 signature: signature.clone(), 758 }); 759 entry.insert(FuncOrDataId::Func(id)); 760 Ok((id, self.functions[id].linkage)) 761 } 762 } 763 } 764 765 /// Declare an anonymous function in this module. 766 pub fn declare_anonymous_function( 767 &mut self, 768 signature: &ir::Signature, 769 ) -> ModuleResult<FuncId> { 770 let id = self.functions.push(FunctionDeclaration { 771 name: None, 772 linkage: Linkage::Local, 773 signature: signature.clone(), 774 }); 775 Ok(id) 776 } 777 778 /// Declare a data object in this module. 779 pub fn declare_data( 780 &mut self, 781 name: &str, 782 linkage: Linkage, 783 writable: bool, 784 tls: bool, 785 ) -> ModuleResult<(DataId, Linkage)> { 786 // TODO: Can we avoid allocating names so often? 787 use super::hash_map::Entry::*; 788 match self.names.entry(name.to_owned()) { 789 Occupied(entry) => match *entry.get() { 790 FuncOrDataId::Data(id) => { 791 let existing = &mut self.data_objects[id]; 792 existing.merge(linkage, writable, tls); 793 Ok((id, existing.linkage)) 794 } 795 796 FuncOrDataId::Func(..) => { 797 Err(ModuleError::IncompatibleDeclaration(name.to_owned())) 798 } 799 }, 800 Vacant(entry) => { 801 let id = self.data_objects.push(DataDeclaration { 802 name: Some(name.to_owned()), 803 linkage, 804 writable, 805 tls, 806 }); 807 entry.insert(FuncOrDataId::Data(id)); 808 Ok((id, self.data_objects[id].linkage)) 809 } 810 } 811 } 812 813 /// Declare an anonymous data object in this module. 814 pub fn declare_anonymous_data(&mut self, writable: bool, tls: bool) -> ModuleResult<DataId> { 815 let id = self.data_objects.push(DataDeclaration { 816 name: None, 817 linkage: Linkage::Local, 818 writable, 819 tls, 820 }); 821 Ok(id) 822 } 823 } 824 825 /// A `Module` is a utility for collecting functions and data objects, and linking them together. 826 pub trait Module { 827 /// Return the `TargetIsa` to compile for. 828 fn isa(&self) -> &dyn isa::TargetIsa; 829 830 /// Get all declarations in this module. 831 fn declarations(&self) -> &ModuleDeclarations; 832 833 /// Get the module identifier for a given name, if that name 834 /// has been declared. 835 fn get_name(&self, name: &str) -> Option<FuncOrDataId> { 836 self.declarations().get_name(name) 837 } 838 839 /// Return the target information needed by frontends to produce Cranelift IR 840 /// for the current target. 841 fn target_config(&self) -> isa::TargetFrontendConfig { 842 self.isa().frontend_config() 843 } 844 845 /// Create a new `Context` initialized for use with this `Module`. 846 /// 847 /// This ensures that the `Context` is initialized with the default calling 848 /// convention for the `TargetIsa`. 849 fn make_context(&self) -> Context { 850 let mut ctx = Context::new(); 851 ctx.func.signature.call_conv = self.isa().default_call_conv(); 852 ctx 853 } 854 855 /// Clear the given `Context` and reset it for use with a new function. 856 /// 857 /// This ensures that the `Context` is initialized with the default calling 858 /// convention for the `TargetIsa`. 859 fn clear_context(&self, ctx: &mut Context) { 860 ctx.clear(); 861 ctx.func.signature.call_conv = self.isa().default_call_conv(); 862 } 863 864 /// Create a new empty `Signature` with the default calling convention for 865 /// the `TargetIsa`, to which parameter and return types can be added for 866 /// declaring a function to be called by this `Module`. 867 fn make_signature(&self) -> ir::Signature { 868 ir::Signature::new(self.isa().default_call_conv()) 869 } 870 871 /// Clear the given `Signature` and reset for use with a new function. 872 /// 873 /// This ensures that the `Signature` is initialized with the default 874 /// calling convention for the `TargetIsa`. 875 fn clear_signature(&self, sig: &mut ir::Signature) { 876 sig.clear(self.isa().default_call_conv()); 877 } 878 879 /// Declare a function in this module. 880 fn declare_function( 881 &mut self, 882 name: &str, 883 linkage: Linkage, 884 signature: &ir::Signature, 885 ) -> ModuleResult<FuncId>; 886 887 /// Declare an anonymous function in this module. 888 fn declare_anonymous_function(&mut self, signature: &ir::Signature) -> ModuleResult<FuncId>; 889 890 /// Declare a data object in this module. 891 fn declare_data( 892 &mut self, 893 name: &str, 894 linkage: Linkage, 895 writable: bool, 896 tls: bool, 897 ) -> ModuleResult<DataId>; 898 899 /// Declare an anonymous data object in this module. 900 fn declare_anonymous_data(&mut self, writable: bool, tls: bool) -> ModuleResult<DataId>; 901 902 /// Use this when you're building the IR of a function to reference a function. 903 /// 904 /// TODO: Coalesce redundant decls and signatures. 905 /// TODO: Look into ways to reduce the risk of using a FuncRef in the wrong function. 906 fn declare_func_in_func(&mut self, func_id: FuncId, func: &mut ir::Function) -> ir::FuncRef { 907 let decl = &self.declarations().functions[func_id]; 908 let signature = func.import_signature(decl.signature.clone()); 909 let user_name_ref = func.declare_imported_user_function(ir::UserExternalName { 910 namespace: 0, 911 index: func_id.as_u32(), 912 }); 913 let colocated = decl.linkage.is_final(); 914 func.import_function(ir::ExtFuncData { 915 name: ir::ExternalName::user(user_name_ref), 916 signature, 917 colocated, 918 patchable: false, 919 }) 920 } 921 922 /// Use this when you're building the IR of a function to reference a data object. 923 /// 924 /// TODO: Same as above. 925 fn declare_data_in_func(&self, data: DataId, func: &mut ir::Function) -> ir::GlobalValue { 926 let decl = &self.declarations().data_objects[data]; 927 let colocated = decl.linkage.is_final(); 928 let user_name_ref = func.declare_imported_user_function(ir::UserExternalName { 929 namespace: 1, 930 index: data.as_u32(), 931 }); 932 func.create_global_value(ir::GlobalValueData::Symbol { 933 name: ir::ExternalName::user(user_name_ref), 934 offset: ir::immediates::Imm64::new(0), 935 colocated, 936 tls: decl.tls, 937 }) 938 } 939 940 /// TODO: Same as above. 941 fn declare_func_in_data(&self, func_id: FuncId, data: &mut DataDescription) -> ir::FuncRef { 942 data.import_function(ModuleRelocTarget::user(0, func_id.as_u32())) 943 } 944 945 /// TODO: Same as above. 946 fn declare_data_in_data(&self, data_id: DataId, data: &mut DataDescription) -> ir::GlobalValue { 947 data.import_global_value(ModuleRelocTarget::user(1, data_id.as_u32())) 948 } 949 950 /// Define a function, producing the function body from the given `Context`. 951 /// 952 /// Returns the size of the function's code and constant data. 953 /// 954 /// Unlike [`define_function_with_control_plane`] this uses a default [`ControlPlane`] for 955 /// convenience. 956 /// 957 /// Note: After calling this function the given `Context` will contain the compiled function. 958 /// 959 /// [`define_function_with_control_plane`]: Self::define_function_with_control_plane 960 fn define_function(&mut self, func: FuncId, ctx: &mut Context) -> ModuleResult<()> { 961 self.define_function_with_control_plane(func, ctx, &mut ControlPlane::default()) 962 } 963 964 /// Define a function, producing the function body from the given `Context`. 965 /// 966 /// Returns the size of the function's code and constant data. 967 /// 968 /// Note: After calling this function the given `Context` will contain the compiled function. 969 fn define_function_with_control_plane( 970 &mut self, 971 func: FuncId, 972 ctx: &mut Context, 973 ctrl_plane: &mut ControlPlane, 974 ) -> ModuleResult<()>; 975 976 /// Define a function, taking the function body from the given `bytes`. 977 /// 978 /// This function is generally only useful if you need to precisely specify 979 /// the emitted instructions for some reason; otherwise, you should use 980 /// `define_function`. 981 /// 982 /// Returns the size of the function's code. 983 fn define_function_bytes( 984 &mut self, 985 func_id: FuncId, 986 alignment: u64, 987 bytes: &[u8], 988 relocs: &[ModuleReloc], 989 ) -> ModuleResult<()>; 990 991 /// Define a data object, producing the data contents from the given `DataDescription`. 992 fn define_data(&mut self, data_id: DataId, data: &DataDescription) -> ModuleResult<()>; 993 } 994 995 impl<M: Module + ?Sized> Module for &mut M { 996 fn isa(&self) -> &dyn isa::TargetIsa { 997 (**self).isa() 998 } 999 1000 fn declarations(&self) -> &ModuleDeclarations { 1001 (**self).declarations() 1002 } 1003 1004 fn get_name(&self, name: &str) -> Option<FuncOrDataId> { 1005 (**self).get_name(name) 1006 } 1007 1008 fn target_config(&self) -> isa::TargetFrontendConfig { 1009 (**self).target_config() 1010 } 1011 1012 fn make_context(&self) -> Context { 1013 (**self).make_context() 1014 } 1015 1016 fn clear_context(&self, ctx: &mut Context) { 1017 (**self).clear_context(ctx) 1018 } 1019 1020 fn make_signature(&self) -> ir::Signature { 1021 (**self).make_signature() 1022 } 1023 1024 fn clear_signature(&self, sig: &mut ir::Signature) { 1025 (**self).clear_signature(sig) 1026 } 1027 1028 fn declare_function( 1029 &mut self, 1030 name: &str, 1031 linkage: Linkage, 1032 signature: &ir::Signature, 1033 ) -> ModuleResult<FuncId> { 1034 (**self).declare_function(name, linkage, signature) 1035 } 1036 1037 fn declare_anonymous_function(&mut self, signature: &ir::Signature) -> ModuleResult<FuncId> { 1038 (**self).declare_anonymous_function(signature) 1039 } 1040 1041 fn declare_data( 1042 &mut self, 1043 name: &str, 1044 linkage: Linkage, 1045 writable: bool, 1046 tls: bool, 1047 ) -> ModuleResult<DataId> { 1048 (**self).declare_data(name, linkage, writable, tls) 1049 } 1050 1051 fn declare_anonymous_data(&mut self, writable: bool, tls: bool) -> ModuleResult<DataId> { 1052 (**self).declare_anonymous_data(writable, tls) 1053 } 1054 1055 fn declare_func_in_func(&mut self, func: FuncId, in_func: &mut ir::Function) -> ir::FuncRef { 1056 (**self).declare_func_in_func(func, in_func) 1057 } 1058 1059 fn declare_data_in_func(&self, data: DataId, func: &mut ir::Function) -> ir::GlobalValue { 1060 (**self).declare_data_in_func(data, func) 1061 } 1062 1063 fn declare_func_in_data(&self, func_id: FuncId, data: &mut DataDescription) -> ir::FuncRef { 1064 (**self).declare_func_in_data(func_id, data) 1065 } 1066 1067 fn declare_data_in_data(&self, data_id: DataId, data: &mut DataDescription) -> ir::GlobalValue { 1068 (**self).declare_data_in_data(data_id, data) 1069 } 1070 1071 fn define_function(&mut self, func: FuncId, ctx: &mut Context) -> ModuleResult<()> { 1072 (**self).define_function(func, ctx) 1073 } 1074 1075 fn define_function_with_control_plane( 1076 &mut self, 1077 func: FuncId, 1078 ctx: &mut Context, 1079 ctrl_plane: &mut ControlPlane, 1080 ) -> ModuleResult<()> { 1081 (**self).define_function_with_control_plane(func, ctx, ctrl_plane) 1082 } 1083 1084 fn define_function_bytes( 1085 &mut self, 1086 func_id: FuncId, 1087 alignment: u64, 1088 bytes: &[u8], 1089 relocs: &[ModuleReloc], 1090 ) -> ModuleResult<()> { 1091 (**self).define_function_bytes(func_id, alignment, bytes, relocs) 1092 } 1093 1094 fn define_data(&mut self, data_id: DataId, data: &DataDescription) -> ModuleResult<()> { 1095 (**self).define_data(data_id, data) 1096 } 1097 } 1098 1099 impl<M: Module + ?Sized> Module for Box<M> { 1100 fn isa(&self) -> &dyn isa::TargetIsa { 1101 (**self).isa() 1102 } 1103 1104 fn declarations(&self) -> &ModuleDeclarations { 1105 (**self).declarations() 1106 } 1107 1108 fn get_name(&self, name: &str) -> Option<FuncOrDataId> { 1109 (**self).get_name(name) 1110 } 1111 1112 fn target_config(&self) -> isa::TargetFrontendConfig { 1113 (**self).target_config() 1114 } 1115 1116 fn make_context(&self) -> Context { 1117 (**self).make_context() 1118 } 1119 1120 fn clear_context(&self, ctx: &mut Context) { 1121 (**self).clear_context(ctx) 1122 } 1123 1124 fn make_signature(&self) -> ir::Signature { 1125 (**self).make_signature() 1126 } 1127 1128 fn clear_signature(&self, sig: &mut ir::Signature) { 1129 (**self).clear_signature(sig) 1130 } 1131 1132 fn declare_function( 1133 &mut self, 1134 name: &str, 1135 linkage: Linkage, 1136 signature: &ir::Signature, 1137 ) -> ModuleResult<FuncId> { 1138 (**self).declare_function(name, linkage, signature) 1139 } 1140 1141 fn declare_anonymous_function(&mut self, signature: &ir::Signature) -> ModuleResult<FuncId> { 1142 (**self).declare_anonymous_function(signature) 1143 } 1144 1145 fn declare_data( 1146 &mut self, 1147 name: &str, 1148 linkage: Linkage, 1149 writable: bool, 1150 tls: bool, 1151 ) -> ModuleResult<DataId> { 1152 (**self).declare_data(name, linkage, writable, tls) 1153 } 1154 1155 fn declare_anonymous_data(&mut self, writable: bool, tls: bool) -> ModuleResult<DataId> { 1156 (**self).declare_anonymous_data(writable, tls) 1157 } 1158 1159 fn declare_func_in_func(&mut self, func: FuncId, in_func: &mut ir::Function) -> ir::FuncRef { 1160 (**self).declare_func_in_func(func, in_func) 1161 } 1162 1163 fn declare_data_in_func(&self, data: DataId, func: &mut ir::Function) -> ir::GlobalValue { 1164 (**self).declare_data_in_func(data, func) 1165 } 1166 1167 fn declare_func_in_data(&self, func_id: FuncId, data: &mut DataDescription) -> ir::FuncRef { 1168 (**self).declare_func_in_data(func_id, data) 1169 } 1170 1171 fn declare_data_in_data(&self, data_id: DataId, data: &mut DataDescription) -> ir::GlobalValue { 1172 (**self).declare_data_in_data(data_id, data) 1173 } 1174 1175 fn define_function(&mut self, func: FuncId, ctx: &mut Context) -> ModuleResult<()> { 1176 (**self).define_function(func, ctx) 1177 } 1178 1179 fn define_function_with_control_plane( 1180 &mut self, 1181 func: FuncId, 1182 ctx: &mut Context, 1183 ctrl_plane: &mut ControlPlane, 1184 ) -> ModuleResult<()> { 1185 (**self).define_function_with_control_plane(func, ctx, ctrl_plane) 1186 } 1187 1188 fn define_function_bytes( 1189 &mut self, 1190 func_id: FuncId, 1191 alignment: u64, 1192 bytes: &[u8], 1193 relocs: &[ModuleReloc], 1194 ) -> ModuleResult<()> { 1195 (**self).define_function_bytes(func_id, alignment, bytes, relocs) 1196 } 1197 1198 fn define_data(&mut self, data_id: DataId, data: &DataDescription) -> ModuleResult<()> { 1199 (**self).define_data(data_id, data) 1200 } 1201 } 1202