Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Enable fat pointer comparison for slices and add check for vtable ones #1195

Merged
merged 6 commits into from
May 16, 2022

Conversation

celinval
Copy link
Contributor

@celinval celinval commented May 12, 2022

Description of changes:

Currently, we don't support fat pointer comparison. This issue adds support for slice fat pointer comparison and a proper check that will fail the verification if its trying to compare vtable fat pointers.

Resolved issues:

Resolves #327

Call-outs:

I documented in the original issue #327 why I don't think we should compare vtable fat pointers. In a nutshell, this operation is not UB but it is unstable. I.e.: The behavior may depend on the compilation / linking process.

I kept a similar behavior to thin pointers when comparing the data pointer portion of the fat pointer. Comparing pointers that have different precedence will also fail.

We could add optional support to the operations mentioned above using uninterpreted functions, but I will defer that to when we see demand for it.

Testing:

  • How is this change tested? New tests

  • Is this a refactor change? No

Checklist

  • Each commit message has a non-empty body, explaining why the change was made
  • Methods or procedures are documented
  • Regression or unit tests are included, or existing tests cover the modified code
  • My PR is restricted to a single feature or bugfix

By submitting this pull request, I confirm that my contribution is made under the terms of the Apache 2.0 and MIT licenses.

@celinval celinval requested a review from a team as a code owner May 12, 2022 20:21
Copy link
Contributor

@zhassan-aws zhassan-aws left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Neat! Thanks Celina!

BinOp::Gt => self.codegen_operand(e1).gt(self.codegen_operand(e2)),
_ => unreachable!(),
}
let left = self.codegen_operand(e1);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice refactoring!

// Codegen an assertion failure since vtable comparison is not stable.
let ret_type = Type::Bool;
let body = vec![
self.codegen_assert_false(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is it not possible to use codegen_unimplemented instead?

EDIT: I now see that this is due to the property class. Perhaps this needs refactoring to add a property-class parameter in a follow-up PR.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think this should use codegen_unimplemented() though. This is by design and it's meant to alert users of behaviour instability.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we should only use codegen_unimplemented for things that Kani can't yet handle.

Copy link
Contributor

@zhassan-aws zhassan-aws May 13, 2022

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Perhaps we can put the code under codegen_unimplemented in a function with a more generic name (e.g. codegen_assert_assume_false), and call that function from codegen_unimplemented and codegen_unstable?

EDIT: Not necessarily in this PR.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

My bad, I think we already have that. codegen_fatal_error() should do it. It uses Abort instead of assume.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It actually takes Span instead of Location. So I'll leave this change for another time. Sorry!

// It is different if any is different.
BinOp::Ne => data_cmp.or(metadata_cmp),
// If data is different, only compare data.
// If data is equal, apply operator to metadata.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Any references for this logic?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sure. Let me find the PR that implemented this on the rust compiler side.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, we need a reference here.

_ => unreachable!(),
}
let left = self.codegen_operand(e1);
let right = self.codegen_operand(e2);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nit: left_op and right_op.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

sure

self.codegen_assert_false(
PropertyClass::KaniCheck,
format!("Reached unstable vtable comparison '{:?}'", op).as_str(),
Location::None,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is there a way to get the locations for these statements? If so, I suggest adding them.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I can try to refactor things a bit to pass down the location.

// It is different if any is different.
BinOp::Ne => data_cmp.or(metadata_cmp),
// If data is different, only compare data.
// If data is equal, apply operator to metadata.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, we need a reference here.

celinval added 4 commits May 14, 2022 15:34
1. This only works for pointers with the same provenance.
2. We should model vtable comparison a bit differently. Follow up commit
    - Rename variables.
    - Add location to the assertion and other statements.
    - Added a link to the compiler PR that implemented fat pointer
      comparison.

I did not change the assert / assume logic to a different function yet.
@celinval celinval merged commit 25dfce5 into model-checking:main May 16, 2022
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Kani does not support fat pointer comparison
3 participants