Skip to content

Commit 2c2a35d

Browse files
Dwight GuthBaltoli
Dwight Guth
andauthored
Fix kompile -O3 (#1009)
There exists a bug in `kompile -O3` which occurs if a specialized step function has a residual of collection sort (i.e., its rhs has a function that returns a collection) and garbage collection is triggered immediately after this rule applies. In this case, the collection in question has been allocated with koreAllocAlwaysGC, and the memory it is using gets freed. As a result, subsequent allocations can write to that memory and corrupt the term pointed to by that collection pointer. The fix is to detect when this occurs just prior to invoking the garbage collector, and copying the collection pointer into a block with a block header, which then gets passed as the updated garbage collection root to the gc. GC keeps that memory alive, and then the post-gc address seen by the function that invoked the collector corresponds to the migrated pointer on the main kore heap. We implement this with the following steps: 1. Create a family of functions `store_<cat>_for_gc` and `load_<cat>_for_gc` which handle the process of storing and loading the collection pointer to the heap when the GC is invoked. 2. Create new reserved symbols used to store these values for all collection sort categories present in the definition. 3. Ensure that the garbage collector knows that a root is a block containing a collection rather than the collection pointer itself. --------- Co-authored-by: Bruce Collie <[email protected]>
1 parent ec7cba1 commit 2c2a35d

File tree

11 files changed

+3089
-60
lines changed

11 files changed

+3089
-60
lines changed

include/kllvm/ast/AST.h

+3
Original file line numberDiff line numberDiff line change
@@ -222,6 +222,7 @@ class kore_symbol {
222222
/* A unique integer representing the layout of the symbol in memory.
223223
See create_term.cpp for more information about the layout of K terms. */
224224
uint16_t layout_{};
225+
bool instantiated_ = false;
225226

226227
public:
227228
static ptr<kore_symbol> create(std::string const &name) {
@@ -1017,6 +1018,8 @@ void read_multimap(
10171018

10181019
sptr<kore_pattern> strip_raw_term(sptr<kore_pattern> const &term);
10191020

1021+
std::string get_raw_symbol_name(sort_category);
1022+
10201023
namespace detail {
10211024

10221025
template <typename T>

include/runtime/collect.h

+9-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,15 @@ void migrate_rangemap(void *m);
3636
void migrate_set(void *s);
3737
void migrate_collection_node(void **node_ptr);
3838
void set_kore_memory_functions_for_gmp(void);
39-
void kore_collect(void **, uint8_t, layoutitem *);
39+
void kore_collect(void **, uint8_t, layoutitem *, bool *);
40+
bool store_map_for_gc(void **, map *);
41+
bool store_set_for_gc(void **, set *);
42+
bool store_list_for_gc(void **, list *);
43+
bool store_rangemap_for_gc(void **, rangemap *);
44+
map *load_map_for_gc(void **, bool);
45+
set *load_set_for_gc(void **, bool);
46+
list *load_list_for_gc(void **, bool);
47+
rangemap *load_rangemap_for_gc(void **, bool);
4048
}
4149

4250
#ifdef GC_DBG

include/runtime/collections/RBTree.h

+11
Original file line numberDiff line numberDiff line change
@@ -130,6 +130,17 @@ class RBTree {
130130
root_ = t.root_;
131131
}
132132

133+
RBTree(RBTree const &other) = default;
134+
135+
RBTree(RBTree &&other) { other.root_.swap(root_); }
136+
137+
RBTree &operator=(RBTree const &other) = default;
138+
139+
RBTree &operator=(RBTree &&other) {
140+
other.root_.swap(root_);
141+
return *this;
142+
}
143+
133144
// Return true if this tree is empty.
134145
[[nodiscard]] bool empty() const { return root_->is_leaf(); }
135146

include/runtime/collections/rangemap.h

+8
Original file line numberDiff line numberDiff line change
@@ -328,6 +328,14 @@ class RangeMap {
328328
treemap_ = m.treemap_;
329329
}
330330

331+
RangeMap(RangeMap const &other) = default;
332+
333+
RangeMap(RangeMap &&other) = default;
334+
335+
RangeMap &operator=(RangeMap const &other) = default;
336+
337+
RangeMap &operator=(RangeMap &&other) = default;
338+
331339
// Getter for the rb-tree underlying this rangemap.
332340
[[nodiscard]] rb_tree::RBTree<Range<T>, V> treemap() const {
333341
return treemap_;

lib/ast/AST.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -319,6 +319,9 @@ bool kore_symbol::is_builtin() const {
319319
}
320320

321321
void kore_symbol::instantiate_symbol(kore_symbol_declaration *decl) {
322+
if (instantiated_) {
323+
return;
324+
}
322325
std::vector<sptr<kore_sort>> instantiated;
323326
int i = 0;
324327
kore_sort::substitution vars;
@@ -332,6 +335,7 @@ void kore_symbol::instantiate_symbol(kore_symbol_declaration *decl) {
332335
sort_ = return_sort->substitute(vars);
333336

334337
arguments_ = instantiated;
338+
instantiated_ = true;
335339
}
336340

337341
std::string kore_variable::get_name() const {

lib/ast/definition.cpp

+81-30
Original file line numberDiff line numberDiff line change
@@ -70,15 +70,37 @@ void kore_definition::add_module(sptr<kore_module> module) {
7070
modules_.push_back(std::move(module));
7171
}
7272

73+
std::string get_raw_symbol_name(sort_category cat) {
74+
return "rawCollection_" + std::to_string((int)cat);
75+
}
76+
7377
void kore_definition::insert_reserved_symbols() {
7478
auto mod = kore_module::create("K-RAW-TERM");
7579
auto decl = kore_symbol_declaration::create("rawTerm", true);
76-
auto sort = kore_composite_sort::create("SortKItem");
80+
auto kitem = kore_composite_sort::create("SortKItem");
7781

78-
decl->get_symbol()->add_sort(sort);
79-
decl->get_symbol()->add_argument(sort);
82+
decl->get_symbol()->add_sort(kitem);
83+
decl->get_symbol()->add_argument(kitem);
8084
mod->add_declaration(std::move(decl));
8185

86+
for (auto const &cat : hooked_sorts_) {
87+
switch (cat.first.cat) {
88+
case sort_category::Map:
89+
case sort_category::List:
90+
case sort_category::Set:
91+
case sort_category::RangeMap: {
92+
auto decl = kore_symbol_declaration::create(
93+
get_raw_symbol_name(cat.first.cat), true);
94+
auto sort = cat.second;
95+
decl->get_symbol()->add_sort(kitem);
96+
decl->get_symbol()->add_argument(sort);
97+
mod->add_declaration(std::move(decl));
98+
break;
99+
}
100+
default: break;
101+
}
102+
}
103+
82104
add_module(std::move(mod));
83105
}
84106

@@ -121,24 +143,11 @@ SymbolMap kore_definition::get_overloads() const {
121143

122144
// NOLINTNEXTLINE(*-function-cognitive-complexity)
123145
void kore_definition::preprocess() {
124-
insert_reserved_symbols();
125-
126146
for (auto *axiom : axioms_) {
127147
axiom->pattern_ = axiom->pattern_->expand_aliases(this);
128148
}
129149
auto symbols = std::map<std::string, std::vector<kore_symbol *>>{};
130150
unsigned next_ordinal = 0;
131-
for (auto const &decl : symbol_declarations_) {
132-
if (decl.second->attributes().contains(
133-
attribute_set::key::FreshGenerator)) {
134-
auto sort = decl.second->get_symbol()->get_sort();
135-
if (sort->is_concrete()) {
136-
fresh_functions_[dynamic_cast<kore_composite_sort *>(sort.get())
137-
->get_name()]
138-
= decl.second->get_symbol();
139-
}
140-
}
141-
}
142151
for (auto iter = axioms_.begin(); iter != axioms_.end();) {
143152
auto *axiom = *iter;
144153
axiom->ordinal_ = next_ordinal;
@@ -163,12 +172,68 @@ void kore_definition::preprocess() {
163172
}
164173
}
165174
}
175+
166176
for (auto const &entry : symbols) {
167177
for (auto *symbol : entry.second) {
168178
auto *decl = symbol_declarations_.at(symbol->get_name());
169179
symbol->instantiate_symbol(decl);
170180
}
171181
}
182+
183+
for (auto const &entry : symbols) {
184+
for (auto *symbol : entry.second) {
185+
for (auto const &sort : symbol->get_arguments()) {
186+
if (sort->is_concrete()) {
187+
hooked_sorts_[dynamic_cast<kore_composite_sort *>(sort.get())
188+
->get_category(this)]
189+
= std::dynamic_pointer_cast<kore_composite_sort>(sort);
190+
}
191+
}
192+
if (symbol->get_sort()->is_concrete()) {
193+
hooked_sorts_[dynamic_cast<kore_composite_sort *>(
194+
symbol->get_sort().get())
195+
->get_category(this)]
196+
= std::dynamic_pointer_cast<kore_composite_sort>(
197+
symbol->get_sort());
198+
}
199+
}
200+
}
201+
202+
insert_reserved_symbols();
203+
204+
for (auto &module : modules_) {
205+
auto const &declarations = module->get_declarations();
206+
for (auto const &declaration : declarations) {
207+
auto *decl = dynamic_cast<kore_symbol_declaration *>(declaration.get());
208+
if (decl == nullptr) {
209+
continue;
210+
}
211+
if (decl->is_hooked() && decl->get_object_sort_variables().empty()) {
212+
kore_symbol *symbol = decl->get_symbol();
213+
symbols.emplace(symbol->get_name(), std::vector<kore_symbol *>{symbol});
214+
}
215+
}
216+
}
217+
218+
for (auto const &entry : symbols) {
219+
for (auto *symbol : entry.second) {
220+
auto *decl = symbol_declarations_.at(symbol->get_name());
221+
symbol->instantiate_symbol(decl);
222+
}
223+
}
224+
225+
for (auto const &decl : symbol_declarations_) {
226+
if (decl.second->attributes().contains(
227+
attribute_set::key::FreshGenerator)) {
228+
auto sort = decl.second->get_symbol()->get_sort();
229+
if (sort->is_concrete()) {
230+
fresh_functions_[dynamic_cast<kore_composite_sort *>(sort.get())
231+
->get_name()]
232+
= decl.second->get_symbol();
233+
}
234+
}
235+
}
236+
172237
uint32_t next_symbol = 0;
173238
uint16_t next_layout = 1;
174239
auto instantiations
@@ -202,20 +267,6 @@ void kore_definition::preprocess() {
202267
for (auto const &entry : symbols) {
203268
auto range = variables.at(entry.first);
204269
for (auto *symbol : entry.second) {
205-
for (auto const &sort : symbol->get_arguments()) {
206-
if (sort->is_concrete()) {
207-
hooked_sorts_[dynamic_cast<kore_composite_sort *>(sort.get())
208-
->get_category(this)]
209-
= std::dynamic_pointer_cast<kore_composite_sort>(sort);
210-
}
211-
}
212-
if (symbol->get_sort()->is_concrete()) {
213-
hooked_sorts_[dynamic_cast<kore_composite_sort *>(
214-
symbol->get_sort().get())
215-
->get_category(this)]
216-
= std::dynamic_pointer_cast<kore_composite_sort>(
217-
symbol->get_sort());
218-
}
219270
if (!symbol->is_concrete()) {
220271
if (symbol->is_polymorphic()) {
221272
symbol->first_tag_ = range.first;

0 commit comments

Comments
 (0)