forked from carbon-language/carbon-lang
-
Notifications
You must be signed in to change notification settings - Fork 0
/
heap.cpp
94 lines (82 loc) · 3.21 KB
/
heap.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
// Part of the Carbon Language project, under the Apache License v2.0 with LLVM
// Exceptions. See /LICENSE for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
#include "explorer/interpreter/heap.h"
#include "explorer/common/error_builders.h"
#include "explorer/interpreter/value.h"
#include "llvm/ADT/StringExtras.h"
#include "llvm/Support/Error.h"
namespace Carbon {
auto Heap::AllocateValue(Nonnull<const Value*> v) -> AllocationId {
// Putting the following two side effects together in this function
// ensures that we don't do anything else in between, which is really bad!
// Consider whether to include a copy of the input v in this function
// or to leave it up to the caller.
AllocationId a(values_.size());
values_.push_back(v);
if (v->kind() == Carbon::Value::Kind::UninitializedValue) {
states_.push_back(ValueState::Uninitialized);
} else {
states_.push_back(ValueState::Alive);
}
return a;
}
auto Heap::Read(const Address& a, SourceLocation source_loc) const
-> ErrorOr<Nonnull<const Value*>> {
CARBON_RETURN_IF_ERROR(this->CheckInit(a.allocation_, source_loc));
CARBON_RETURN_IF_ERROR(this->CheckAlive(a.allocation_, source_loc));
Nonnull<const Value*> value = values_[a.allocation_.index_];
return value->GetMember(arena_, a.field_path_, source_loc, value);
}
auto Heap::Write(const Address& a, Nonnull<const Value*> v,
SourceLocation source_loc) -> ErrorOr<Success> {
CARBON_RETURN_IF_ERROR(this->CheckAlive(a.allocation_, source_loc));
if (states_[a.allocation_.index_] == ValueState::Uninitialized) {
states_[a.allocation_.index_] = ValueState::Alive;
}
CARBON_ASSIGN_OR_RETURN(values_[a.allocation_.index_],
values_[a.allocation_.index_]->SetField(
arena_, a.field_path_, v, source_loc));
return Success();
}
auto Heap::CheckAlive(AllocationId allocation, SourceLocation source_loc) const
-> ErrorOr<Success> {
if (states_[allocation.index_] == ValueState::Dead) {
return RuntimeError(source_loc)
<< "undefined behavior: access to dead value "
<< *values_[allocation.index_];
}
return Success();
}
auto Heap::CheckInit(AllocationId allocation, SourceLocation source_loc) const
-> ErrorOr<Success> {
if (states_[allocation.index_] == ValueState::Uninitialized) {
return RuntimeError(source_loc)
<< "undefined behavior: access to uninitialized value "
<< *values_[allocation.index_];
}
return Success();
}
void Heap::Deallocate(AllocationId allocation) {
if (states_[allocation.index_] != ValueState::Dead) {
states_[allocation.index_] = ValueState::Dead;
} else {
CARBON_FATAL() << "deallocating an already dead value: "
<< *values_[allocation.index_];
}
}
void Heap::Deallocate(const Address& a) { Deallocate(a.allocation_); }
void Heap::Print(llvm::raw_ostream& out) const {
llvm::ListSeparator sep;
for (size_t i = 0; i < values_.size(); ++i) {
out << sep;
out << i << ": ";
if (states_[i] == ValueState::Uninitialized) {
out << "!";
} else if (states_[i] == ValueState::Dead) {
out << "!!";
}
out << *values_[i];
}
}
} // namespace Carbon