1use std::ops::Range;
2
3use rustc_abi::{Align, HasDataLayout, Primitive, Scalar, Size, WrappingRange};
4use rustc_codegen_ssa::common;
5use rustc_codegen_ssa::traits::*;
6use rustc_hir::LangItem;
7use rustc_hir::attrs::Linkage;
8use rustc_hir::def::DefKind;
9use rustc_hir::def_id::{DefId, LOCAL_CRATE};
10use rustc_middle::middle::codegen_fn_attrs::{CodegenFnAttrFlags, CodegenFnAttrs};
11use rustc_middle::mir::interpret::{
12 Allocation, ConstAllocation, ErrorHandled, InitChunk, Pointer, Scalar as InterpScalar,
13 read_target_uint,
14};
15use rustc_middle::mir::mono::MonoItem;
16use rustc_middle::ty::layout::{HasTypingEnv, LayoutOf};
17use rustc_middle::ty::{self, Instance};
18use rustc_middle::{bug, span_bug};
19use rustc_span::Symbol;
20use rustc_target::spec::Arch;
21use tracing::{debug, instrument, trace};
22
23use crate::common::CodegenCx;
24use crate::errors::SymbolAlreadyDefined;
25use crate::llvm::{self, Type, Value};
26use crate::type_of::LayoutLlvmExt;
27use crate::{base, debuginfo};
28
29pub(crate) fn const_alloc_to_llvm<'ll>(
30 cx: &CodegenCx<'ll, '_>,
31 alloc: &Allocation,
32 is_static: bool,
33) -> &'ll Value {
34 if !is_static {
42 assert!(alloc.len() != 0);
43 }
44 let mut llvals = Vec::with_capacity(alloc.provenance().ptrs().len() + 1);
45 let dl = cx.data_layout();
46 let pointer_size = dl.pointer_size();
47 let pointer_size_bytes = pointer_size.bytes() as usize;
48
49 fn append_chunks_of_init_and_uninit_bytes<'ll, 'a, 'b>(
52 llvals: &mut Vec<&'ll Value>,
53 cx: &'a CodegenCx<'ll, 'b>,
54 alloc: &'a Allocation,
55 range: Range<usize>,
56 ) {
57 let chunks = alloc.init_mask().range_as_init_chunks(range.clone().into());
58
59 let chunk_to_llval = move |chunk| match chunk {
60 InitChunk::Init(range) => {
61 let range = (range.start.bytes() as usize)..(range.end.bytes() as usize);
62 let bytes = alloc.inspect_with_uninit_and_ptr_outside_interpreter(range);
63 cx.const_bytes(bytes)
64 }
65 InitChunk::Uninit(range) => {
66 let len = range.end.bytes() - range.start.bytes();
67 cx.const_undef(cx.type_array(cx.type_i8(), len))
68 }
69 };
70
71 let max = cx.sess().opts.unstable_opts.uninit_const_chunk_threshold;
76 let allow_uninit_chunks = chunks.clone().take(max.saturating_add(1)).count() <= max;
77
78 if allow_uninit_chunks {
79 llvals.extend(chunks.map(chunk_to_llval));
80 } else {
81 let bytes = alloc.inspect_with_uninit_and_ptr_outside_interpreter(range);
84 llvals.push(cx.const_bytes(bytes));
85 }
86 }
87
88 let mut next_offset = 0;
89 for &(offset, prov) in alloc.provenance().ptrs().iter() {
90 let offset = offset.bytes();
91 assert_eq!(offset as usize as u64, offset);
92 let offset = offset as usize;
93 if offset > next_offset {
94 append_chunks_of_init_and_uninit_bytes(&mut llvals, cx, alloc, next_offset..offset);
98 }
99 let ptr_offset = read_target_uint(
100 dl.endian,
101 alloc.inspect_with_uninit_and_ptr_outside_interpreter(
105 offset..(offset + pointer_size_bytes),
106 ),
107 )
108 .expect("const_alloc_to_llvm: could not read relocation pointer")
109 as u64;
110
111 let address_space = cx.tcx.global_alloc(prov.alloc_id()).address_space(cx);
112
113 llvals.push(cx.scalar_to_backend(
114 InterpScalar::from_pointer(Pointer::new(prov, Size::from_bytes(ptr_offset)), &cx.tcx),
115 Scalar::Initialized {
116 value: Primitive::Pointer(address_space),
117 valid_range: WrappingRange::full(pointer_size),
118 },
119 cx.type_ptr_ext(address_space),
120 ));
121 next_offset = offset + pointer_size_bytes;
122 }
123 if alloc.len() >= next_offset {
124 let range = next_offset..alloc.len();
125 append_chunks_of_init_and_uninit_bytes(&mut llvals, cx, alloc, range);
129 }
130
131 if let &[data] = &*llvals { data } else { cx.const_struct(&llvals, true) }
137}
138
139fn codegen_static_initializer<'ll, 'tcx>(
140 cx: &CodegenCx<'ll, 'tcx>,
141 def_id: DefId,
142) -> Result<(&'ll Value, ConstAllocation<'tcx>), ErrorHandled> {
143 let alloc = cx.tcx.eval_static_initializer(def_id)?;
144 Ok((const_alloc_to_llvm(cx, alloc.inner(), true), alloc))
145}
146
147fn set_global_alignment<'ll>(cx: &CodegenCx<'ll, '_>, gv: &'ll Value, mut align: Align) {
148 if let Some(min_global) = cx.sess().target.min_global_align {
152 align = Ord::max(align, min_global);
153 }
154 llvm::set_alignment(gv, align);
155}
156
157fn check_and_apply_linkage<'ll, 'tcx>(
158 cx: &CodegenCx<'ll, 'tcx>,
159 attrs: &CodegenFnAttrs,
160 llty: &'ll Type,
161 sym: &str,
162 def_id: DefId,
163) -> &'ll Value {
164 if let Some(linkage) = attrs.import_linkage {
165 debug!("get_static: sym={} linkage={:?}", sym, linkage);
166
167 let g1 = if matches!(attrs.import_linkage, Some(Linkage::ExternalWeak)) {
170 let instance = Instance::mono(cx.tcx, def_id);
174 if let ty::Adt(struct_def, args) = instance.ty(cx.tcx, cx.typing_env()).kind()
175 && cx.tcx.is_lang_item(struct_def.did(), LangItem::Option)
176 && let ty::FnPtr(sig, header) = args.type_at(0).kind()
177 {
178 let fn_sig = sig.with(*header);
179
180 let fn_abi = cx.fn_abi_of_fn_ptr(fn_sig, ty::List::empty());
181 cx.declare_fn(sym, &fn_abi, None)
182 } else {
183 cx.declare_global(sym, cx.type_i8())
184 }
185 } else {
186 cx.declare_global(sym, cx.type_i8())
187 };
188 llvm::set_linkage(g1, base::linkage_to_llvm(linkage));
189
190 let real_name =
197 format!("_rust_extern_with_linkage_{:016x}_{sym}", cx.tcx.stable_crate_id(LOCAL_CRATE));
198 let g2 = cx.define_global(&real_name, llty).unwrap_or_else(|| {
199 cx.sess().dcx().emit_fatal(SymbolAlreadyDefined {
200 span: cx.tcx.def_span(def_id),
201 symbol_name: sym,
202 })
203 });
204 llvm::set_linkage(g2, llvm::Linkage::InternalLinkage);
205 llvm::set_initializer(g2, g1);
206 g2
207 } else if cx.tcx.sess.target.arch == Arch::X86
208 && common::is_mingw_gnu_toolchain(&cx.tcx.sess.target)
209 && let Some(dllimport) = crate::common::get_dllimport(cx.tcx, def_id, sym)
210 {
211 cx.declare_global(&common::i686_decorated_name(dllimport, true, true, false), llty)
212 } else {
213 cx.declare_global(sym, llty)
216 }
217}
218
219impl<'ll> CodegenCx<'ll, '_> {
220 pub(crate) fn const_bitcast(&self, val: &'ll Value, ty: &'ll Type) -> &'ll Value {
221 unsafe { llvm::LLVMConstBitCast(val, ty) }
222 }
223
224 pub(crate) fn const_pointercast(&self, val: &'ll Value, ty: &'ll Type) -> &'ll Value {
225 unsafe { llvm::LLVMConstPointerCast(val, ty) }
226 }
227
228 pub(crate) fn static_addr_of_mut(
233 &self,
234 cv: &'ll Value,
235 align: Align,
236 kind: Option<&str>,
237 ) -> &'ll Value {
238 let gv = match kind {
239 Some(kind) if !self.tcx.sess.fewer_names() => {
240 let name = self.generate_local_symbol_name(kind);
241 let gv = self.define_global(&name, self.val_ty(cv)).unwrap_or_else(|| {
242 bug!("symbol `{}` is already defined", name);
243 });
244 gv
245 }
246 _ => self.define_global("", self.val_ty(cv)).unwrap_or_else(|| {
247 bug!("anonymous global symbol is already defined");
248 }),
249 };
250 llvm::set_linkage(gv, llvm::Linkage::PrivateLinkage);
251 llvm::set_initializer(gv, cv);
252 set_global_alignment(self, gv, align);
253 llvm::set_unnamed_address(gv, llvm::UnnamedAddr::Global);
254 gv
255 }
256
257 pub(crate) fn static_addr_of_impl(
261 &self,
262 cv: &'ll Value,
263 align: Align,
264 kind: Option<&str>,
265 ) -> &'ll Value {
266 if let Some(&gv) = self.const_globals.borrow().get(&cv) {
267 unsafe {
268 let llalign = align.bytes() as u32;
271 if llalign > llvm::LLVMGetAlignment(gv) {
272 llvm::LLVMSetAlignment(gv, llalign);
273 }
274 }
275 return gv;
276 }
277 let gv = self.static_addr_of_mut(cv, align, kind);
278 llvm::set_global_constant(gv, true);
279
280 self.const_globals.borrow_mut().insert(cv, gv);
281 gv
282 }
283
284 #[instrument(level = "debug", skip(self))]
285 pub(crate) fn get_static(&self, def_id: DefId) -> &'ll Value {
286 let instance = Instance::mono(self.tcx, def_id);
287 trace!(?instance);
288
289 let DefKind::Static { nested, .. } = self.tcx.def_kind(def_id) else { bug!() };
290 let llty = if nested {
293 self.type_i8()
294 } else {
295 let ty = instance.ty(self.tcx, self.typing_env());
296 trace!(?ty);
297 self.layout_of(ty).llvm_type(self)
298 };
299 self.get_static_inner(def_id, llty)
300 }
301
302 #[instrument(level = "debug", skip(self, llty))]
303 fn get_static_inner(&self, def_id: DefId, llty: &'ll Type) -> &'ll Value {
304 let instance = Instance::mono(self.tcx, def_id);
305 if let Some(&g) = self.instances.borrow().get(&instance) {
306 trace!("used cached value");
307 return g;
308 }
309
310 let defined_in_current_codegen_unit =
311 self.codegen_unit.items().contains_key(&MonoItem::Static(def_id));
312 assert!(
313 !defined_in_current_codegen_unit,
314 "consts::get_static() should always hit the cache for \
315 statics defined in the same CGU, but did not for `{def_id:?}`"
316 );
317
318 let sym = self.tcx.symbol_name(instance).name;
319 let fn_attrs = self.tcx.codegen_fn_attrs(def_id);
320
321 debug!(?sym, ?fn_attrs);
322
323 let g = if def_id.is_local() && !self.tcx.is_foreign_item(def_id) {
324 if let Some(g) = self.get_declared_value(sym) {
325 if self.val_ty(g) != self.type_ptr() {
326 span_bug!(self.tcx.def_span(def_id), "Conflicting types for static");
327 }
328 }
329
330 let g = self.declare_global(sym, llty);
331
332 if !self.tcx.is_reachable_non_generic(def_id) {
333 llvm::set_visibility(g, llvm::Visibility::Hidden);
334 }
335
336 g
337 } else if let Some(classname) = fn_attrs.objc_class {
338 self.get_objc_classref(classname)
339 } else if let Some(methname) = fn_attrs.objc_selector {
340 self.get_objc_selref(methname)
341 } else {
342 check_and_apply_linkage(self, fn_attrs, llty, sym, def_id)
343 };
344
345 if fn_attrs.flags.contains(CodegenFnAttrFlags::THREAD_LOCAL) {
352 llvm::set_thread_local_mode(g, self.tls_model);
353 }
354
355 let dso_local = self.assume_dso_local(g, true);
356
357 if !def_id.is_local() {
358 let needs_dll_storage_attr = self.use_dll_storage_attrs
359 && !self.tcx.is_foreign_item(def_id)
360 && !dso_local
363 && !self.tcx.sess.opts.cg.linker_plugin_lto.enabled();
368
369 assert!(
372 !(self.tcx.sess.opts.cg.linker_plugin_lto.enabled()
373 && self.tcx.sess.target.is_like_windows
374 && self.tcx.sess.opts.cg.prefer_dynamic)
375 );
376
377 if needs_dll_storage_attr {
378 if !self.tcx.is_codegened_item(def_id) {
389 llvm::set_dllimport_storage_class(g);
390 }
391 }
392 }
393
394 if self.use_dll_storage_attrs
395 && let Some(library) = self.tcx.native_library(def_id)
396 && library.kind.is_dllimport()
397 {
398 llvm::set_dllimport_storage_class(g);
400 }
401
402 self.instances.borrow_mut().insert(instance, g);
403 g
404 }
405
406 fn codegen_static_item(&mut self, def_id: DefId) {
407 assert!(
408 llvm::LLVMGetInitializer(
409 self.instances.borrow().get(&Instance::mono(self.tcx, def_id)).unwrap()
410 )
411 .is_none()
412 );
413 let attrs = self.tcx.codegen_fn_attrs(def_id);
414
415 let Ok((v, alloc)) = codegen_static_initializer(self, def_id) else {
416 return;
418 };
419 let alloc = alloc.inner();
420
421 let val_llty = self.val_ty(v);
422
423 let g = self.get_static_inner(def_id, val_llty);
424 let llty = self.get_type_of_global(g);
425
426 let g = if val_llty == llty {
427 g
428 } else {
429 let name = String::from_utf8(llvm::get_value_name(g))
438 .expect("we declare our statics with a utf8-valid name");
439 llvm::set_value_name(g, b"");
440
441 let linkage = llvm::get_linkage(g);
442 let visibility = llvm::get_visibility(g);
443
444 let new_g = self.declare_global(&name, val_llty);
445
446 llvm::set_linkage(new_g, linkage);
447 llvm::set_visibility(new_g, visibility);
448
449 self.renamed_statics.borrow_mut().insert(def_id, new_g);
455
456 self.statics_to_rauw.borrow_mut().push((g, new_g));
460 new_g
461 };
462
463 set_global_alignment(self, g, alloc.align);
465 llvm::set_initializer(g, v);
466
467 self.assume_dso_local(g, true);
468
469 if alloc.mutability.is_not() {
471 llvm::set_global_constant(g, true);
472 }
473
474 debuginfo::build_global_var_di_node(self, def_id, g);
475
476 if attrs.flags.contains(CodegenFnAttrFlags::THREAD_LOCAL) {
477 llvm::set_thread_local_mode(g, self.tls_model);
478 }
479
480 if self.tcx.sess.target.is_like_wasm
484 && attrs
485 .link_section
486 .map(|link_section| !link_section.as_str().starts_with(".init_array"))
487 .unwrap_or(true)
488 {
489 if let Some(section) = attrs.link_section {
490 let section = self.create_metadata(section.as_str().as_bytes());
491 assert!(alloc.provenance().ptrs().is_empty());
492
493 let bytes = alloc.inspect_with_uninit_and_ptr_outside_interpreter(0..alloc.len());
497 let alloc = self.create_metadata(bytes);
498 let data = [section, alloc];
499 self.module_add_named_metadata_node(self.llmod(), c"wasm.custom_sections", &data);
500 }
501 } else {
502 base::set_link_section(g, attrs);
503 }
504
505 base::set_variable_sanitizer_attrs(g, attrs);
506
507 if attrs.flags.contains(CodegenFnAttrFlags::USED_COMPILER) {
508 assert!(!attrs.flags.contains(CodegenFnAttrFlags::USED_LINKER));
510
511 self.add_compiler_used_global(g);
528 }
529 if attrs.flags.contains(CodegenFnAttrFlags::USED_LINKER) {
530 assert!(!attrs.flags.contains(CodegenFnAttrFlags::USED_COMPILER));
532
533 self.add_used_global(g);
534 }
535 }
536
537 pub(crate) fn add_used_global(&mut self, global: &'ll Value) {
539 self.used_statics.push(global);
540 }
541
542 pub(crate) fn add_compiler_used_global(&self, global: &'ll Value) {
545 self.compiler_used_statics.borrow_mut().push(global);
546 }
547
548 fn define_objc_classname(&self, classname: &str) -> &'ll Value {
552 assert_eq!(self.objc_abi_version(), 1);
553
554 let llval = self.null_terminate_const_bytes(classname.as_bytes());
555 let llty = self.val_ty(llval);
556 let sym = self.generate_local_symbol_name("OBJC_CLASS_NAME_");
557 let g = self.define_global(&sym, llty).unwrap_or_else(|| {
558 bug!("symbol `{}` is already defined", sym);
559 });
560 set_global_alignment(self, g, self.tcx.data_layout.i8_align);
561 llvm::set_initializer(g, llval);
562 llvm::set_linkage(g, llvm::Linkage::PrivateLinkage);
563 llvm::set_section(g, c"__TEXT,__cstring,cstring_literals");
564 llvm::LLVMSetGlobalConstant(g, llvm::TRUE);
565 llvm::LLVMSetUnnamedAddress(g, llvm::UnnamedAddr::Global);
566 self.add_compiler_used_global(g);
567
568 g
569 }
570
571 fn get_objc_class_t(&self) -> &'ll Type {
575 if let Some(class_t) = self.objc_class_t.get() {
576 return class_t;
577 }
578
579 assert_eq!(self.objc_abi_version(), 2);
580
581 let class_t = self.type_named_struct("struct._class_t");
590 let els = [self.type_ptr(); 5];
591 let packed = false;
592 self.set_struct_body(class_t, &els, packed);
593
594 self.objc_class_t.set(Some(class_t));
595 class_t
596 }
597
598 fn get_objc_classref(&self, classname: Symbol) -> &'ll Value {
603 let mut classrefs = self.objc_classrefs.borrow_mut();
604 if let Some(classref) = classrefs.get(&classname).copied() {
605 return classref;
606 }
607
608 let g = match self.objc_abi_version() {
609 1 => {
610 let llval = self.define_objc_classname(classname.as_str());
613 let llty = self.type_ptr();
614 let sym = self.generate_local_symbol_name("OBJC_CLASS_REFERENCES_");
615 let g = self.define_global(&sym, llty).unwrap_or_else(|| {
616 bug!("symbol `{}` is already defined", sym);
617 });
618 set_global_alignment(self, g, self.tcx.data_layout.pointer_align().abi);
619 llvm::set_initializer(g, llval);
620 llvm::set_linkage(g, llvm::Linkage::PrivateLinkage);
621 llvm::set_section(g, c"__OBJC,__cls_refs,literal_pointers,no_dead_strip");
622 self.add_compiler_used_global(g);
623 g
624 }
625 2 => {
626 let llval = {
629 let extern_sym = format!("OBJC_CLASS_$_{}", classname.as_str());
630 let extern_llty = self.get_objc_class_t();
631 self.declare_global(&extern_sym, extern_llty)
632 };
633 let llty = self.type_ptr();
634 let sym = self.generate_local_symbol_name("OBJC_CLASSLIST_REFERENCES_$_");
635 let g = self.define_global(&sym, llty).unwrap_or_else(|| {
636 bug!("symbol `{}` is already defined", sym);
637 });
638 set_global_alignment(self, g, self.tcx.data_layout.pointer_align().abi);
639 llvm::set_initializer(g, llval);
640 llvm::set_linkage(g, llvm::Linkage::InternalLinkage);
641 llvm::set_section(g, c"__DATA,__objc_classrefs,regular,no_dead_strip");
642 self.add_compiler_used_global(g);
643 g
644 }
645 _ => unreachable!(),
646 };
647
648 classrefs.insert(classname, g);
649 g
650 }
651
652 fn get_objc_selref(&self, methname: Symbol) -> &'ll Value {
661 let mut selrefs = self.objc_selrefs.borrow_mut();
662 if let Some(selref) = selrefs.get(&methname).copied() {
663 return selref;
664 }
665
666 let abi_version = self.objc_abi_version();
667
668 let methname_llval = self.null_terminate_const_bytes(methname.as_str().as_bytes());
671 let methname_llty = self.val_ty(methname_llval);
672 let methname_sym = self.generate_local_symbol_name("OBJC_METH_VAR_NAME_");
673 let methname_g = self.define_global(&methname_sym, methname_llty).unwrap_or_else(|| {
674 bug!("symbol `{}` is already defined", methname_sym);
675 });
676 set_global_alignment(self, methname_g, self.tcx.data_layout.i8_align);
677 llvm::set_initializer(methname_g, methname_llval);
678 llvm::set_linkage(methname_g, llvm::Linkage::PrivateLinkage);
679 llvm::set_section(
680 methname_g,
681 match abi_version {
682 1 => c"__TEXT,__cstring,cstring_literals",
683 2 => c"__TEXT,__objc_methname,cstring_literals",
684 _ => unreachable!(),
685 },
686 );
687 llvm::LLVMSetGlobalConstant(methname_g, llvm::TRUE);
688 llvm::LLVMSetUnnamedAddress(methname_g, llvm::UnnamedAddr::Global);
689 self.add_compiler_used_global(methname_g);
690
691 let selref_llval = methname_g;
696 let selref_llty = self.type_ptr();
697 let selref_sym = self.generate_local_symbol_name("OBJC_SELECTOR_REFERENCES_");
698 let selref_g = self.define_global(&selref_sym, selref_llty).unwrap_or_else(|| {
699 bug!("symbol `{}` is already defined", selref_sym);
700 });
701 set_global_alignment(self, selref_g, self.tcx.data_layout.pointer_align().abi);
702 llvm::set_initializer(selref_g, selref_llval);
703 llvm::set_externally_initialized(selref_g, true);
704 llvm::set_linkage(
705 selref_g,
706 match abi_version {
707 1 => llvm::Linkage::PrivateLinkage,
708 2 => llvm::Linkage::InternalLinkage,
709 _ => unreachable!(),
710 },
711 );
712 llvm::set_section(
713 selref_g,
714 match abi_version {
715 1 => c"__OBJC,__message_refs,literal_pointers,no_dead_strip",
716 2 => c"__DATA,__objc_selrefs,literal_pointers,no_dead_strip",
717 _ => unreachable!(),
718 },
719 );
720 self.add_compiler_used_global(selref_g);
721
722 selrefs.insert(methname, selref_g);
723 selref_g
724 }
725
726 pub(crate) fn define_objc_module_info(&mut self) {
732 assert_eq!(self.objc_abi_version(), 1);
733
734 let llty = self.type_named_struct("struct._objc_module");
742 let i32_llty = self.type_i32();
743 let ptr_llty = self.type_ptr();
744 let packed = false;
745 self.set_struct_body(llty, &[i32_llty, i32_llty, ptr_llty, ptr_llty], packed);
746
747 let version = self.const_uint(i32_llty, 7);
748 let size = self.const_uint(i32_llty, 16);
749 let name = self.define_objc_classname("");
750 let symtab = self.const_null(ptr_llty);
751 let llval = crate::common::named_struct(llty, &[version, size, name, symtab]);
752
753 let sym = "OBJC_MODULES";
754 let g = self.define_global(&sym, llty).unwrap_or_else(|| {
755 bug!("symbol `{}` is already defined", sym);
756 });
757 set_global_alignment(self, g, self.tcx.data_layout.pointer_align().abi);
758 llvm::set_initializer(g, llval);
759 llvm::set_linkage(g, llvm::Linkage::PrivateLinkage);
760 llvm::set_section(g, c"__OBJC,__module_info,regular,no_dead_strip");
761
762 self.add_compiler_used_global(g);
763 }
764}
765
766impl<'ll> StaticCodegenMethods for CodegenCx<'ll, '_> {
767 fn static_addr_of(&self, cv: &'ll Value, align: Align, kind: Option<&str>) -> &'ll Value {
772 let gv = self.static_addr_of_impl(cv, align, kind);
773 self.const_pointercast(gv, self.type_ptr())
776 }
777
778 fn codegen_static(&mut self, def_id: DefId) {
779 self.codegen_static_item(def_id)
780 }
781}