rustc_codegen_ssa/mir/
constant.rs1use rustc_abi::BackendRepr;
2use rustc_middle::mir::interpret::ErrorHandled;
3use rustc_middle::ty::layout::{HasTyCtxt, HasTypingEnv};
4use rustc_middle::ty::{self, Ty};
5use rustc_middle::{bug, mir, span_bug};
6
7use super::FunctionCx;
8use crate::errors;
9use crate::mir::operand::OperandRef;
10use crate::traits::*;
11
12impl<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> FunctionCx<'a, 'tcx, Bx> {
13 pub(crate) fn eval_mir_constant_to_operand(
14 &self,
15 bx: &mut Bx,
16 constant: &mir::ConstOperand<'tcx>,
17 ) -> OperandRef<'tcx, Bx::Value> {
18 let val = self.eval_mir_constant(constant);
19 let ty = self.monomorphize(constant.ty());
20 OperandRef::from_const(bx, val, ty)
21 }
22
23 pub fn eval_mir_constant(&self, constant: &mir::ConstOperand<'tcx>) -> mir::ConstValue {
24 self.monomorphize(constant.const_)
27 .eval(self.cx.tcx(), self.cx.typing_env(), constant.span)
28 .expect("erroneous constant missed by mono item collection")
29 }
30
31 fn eval_unevaluated_mir_constant_to_valtree(
38 &self,
39 constant: &mir::ConstOperand<'tcx>,
40 ) -> Result<Result<ty::ValTree<'tcx>, Ty<'tcx>>, ErrorHandled> {
41 let uv = match self.monomorphize(constant.const_) {
42 mir::Const::Unevaluated(uv, _) => uv.shrink(),
43 mir::Const::Ty(_, c) => match c.kind() {
44 ty::ConstKind::Value(cv) => return Ok(Ok(cv.valtree)),
47 other => span_bug!(constant.span, "{other:#?}"),
48 },
49 other => span_bug!(constant.span, "{other:#?}"),
58 };
59 let uv = self.monomorphize(uv);
60 self.cx.tcx().const_eval_resolve_for_typeck(self.cx.typing_env(), uv, constant.span)
61 }
62
63 pub fn immediate_const_vector(
65 &mut self,
66 bx: &Bx,
67 constant: &mir::ConstOperand<'tcx>,
68 ) -> (Bx::Value, Ty<'tcx>) {
69 let ty = self.monomorphize(constant.ty());
70 assert!(ty.is_simd());
71 let field_ty = ty.simd_size_and_type(bx.tcx()).1;
72
73 let val = self
74 .eval_unevaluated_mir_constant_to_valtree(constant)
75 .ok()
76 .map(|x| x.ok())
77 .flatten()
78 .map(|val| {
79 let fields = val.to_branch();
81 assert_eq!(fields.len(), 1);
82 let array = fields[0].to_branch();
83 let values: Vec<_> = array
85 .iter()
86 .map(|field| {
87 let Some(prim) = field.try_to_scalar() else {
88 bug!("field is not a scalar {:?}", field)
89 };
90 let layout = bx.layout_of(field_ty);
91 let BackendRepr::Scalar(scalar) = layout.backend_repr else {
92 bug!("from_const: invalid ByVal layout: {:#?}", layout);
93 };
94 bx.scalar_to_backend(prim, scalar, bx.immediate_backend_type(layout))
95 })
96 .collect();
97 bx.const_vector(&values)
98 })
99 .unwrap_or_else(|| {
100 bx.tcx().dcx().emit_err(errors::ShuffleIndicesEvaluation { span: constant.span });
101 let llty = bx.backend_type(bx.layout_of(ty));
103 bx.const_undef(llty)
104 });
105 (val, ty)
106 }
107}