Commit 6f66a89d authored by David Renshaw's avatar David Renshaw

Fix bug where struct equality could ignore some pointer fields.

parent cfd813e4
...@@ -402,6 +402,38 @@ KJ_TEST("Pointer list unequal to struct list") { ...@@ -402,6 +402,38 @@ KJ_TEST("Pointer list unequal to struct list") {
EXPECT_EQ(Equality::NOT_EQUAL, message1.getRoot<AnyList>().equals(message2.getRoot<AnyList>())); EXPECT_EQ(Equality::NOT_EQUAL, message1.getRoot<AnyList>().equals(message2.getRoot<AnyList>()));
} }
KJ_TEST("Truncating non-null pointer fields does not preserve equality") {
AlignedData<3> segment1 = {{
// list with one data word and one pointer field
0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x01, 0x00,
// data word
0xab, 0xab, 0xab, 0xab, 0xab, 0xab, 0xab, 0xab,
// non-null pointer to zero-sized struct
0xfc, 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00,
}};
kj::ArrayPtr<const word> segments1[1] = {
kj::arrayPtr(segment1.words, 3)
};
SegmentArrayMessageReader message1(kj::arrayPtr(segments1, 1));
AlignedData<2> segment2 = {{
// list with one data word and zero pointers
0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00,
// data word
0xab, 0xab, 0xab, 0xab, 0xab, 0xab, 0xab, 0xab,
}};
kj::ArrayPtr<const word> segments2[1] = {
kj::arrayPtr(segment2.words, 2)
};
SegmentArrayMessageReader message2(kj::arrayPtr(segments2, 1));
EXPECT_EQ(Equality::NOT_EQUAL,
message1.getRoot<AnyPointer>().equals(message2.getRoot<AnyPointer>()));
}
} // namespace } // namespace
} // namespace _ (private) } // namespace _ (private)
} // namespace capnp } // namespace capnp
...@@ -101,12 +101,25 @@ Equality AnyStruct::Reader::equals(AnyStruct::Reader right) { ...@@ -101,12 +101,25 @@ Equality AnyStruct::Reader::equals(AnyStruct::Reader right) {
} }
auto ptrsL = getPointerSection(); auto ptrsL = getPointerSection();
size_t ptrsSizeL = ptrsL.size();
while (ptrsSizeL > 0 && ptrsL[ptrsSizeL - 1].isNull()) {
-- ptrsSizeL;
}
auto ptrsR = right.getPointerSection(); auto ptrsR = right.getPointerSection();
size_t ptrsSizeR = ptrsR.size();
while (ptrsSizeR > 0 && ptrsR[ptrsSizeR - 1].isNull()) {
-- ptrsSizeR;
}
if(ptrsSizeL != ptrsSizeR) {
return Equality::NOT_EQUAL;
}
size_t i = 0; size_t i = 0;
auto eqResult = Equality::EQUAL; auto eqResult = Equality::EQUAL;
for(; i < kj::min(ptrsL.size(), ptrsR.size()); i++) { for (; i < ptrsSizeL; i++) {
auto l = ptrsL[i]; auto l = ptrsL[i];
auto r = ptrsR[i]; auto r = ptrsR[i];
switch(l.equals(r)) { switch(l.equals(r)) {
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment