Skip to content

Commit 37a43ea

Browse files
committed
fix(opcode/Ifacmp): compare objects by data pointer only
We store class instances as Rc<dyn ClassInstanc> values. The pointers to the inner values consist of 1. a pointer to the vtable of the trait 2. a pointer to the data itself Due to rust internals, the vtable pointer can be different for Rc<dyn ClassInstance> values that point to the same *data*, effectively making any comparison involing the vtable pointer moot. This uses a function rust just introduced to compare such pointers. (cf. rust-lang/rust#106447, rust-lang/rust#117717, rust-lang/rust#116325)
1 parent fa5d323 commit 37a43ea

File tree

4 files changed

+25
-3
lines changed

4 files changed

+25
-3
lines changed

src/executor/op_code.rs

+8-2
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,12 @@ use crate::{
1919
heap::Heap,
2020
};
2121

22+
/// Explicitly compare only the data part of fat/trait/dyn Trait pointers.
23+
#[allow(clippy::ptr_eq)]
24+
fn trait_pointer_eq<T: ?Sized, U: ?Sized>(p: *const T, q: *const U) -> bool {
25+
(p as *const ()) == (q as *const ())
26+
}
27+
2228
#[derive(Clone, Debug)]
2329
pub enum ArrayReferenceKinds {
2430
Boolean,
@@ -1268,7 +1274,7 @@ got: {:?}",
12681274
},
12691275
(None, None) => Update::None,
12701276
(Some(op1), Some(op2)) => {
1271-
if !Rc::<dyn ClassInstance>::ptr_eq(&op1, &op2) {
1277+
if !trait_pointer_eq(op1.as_ref(), op2.as_ref()) {
12721278
Update::GoTo(*size, *direction)
12731279
} else {
12741280
Update::None
@@ -1286,7 +1292,7 @@ got: {:?}",
12861292
(Some(_), None) | (None, Some(_)) => Update::None,
12871293
(None, None) => Update::GoTo(*size, *direction),
12881294
(Some(op1), Some(op2)) => {
1289-
if Rc::<dyn ClassInstance>::ptr_eq(&op1, &op2) {
1295+
if trait_pointer_eq(op1.as_ref(), op2.as_ref()) {
12901296
Update::GoTo(*size, *direction)
12911297
} else {
12921298
Update::None

tests/control_flow.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,9 @@ fn control_flow() -> Result<(), Box<dyn std::error::Error>> {
1212
.stdout(predicate::str::contains("a / 2 == 5\n"))
1313
.stdout(predicate::str::contains("l / 2 != 5\n"))
1414
.stdout(predicate::str::contains("d:\n15\n"))
15-
.stdout(predicate::str::contains("f > 10"));
15+
.stdout(predicate::str::contains("f > 10"))
16+
.stdout(predicate::str::contains("s1 == s2"))
17+
.stdout(predicate::str::contains("s1 != s3"));
1618

1719
Ok(())
1820
}

tests/data/control_flow/Main.class

289 Bytes
Binary file not shown.

tests/data/control_flow/Main.java

+14
Original file line numberDiff line numberDiff line change
@@ -37,5 +37,19 @@ public static void main(String[] args) {
3737
} else {
3838
System.out.println("f == 10");
3939
}
40+
41+
String s1 = "Hello World";
42+
String s2 = s1;
43+
String s3 = "Goodbye";
44+
if (s1 == s2) {
45+
System.out.println("s1 == s2");
46+
} else {
47+
System.out.println("s1 != s2");
48+
}
49+
if (s1 != s3) {
50+
System.out.println("s1 != s3");
51+
} else {
52+
System.out.println("s1 == s3");
53+
}
4054
}
4155
}

0 commit comments

Comments
 (0)