@@ -43,15 +43,20 @@ namespace xls {
43
43
// This class implements a cache with a simple state machine; each key has
44
44
// either no recorded value (kUnknown), a value that may be out-of-date
45
45
// (kUnverified), a value that is valid if all inputs prove to be up-to-date
46
- // (kInputsUnverified), or a value that is known to be current & correct
47
- // (kKnown).
46
+ // (kInputsUnverified), a value that is known to be current & correct
47
+ // (kKnown), or information with a forced value (kForced) .
48
48
//
49
49
// When a `key` is queried, we query for the values for all of its inputs, and
50
50
// then re-compute the value for `key` if absent or unverified. If `key` was in
51
51
// state kUnverified & this does change its associated value, we mark any direct
52
52
// users that were in state kInputsUnverified as kUnverified, since their inputs
53
53
// have changed.
54
54
//
55
+ // Any node that is in the kForced state will remain in the same state
56
+ // regardless of other changes to the graph. Changing the value a node is Forced
57
+ // to will invalidate all its users and force a recalculation of downstream
58
+ // values.
59
+ //
55
60
// NOTE: If `key` is in state kInputsUnverified after we queried all of its
56
61
// inputs, then their values did not change, so we have verified that
57
62
// `key`'s value is up-to-date without having to recompute it! This is the
@@ -93,8 +98,45 @@ class LazyDagCache {
93
98
LazyDagCache<Key, Value>& operator =(LazyDagCache<Key, Value>&& other) =
94
99
delete ;
95
100
101
+ // Erase all knowledge of the values of all keys.
96
102
void Clear () { cache_.clear (); }
97
- void Forget (const Key& key) { cache_.erase (key); }
103
+ // Erase all knowledge of the value of all keys except for 'Forced' values.
104
+ void ClearNonForced () {
105
+ absl::erase_if (cache_, [](const auto & v) {
106
+ return v.second .state != CacheState::kForced ;
107
+ });
108
+ }
109
+
110
+ // Entirely remove knowledge of this key. This includes erasing any Forced
111
+ // data.
112
+ void Forget (const Key& key) {
113
+ cache_.erase (key);
114
+ for (const Key& user : provider_->GetUsers (key)) {
115
+ MarkInputsUnverified (user);
116
+ }
117
+ }
118
+
119
+ // Set the key as having the immutable, authoritative 'value'.
120
+ //
121
+ // *This is a dangerous operation and should be used with care.* It tells the
122
+ // cache to never call the ComputeValue callback and to instead consider
123
+ // 'value' to be associated with 'key' now and forever. This knowledge may
124
+ // only be removed by calling 'Forget' or 'Clear'.
125
+ void SetForced (const Key& key, std::unique_ptr<Value> value) {
126
+ cache_.insert_or_assign (key, CacheEntry{.state = CacheState::kForced ,
127
+ .value = std::move (value)});
128
+ MarkUsersUnverified (key);
129
+ }
130
+
131
+ // Set the key as having the immutable, authoritative 'value'.
132
+ //
133
+ // *This is a dangerous operation and should be used with care.* It tells the
134
+ // cache to never call the ComputeValue callback and to instead consider
135
+ // 'value' to be associated with 'key' now and forever. This knowledge may
136
+ // only be removed by calling 'Forget' or 'Clear'.
137
+ void SetForced (const Key& key, Value value) {
138
+ SetForced (key, std::make_unique<Value>(std::move (value)));
139
+ }
98
140
99
141
void AddUnverified (const Key& key, Value value) {
100
142
cache_.insert_or_assign (
@@ -106,6 +148,15 @@ class LazyDagCache {
106
148
.value = std::move (value)});
107
149
}
108
150
151
+ // Request recomputation of any users of this key.
152
+ //
153
+ // Mark as full unverified to force recomputation.
154
+ void MarkUsersUnverified (const Key& key) {
155
+ for (const Key& user : provider_->GetUsers (key)) {
156
+ MarkUnverified (user);
157
+ }
158
+ }
159
+
109
160
void MarkUnverified (const Key& key) {
110
161
auto it = cache_.find (key);
111
162
if (it == cache_.end ()) {
@@ -115,6 +166,11 @@ class LazyDagCache {
115
166
if (state == CacheState::kUnverified ) {
116
167
return ;
117
168
}
169
+ if (state == CacheState::kForced ) {
170
+ VLOG (1 ) << " Mark unverified called on forced entry "
171
+ << provider_->GetName (key);
172
+ return ;
173
+ }
118
174
state = CacheState::kUnverified ;
119
175
for (const Key& user : provider_->GetUsers (key)) {
120
176
if (GetCacheState (user) == CacheState::kKnown ) {
@@ -155,7 +211,8 @@ class LazyDagCache {
155
211
Value* QueryValue (const Key& key) {
156
212
// If `key` is already known, return a pointer to the cached value.
157
213
if (auto it = cache_.find (key);
158
- it != cache_.end () && it->second .state == CacheState::kKnown ) {
214
+ it != cache_.end () && (it->second .state == CacheState::kKnown ||
215
+ it->second .state == CacheState::kForced )) {
159
216
return it->second .value .get ();
160
217
}
161
218
@@ -189,6 +246,7 @@ class LazyDagCache {
189
246
return cached_value.get ();
190
247
}
191
248
249
+ CHECK_NE (state, CacheState::kForced );
192
250
state = CacheState::kKnown ;
193
251
cached_value = std::make_unique<Value>(*std::move (new_value));
194
252
@@ -208,7 +266,7 @@ class LazyDagCache {
208
266
// all nodes in the DAG.
209
267
absl::Status EagerlyPopulate (absl::Span<const Key> topo_sorted_keys) {
210
268
for (const Key& key : topo_sorted_keys) {
211
- if (GetCacheState (key) == CacheState::kKnown ) {
269
+ if (GetNonForcedCacheState (key) == CacheState::kKnown ) {
212
270
continue ;
213
271
}
214
272
std::vector<const Value*> input_values;
@@ -232,6 +290,8 @@ class LazyDagCache {
232
290
// consistent regardless. This is an expensive operation, intended for use in
233
291
// tests. `topo_sorted_keys` must be a topological sort of the keys for all
234
292
// nodes in the DAG.
293
+ //
294
+ // Note that Forced values are always considered consistent.
235
295
absl::Status CheckConsistency (absl::Span<const Key> topo_sorted_keys) const ;
236
296
237
297
enum class CacheState : uint8_t {
@@ -241,6 +301,14 @@ class LazyDagCache {
241
301
kUnverified ,
242
302
kInputsUnverified ,
243
303
kKnown ,
304
+ // A value has been provided via external information and should be
305
+ // considered authoritatively known.
306
+ //
307
+ // This node can never be put to unverified state.
308
+ //
309
+ // It is possible that the value this key is forced to is one that cannot be
310
+ // arrived at through the normal update sequence.
311
+ kForced ,
244
312
};
245
313
246
314
template <typename Sink>
@@ -258,6 +326,9 @@ class LazyDagCache {
258
326
case CacheState::kKnown :
259
327
absl::Format (&sink, " KNOWN" );
260
328
return ;
329
+ case CacheState::kForced :
330
+ absl::Format (&sink, " FORCED" );
331
+ return ;
261
332
}
262
333
LOG (FATAL) << " Unknown CacheState: " << static_cast <int >(state);
263
334
}
@@ -272,6 +343,14 @@ class LazyDagCache {
272
343
}
273
344
return it->second .state ;
274
345
}
346
+ // Get the cache state with forced values being considered kKnown
347
+ CacheState GetNonForcedCacheState (const Key& key) const {
348
+ CacheState s = GetCacheState (key);
349
+ if (s == CacheState::kForced ) {
350
+ return CacheState::kKnown ;
351
+ }
352
+ return s;
353
+ }
275
354
const Value* GetCachedValue (const Key& key) const {
276
355
auto it = cache_.find (key);
277
356
if (it == cache_.end () || it->second .value == nullptr ) {
@@ -320,9 +399,10 @@ absl::Status LazyDagCache<Key, Value>::CheckConsistency(
320
399
321
400
if (state == CacheState::kKnown ) {
322
401
for (const Key& input : provider_->GetInputs (key)) {
323
- XLS_RET_CHECK_EQ (GetCacheState (input), CacheState::kKnown )
402
+ XLS_RET_CHECK_EQ (GetNonForcedCacheState (input), CacheState::kKnown )
324
403
<< " Non-KNOWN input for KNOWN key " << provider_->GetName (key)
325
- << " : " << provider_->GetName (input);
404
+ << " : " << provider_->GetName (input) << " (input is "
405
+ << GetCacheState (input) << " )" ;
326
406
}
327
407
}
328
408
if (state == CacheState::kInputsUnverified ) {
@@ -333,6 +413,8 @@ absl::Status LazyDagCache<Key, Value>::CheckConsistency(
333
413
}
334
414
}
335
415
416
+ // NB state FORCED & UNVERIFIED has no requirements on its inputs.
417
+
336
418
if (absl::c_any_of (provider_->GetInputs (key), [&](const Key& input) {
337
419
return !correct_values.contains (input);
338
420
})) {
@@ -341,6 +423,12 @@ absl::Status LazyDagCache<Key, Value>::CheckConsistency(
341
423
continue ;
342
424
}
343
425
426
+ if (state == CacheState::kForced ) {
427
+ // Forced values are definitionally correct.
428
+ correct_values.insert (key);
429
+ continue ;
430
+ }
431
+
344
432
std::vector<const Value*> input_values;
345
433
absl::Span<const Key> inputs = provider_->GetInputs (key);
346
434
input_values.reserve (inputs.size ());
0 commit comments