mirror of
https://github.com/rust-lang/rust.git
synced 2025-01-27 07:03:45 +00:00
Auto merge of #44252 - eddyb:what-is-dead-may-never-die, r=nikomatsakis
Better StorageLive / StorageDead placement for constants. Fixes problems in miri (see https://github.com/solson/miri/pull/324#issuecomment-326555552) caused by the new scope rules in #43932. What I've tried to do here is always have a `StorageLive` but no `StorageDead` for `'static` slots. It might not work perfectly in all cases, but it should unblock miri. r? @nikomatsakis cc @oli-obk
This commit is contained in:
commit
23ade23cbc
@ -31,6 +31,7 @@ use ty::fast_reject::SimplifiedType;
|
|||||||
use util::nodemap::{DefIdSet, NodeSet};
|
use util::nodemap::{DefIdSet, NodeSet};
|
||||||
use util::common::{profq_msg, ProfileQueriesMsg};
|
use util::common::{profq_msg, ProfileQueriesMsg};
|
||||||
|
|
||||||
|
use rustc_data_structures::indexed_set::IdxSetBuf;
|
||||||
use rustc_data_structures::indexed_vec::IndexVec;
|
use rustc_data_structures::indexed_vec::IndexVec;
|
||||||
use rustc_data_structures::fx::FxHashMap;
|
use rustc_data_structures::fx::FxHashMap;
|
||||||
use std::cell::{RefCell, RefMut, Cell};
|
use std::cell::{RefCell, RefMut, Cell};
|
||||||
@ -1019,7 +1020,7 @@ define_maps! { <'tcx>
|
|||||||
/// Maps DefId's that have an associated Mir to the result
|
/// Maps DefId's that have an associated Mir to the result
|
||||||
/// of the MIR qualify_consts pass. The actual meaning of
|
/// of the MIR qualify_consts pass. The actual meaning of
|
||||||
/// the value isn't known except to the pass itself.
|
/// the value isn't known except to the pass itself.
|
||||||
[] fn mir_const_qualif: MirConstQualif(DefId) -> u8,
|
[] fn mir_const_qualif: MirConstQualif(DefId) -> (u8, Rc<IdxSetBuf<mir::Local>>),
|
||||||
|
|
||||||
/// Fetch the MIR for a given def-id up till the point where it is
|
/// Fetch the MIR for a given def-id up till the point where it is
|
||||||
/// ready for const evaluation.
|
/// ready for const evaluation.
|
||||||
|
@ -38,6 +38,7 @@ use syntax::ext::base::SyntaxExtension;
|
|||||||
use syntax::parse::filemap_to_stream;
|
use syntax::parse::filemap_to_stream;
|
||||||
use syntax::symbol::Symbol;
|
use syntax::symbol::Symbol;
|
||||||
use syntax_pos::{Span, NO_EXPANSION};
|
use syntax_pos::{Span, NO_EXPANSION};
|
||||||
|
use rustc_data_structures::indexed_set::IdxSetBuf;
|
||||||
use rustc::hir::svh::Svh;
|
use rustc::hir::svh::Svh;
|
||||||
use rustc::hir;
|
use rustc::hir;
|
||||||
|
|
||||||
@ -106,7 +107,9 @@ provide! { <'tcx> tcx, def_id, cdata,
|
|||||||
mir
|
mir
|
||||||
}
|
}
|
||||||
generator_sig => { cdata.generator_sig(def_id.index, tcx) }
|
generator_sig => { cdata.generator_sig(def_id.index, tcx) }
|
||||||
mir_const_qualif => { cdata.mir_const_qualif(def_id.index) }
|
mir_const_qualif => {
|
||||||
|
(cdata.mir_const_qualif(def_id.index), Rc::new(IdxSetBuf::new_empty(0)))
|
||||||
|
}
|
||||||
typeck_tables_of => { cdata.item_body_tables(def_id.index, tcx) }
|
typeck_tables_of => { cdata.item_body_tables(def_id.index, tcx) }
|
||||||
closure_kind => { cdata.closure_kind(def_id.index) }
|
closure_kind => { cdata.closure_kind(def_id.index) }
|
||||||
fn_sig => { cdata.fn_sig(def_id.index, tcx) }
|
fn_sig => { cdata.fn_sig(def_id.index, tcx) }
|
||||||
|
@ -793,7 +793,7 @@ impl<'a, 'b: 'a, 'tcx: 'b> IsolatedEncoder<'a, 'b, 'tcx> {
|
|||||||
let kind = match impl_item.kind {
|
let kind = match impl_item.kind {
|
||||||
ty::AssociatedKind::Const => {
|
ty::AssociatedKind::Const => {
|
||||||
EntryKind::AssociatedConst(container,
|
EntryKind::AssociatedConst(container,
|
||||||
self.tcx.at(ast_item.span).mir_const_qualif(def_id))
|
self.tcx.at(ast_item.span).mir_const_qualif(def_id).0)
|
||||||
}
|
}
|
||||||
ty::AssociatedKind::Method => {
|
ty::AssociatedKind::Method => {
|
||||||
let fn_data = if let hir::ImplItemKind::Method(ref sig, body) = ast_item.node {
|
let fn_data = if let hir::ImplItemKind::Method(ref sig, body) = ast_item.node {
|
||||||
@ -912,7 +912,7 @@ impl<'a, 'b: 'a, 'tcx: 'b> IsolatedEncoder<'a, 'b, 'tcx> {
|
|||||||
hir::ItemStatic(_, hir::MutMutable, _) => EntryKind::MutStatic,
|
hir::ItemStatic(_, hir::MutMutable, _) => EntryKind::MutStatic,
|
||||||
hir::ItemStatic(_, hir::MutImmutable, _) => EntryKind::ImmStatic,
|
hir::ItemStatic(_, hir::MutImmutable, _) => EntryKind::ImmStatic,
|
||||||
hir::ItemConst(..) => {
|
hir::ItemConst(..) => {
|
||||||
EntryKind::Const(tcx.at(item.span).mir_const_qualif(def_id))
|
EntryKind::Const(tcx.at(item.span).mir_const_qualif(def_id).0)
|
||||||
}
|
}
|
||||||
hir::ItemFn(_, _, constness, .., body) => {
|
hir::ItemFn(_, _, constness, .., body) => {
|
||||||
let data = FnData {
|
let data = FnData {
|
||||||
@ -1256,7 +1256,7 @@ impl<'a, 'b: 'a, 'tcx: 'b> IsolatedEncoder<'a, 'b, 'tcx> {
|
|||||||
let body = tcx.hir.body_owned_by(id);
|
let body = tcx.hir.body_owned_by(id);
|
||||||
|
|
||||||
Entry {
|
Entry {
|
||||||
kind: EntryKind::Const(tcx.mir_const_qualif(def_id)),
|
kind: EntryKind::Const(tcx.mir_const_qualif(def_id).0),
|
||||||
visibility: self.lazy(&ty::Visibility::Public),
|
visibility: self.lazy(&ty::Visibility::Public),
|
||||||
span: self.lazy(&tcx.def_span(def_id)),
|
span: self.lazy(&tcx.def_span(def_id)),
|
||||||
attributes: LazySeq::empty(),
|
attributes: LazySeq::empty(),
|
||||||
|
@ -97,12 +97,12 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
|||||||
ExprKind::Box { value } => {
|
ExprKind::Box { value } => {
|
||||||
let value = this.hir.mirror(value);
|
let value = this.hir.mirror(value);
|
||||||
let result = this.temp(expr.ty, expr_span);
|
let result = this.temp(expr.ty, expr_span);
|
||||||
if let Some(scope) = scope {
|
|
||||||
// schedule a shallow free of that memory, lest we unwind:
|
|
||||||
this.cfg.push(block, Statement {
|
this.cfg.push(block, Statement {
|
||||||
source_info,
|
source_info,
|
||||||
kind: StatementKind::StorageLive(result.clone())
|
kind: StatementKind::StorageLive(result.clone())
|
||||||
});
|
});
|
||||||
|
if let Some(scope) = scope {
|
||||||
|
// schedule a shallow free of that memory, lest we unwind:
|
||||||
this.schedule_drop(expr_span, scope, &result, value.ty);
|
this.schedule_drop(expr_span, scope, &result, value.ty);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -50,7 +50,7 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
|||||||
let expr_ty = expr.ty.clone();
|
let expr_ty = expr.ty.clone();
|
||||||
let temp = this.temp(expr_ty.clone(), expr_span);
|
let temp = this.temp(expr_ty.clone(), expr_span);
|
||||||
|
|
||||||
if !expr_ty.is_never() && temp_lifetime.is_some() {
|
if !expr_ty.is_never() {
|
||||||
this.cfg.push(block, Statement {
|
this.cfg.push(block, Statement {
|
||||||
source_info,
|
source_info,
|
||||||
kind: StatementKind::StorageLive(temp.clone())
|
kind: StatementKind::StorageLive(temp.clone())
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
//! diagnostics as to why a constant rvalue wasn't promoted.
|
//! diagnostics as to why a constant rvalue wasn't promoted.
|
||||||
|
|
||||||
use rustc_data_structures::bitvec::BitVector;
|
use rustc_data_structures::bitvec::BitVector;
|
||||||
|
use rustc_data_structures::indexed_set::IdxSetBuf;
|
||||||
use rustc_data_structures::indexed_vec::{IndexVec, Idx};
|
use rustc_data_structures::indexed_vec::{IndexVec, Idx};
|
||||||
use rustc::hir;
|
use rustc::hir;
|
||||||
use rustc::hir::map as hir_map;
|
use rustc::hir::map as hir_map;
|
||||||
@ -33,6 +34,7 @@ use syntax::feature_gate::UnstableFeatures;
|
|||||||
use syntax_pos::{Span, DUMMY_SP};
|
use syntax_pos::{Span, DUMMY_SP};
|
||||||
|
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
|
use std::rc::Rc;
|
||||||
use std::usize;
|
use std::usize;
|
||||||
|
|
||||||
use super::promote_consts::{self, Candidate, TempState};
|
use super::promote_consts::{self, Candidate, TempState};
|
||||||
@ -127,7 +129,6 @@ struct Qualifier<'a, 'gcx: 'a+'tcx, 'tcx: 'a> {
|
|||||||
|
|
||||||
impl<'a, 'tcx> Qualifier<'a, 'tcx, 'tcx> {
|
impl<'a, 'tcx> Qualifier<'a, 'tcx, 'tcx> {
|
||||||
fn new(tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
fn new(tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
||||||
param_env: ty::ParamEnv<'tcx>,
|
|
||||||
def_id: DefId,
|
def_id: DefId,
|
||||||
mir: &'a Mir<'tcx>,
|
mir: &'a Mir<'tcx>,
|
||||||
mode: Mode)
|
mode: Mode)
|
||||||
@ -142,7 +143,7 @@ impl<'a, 'tcx> Qualifier<'a, 'tcx, 'tcx> {
|
|||||||
mir,
|
mir,
|
||||||
rpo,
|
rpo,
|
||||||
tcx,
|
tcx,
|
||||||
param_env,
|
param_env: tcx.param_env(def_id),
|
||||||
temp_qualif: IndexVec::from_elem(None, &mir.local_decls),
|
temp_qualif: IndexVec::from_elem(None, &mir.local_decls),
|
||||||
return_qualif: None,
|
return_qualif: None,
|
||||||
qualif: Qualif::empty(),
|
qualif: Qualif::empty(),
|
||||||
@ -368,7 +369,7 @@ impl<'a, 'tcx> Qualifier<'a, 'tcx, 'tcx> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Qualify a whole const, static initializer or const fn.
|
/// Qualify a whole const, static initializer or const fn.
|
||||||
fn qualify_const(&mut self) -> Qualif {
|
fn qualify_const(&mut self) -> (Qualif, Rc<IdxSetBuf<Local>>) {
|
||||||
debug!("qualifying {} {:?}", self.mode, self.def_id);
|
debug!("qualifying {} {:?}", self.mode, self.def_id);
|
||||||
|
|
||||||
let mir = self.mir;
|
let mir = self.mir;
|
||||||
@ -390,7 +391,7 @@ impl<'a, 'tcx> Qualifier<'a, 'tcx, 'tcx> {
|
|||||||
|
|
||||||
// Non-terminating calls cannot produce any value.
|
// Non-terminating calls cannot produce any value.
|
||||||
TerminatorKind::Call { destination: None, .. } => {
|
TerminatorKind::Call { destination: None, .. } => {
|
||||||
return Qualif::empty();
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
TerminatorKind::SwitchInt {..} |
|
TerminatorKind::SwitchInt {..} |
|
||||||
@ -472,7 +473,25 @@ impl<'a, 'tcx> Qualifier<'a, 'tcx, 'tcx> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
self.qualif
|
|
||||||
|
// Collect all the temps we need to promote.
|
||||||
|
let mut promoted_temps = IdxSetBuf::new_empty(self.temp_promotion_state.len());
|
||||||
|
|
||||||
|
for candidate in &self.promotion_candidates {
|
||||||
|
match *candidate {
|
||||||
|
Candidate::Ref(Location { block: bb, statement_index: stmt_idx }) => {
|
||||||
|
match self.mir[bb].statements[stmt_idx].kind {
|
||||||
|
StatementKind::Assign(_, Rvalue::Ref(_, _, Lvalue::Local(index))) => {
|
||||||
|
promoted_temps.add(&index);
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Candidate::ShuffleIndices(_) => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
(self.qualif, Rc::new(promoted_temps))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -516,6 +535,7 @@ impl<'a, 'tcx> Visitor<'tcx> for Qualifier<'a, 'tcx, 'tcx> {
|
|||||||
span_err!(self.tcx.sess, self.span, E0625,
|
span_err!(self.tcx.sess, self.span, E0625,
|
||||||
"thread-local statics cannot be \
|
"thread-local statics cannot be \
|
||||||
accessed at compile-time");
|
accessed at compile-time");
|
||||||
|
self.add(Qualif::NOT_CONST);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -598,7 +618,7 @@ impl<'a, 'tcx> Visitor<'tcx> for Qualifier<'a, 'tcx, 'tcx> {
|
|||||||
if self.tcx.trait_of_item(def_id).is_some() {
|
if self.tcx.trait_of_item(def_id).is_some() {
|
||||||
self.add_type(constant.ty);
|
self.add_type(constant.ty);
|
||||||
} else {
|
} else {
|
||||||
let bits = self.tcx.at(constant.span).mir_const_qualif(def_id);
|
let (bits, _) = self.tcx.at(constant.span).mir_const_qualif(def_id);
|
||||||
|
|
||||||
let qualif = Qualif::from_bits(bits).expect("invalid mir_const_qualif");
|
let qualif = Qualif::from_bits(bits).expect("invalid mir_const_qualif");
|
||||||
self.add(qualif);
|
self.add(qualif);
|
||||||
@ -702,7 +722,6 @@ impl<'a, 'tcx> Visitor<'tcx> for Qualifier<'a, 'tcx, 'tcx> {
|
|||||||
|
|
||||||
// We might have a candidate for promotion.
|
// We might have a candidate for promotion.
|
||||||
let candidate = Candidate::Ref(location);
|
let candidate = Candidate::Ref(location);
|
||||||
if self.mode == Mode::Fn || self.mode == Mode::ConstFn {
|
|
||||||
if !self.qualif.intersects(Qualif::NEVER_PROMOTE) {
|
if !self.qualif.intersects(Qualif::NEVER_PROMOTE) {
|
||||||
// We can only promote direct borrows of temps.
|
// We can only promote direct borrows of temps.
|
||||||
if let Lvalue::Local(local) = *lvalue {
|
if let Lvalue::Local(local) = *lvalue {
|
||||||
@ -712,7 +731,6 @@ impl<'a, 'tcx> Visitor<'tcx> for Qualifier<'a, 'tcx, 'tcx> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
Rvalue::Cast(CastKind::Misc, ref operand, cast_ty) => {
|
Rvalue::Cast(CastKind::Misc, ref operand, cast_ty) => {
|
||||||
let operand_ty = operand.ty(self.mir, self.tcx);
|
let operand_ty = operand.ty(self.mir, self.tcx);
|
||||||
@ -999,7 +1017,7 @@ pub fn provide(providers: &mut Providers) {
|
|||||||
|
|
||||||
fn mir_const_qualif<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
fn mir_const_qualif<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
||||||
def_id: DefId)
|
def_id: DefId)
|
||||||
-> u8 {
|
-> (u8, Rc<IdxSetBuf<Local>>) {
|
||||||
// NB: This `borrow()` is guaranteed to be valid (i.e., the value
|
// NB: This `borrow()` is guaranteed to be valid (i.e., the value
|
||||||
// cannot yet be stolen), because `mir_validated()`, which steals
|
// cannot yet be stolen), because `mir_validated()`, which steals
|
||||||
// from `mir_const(), forces this query to execute before
|
// from `mir_const(), forces this query to execute before
|
||||||
@ -1007,13 +1025,13 @@ fn mir_const_qualif<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
|||||||
let mir = &tcx.mir_const(def_id).borrow();
|
let mir = &tcx.mir_const(def_id).borrow();
|
||||||
|
|
||||||
if mir.return_ty.references_error() {
|
if mir.return_ty.references_error() {
|
||||||
return Qualif::NOT_CONST.bits();
|
tcx.sess.delay_span_bug(mir.span, "mir_const_qualif: Mir had errors");
|
||||||
|
return (Qualif::NOT_CONST.bits(), Rc::new(IdxSetBuf::new_empty(0)));
|
||||||
}
|
}
|
||||||
|
|
||||||
let param_env = tcx.param_env(def_id);
|
let mut qualifier = Qualifier::new(tcx, def_id, mir, Mode::Const);
|
||||||
|
let (qualif, promoted_temps) = qualifier.qualify_const();
|
||||||
let mut qualifier = Qualifier::new(tcx, param_env, def_id, mir, Mode::Const);
|
(qualif.bits(), promoted_temps)
|
||||||
qualifier.qualify_const().bits()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct QualifyAndPromoteConstants;
|
pub struct QualifyAndPromoteConstants;
|
||||||
@ -1023,8 +1041,15 @@ impl MirPass for QualifyAndPromoteConstants {
|
|||||||
tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
||||||
src: MirSource,
|
src: MirSource,
|
||||||
mir: &mut Mir<'tcx>) {
|
mir: &mut Mir<'tcx>) {
|
||||||
|
// There's not really any point in promoting errorful MIR.
|
||||||
|
if mir.return_ty.references_error() {
|
||||||
|
tcx.sess.delay_span_bug(mir.span, "QualifyAndPromoteConstants: Mir had errors");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
let id = src.item_id();
|
let id = src.item_id();
|
||||||
let def_id = tcx.hir.local_def_id(id);
|
let def_id = tcx.hir.local_def_id(id);
|
||||||
|
let mut const_promoted_temps = None;
|
||||||
let mode = match src {
|
let mode = match src {
|
||||||
MirSource::Fn(_) => {
|
MirSource::Fn(_) => {
|
||||||
if tcx.is_const_fn(def_id) {
|
if tcx.is_const_fn(def_id) {
|
||||||
@ -1033,20 +1058,21 @@ impl MirPass for QualifyAndPromoteConstants {
|
|||||||
Mode::Fn
|
Mode::Fn
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
MirSource::Const(_) => {
|
||||||
|
const_promoted_temps = Some(tcx.mir_const_qualif(def_id).1);
|
||||||
|
Mode::Const
|
||||||
|
}
|
||||||
MirSource::Static(_, hir::MutImmutable) => Mode::Static,
|
MirSource::Static(_, hir::MutImmutable) => Mode::Static,
|
||||||
MirSource::Static(_, hir::MutMutable) => Mode::StaticMut,
|
MirSource::Static(_, hir::MutMutable) => Mode::StaticMut,
|
||||||
MirSource::GeneratorDrop(_) |
|
MirSource::GeneratorDrop(_) |
|
||||||
MirSource::Const(_) |
|
|
||||||
MirSource::Promoted(..) => return
|
MirSource::Promoted(..) => return
|
||||||
};
|
};
|
||||||
let param_env = tcx.param_env(def_id);
|
|
||||||
|
|
||||||
if mode == Mode::Fn || mode == Mode::ConstFn {
|
if mode == Mode::Fn || mode == Mode::ConstFn {
|
||||||
// This is ugly because Qualifier holds onto mir,
|
// This is ugly because Qualifier holds onto mir,
|
||||||
// which can't be mutated until its scope ends.
|
// which can't be mutated until its scope ends.
|
||||||
let (temps, candidates) = {
|
let (temps, candidates) = {
|
||||||
let mut qualifier = Qualifier::new(tcx, param_env,
|
let mut qualifier = Qualifier::new(tcx, def_id, mir, mode);
|
||||||
def_id, mir, mode);
|
|
||||||
if mode == Mode::ConstFn {
|
if mode == Mode::ConstFn {
|
||||||
// Enforce a constant-like CFG for `const fn`.
|
// Enforce a constant-like CFG for `const fn`.
|
||||||
qualifier.qualify_const();
|
qualifier.qualify_const();
|
||||||
@ -1062,8 +1088,37 @@ impl MirPass for QualifyAndPromoteConstants {
|
|||||||
// Do the actual promotion, now that we know what's viable.
|
// Do the actual promotion, now that we know what's viable.
|
||||||
promote_consts::promote_candidates(mir, tcx, temps, candidates);
|
promote_consts::promote_candidates(mir, tcx, temps, candidates);
|
||||||
} else {
|
} else {
|
||||||
let mut qualifier = Qualifier::new(tcx, param_env, def_id, mir, mode);
|
let promoted_temps = if mode == Mode::Const {
|
||||||
qualifier.qualify_const();
|
// Already computed by `mir_const_qualif`.
|
||||||
|
const_promoted_temps.unwrap()
|
||||||
|
} else {
|
||||||
|
Qualifier::new(tcx, def_id, mir, mode).qualify_const().1
|
||||||
|
};
|
||||||
|
|
||||||
|
// In `const` and `static` everything without `StorageDead`
|
||||||
|
// is `'static`, we don't have to create promoted MIR fragments,
|
||||||
|
// just remove `Drop` and `StorageDead` on "promoted" locals.
|
||||||
|
for block in mir.basic_blocks_mut() {
|
||||||
|
block.statements.retain(|statement| {
|
||||||
|
match statement.kind {
|
||||||
|
StatementKind::StorageDead(Lvalue::Local(index)) => {
|
||||||
|
!promoted_temps.contains(&index)
|
||||||
|
}
|
||||||
|
_ => true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
let terminator = block.terminator_mut();
|
||||||
|
match terminator.kind {
|
||||||
|
TerminatorKind::Drop { location: Lvalue::Local(index), target, .. } => {
|
||||||
|
if promoted_temps.contains(&index) {
|
||||||
|
terminator.kind = TerminatorKind::Goto {
|
||||||
|
target,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Statics must be Sync.
|
// Statics must be Sync.
|
||||||
|
Loading…
Reference in New Issue
Block a user