Fix pointer tracking in AST serialization

Reviewers: teon.banek, llugovic

Reviewed By: teon.banek

Subscribers: pullbot

Differential Revision: https://phabricator.memgraph.io/D1640
This commit is contained in:
Marin Tomic 2018-10-09 16:43:55 +02:00
parent e4726f54ec
commit ce6085fa06
5 changed files with 136 additions and 119 deletions

View File

@ -272,6 +272,8 @@ produces:
;; Extra arguments to the generated save function. List of (name cpp-type). ;; Extra arguments to the generated save function. List of (name cpp-type).
(save-args nil :read-only t) (save-args nil :read-only t)
(load-args nil :read-only t) (load-args nil :read-only t)
;; Function to be called after saving the instance. Lambda taking builder name as only argument.
(post-save nil :read-only t)
(construct nil :read-only t) (construct nil :read-only t)
;; Explicit instantiation of template to generate schema with enum. ;; Explicit instantiation of template to generate schema with enum.
(type-args nil :read-only t) (type-args nil :read-only t)
@ -1019,7 +1021,11 @@ Proto schema."
(cpp-variable-name (first name-and-type))) (cpp-variable-name (first name-and-type)))
(capnp-extra-args (find-cpp-class parent) :save)))))) (capnp-extra-args (find-cpp-class parent) :save))))))
;; Save members ;; Save members
(write-string (capnp-save-members cpp-class builder :instance-access "self.") cpp-out)))) (write-string (capnp-save-members cpp-class builder :instance-access "self.") cpp-out)
;; Call post-save function if necessary
(let ((capnp-opts (cpp-class-capnp-opts cpp-class)))
(when (capnp-opts-post-save capnp-opts)
(write-string (cpp-code (funcall (capnp-opts-post-save capnp-opts) builder)) cpp-out))))))
(with-output-to-string (cpp-out) (with-output-to-string (cpp-out)
(let ((subclasses (capnp-union-subclasses cpp-class)) (let ((subclasses (capnp-union-subclasses cpp-class))
(builder (if (capnp-union-parents-rec cpp-class) (builder (if (capnp-union-parents-rec cpp-class)

View File

@ -5,7 +5,7 @@
namespace query { namespace query {
AstStorage::AstStorage() { AstStorage::AstStorage() {
std::unique_ptr<Query> root(new Query(next_uid_++)); std::unique_ptr<Query> root(new Query(++max_existing_uid_));
root_idx_ = 0; root_idx_ = 0;
storage_.emplace_back(std::move(root)); storage_.emplace_back(std::move(root));
} }

View File

@ -56,16 +56,14 @@ cpp<#
(defun load-ast-pointer (type) (defun load-ast-pointer (type)
(lambda (reader member capnp-name) (lambda (reader member capnp-name)
(let ((cpp-type (lcp::cpp-type-name type))) #>cpp
#>cpp if (${reader}.has${capnp-name}()) {
if (${reader}.has${capnp-name}()) { ${member} = static_cast<${type}>(
std::unique_ptr<${cpp-type}> tmp; Load(storage, ${reader}.get${capnp-name}(), loaded_uids));
Load(&tmp, ${reader}.get${capnp-name}(), storage, loaded_uids); } else {
${member} = storage->Take(std::move(tmp)); ${member} = nullptr;
} else { }
${member} = nullptr; cpp<#))
}
cpp<#)))
(defun save-ast-vector (type) (defun save-ast-vector (type)
(lcp:capnp-save-vector "capnp::Tree" type (lcp:capnp-save-vector "capnp::Tree" type
@ -78,11 +76,9 @@ cpp<#
(format (format
nil nil
"[storage, loaded_uids](const auto &reader) { "[storage, loaded_uids](const auto &reader) {
std::unique_ptr<~A> tmp; return static_cast<~A>(Load(storage, reader, loaded_uids));
Load(&tmp, reader, storage, loaded_uids);
return storage->Take(std::move(tmp));
}" }"
(remove #\* type)))) type)))
(defun save-property-map (builder member capnp-name) (defun save-property-map (builder member capnp-name)
#>cpp #>cpp
@ -102,13 +98,11 @@ cpp<#
(defun load-property-map (reader member capnp-name) (defun load-property-map (reader member capnp-name)
#>cpp #>cpp
for (const auto &entry : ${reader}.getEntries()) { for (const auto &entry : ${reader}.getEntries()) {
std::unique_ptr<Expression> value;
Load(&value, entry.getValue(), storage, loaded_uids);
std::string prop_name = entry.getKey().getFirst(); std::string prop_name = entry.getKey().getFirst();
storage::Property prop_id; storage::Property prop_id;
storage::Load(&prop_id, entry.getKey().getSecond()); storage::Load(&prop_id, entry.getKey().getSecond());
${member}.emplace(std::make_pair(prop_name, prop_id), ${member}.emplace(std::make_pair(prop_name, prop_id),
storage->Take(std::move(value))); static_cast<Expression *>(Load(storage, entry.getValue(), loaded_uids)));
} }
cpp<#) cpp<#)
@ -144,26 +138,20 @@ class AstStorage {
template <typename T, typename... Args> template <typename T, typename... Args>
T *Create(Args &&... args) { T *Create(Args &&... args) {
// Never call create for a Query. Call query() instead. T *ptr = new T(++max_existing_uid_, std::forward<Args>(args)...);
static_assert(!std::is_same<T, Query>::value, "Call query() instead");
T *ptr = new T(next_uid_++, std::forward<Args>(args)...);
std::unique_ptr<T> tmp(ptr); std::unique_ptr<T> tmp(ptr);
if (std::is_same<T, Query>::value) {
root_idx_ = storage_.size();
}
storage_.emplace_back(std::move(tmp)); storage_.emplace_back(std::move(tmp));
return ptr; return ptr;
} }
template <typename T>
T *Take(std::unique_ptr<T> &&node) {
T *ret = node.get();
storage_.emplace_back(std::move(node));
return ret;
}
Query *query() const; Query *query() const;
// Public only for serialization access // Public only for serialization access
std::vector<std::unique_ptr<Tree>> storage_; std::vector<std::unique_ptr<Tree>> storage_;
int next_uid_ = 0; int max_existing_uid_ = -1;
size_t root_idx_; size_t root_idx_;
}; };
cpp<# cpp<#
@ -192,7 +180,18 @@ cpp<#
(:serialize :capnp :base t (:serialize :capnp :base t
:save-args '((saved-uids "std::vector<int32_t> *")) :save-args '((saved-uids "std::vector<int32_t> *"))
:load-args '((storage "AstStorage *") :load-args '((storage "AstStorage *")
(loaded-uids "std::vector<int32_t> *")))) (loaded-uids "std::vector<int32_t> *"))
:post-save (lambda (builder)
(declare (ignore builder))
;; This is a bit hacky because it relies on the fact that parent class
;; serialization is inlined so we can short-circuit and avoid serializing
;; the derived class.
#>cpp
if (utils::Contains(*saved_uids, self.uid_)) {
return;
}
saved_uids->push_back(self.uid_);
cpp<#)))
(lcp:define-class expression (tree) (lcp:define-class expression (tree)
() ()
@ -217,7 +216,7 @@ cpp<#
((expression "Expression *" :initval "nullptr" :scope :public ((expression "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression"))) :capnp-load (load-ast-pointer "Expression *")))
(:public (:public
#>cpp #>cpp
Where() = default; Where() = default;
@ -249,11 +248,11 @@ cpp<#
((expression1 "Expression *" :initval "nullptr" :scope :public ((expression1 "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(expression2 "Expression *" :initval "nullptr" :scope :public (expression2 "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression"))) :capnp-load (load-ast-pointer "Expression *")))
(:abstractp t) (:abstractp t)
(:public (:public
#>cpp #>cpp
@ -277,7 +276,7 @@ cpp<#
((expression "Expression *" :initval "nullptr" :scope :public ((expression "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression"))) :capnp-load (load-ast-pointer "Expression *")))
(:abstractp t) (:abstractp t)
(:public (:public
#>cpp #>cpp
@ -426,15 +425,15 @@ cpp<#
((list "Expression *" :initval "nullptr" :scope :public ((list "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(lower-bound "Expression *" :initval "nullptr" :scope :public (lower-bound "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(upper-bound "Expression *" :initval "nullptr" :scope :public (upper-bound "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression"))) :capnp-load (load-ast-pointer "Expression *")))
(:public (:public
#>cpp #>cpp
ListSlicingOperator() = default; ListSlicingOperator() = default;
@ -479,16 +478,16 @@ cpp<#
((condition "Expression *" :scope :public ((condition "Expression *" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression") :capnp-load (load-ast-pointer "Expression *")
:documentation "None of the expressions should be nullptr. If there is no else_expression, you should make it null PrimitiveLiteral.") :documentation "None of the expressions should be nullptr. If there is no else_expression, you should make it null PrimitiveLiteral.")
(then-expression "Expression *" :scope :public (then-expression "Expression *" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(else-expression "Expression *" :scope :public (else-expression "Expression *" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression"))) :capnp-load (load-ast-pointer "Expression *")))
(:public (:public
#>cpp #>cpp
IfOperator() = default; IfOperator() = default;
@ -689,7 +688,7 @@ cpp<#
((expression "Expression *" :initval "nullptr" :scope :public ((expression "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(property-name "std::string" :scope :public) (property-name "std::string" :scope :public)
(property "storage::Property" :scope :public)) (property "storage::Property" :scope :public))
(:public (:public
@ -734,7 +733,7 @@ cpp<#
((expression "Expression *" :initval "nullptr" :scope :public ((expression "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(labels "std::vector<storage::Label>" :scope :public (labels "std::vector<storage::Label>" :scope :public
:capnp-save (lcp:capnp-save-vector :capnp-save (lcp:capnp-save-vector
"storage::capnp::Common" "storage::capnp::Common"
@ -829,27 +828,27 @@ cpp<#
((accumulator "Identifier *" :initval "nullptr" :scope :public ((accumulator "Identifier *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Identifier") :capnp-load (load-ast-pointer "Identifier *")
:documentation "Identifier for the accumulating variable") :documentation "Identifier for the accumulating variable")
(initializer "Expression *" :initval "nullptr" :scope :public (initializer "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression") :capnp-load (load-ast-pointer "Expression *")
:documentation "Expression which produces the initial accumulator value.") :documentation "Expression which produces the initial accumulator value.")
(identifier "Identifier *" :initval "nullptr" :scope :public (identifier "Identifier *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Identifier") :capnp-load (load-ast-pointer "Identifier *")
:documentation "Identifier for the list element.") :documentation "Identifier for the list element.")
(list "Expression *" :initval "nullptr" :scope :public (list "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression") :capnp-load (load-ast-pointer "Expression *")
:documentation "Expression which produces a list to be reduced.") :documentation "Expression which produces a list to be reduced.")
(expression "Expression *" :initval "nullptr" :scope :public (expression "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression") :capnp-load (load-ast-pointer "Expression *")
:documentation "Expression which does the reduction, i.e. produces the new accumulator value.")) :documentation "Expression which does the reduction, i.e. produces the new accumulator value."))
(:public (:public
#>cpp #>cpp
@ -893,17 +892,17 @@ cpp<#
((identifier "Identifier *" :initval "nullptr" :scope :public ((identifier "Identifier *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Identifier") :capnp-load (load-ast-pointer "Identifier *")
:documentation "Identifier for the list element.") :documentation "Identifier for the list element.")
(list "Expression *" :initval "nullptr" :scope :public (list "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression") :capnp-load (load-ast-pointer "Expression *")
:documentation "Expression which produces a list which will be extracted.") :documentation "Expression which produces a list which will be extracted.")
(expression "Expression *" :initval "nullptr" :scope :public (expression "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression") :capnp-load (load-ast-pointer "Expression *")
:documentation "Expression which produces the new value for list element.")) :documentation "Expression which produces the new value for list element."))
(:public (:public
#>cpp #>cpp
@ -943,15 +942,15 @@ cpp<#
((identifier "Identifier *" :initval "nullptr" :scope :public ((identifier "Identifier *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Identifier")) :capnp-load (load-ast-pointer "Identifier *"))
(list-expression "Expression *" :initval "nullptr" :scope :public (list-expression "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(where "Where *" :initval "nullptr" :scope :public (where "Where *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Where"))) :capnp-load (load-ast-pointer "Where *")))
(:public (:public
#>cpp #>cpp
All() = default; All() = default;
@ -993,15 +992,15 @@ cpp<#
((identifier "Identifier *" :initval "nullptr" :scope :public ((identifier "Identifier *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Identifier")) :capnp-load (load-ast-pointer "Identifier *"))
(list-expression "Expression *" :initval "nullptr" :scope :public (list-expression "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(where "Where *" :initval "nullptr" :scope :public (where "Where *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Where"))) :capnp-load (load-ast-pointer "Where *")))
(:public (:public
#>cpp #>cpp
Single() = default; Single() = default;
@ -1067,7 +1066,7 @@ cpp<#
(expression "Expression *" :initval "nullptr" :scope :public (expression "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(token-position :int32_t :initval -1 :scope :public (token-position :int32_t :initval -1 :scope :public
:documentation "This field contains token position of first token in named expression used to create name_. If NamedExpression object is not created from query or it is aliased leave this value at -1.")) :documentation "This field contains token position of first token in named expression used to create name_. If NamedExpression object is not created from query or it is aliased leave this value at -1."))
(:public (:public
@ -1110,7 +1109,7 @@ cpp<#
((identifier "Identifier *" :initval "nullptr" :scope :public ((identifier "Identifier *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Identifier"))) :capnp-load (load-ast-pointer "Identifier *")))
(:abstractp t) (:abstractp t)
(:public (:public
#>cpp #>cpp
@ -1195,12 +1194,12 @@ cpp<#
(lower-bound "Expression *" :initval "nullptr" :scope :public (lower-bound "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression") :capnp-load (load-ast-pointer "Expression *")
:documentation "Evaluates to lower bound in variable length expands.") :documentation "Evaluates to lower bound in variable length expands.")
(upper-bound "Expression *" :initval "nullptr" :scope :public (upper-bound "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression") :capnp-load (load-ast-pointer "Expression *")
:documentation "Evaluated to upper bound in variable length expands.") :documentation "Evaluated to upper bound in variable length expands.")
(filter-lambda "Lambda" :scope :public (filter-lambda "Lambda" :scope :public
:documentation "Filter lambda for variable length expands. Can have an empty expression, but identifiers must be valid, because an optimization pass may inline other expressions into this lambda.") :documentation "Filter lambda for variable length expands. Can have an empty expression, but identifiers must be valid, because an optimization pass may inline other expressions into this lambda.")
@ -1209,7 +1208,7 @@ cpp<#
(total-weight "Identifier *" :initval "nullptr" :scope :public (total-weight "Identifier *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Identifier") :capnp-load (load-ast-pointer "Identifier *")
:documentation "Variable where the total weight for weighted shortest path will be stored.")) :documentation "Variable where the total weight for weighted shortest path will be stored."))
(:public (:public
(lcp:define-enum type (lcp:define-enum type
@ -1222,17 +1221,17 @@ cpp<#
((inner-edge "Identifier *" :initval "nullptr" ((inner-edge "Identifier *" :initval "nullptr"
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Identifier") :capnp-load (load-ast-pointer "Identifier *")
:documentation "Argument identifier for the edge currently being traversed.") :documentation "Argument identifier for the edge currently being traversed.")
(inner-node "Identifier *" :initval "nullptr" (inner-node "Identifier *" :initval "nullptr"
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Identifier") :capnp-load (load-ast-pointer "Identifier *")
:documentation "Argument identifier for the destination node of the edge.") :documentation "Argument identifier for the destination node of the edge.")
(expression "Expression *" :initval "nullptr" (expression "Expression *" :initval "nullptr"
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression") :capnp-load (load-ast-pointer "Expression *")
:documentation "Evaluates the result of the lambda.")) :documentation "Evaluates the result of the lambda."))
(:documentation "Lambda for use in filtering or weight calculation during variable expand.") (:documentation "Lambda for use in filtering or weight calculation during variable expand.")
(:serialize :capnp (:serialize :capnp
@ -1323,7 +1322,7 @@ cpp<#
((identifier "Identifier *" :initval "nullptr" :scope :public ((identifier "Identifier *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Identifier")) :capnp-load (load-ast-pointer "Identifier *"))
(atoms "std::vector<PatternAtom *>" (atoms "std::vector<PatternAtom *>"
:scope :public :scope :public
:capnp-type "List(Tree)" :capnp-type "List(Tree)"
@ -1426,7 +1425,7 @@ cpp<#
((single-query "SingleQuery *" :initval "nullptr" :scope :public ((single-query "SingleQuery *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "SingleQuery")) :capnp-load (load-ast-pointer "SingleQuery *"))
(distinct :bool :initval "false" :scope :public) (distinct :bool :initval "false" :scope :public)
(union-symbols "std::vector<Symbol>" :scope :public (union-symbols "std::vector<Symbol>" :scope :public
:documentation "Holds symbols that are created during symbol generation phase. These symbols are used when UNION/UNION ALL combines single query results.")) :documentation "Holds symbols that are created during symbol generation phase. These symbols are used when UNION/UNION ALL combines single query results."))
@ -1472,7 +1471,7 @@ cpp<#
(single-query "SingleQuery *" :initval "nullptr" :scope :public (single-query "SingleQuery *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "SingleQuery") :capnp-load (load-ast-pointer "SingleQuery *")
:documentation "First and potentially only query.") :documentation "First and potentially only query.")
(cypher-unions "std::vector<CypherUnion *>" (cypher-unions "std::vector<CypherUnion *>"
:scope :public :scope :public
@ -1499,7 +1498,7 @@ cpp<#
// Creates deep copy of whole ast. // Creates deep copy of whole ast.
Query *Clone(AstStorage &storage) const override { Query *Clone(AstStorage &storage) const override {
auto *query = storage.query(); auto *query = storage.Create<Query>();
query->single_query_ = single_query_->Clone(storage); query->single_query_ = single_query_->Clone(storage);
for (auto *cypher_union : cypher_unions_) { for (auto *cypher_union : cypher_unions_) {
query->cypher_unions_.push_back(cypher_union->Clone(storage)); query->cypher_unions_.push_back(cypher_union->Clone(storage));
@ -1567,7 +1566,7 @@ cpp<#
(where "Where *" :initval "nullptr" :scope :public (where "Where *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Where")) :capnp-load (load-ast-pointer "Where *"))
(optional :bool :initval "false" :scope :public)) (optional :bool :initval "false" :scope :public))
(:public (:public
#>cpp #>cpp
@ -1622,7 +1621,7 @@ cpp<#
(expression "Expression *" (expression "Expression *"
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression"))) :capnp-load (load-ast-pointer "Expression *")))
(:serialize :capnp (:serialize :capnp
:save-args '((saved-uids "std::vector<int32_t> *")) :save-args '((saved-uids "std::vector<int32_t> *"))
:load-args '((storage "AstStorage *") :load-args '((storage "AstStorage *")
@ -1658,12 +1657,12 @@ cpp<#
(skip "Expression *" :initval "nullptr" (skip "Expression *" :initval "nullptr"
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression") :capnp-load (load-ast-pointer "Expression *")
:documentation "Optional expression on how many results to skip.") :documentation "Optional expression on how many results to skip.")
(limit "Expression *" :initval "nullptr" (limit "Expression *" :initval "nullptr"
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression") :capnp-load (load-ast-pointer "Expression *")
:documentation "Optional expression on how many results to produce.")) :documentation "Optional expression on how many results to produce."))
(:documentation "Contents common to @c Return and @c With clauses.") (:documentation "Contents common to @c Return and @c With clauses.")
(:serialize :capnp (:serialize :capnp
@ -1730,7 +1729,7 @@ cpp<#
(where "Where *" :initval "nullptr" :scope :public (where "Where *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Where"))) :capnp-load (load-ast-pointer "Where *")))
(:public (:public
#>cpp #>cpp
With() = default; With() = default;
@ -1825,11 +1824,11 @@ cpp<#
((property-lookup "PropertyLookup *" :initval "nullptr" :scope :public ((property-lookup "PropertyLookup *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "PropertyLookup")) :capnp-load (load-ast-pointer "PropertyLookup *"))
(expression "Expression *" :initval "nullptr" :scope :public (expression "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression"))) :capnp-load (load-ast-pointer "Expression *")))
(:public (:public
#>cpp #>cpp
SetProperty() = default; SetProperty() = default;
@ -1865,11 +1864,11 @@ cpp<#
((identifier "Identifier *" :initval "nullptr" :scope :public ((identifier "Identifier *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Identifier")) :capnp-load (load-ast-pointer "Identifier *"))
(expression "Expression *" :initval "nullptr" :scope :public (expression "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(update :bool :initval "false" :scope :public)) (update :bool :initval "false" :scope :public))
(:public (:public
#>cpp #>cpp
@ -1908,7 +1907,7 @@ cpp<#
((identifier "Identifier *" :initval "nullptr" :scope :public ((identifier "Identifier *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Identifier")) :capnp-load (load-ast-pointer "Identifier *"))
(labels "std::vector<storage::Label>" :scope :public (labels "std::vector<storage::Label>" :scope :public
:capnp-save (lcp:capnp-save-vector :capnp-save (lcp:capnp-save-vector
"storage::capnp::Common" "storage::capnp::Common"
@ -1949,7 +1948,7 @@ cpp<#
((property-lookup "PropertyLookup *" :initval "nullptr" :scope :public ((property-lookup "PropertyLookup *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "PropertyLookup"))) :capnp-load (load-ast-pointer "PropertyLookup *")))
(:public (:public
#>cpp #>cpp
RemoveProperty() = default; RemoveProperty() = default;
@ -1982,7 +1981,7 @@ cpp<#
((identifier "Identifier *" :initval "nullptr" :scope :public ((identifier "Identifier *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Identifier")) :capnp-load (load-ast-pointer "Identifier *"))
(labels "std::vector<storage::Label>" :scope :public (labels "std::vector<storage::Label>" :scope :public
:capnp-save (lcp:capnp-save-vector :capnp-save (lcp:capnp-save-vector
"storage::capnp::Common" "storage::capnp::Common"
@ -2023,7 +2022,7 @@ cpp<#
((pattern "Pattern *" :initval "nullptr" :scope :public ((pattern "Pattern *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Pattern")) :capnp-load (load-ast-pointer "Pattern *"))
(on-match "std::vector<Clause *>" (on-match "std::vector<Clause *>"
:scope :public :scope :public
:capnp-type "List(Tree)" :capnp-type "List(Tree)"
@ -2094,7 +2093,7 @@ cpp<#
((named-expression "NamedExpression *" :initval "nullptr" :scope :public ((named-expression "NamedExpression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "NamedExpression"))) :capnp-load (load-ast-pointer "NamedExpression *")))
(:public (:public
#>cpp #>cpp
Unwind() = default; Unwind() = default;
@ -2161,7 +2160,7 @@ cpp<#
(password "Expression *" :initval "nullptr" :scope :public (password "Expression *" :initval "nullptr" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(privileges "std::vector<Privilege>" :scope :public (privileges "std::vector<Privilege>" :scope :public
:capnp-save (lambda (builder member capnp-name) :capnp-save (lambda (builder member capnp-name)
#>cpp #>cpp
@ -2293,23 +2292,23 @@ cpp<#
(stream-uri "Expression *" :scope :public (stream-uri "Expression *" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(stream-topic "Expression *" :scope :public (stream-topic "Expression *" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(transform-uri "Expression *" :scope :public (transform-uri "Expression *" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(batch-interval-in-ms "Expression *" :scope :public (batch-interval-in-ms "Expression *" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression")) :capnp-load (load-ast-pointer "Expression *"))
(batch-size "Expression *" :scope :public (batch-size "Expression *" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression"))) :capnp-load (load-ast-pointer "Expression *")))
(:public (:public
#>cpp #>cpp
CreateStream() = default; CreateStream() = default;
@ -2399,7 +2398,7 @@ cpp<#
(limit-batches "Expression *" :scope :public (limit-batches "Expression *" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression"))) :capnp-load (load-ast-pointer "Expression *")))
(:public (:public
#>cpp #>cpp
StartStopStream() = default; StartStopStream() = default;
@ -2460,7 +2459,7 @@ cpp<#
(limit-batches "Expression *" :scope :public (limit-batches "Expression *" :scope :public
:capnp-type "Tree" :capnp-init nil :capnp-type "Tree" :capnp-init nil
:capnp-save #'save-ast-pointer :capnp-save #'save-ast-pointer
:capnp-load (load-ast-pointer "Expression"))) :capnp-load (load-ast-pointer "Expression *")))
(:public (:public
#>cpp #>cpp
TestStream() = default; TestStream() = default;

View File

@ -20,12 +20,26 @@ cpp<#
#>cpp #>cpp
Tree *Load(AstStorage *ast, const capnp::Tree::Reader &tree, Tree *Load(AstStorage *ast, const capnp::Tree::Reader &tree,
std::vector<int> *loaded_uids) { std::vector<int> *loaded_uids) {
ast->storage_.clear(); // Check if element already deserialized and if yes, return existing
// element from storage.
auto uid = tree.getUid();
if (utils::Contains(*loaded_uids, uid)) {
auto found = std::find_if(ast->storage_.begin(), ast->storage_.end(),
[&](const auto &n) { return n->uid_ == uid; });
CHECK(found != ast->storage_.end());
return found->get();
}
std::unique_ptr<Tree> root; std::unique_ptr<Tree> root;
::query::Load(&root, tree, ast, loaded_uids); ::query::Load(&root, tree, ast, loaded_uids);
ast->root_idx_ = ast->storage_.size();
if (dynamic_cast<Query *>(root.get())) {
ast->root_idx_ = ast->storage_.size();
}
ast->storage_.emplace_back(std::move(root)); ast->storage_.emplace_back(std::move(root));
return ast->storage_[ast->root_idx_].get(); loaded_uids->emplace_back(uid);
ast->max_existing_uid_ = std::max(ast->max_existing_uid_, uid);
return ast->storage_.back().get();
} }
cpp<#) cpp<#)

View File

@ -245,16 +245,15 @@ can serve as inputs to others and thus a sequence of operations is formed.")
(defun load-ast-pointer (ast-type) (defun load-ast-pointer (ast-type)
(lambda (reader member capnp-name) (lambda (reader member capnp-name)
(let ((cpp-type (remove #\* ast-type))) #>cpp
#>cpp if (${reader}.has${capnp-name}()) {
if (${reader}.has${capnp-name}()) { ${member} = static_cast<${ast-type}>(Load(&helper->ast_storage,
std::unique_ptr<${cpp-type}> tmp; ${reader}.get${capnp-name}(),
Load(&tmp, ${reader}.get${capnp-name}(), &helper->ast_storage, &helper->loaded_ast_uids); &helper->loaded_ast_uids));
${member} = helper->ast_storage.Take(std::move(tmp)); } else {
} else { ${member} = nullptr;
${member} = nullptr; }
} cpp<#))
cpp<#)))
(defun save-ast-vector (ast-type) (defun save-ast-vector (ast-type)
(lcp:capnp-save-vector "::query::capnp::Tree" ast-type (lcp:capnp-save-vector "::query::capnp::Tree" ast-type
@ -267,11 +266,10 @@ can serve as inputs to others and thus a sequence of operations is formed.")
(format (format
nil nil
"[helper](const auto &reader) { "[helper](const auto &reader) {
std::unique_ptr<~A> tmp; return static_cast<~A>(Load(&helper->ast_storage, reader,
Load(&tmp, reader, &helper->ast_storage, &helper->loaded_ast_uids); &helper->loaded_ast_uids));
return helper->ast_storage.Take(std::move(tmp));
}" }"
(remove #\* ast-type)))) ast-type)))
(defun save-operator-pointer (builder member-name capnp-name) (defun save-operator-pointer (builder member-name capnp-name)
(declare (ignore capnp-name)) (declare (ignore capnp-name))
@ -563,14 +561,15 @@ given label.
(defun load-optional-bound (reader member capnp-name) (defun load-optional-bound (reader member capnp-name)
(let ((load-bound (let ((load-bound
"[helper](const auto &reader) { "[helper](const auto &reader) {
auto type = reader.getType() == ::utils::capnp::Bound<::query::capnp::Tree>::Type::INCLUSIVE auto type = reader.getType() ==
? utils::BoundType::INCLUSIVE : utils::BoundType::EXCLUSIVE; ::utils::capnp::Bound<::query::capnp::Tree>::Type::INCLUSIVE
std::unique_ptr<Tree> tmp; ? utils::BoundType::INCLUSIVE
Load(&tmp, reader.getValue(), &helper->ast_storage, &helper->loaded_ast_uids); : utils::BoundType::EXCLUSIVE;
auto *value = static_cast<Expression *>(helper->ast_storage.Take(std::move(tmp))); auto *value = static_cast<Expression *>(
return utils::Bound<Expression *>(value, type); Load(&helper->ast_storage, reader.getValue(), &helper->loaded_ast_uids));
}")) return utils::Bound<Expression *>(value, type);
}"))
(funcall (lcp:capnp-load-optional "::utils::capnp::Bound<::query::capnp::Tree>" (funcall (lcp:capnp-load-optional "::utils::capnp::Bound<::query::capnp::Tree>"
"utils::Bound<Expression *>" "utils::Bound<Expression *>"
load-bound) load-bound)
@ -834,9 +833,8 @@ pulled.")
:capnp-load (lambda (reader member capnp-name) :capnp-load (lambda (reader member capnp-name)
#>cpp #>cpp
if (${reader}.hasExpression()) { if (${reader}.hasExpression()) {
std::unique_ptr<Expression> tmp; ${member} = static_cast<Expression *>(
Load(&tmp, ${reader}.getExpression(), ast_storage, loaded_ast_uids); Load(ast_storage, ${reader}.getExpression(), loaded_ast_uids));
${member} = ast_storage->Take(std::move(tmp));
} else { } else {
${member} = nullptr; ${member} = nullptr;
} }