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
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
|
/********************* */
/*! \file stacking_vector.h
** \verbatim
** Top contributors (to current version):
** Morgan Deters, Tim King, Dejan Jovanovic
** This file is part of the CVC4 project.
** Copyright (c) 2009-2016 by the authors listed in the file AUTHORS
** in the top-level source directory) and their institutional affiliations.
** All rights reserved. See the file COPYING in the top-level source
** directory for licensing information.\endverbatim
**
** \brief Backtrackable vector using an undo stack
**
** Backtrackable vector using an undo stack rather than storing items in
** a CDVector<>.
**/
#include "cvc4_private.h"
#ifndef __CVC4__CONTEXT__STACKING_VECTOR_H
#define __CVC4__CONTEXT__STACKING_VECTOR_H
#include <utility>
#include <vector>
#include "context/cdo.h"
namespace CVC4 {
namespace context {
template <class T>
class StackingVector : context::ContextNotifyObj {
/** Our underlying map type. */
typedef std::vector<T> VectorType;
/** Our map of indices to their values. */
VectorType d_map;
/** Our undo stack for changes made to d_map. */
std::vector<std::pair<size_t, T> > d_trace;
/** Our current offset in the d_trace stack (context-dependent). */
context::CDO<size_t> d_offset;
public:
typedef typename VectorType::const_iterator const_iterator;
StackingVector(context::Context* ctxt) :
context::ContextNotifyObj(ctxt),
d_offset(ctxt, 0) {
}
~StackingVector() throw() { }
/**
* Return a value from the vector. If n is not a key in
* the map, this function returns a default-constructed T.
*/
T operator[](size_t n) const {
return n < d_map.size() ? d_map[n] : T();
}
//T& operator[](ArgType n) { return d_map[n]; }// not permitted--bypasses set() logic
/**
* Set the value in the key-value map for Node n to newValue.
*/
void set(size_t n, const T& newValue);
/**
* Return the current size of the vector. Note that once a certain
* size is achieved, the size never goes down again, although the
* elements off the old end of the vector will be replaced with
* default-constructed T values.
*/
size_t size() const {
return d_map.size();
}
/**
* Called by the Context when a pop occurs. Cancels everything to the
* current context level. Overrides ContextNotifyObj::notify().
*/
void contextNotifyPop();
};/* class StackingVector<> */
template <class T>
void StackingVector<T>::set(size_t n, const T& newValue) {
Trace("sv") << "SV setting " << n << " : " << newValue << " @ " << d_trace.size() << std::endl;
if(n >= d_map.size()) {
d_map.resize(n + 1);
}
T& ref = d_map[n];
d_trace.push_back(std::make_pair(n, T(ref)));
d_offset = d_trace.size();
ref = newValue;
}
template <class T>
void StackingVector<T>::contextNotifyPop() {
Trace("sv") << "SV cancelling : " << d_offset << " < " << d_trace.size() << " ?" << std::endl;
while(d_offset < d_trace.size()) {
std::pair<size_t, T> p = d_trace.back();
Trace("sv") << "SV cancelling: " << p.first << " back to " << p.second << std::endl;
d_map[p.first] = p.second;
d_trace.pop_back();
}
Trace("sv") << "SV cancelling finished." << std::endl;
}
}/* CVC4::context namespace */
}/* CVC4 namespace */
#endif /*__CVC4__CONTEXT__STACKING_VECTOR_H */
|