2025-09-17 05:56:33 +09:00
|
|
|
use crate::mir::function::MirFunction;
|
|
|
|
|
use crate::mir::verification::utils;
|
2025-11-21 06:25:17 +09:00
|
|
|
use crate::mir::verification_types::VerificationError;
|
|
|
|
|
use crate::mir::{BasicBlockId, ValueId};
|
2025-09-17 05:56:33 +09:00
|
|
|
use std::collections::{HashMap, HashSet};
|
|
|
|
|
|
|
|
|
|
/// Verify CFG references and reachability
|
|
|
|
|
pub fn check_control_flow(function: &MirFunction) -> Result<(), Vec<VerificationError>> {
|
|
|
|
|
let mut errors = Vec::new();
|
|
|
|
|
for (block_id, block) in &function.blocks {
|
|
|
|
|
for successor in &block.successors {
|
|
|
|
|
if !function.blocks.contains_key(successor) {
|
|
|
|
|
errors.push(VerificationError::ControlFlowError {
|
|
|
|
|
block: *block_id,
|
|
|
|
|
reason: format!("References non-existent block {}", successor),
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-11-18 06:11:17 +09:00
|
|
|
// Unreachable blocks are allowed in MIR.
|
|
|
|
|
// They are created intentionally by break/continue/return statements via
|
|
|
|
|
// switch_to_unreachable_block_with_void() to continue SSA construction after
|
|
|
|
|
// control flow terminators. This is standard practice (see LLVM's `unreachable`).
|
|
|
|
|
// Dead code elimination pass (TODO) will remove them during optimization.
|
|
|
|
|
|
2025-11-21 06:25:17 +09:00
|
|
|
if errors.is_empty() {
|
|
|
|
|
Ok(())
|
|
|
|
|
} else {
|
|
|
|
|
Err(errors)
|
|
|
|
|
}
|
2025-09-17 05:56:33 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Verify that merge blocks do not use predecessor-defined values directly (must go through Phi)
|
|
|
|
|
pub fn check_merge_uses(function: &MirFunction) -> Result<(), Vec<VerificationError>> {
|
2025-11-21 06:25:17 +09:00
|
|
|
if crate::config::env::verify_allow_no_phi() {
|
|
|
|
|
return Ok(());
|
|
|
|
|
}
|
2025-09-17 05:56:33 +09:00
|
|
|
let mut errors = Vec::new();
|
|
|
|
|
let preds = utils::compute_predecessors(function);
|
|
|
|
|
let def_block = utils::compute_def_blocks(function);
|
|
|
|
|
let dominators = utils::compute_dominators(function);
|
|
|
|
|
let mut phi_dsts_in_block: HashMap<BasicBlockId, HashSet<ValueId>> = HashMap::new();
|
|
|
|
|
for (bid, block) in &function.blocks {
|
|
|
|
|
let set = phi_dsts_in_block.entry(*bid).or_default();
|
2025-11-24 15:02:51 +09:00
|
|
|
for sp in block.all_spanned_instructions() {
|
|
|
|
|
if let crate::mir::MirInstruction::Phi { dst, .. } = sp.inst {
|
2025-11-21 06:25:17 +09:00
|
|
|
set.insert(*dst);
|
|
|
|
|
}
|
2025-09-17 05:56:33 +09:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
for (bid, block) in &function.blocks {
|
2025-11-21 06:25:17 +09:00
|
|
|
let Some(pred_list) = preds.get(bid) else {
|
|
|
|
|
continue;
|
|
|
|
|
};
|
|
|
|
|
if pred_list.len() < 2 {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2025-09-17 05:56:33 +09:00
|
|
|
let phi_dsts = phi_dsts_in_block.get(bid);
|
|
|
|
|
let doms_of_block = dominators.get(bid).unwrap();
|
2025-11-24 15:02:51 +09:00
|
|
|
for sp in block.all_spanned_instructions() {
|
|
|
|
|
if let crate::mir::MirInstruction::Phi { .. } = sp.inst {
|
2025-11-21 06:25:17 +09:00
|
|
|
continue;
|
|
|
|
|
}
|
2025-11-24 15:02:51 +09:00
|
|
|
for used in sp.inst.used_values() {
|
2025-09-17 05:56:33 +09:00
|
|
|
if let Some(&db) = def_block.get(&used) {
|
|
|
|
|
if !doms_of_block.contains(&db) {
|
|
|
|
|
let is_phi_dst = phi_dsts.map(|s| s.contains(&used)).unwrap_or(false);
|
|
|
|
|
if !is_phi_dst {
|
2025-11-21 06:25:17 +09:00
|
|
|
errors.push(VerificationError::MergeUsesPredecessorValue {
|
|
|
|
|
value: used,
|
|
|
|
|
merge_block: *bid,
|
|
|
|
|
pred_block: db,
|
|
|
|
|
});
|
2025-09-17 05:56:33 +09:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-11-21 06:25:17 +09:00
|
|
|
if errors.is_empty() {
|
|
|
|
|
Ok(())
|
|
|
|
|
} else {
|
|
|
|
|
Err(errors)
|
|
|
|
|
}
|
2025-09-17 05:56:33 +09:00
|
|
|
}
|