Skip to content

Commit c4675e2

Browse files
theo25rv-jenkins
andauthored
Adding a side condition exit event to the proof hint trace (#982)
The new event carries the result of the side condition check. --------- Co-authored-by: rv-jenkins <[email protected]>
1 parent 8918d3e commit c4675e2

File tree

9 files changed

+768
-342
lines changed

9 files changed

+768
-342
lines changed

bindings/python/ast.cpp

+8
Original file line numberDiff line numberDiff line change
@@ -392,6 +392,14 @@ void bind_proof_trace(py::module_ &m) {
392392
LLVMSideConditionEvent, std::shared_ptr<LLVMSideConditionEvent>>(
393393
proof_trace, "LLVMSideConditionEvent", llvm_rewrite_event);
394394

395+
py::class_<
396+
LLVMSideConditionEndEvent, std::shared_ptr<LLVMSideConditionEndEvent>>(
397+
proof_trace, "LLVMSideConditionEndEvent", llvm_step_event)
398+
.def_property_readonly(
399+
"rule_ordinal", &LLVMSideConditionEndEvent::getRuleOrdinal)
400+
.def_property_readonly(
401+
"check_result", &LLVMSideConditionEndEvent::getKOREPattern);
402+
395403
py::class_<LLVMFunctionEvent, std::shared_ptr<LLVMFunctionEvent>>(
396404
proof_trace, "LLVMFunctionEvent", llvm_step_event)
397405
.def_property_readonly("name", &LLVMFunctionEvent::getName)

docs/proof-trace.md

+24-22
Original file line numberDiff line numberDiff line change
@@ -23,36 +23,38 @@ Here is a BNF styled description of the format:
2323
```
2424
proof_trace ::= header event*
2525
26-
header ::= "HINT" <4-byte version number>
26+
header ::= "HINT" <4-byte version number>
2727
28-
event ::= hook
29-
| function
30-
| rule
31-
| side_cond
32-
| config
28+
event ::= hook
29+
| function
30+
| rule
31+
| side_cond_entry
32+
| side_cond_exit
33+
| config
3334
34-
argument ::= hook
35-
| function
36-
| rule
37-
| kore_term
35+
argument ::= hook
36+
| function
37+
| rule
38+
| kore_term
3839
39-
name ::= string
40-
location ::= string
41-
function ::= WORD(0xDD) name location arg* WORD(0x11)
40+
name ::= string
41+
location ::= string
42+
function ::= WORD(0xDD) name location arg* WORD(0x11)
4243
43-
hook ::= WORD(0xAA) name location arg* WORD(0xBB) kore_term
44+
hook ::= WORD(0xAA) name location arg* WORD(0xBB) kore_term
4445
45-
ordinal ::= uint64
46-
arity ::= uint64
47-
variable ::= name kore_term WORD(0xCC)
48-
rule ::= WORD(0x22) ordinal arity variable*
46+
ordinal ::= uint64
47+
arity ::= uint64
48+
variable ::= name kore_term WORD(0xCC)
49+
rule ::= WORD(0x22) ordinal arity variable*
4950
50-
side_cond ::= WORD(0xEE) ordinal arity variable*
51+
side_cond_entry ::= WORD(0xEE) ordinal arity variable*
52+
side_cond_exit ::= WORD(0x33) ordinal kore_term WORD(0xCC)
5153
52-
config ::= WORD(0xFF) kore_term WORD(0xCC)
54+
config ::= WORD(0xFF) kore_term WORD(0xCC)
5355
54-
string ::= <c-style null terminated string>
55-
uint64 ::= <64-bit unsigned little endian integer>
56+
string ::= <c-style null terminated string>
57+
uint64 ::= <64-bit unsigned little endian integer>
5658
```
5759

5860
## Notes

include/kllvm/binary/ProofTraceParser.h

+59-1
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ constexpr uint64_t hook_event_sentinel = detail::word(0xAA);
3232
constexpr uint64_t hook_result_sentinel = detail::word(0xBB);
3333
constexpr uint64_t rule_event_sentinel = detail::word(0x22);
3434
constexpr uint64_t side_condition_event_sentinel = detail::word(0xEE);
35+
constexpr uint64_t side_condition_end_sentinel = detail::word(0x33);
3536

3637
class LLVMStepEvent : public std::enable_shared_from_this<LLVMStepEvent> {
3738
public:
@@ -94,6 +95,34 @@ class LLVMSideConditionEvent : public LLVMRewriteEvent {
9495
virtual void print(std::ostream &Out, unsigned indent = 0u) const override;
9596
};
9697

98+
class LLVMSideConditionEndEvent : public LLVMStepEvent {
99+
private:
100+
uint64_t ruleOrdinal;
101+
sptr<KOREPattern> korePattern;
102+
uint64_t patternLength;
103+
104+
LLVMSideConditionEndEvent(uint64_t _ruleOrdinal)
105+
: ruleOrdinal(_ruleOrdinal)
106+
, korePattern(nullptr)
107+
, patternLength(0u) { }
108+
109+
public:
110+
static sptr<LLVMSideConditionEndEvent> Create(uint64_t _ruleOrdinal) {
111+
return sptr<LLVMSideConditionEndEvent>(
112+
new LLVMSideConditionEndEvent(_ruleOrdinal));
113+
}
114+
115+
uint64_t getRuleOrdinal() const { return ruleOrdinal; }
116+
sptr<KOREPattern> getKOREPattern() const { return korePattern; }
117+
uint64_t getPatternLength() const { return patternLength; }
118+
void setKOREPattern(sptr<KOREPattern> _korePattern, uint64_t _patternLength) {
119+
korePattern = _korePattern;
120+
patternLength = _patternLength;
121+
}
122+
123+
virtual void print(std::ostream &Out, unsigned indent = 0u) const override;
124+
};
125+
97126
class LLVMEvent;
98127

99128
class LLVMFunctionEvent : public LLVMStepEvent {
@@ -210,7 +239,7 @@ class LLVMRewriteTrace {
210239

211240
class ProofTraceParser {
212241
public:
213-
static constexpr uint32_t expectedVersion = 4u;
242+
static constexpr uint32_t expectedVersion = 5u;
214243

215244
private:
216245
bool verbose;
@@ -499,6 +528,33 @@ class ProofTraceParser {
499528
return event;
500529
}
501530

531+
template <typename It>
532+
sptr<LLVMSideConditionEndEvent> parse_side_condition_end(It &ptr, It end) {
533+
if (!check_word(ptr, end, side_condition_end_sentinel)) {
534+
return nullptr;
535+
}
536+
537+
uint64_t ordinal;
538+
if (!parse_ordinal(ptr, end, ordinal)) {
539+
return nullptr;
540+
}
541+
542+
auto event = LLVMSideConditionEndEvent::Create(ordinal);
543+
544+
uint64_t pattern_len;
545+
auto kore_term = parse_kore_term(ptr, end, pattern_len);
546+
if (!kore_term) {
547+
return nullptr;
548+
}
549+
event->setKOREPattern(kore_term, pattern_len);
550+
551+
if (!check_word(ptr, end, kore_end_sentinel)) {
552+
return nullptr;
553+
}
554+
555+
return event;
556+
}
557+
502558
template <typename It>
503559
bool parse_argument(It &ptr, It end, LLVMEvent &event) {
504560
if (std::distance(ptr, end) >= 1u && detail::peek(ptr) == '\x7F') {
@@ -565,6 +621,8 @@ class ProofTraceParser {
565621

566622
case side_condition_event_sentinel: return parse_side_condition(ptr, end);
567623

624+
case side_condition_end_sentinel: return parse_side_condition_end(ptr, end);
625+
568626
default: return nullptr;
569627
}
570628
}

include/kllvm/codegen/ProofEvent.h

+5-1
Original file line numberDiff line numberDiff line change
@@ -121,10 +121,14 @@ class ProofEvent {
121121
[[nodiscard]] llvm::BasicBlock *
122122
functionEvent_post(llvm::BasicBlock *current_block);
123123

124-
[[nodiscard]] llvm::BasicBlock *sideConditionEvent(
124+
[[nodiscard]] llvm::BasicBlock *sideConditionEvent_pre(
125125
KOREAxiomDeclaration *axiom, std::vector<llvm::Value *> const &args,
126126
llvm::BasicBlock *current_block);
127127

128+
[[nodiscard]] llvm::BasicBlock *sideConditionEvent_post(
129+
KOREAxiomDeclaration *axiom, llvm::Value *check_result,
130+
llvm::BasicBlock *current_block);
131+
128132
public:
129133
ProofEvent(KOREDefinition *Definition, llvm::Module *Module)
130134
: Definition(Definition)

lib/binary/ProofTraceParser.cpp

+10-1
Original file line numberDiff line numberDiff line change
@@ -24,10 +24,19 @@ void LLVMRuleEvent::print(std::ostream &Out, unsigned indent) const {
2424
void LLVMSideConditionEvent::print(std::ostream &Out, unsigned indent) const {
2525
std::string Indent(indent * indent_size, ' ');
2626
Out << fmt::format(
27-
"{}side condition: {} {}\n", Indent, ruleOrdinal, substitution.size());
27+
"{}side condition entry: {} {}\n", Indent, ruleOrdinal,
28+
substitution.size());
2829
printSubstitution(Out, indent + 1U);
2930
}
3031

32+
void LLVMSideConditionEndEvent::print(
33+
std::ostream &Out, unsigned indent) const {
34+
std::string Indent(indent * indent_size, ' ');
35+
Out << fmt::format(
36+
"{}side condition exit: {} kore[{}]\n", Indent, ruleOrdinal,
37+
patternLength);
38+
}
39+
3140
void LLVMFunctionEvent::print(std::ostream &Out, unsigned indent) const {
3241
std::string Indent(indent * indent_size, ' ');
3342
Out << fmt::format("{}function: {} ({})\n", Indent, name, relativePosition);

lib/codegen/Decision.cpp

+9-1
Original file line numberDiff line numberDiff line change
@@ -429,7 +429,7 @@ void FunctionNode::codegen(Decision *d) {
429429
ProofEvent p(d->Definition, d->Module);
430430
size_t ordinal = std::stoll(function.substr(15));
431431
KOREAxiomDeclaration *axiom = d->Definition->getAxiomByOrdinal(ordinal);
432-
d->CurrentBlock = p.sideConditionEvent(axiom, args, d->CurrentBlock);
432+
d->CurrentBlock = p.sideConditionEvent_pre(axiom, args, d->CurrentBlock);
433433
}
434434

435435
CreateTerm creator(
@@ -438,6 +438,14 @@ void FunctionNode::codegen(Decision *d) {
438438
function, cat, args, function.substr(0, 5) == "hook_", false);
439439
Call->setName(name.substr(0, max_name_length));
440440
d->store(std::make_pair(name, type), Call);
441+
442+
if (isSideCondition) {
443+
ProofEvent p(d->Definition, d->Module);
444+
size_t ordinal = std::stoll(function.substr(15));
445+
KOREAxiomDeclaration *axiom = d->Definition->getAxiomByOrdinal(ordinal);
446+
d->CurrentBlock = p.sideConditionEvent_post(axiom, Call, d->CurrentBlock);
447+
}
448+
441449
if (d->FailPattern) {
442450
std::string debugName = function;
443451
if (function.substr(0, 5) == "hook_") {

lib/codegen/ProofEvent.cpp

+27-2
Original file line numberDiff line numberDiff line change
@@ -304,15 +304,15 @@ ProofEvent::functionEvent_post(llvm::BasicBlock *current_block) {
304304
return merge_block;
305305
}
306306

307-
llvm::BasicBlock *ProofEvent::sideConditionEvent(
307+
llvm::BasicBlock *ProofEvent::sideConditionEvent_pre(
308308
KOREAxiomDeclaration *axiom, std::vector<llvm::Value *> const &args,
309309
llvm::BasicBlock *current_block) {
310310
if (!ProofHintInstrumentation) {
311311
return current_block;
312312
}
313313

314314
auto [true_block, merge_block, outputFile]
315-
= eventPrelude("side_condition", current_block);
315+
= eventPrelude("side_condition_pre", current_block);
316316

317317
size_t ordinal = axiom->getOrdinal();
318318
size_t arity = args.size();
@@ -343,4 +343,29 @@ llvm::BasicBlock *ProofEvent::sideConditionEvent(
343343
return merge_block;
344344
}
345345

346+
llvm::BasicBlock *ProofEvent::sideConditionEvent_post(
347+
KOREAxiomDeclaration *axiom, llvm::Value *check_result,
348+
llvm::BasicBlock *current_block) {
349+
if (!ProofHintInstrumentation) {
350+
return current_block;
351+
}
352+
353+
auto [true_block, merge_block, outputFile]
354+
= eventPrelude("side_condition_post", current_block);
355+
356+
size_t ordinal = axiom->getOrdinal();
357+
358+
auto check_result_sort = std::dynamic_pointer_cast<KORECompositeSort>(
359+
axiom->getRequires()->getSort());
360+
361+
emitWriteUInt64(outputFile, detail::word(0x33), true_block);
362+
emitWriteUInt64(outputFile, ordinal, true_block);
363+
emitSerializeTerm(*check_result_sort, outputFile, check_result, true_block);
364+
emitWriteUInt64(outputFile, detail::word(0xCC), true_block);
365+
366+
llvm::BranchInst::Create(merge_block, true_block);
367+
368+
return merge_block;
369+
}
370+
346371
} // namespace kllvm

runtime/util/util.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,7 @@ block *constructRawTerm(void *subject, char const *sort, bool raw_value) {
3939
}
4040

4141
void printProofHintHeader(FILE *file) {
42-
uint32_t version = 4;
42+
uint32_t version = 5;
4343
fmt::print(file, "HINT");
4444
fwrite(&version, sizeof(version), 1, file);
4545
}

0 commit comments

Comments
 (0)