use std::collections::HashMap;
use crate::cfg::BodyId;
use crate::ssa::ir::{FieldId, SsaBody, SsaInst, SsaOp, SsaValue};
use super::domain::{AbsLoc, LOC_TOP, LocId, LocInterner, PointsToSet, PtrProxyHint};
const MAX_FIXPOINT_ITERS: usize = 8;
fn is_container_read_callee(callee: &str) -> bool {
let bare = match callee.rsplit_once('.') {
Some((_, m)) => m,
None => callee,
};
matches!(
bare,
"shift"
| "pop"
| "peek"
| "front"
| "back"
| "first"
| "last"
| "head"
| "tail"
| "dequeue"
| "remove"
| "popleft"
| "__index_get__"
)
}
pub fn is_container_write_callee(callee: &str) -> bool {
let bare = match callee.rsplit_once('.') {
Some((_, m)) => m,
None => callee,
};
matches!(
bare,
"push"
| "pushback"
| "push_back"
| "pushfront"
| "push_front"
| "append"
| "add"
| "insert"
| "enqueue"
| "unshift"
| "__index_set__"
)
}
pub fn is_container_read_callee_pub(callee: &str) -> bool {
is_container_read_callee(callee)
}
pub fn extract_field_points_to(
body: &SsaBody,
facts: &PointsToFacts,
) -> crate::summary::points_to::FieldPointsToSummary {
use crate::summary::points_to::FieldPointsToSummary;
let mut out = FieldPointsToSummary::empty();
if body.field_interner.is_empty() && body.field_writes.is_empty() {
return out;
}
let field_name = |field: FieldId| -> Option<String> {
if field == FieldId::ELEM {
Some("<elem>".to_string())
} else if (field.0 as usize) < body.field_interner.len() {
Some(body.field_interner.resolve(field).to_string())
} else {
None
}
};
let record =
|loc: LocId, name: &str, out: &mut FieldPointsToSummary, is_write: bool| match facts
.interner
.resolve(loc)
{
crate::pointer::AbsLoc::Param(_, idx) => {
if is_write {
out.add_write(*idx as u32, name);
} else {
out.add_read(*idx as u32, name);
}
}
crate::pointer::AbsLoc::SelfParam(_) => {
if is_write {
out.add_write(u32::MAX, name);
} else {
out.add_read(u32::MAX, name);
}
}
_ => {}
};
for block in &body.blocks {
for inst in block.body.iter() {
if let SsaOp::FieldProj {
receiver, field, ..
} = &inst.op
{
let pt = facts.pt(*receiver);
if pt.is_empty() || pt.is_top() {
continue;
}
let Some(name) = field_name(*field) else {
continue;
};
for loc in pt.iter() {
record(loc, &name, &mut out, false);
}
}
}
}
for (receiver, field) in body.field_writes.values() {
let pt = facts.pt(*receiver);
if pt.is_empty() || pt.is_top() {
continue;
}
let Some(name) = field_name(*field) else {
continue;
};
for loc in pt.iter() {
record(loc, &name, &mut out, true);
}
}
out
}
#[derive(Clone, Debug)]
pub struct PointsToFacts {
pub body: BodyId,
pub interner: LocInterner,
by_value: Vec<PointsToSet>,
}
impl PointsToFacts {
pub fn empty(body: BodyId) -> Self {
Self {
body,
interner: LocInterner::new(),
by_value: Vec::new(),
}
}
pub fn pt(&self, v: SsaValue) -> &PointsToSet {
let idx = v.0 as usize;
static EMPTY: once_cell::sync::Lazy<PointsToSet> =
once_cell::sync::Lazy::new(PointsToSet::empty);
self.by_value.get(idx).unwrap_or(&EMPTY)
}
pub fn is_trivial(&self) -> bool {
self.by_value.iter().all(|s| s.is_empty())
}
pub fn len(&self) -> usize {
self.by_value.len()
}
pub fn is_empty(&self) -> bool {
self.by_value.is_empty()
}
pub fn proxy_hint(&self, v: SsaValue) -> PtrProxyHint {
let set = self.pt(v);
if set.is_empty() || set.is_top() {
return PtrProxyHint::Other;
}
for id in set.iter() {
match self.interner.resolve(id) {
AbsLoc::Field { .. } => {}
_ => return PtrProxyHint::Other,
}
}
PtrProxyHint::FieldOnly
}
pub fn name_proxy_hints(
&self,
body: &SsaBody,
) -> std::collections::HashMap<String, PtrProxyHint> {
let mut out = std::collections::HashMap::new();
for (idx, def) in body.value_defs.iter().enumerate().rev() {
let Some(name) = def.var_name.as_ref() else {
continue;
};
if out.contains_key(name) {
continue;
}
let hint = self.proxy_hint(SsaValue(idx as u32));
if hint == PtrProxyHint::FieldOnly {
out.insert(name.clone(), hint);
}
}
out
}
}
pub fn analyse_body(body: &SsaBody, body_id: BodyId) -> PointsToFacts {
let mut state = AnalysisState::new(body_id, body.num_values());
for block in &body.blocks {
for inst in block.phis.iter().chain(block.body.iter()) {
state.transfer_inst(body_id, inst);
}
}
for _ in 0..MAX_FIXPOINT_ITERS {
let mut changed = false;
for block in &body.blocks {
for inst in block.phis.iter().chain(block.body.iter()) {
changed |= state.propagate_inst(inst);
}
}
if !changed {
break;
}
}
state.into_facts()
}
struct AnalysisState {
body_id: BodyId,
interner: LocInterner,
pt: Vec<PointsToSet>,
parent: Vec<u32>,
rank: Vec<u8>,
}
impl AnalysisState {
fn new(body_id: BodyId, num_values: usize) -> Self {
Self {
body_id,
interner: LocInterner::new(),
pt: vec![PointsToSet::empty(); num_values],
parent: (0..num_values as u32).collect(),
rank: vec![0; num_values],
}
}
fn find(&mut self, mut v: u32) -> u32 {
if v as usize >= self.parent.len() {
return v;
}
let mut root = v;
while self.parent[root as usize] != root {
root = self.parent[root as usize];
}
while self.parent[v as usize] != root {
let next = self.parent[v as usize];
self.parent[v as usize] = root;
v = next;
}
root
}
fn union(&mut self, a: u32, b: u32) -> u32 {
let ra = self.find(a);
let rb = self.find(b);
if ra == rb {
return ra;
}
let (winner, loser) = match self.rank[ra as usize].cmp(&self.rank[rb as usize]) {
std::cmp::Ordering::Less => (rb, ra),
std::cmp::Ordering::Greater => (ra, rb),
std::cmp::Ordering::Equal => {
self.rank[ra as usize] += 1;
(ra, rb)
}
};
self.parent[loser as usize] = winner;
let loser_pt = std::mem::take(&mut self.pt[loser as usize]);
let _ = self.pt[winner as usize].union_in_place(&loser_pt);
winner
}
fn add_loc(&mut self, ssa: u32, loc: LocId) -> bool {
let rep = self.find(ssa) as usize;
let mut delta = PointsToSet::singleton(loc);
let changed = self.pt[rep].union_in_place(&delta);
let _ = &mut delta;
changed
}
fn copy_pt(&mut self, dst: u32, src: u32) -> bool {
let dr = self.find(dst);
let sr = self.find(src);
if dr == sr {
return false;
}
let src_pt = self.pt[sr as usize].clone();
self.pt[dr as usize].union_in_place(&src_pt)
}
fn transfer_inst(&mut self, body_id: BodyId, inst: &SsaInst) {
let v = inst.value.0;
if (v as usize) >= self.pt.len() {
return;
}
match &inst.op {
SsaOp::Param { index } => {
let loc = self.interner.intern_param(body_id, *index);
self.add_loc(v, loc);
}
SsaOp::SelfParam => {
let loc = self.interner.intern_self_param(body_id);
self.add_loc(v, loc);
}
SsaOp::CatchParam => {
let loc = self.interner.intern_alloc(body_id, v);
self.add_loc(v, loc);
}
SsaOp::Call {
callee, receiver, ..
} => {
let loc = self.interner.intern_alloc(body_id, v);
self.add_loc(v, loc);
if let Some(rcv) = receiver
&& is_container_read_callee(callee)
&& (rcv.0 as usize) < self.parent.len()
{
let rcv_rep = self.find(rcv.0) as usize;
let rcv_pt = self.pt[rcv_rep].clone();
if !rcv_pt.is_empty() && !rcv_pt.is_top() {
for parent_loc in rcv_pt.iter() {
let proj = self.interner.intern_field(parent_loc, FieldId::ELEM);
self.add_loc(v, proj);
}
}
}
}
SsaOp::Assign(uses) => {
for &u in uses {
if (u.0 as usize) < self.parent.len() {
self.union(v, u.0);
}
}
}
SsaOp::Phi(operands) => {
for (_, u) in operands {
if (u.0 as usize) < self.parent.len() {
self.union(v, u.0);
}
}
}
SsaOp::FieldProj { .. } => {
}
SsaOp::Source | SsaOp::Const(_) | SsaOp::Nop | SsaOp::Undef => {
}
}
}
fn propagate_inst(&mut self, inst: &SsaInst) -> bool {
let v = inst.value.0;
if (v as usize) >= self.pt.len() {
return false;
}
match &inst.op {
SsaOp::FieldProj {
receiver, field, ..
} => {
if (receiver.0 as usize) >= self.parent.len() {
return false;
}
let rcv_rep = self.find(receiver.0) as usize;
let mut new_pt = PointsToSet::empty();
let rcv_pt = self.pt[rcv_rep].clone();
if rcv_pt.is_top() {
new_pt.insert(LOC_TOP);
} else if rcv_pt.is_empty() {
return false;
} else {
for parent_loc in rcv_pt.iter() {
let proj = self.interner.intern_field(parent_loc, *field);
new_pt.insert(proj);
}
}
let v_rep = self.find(v) as usize;
self.pt[v_rep].union_in_place(&new_pt)
}
SsaOp::Assign(uses) => {
let mut changed = false;
for &u in uses {
if (u.0 as usize) < self.parent.len() {
changed |= self.copy_pt(v, u.0);
}
}
changed
}
SsaOp::Phi(operands) => {
let mut changed = false;
for (_, u) in operands {
if (u.0 as usize) < self.parent.len() {
changed |= self.copy_pt(v, u.0);
}
}
changed
}
_ => false,
}
}
fn into_facts(mut self) -> PointsToFacts {
let mut by_value = Vec::with_capacity(self.pt.len());
let mut rep_cache: HashMap<u32, PointsToSet> = HashMap::new();
let n = self.pt.len();
for v in 0..n as u32 {
let rep = self.find(v);
let set = rep_cache
.entry(rep)
.or_insert_with(|| self.pt[rep as usize].clone())
.clone();
by_value.push(set);
}
PointsToFacts {
body: self.body_id,
interner: self.interner,
by_value,
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::cfg::Cfg;
use crate::ssa::ir::{
BlockId, FieldId, FieldInterner, SsaBlock, SsaBody, SsaInst, SsaOp, SsaValue, Terminator,
ValueDef,
};
use petgraph::graph::NodeIndex;
use smallvec::{SmallVec, smallvec};
use std::collections::HashMap;
fn body_id() -> BodyId {
BodyId(0)
}
struct BodyBuilder {
defs: Vec<ValueDef>,
body_insts: Vec<SsaInst>,
next_value: u32,
field_interner: FieldInterner,
}
impl BodyBuilder {
fn new() -> Self {
Self {
defs: Vec::new(),
body_insts: Vec::new(),
next_value: 0,
field_interner: FieldInterner::new(),
}
}
fn fresh(&mut self, name: Option<&str>) -> SsaValue {
let v = SsaValue(self.next_value);
self.next_value += 1;
self.defs.push(ValueDef {
var_name: name.map(|s| s.to_string()),
cfg_node: NodeIndex::new(0),
block: BlockId(0),
});
v
}
fn emit(&mut self, value: SsaValue, op: SsaOp, name: Option<&str>) {
self.body_insts.push(SsaInst {
value,
op,
cfg_node: NodeIndex::new(0),
var_name: name.map(|s| s.to_string()),
span: (0, 0),
});
}
fn intern_field(&mut self, name: &str) -> FieldId {
self.field_interner.intern(name)
}
fn build(self) -> SsaBody {
SsaBody {
blocks: vec![SsaBlock {
id: BlockId(0),
phis: vec![],
body: self.body_insts,
terminator: Terminator::Return(None),
preds: SmallVec::new(),
succs: SmallVec::new(),
}],
entry: BlockId(0),
value_defs: self.defs,
cfg_node_map: HashMap::new(),
exception_edges: vec![],
field_interner: self.field_interner,
field_writes: std::collections::HashMap::new(),
synthetic_externals: std::collections::HashSet::new(),
}
}
}
#[test]
fn field_subobject_distinct_from_receiver() {
let mut b = BodyBuilder::new();
let c = b.fresh(Some("c"));
b.emit(c, SsaOp::SelfParam, Some("c"));
let mu_field = b.intern_field("mu");
let m = b.fresh(Some("c.mu"));
b.emit(
m,
SsaOp::FieldProj {
receiver: c,
field: mu_field,
projected_type: None,
},
Some("c.mu"),
);
let body = b.build();
let facts = analyse_body(&body, body_id());
let pt_c = facts.pt(c);
let pt_m = facts.pt(m);
assert_eq!(pt_c.len(), 1, "pt(c) should be a singleton SelfParam");
assert_eq!(pt_m.len(), 1, "pt(c.mu) should be a singleton Field");
assert!(!pt_m.is_top());
for c_loc in pt_c.iter() {
for m_loc in pt_m.iter() {
assert_ne!(c_loc, m_loc, "field and receiver share a location");
}
}
let m_loc = pt_m.iter().next().unwrap();
match facts.interner.resolve(m_loc) {
crate::pointer::AbsLoc::Field { parent, field } => {
assert_eq!(*field, mu_field);
assert_eq!(*parent, pt_c.iter().next().unwrap());
}
other => panic!("expected Field, got {other:?}"),
}
}
#[test]
fn copy_propagation_unifies() {
let mut b = BodyBuilder::new();
let x = b.fresh(Some("x"));
b.emit(x, SsaOp::Param { index: 0 }, Some("x"));
let y = b.fresh(Some("y"));
b.emit(y, SsaOp::Assign(smallvec![x]), Some("y"));
let body = b.build();
let facts = analyse_body(&body, body_id());
assert_eq!(
facts.pt(x),
facts.pt(y),
"Steensgaard unifies pt(y) with pt(x) via the copy"
);
assert!(!facts.pt(y).is_empty());
}
#[test]
fn phi_unifies_branches() {
let mut b = BodyBuilder::new();
let a = b.fresh(Some("a"));
b.emit(a, SsaOp::Param { index: 0 }, Some("a"));
let b_v = b.fresh(Some("b"));
b.emit(b_v, SsaOp::Param { index: 1 }, Some("b"));
let z = b.fresh(Some("z"));
b.emit(
z,
SsaOp::Phi(smallvec![(BlockId(0), a), (BlockId(0), b_v)]),
Some("z"),
);
let body = b.build();
let facts = analyse_body(&body, body_id());
let pt_z = facts.pt(z);
assert_eq!(pt_z, facts.pt(a));
assert_eq!(pt_z, facts.pt(b_v));
assert_eq!(pt_z.len(), 2);
}
#[test]
fn self_referential_field_chain_terminates() {
let mut b = BodyBuilder::new();
let node = b.fresh(Some("node"));
b.emit(node, SsaOp::Param { index: 0 }, Some("node"));
let next_field = b.intern_field("next");
for _ in 0..6 {
let fp = b.fresh(Some("node.next"));
b.emit(
fp,
SsaOp::FieldProj {
receiver: node,
field: next_field,
projected_type: None,
},
Some("node.next"),
);
let new_node = b.fresh(Some("node"));
b.emit(new_node, SsaOp::Assign(smallvec![fp]), Some("node"));
}
let body = b.build();
let facts = analyse_body(&body, body_id());
let pt_node = facts.pt(node);
assert!(!pt_node.is_empty());
}
#[test]
fn source_op_has_empty_pt() {
let mut b = BodyBuilder::new();
let s = b.fresh(Some("s"));
b.emit(s, SsaOp::Source, Some("s"));
let body = b.build();
let facts = analyse_body(&body, body_id());
assert!(facts.pt(s).is_empty());
}
#[test]
fn call_result_is_fresh_alloc() {
let mut b = BodyBuilder::new();
let arg = b.fresh(Some("x"));
b.emit(arg, SsaOp::Param { index: 0 }, Some("x"));
let result = b.fresh(Some("r"));
b.emit(
result,
SsaOp::Call {
callee: "make_thing".into(),
callee_text: None,
args: vec![smallvec![arg]],
receiver: None,
},
Some("r"),
);
let body = b.build();
let facts = analyse_body(&body, body_id());
let pt_arg = facts.pt(arg);
let pt_result = facts.pt(result);
assert!(!pt_result.is_empty());
assert!(!pt_arg.is_empty());
for ra in pt_arg.iter() {
for rr in pt_result.iter() {
assert_ne!(ra, rr);
}
}
}
#[test]
fn smoke_runs_on_lowered_body() {
let body = SsaBody {
blocks: vec![SsaBlock {
id: BlockId(0),
phis: vec![],
body: vec![],
terminator: Terminator::Return(None),
preds: SmallVec::new(),
succs: SmallVec::new(),
}],
entry: BlockId(0),
value_defs: vec![],
cfg_node_map: HashMap::new(),
exception_edges: vec![],
field_interner: FieldInterner::new(),
field_writes: std::collections::HashMap::new(),
synthetic_externals: std::collections::HashSet::new(),
};
let facts = analyse_body(&body, body_id());
assert!(facts.is_trivial());
assert_eq!(facts.len(), 0);
let _ = std::marker::PhantomData::<Cfg>;
}
#[test]
fn proxy_hint_field_only_for_field_proj_value() {
let mut b = BodyBuilder::new();
let c = b.fresh(Some("c"));
b.emit(c, SsaOp::SelfParam, Some("c"));
let mu = b.intern_field("mu");
let m = b.fresh(Some("m"));
b.emit(
m,
SsaOp::FieldProj {
receiver: c,
field: mu,
projected_type: None,
},
Some("m"),
);
let body = b.build();
let facts = analyse_body(&body, BodyId(7));
assert_eq!(
facts.body,
BodyId(7),
"PointsToFacts must preserve caller-supplied BodyId"
);
assert_eq!(facts.proxy_hint(m), crate::pointer::PtrProxyHint::FieldOnly);
assert_eq!(facts.proxy_hint(c), crate::pointer::PtrProxyHint::Other);
}
#[test]
fn container_read_callee_classifier_covers_common_methods() {
for c in [
"shift",
"pop",
"peek",
"front",
"back",
"queue.shift",
"list.pop",
"deque.popleft",
"stack.peek",
"vec.first",
] {
assert!(is_container_read_callee(c), "expected container read: {c}");
}
for c in ["push", "append", "insert", "myMethod", "process"] {
assert!(
!is_container_read_callee(c),
"non-read should classify false: {c}"
);
}
}
#[test]
fn container_write_callee_classifier() {
for c in [
"push",
"pushback",
"push_back",
"append",
"insert",
"enqueue",
"list.append",
] {
assert!(is_container_write_callee(c), "expected write: {c}");
}
for c in ["pop", "shift", "process", "lookup"] {
assert!(
!is_container_write_callee(c),
"non-write should classify false: {c}"
);
}
}
#[test]
fn container_read_call_projects_through_elem_field() {
let mut b = BodyBuilder::new();
let arr = b.fresh(Some("arr"));
b.emit(arr, SsaOp::Param { index: 0 }, Some("arr"));
let e = b.fresh(Some("e"));
b.emit(
e,
SsaOp::Call {
callee: "shift".into(),
callee_text: None,
args: vec![],
receiver: Some(arr),
},
Some("e"),
);
let body = b.build();
let facts = analyse_body(&body, BodyId(0));
let pt_e = facts.pt(e);
let mut saw_elem = false;
for loc in pt_e.iter() {
if let crate::pointer::AbsLoc::Field { field, .. } = facts.interner.resolve(loc)
&& *field == FieldId::ELEM
{
saw_elem = true;
break;
}
}
assert!(
saw_elem,
"container read result should include Field(_, ELEM); got {pt_e:?}"
);
}
#[test]
fn extract_field_points_to_records_param_reads() {
let mut b = BodyBuilder::new();
let obj = b.fresh(Some("obj"));
b.emit(obj, SsaOp::Param { index: 0 }, Some("obj"));
let name_field = b.intern_field("name");
let n = b.fresh(Some("n"));
b.emit(
n,
SsaOp::FieldProj {
receiver: obj,
field: name_field,
projected_type: None,
},
Some("n"),
);
let body = b.build();
let facts = analyse_body(&body, BodyId(0));
let summary = extract_field_points_to(&body, &facts);
let entry = summary
.param_field_reads
.iter()
.find(|(p, _)| *p == 0)
.expect("param 0 read recorded");
assert!(entry.1.iter().any(|s| s == "name"));
}
#[test]
fn extract_field_points_to_records_param_writes() {
let mut b = BodyBuilder::new();
let obj = b.fresh(Some("obj"));
b.emit(obj, SsaOp::Param { index: 0 }, Some("obj"));
let cache_id = b.intern_field("cache");
let rhs = b.fresh(Some("rhs"));
b.emit(rhs, SsaOp::Source, Some("rhs"));
let synth = b.fresh(Some("obj"));
b.emit(synth, SsaOp::Assign(smallvec![rhs]), Some("obj"));
let mut body = b.build();
body.field_writes.insert(synth, (obj, cache_id));
let facts = analyse_body(&body, BodyId(0));
let summary = extract_field_points_to(&body, &facts);
let entry = summary
.param_field_writes
.iter()
.find(|(p, _)| *p == 0)
.expect("param 0 write must be recorded from field_writes");
assert!(
entry.1.iter().any(|s| s == "cache"),
"expected 'cache' in writes; got {:?}",
entry.1,
);
}
#[test]
fn extract_field_points_to_records_self_writes_under_sentinel() {
let mut b = BodyBuilder::new();
let this = b.fresh(Some("this"));
b.emit(this, SsaOp::SelfParam, Some("this"));
let cache_id = b.intern_field("cache");
let rhs = b.fresh(Some("rhs"));
b.emit(rhs, SsaOp::Source, Some("rhs"));
let synth = b.fresh(Some("this"));
b.emit(synth, SsaOp::Assign(smallvec![rhs]), Some("this"));
let mut body = b.build();
body.field_writes.insert(synth, (this, cache_id));
let facts = analyse_body(&body, BodyId(0));
let summary = extract_field_points_to(&body, &facts);
let entry = summary
.param_field_writes
.iter()
.find(|(p, _)| *p == u32::MAX)
.expect("receiver write recorded under u32::MAX sentinel");
assert!(entry.1.iter().any(|s| s == "cache"));
}
#[test]
fn extract_field_points_to_records_elem_writes() {
let mut b = BodyBuilder::new();
let arr = b.fresh(Some("arr"));
b.emit(arr, SsaOp::Param { index: 0 }, Some("arr"));
let rhs = b.fresh(Some("rhs"));
b.emit(rhs, SsaOp::Source, Some("rhs"));
let synth = b.fresh(Some("arr"));
b.emit(synth, SsaOp::Assign(smallvec![rhs]), Some("arr"));
let mut body = b.build();
body.field_writes.insert(synth, (arr, FieldId::ELEM));
let facts = analyse_body(&body, BodyId(0));
let summary = extract_field_points_to(&body, &facts);
let entry = summary
.param_field_writes
.iter()
.find(|(p, _)| *p == 0)
.expect("ELEM write on param 0 recorded");
assert!(
entry.1.iter().any(|s| s == "<elem>"),
"ELEM marker '<elem>' must surface unchanged across the wire",
);
}
#[test]
fn extract_field_points_to_records_self_reads_under_sentinel() {
let mut b = BodyBuilder::new();
let this = b.fresh(Some("this"));
b.emit(this, SsaOp::SelfParam, Some("this"));
let cache = b.intern_field("cache");
let c = b.fresh(Some("c"));
b.emit(
c,
SsaOp::FieldProj {
receiver: this,
field: cache,
projected_type: None,
},
Some("c"),
);
let body = b.build();
let facts = analyse_body(&body, BodyId(0));
let summary = extract_field_points_to(&body, &facts);
let entry = summary
.param_field_reads
.iter()
.find(|(p, _)| *p == u32::MAX)
.expect("receiver read recorded under u32::MAX sentinel");
assert!(entry.1.iter().any(|s| s == "cache"));
}
#[test]
fn subscript_get_classifies_as_container_read() {
assert!(is_container_read_callee_pub("__index_get__"));
assert!(is_container_read_callee_pub("arr.__index_get__"));
}
#[test]
fn subscript_set_classifies_as_container_write() {
assert!(is_container_write_callee("__index_set__"));
assert!(is_container_write_callee("arr.__index_set__"));
}
#[test]
fn subscript_synth_callees_do_not_cross_classify() {
assert!(!is_container_read_callee_pub("__index_set__"));
assert!(!is_container_write_callee("__index_get__"));
}
#[test]
fn name_proxy_hints_collects_field_only_locals() {
let mut b = BodyBuilder::new();
let c = b.fresh(Some("c"));
b.emit(c, SsaOp::SelfParam, Some("c"));
let mu = b.intern_field("mu");
let m = b.fresh(Some("m"));
b.emit(
m,
SsaOp::FieldProj {
receiver: c,
field: mu,
projected_type: None,
},
Some("m"),
);
let body = b.build();
let facts = analyse_body(&body, BodyId(0));
let hints = facts.name_proxy_hints(&body);
assert_eq!(
hints.get("m"),
Some(&crate::pointer::PtrProxyHint::FieldOnly)
);
assert!(
!hints.contains_key("c"),
"root receiver must not appear in the FieldOnly map"
);
}
}