Refactored call expression context in sequence diagram visitor
This commit is contained in:
188
src/sequence_diagram/visitor/call_expression_context.cc
Normal file
188
src/sequence_diagram/visitor/call_expression_context.cc
Normal file
@@ -0,0 +1,188 @@
|
|||||||
|
/**
|
||||||
|
* src/sequence_diagram/visitor/call_expression_context.cc
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021-2022 Bartek Kryza <bkryza@gmail.com>
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "call_expression_context.h"
|
||||||
|
|
||||||
|
namespace clanguml::sequence_diagram::visitor {
|
||||||
|
|
||||||
|
call_expression_context::call_expression_context()
|
||||||
|
: current_class_decl_{nullptr}
|
||||||
|
, current_class_template_decl_{nullptr}
|
||||||
|
, current_class_template_specialization_decl_{nullptr}
|
||||||
|
, current_method_decl_{nullptr}
|
||||||
|
, current_function_decl_{nullptr}
|
||||||
|
, current_function_template_decl_{nullptr}
|
||||||
|
, current_caller_id_{0}
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void call_expression_context::reset()
|
||||||
|
{
|
||||||
|
current_caller_id_ = 0;
|
||||||
|
current_class_decl_ = nullptr;
|
||||||
|
current_class_template_decl_ = nullptr;
|
||||||
|
current_class_template_specialization_decl_ = nullptr;
|
||||||
|
current_method_decl_ = nullptr;
|
||||||
|
current_function_decl_ = nullptr;
|
||||||
|
current_function_template_decl_ = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void call_expression_context::dump()
|
||||||
|
{
|
||||||
|
LOG_DBG("current_caller_id_ = {}", current_caller_id_);
|
||||||
|
LOG_DBG("current_class_decl_ = {}", (void *)current_class_decl_);
|
||||||
|
LOG_DBG("current_class_template_decl_ = {}",
|
||||||
|
(void *)current_class_template_decl_);
|
||||||
|
LOG_DBG("current_class_template_specialization_decl_ = {}",
|
||||||
|
(void *)current_class_template_specialization_decl_);
|
||||||
|
LOG_DBG("current_method_decl_ = {}", (void *)current_method_decl_);
|
||||||
|
LOG_DBG("current_function_decl_ = {}", (void *)current_function_decl_);
|
||||||
|
LOG_DBG("current_function_template_decl_ = {}",
|
||||||
|
(void *)current_function_template_decl_);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool call_expression_context::valid() const
|
||||||
|
{
|
||||||
|
return (current_class_decl_ != nullptr) ||
|
||||||
|
(current_class_template_decl_ != nullptr) ||
|
||||||
|
(current_class_template_specialization_decl_ != nullptr) ||
|
||||||
|
(current_method_decl_ != nullptr) ||
|
||||||
|
(current_function_decl_ != nullptr) ||
|
||||||
|
(current_function_template_decl_ != nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
clang::ASTContext *call_expression_context::get_ast_context()
|
||||||
|
{
|
||||||
|
if (current_class_template_specialization_decl_)
|
||||||
|
return ¤t_class_template_specialization_decl_->getASTContext();
|
||||||
|
|
||||||
|
if (current_class_template_decl_)
|
||||||
|
return ¤t_class_template_decl_->getASTContext();
|
||||||
|
|
||||||
|
if (current_class_decl_)
|
||||||
|
return ¤t_class_decl_->getASTContext();
|
||||||
|
|
||||||
|
if (current_function_template_decl_)
|
||||||
|
return ¤t_function_template_decl_->getASTContext();
|
||||||
|
|
||||||
|
return ¤t_function_decl_->getASTContext();
|
||||||
|
}
|
||||||
|
|
||||||
|
void call_expression_context::update(clang::CXXRecordDecl *cls)
|
||||||
|
{
|
||||||
|
current_class_decl_ = cls;
|
||||||
|
}
|
||||||
|
|
||||||
|
void call_expression_context::update(
|
||||||
|
clang::ClassTemplateSpecializationDecl *clst)
|
||||||
|
{
|
||||||
|
current_class_template_specialization_decl_ = clst;
|
||||||
|
}
|
||||||
|
|
||||||
|
void call_expression_context::update(clang::ClassTemplateDecl *clst)
|
||||||
|
{
|
||||||
|
current_class_template_decl_ = clst;
|
||||||
|
}
|
||||||
|
|
||||||
|
void call_expression_context::update(clang::CXXMethodDecl *method)
|
||||||
|
{
|
||||||
|
current_method_decl_ = method;
|
||||||
|
}
|
||||||
|
|
||||||
|
void call_expression_context::update(clang::FunctionDecl *function)
|
||||||
|
{
|
||||||
|
if (!function->isCXXClassMember())
|
||||||
|
reset();
|
||||||
|
|
||||||
|
current_function_decl_ = function;
|
||||||
|
|
||||||
|
// Check if this function is a part of template function declaration,
|
||||||
|
// If no - reset the current_function_template_decl_
|
||||||
|
if (current_function_template_decl_ &&
|
||||||
|
current_function_template_decl_->getQualifiedNameAsString() !=
|
||||||
|
function->getQualifiedNameAsString()) {
|
||||||
|
current_function_template_decl_ = nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void call_expression_context::update(
|
||||||
|
clang::FunctionTemplateDecl *function_template)
|
||||||
|
{
|
||||||
|
current_function_template_decl_ = function_template;
|
||||||
|
|
||||||
|
if (!function_template->isCXXClassMember())
|
||||||
|
current_class_decl_ = nullptr;
|
||||||
|
|
||||||
|
current_function_template_decl_ = function_template;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool call_expression_context::in_class_method() const
|
||||||
|
{
|
||||||
|
return current_class_decl_ != nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool call_expression_context::in_function() const
|
||||||
|
{
|
||||||
|
return current_class_decl_ == nullptr && current_function_decl_ != nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool call_expression_context::in_function_template() const
|
||||||
|
{
|
||||||
|
return current_function_decl_ != nullptr &&
|
||||||
|
current_function_template_decl_ != nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::int64_t call_expression_context::caller_id() const
|
||||||
|
{
|
||||||
|
return current_caller_id_;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::int64_t call_expression_context::lambda_caller_id() const
|
||||||
|
{
|
||||||
|
if (current_lambda_caller_id_.empty())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return current_lambda_caller_id_.top();
|
||||||
|
}
|
||||||
|
|
||||||
|
void call_expression_context::set_caller_id(std::int64_t id)
|
||||||
|
{
|
||||||
|
LOG_DBG("Setting current caller id to {}", id);
|
||||||
|
current_caller_id_ = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
void call_expression_context::enter_lambda_expression(std::int64_t id)
|
||||||
|
{
|
||||||
|
LOG_DBG("Setting current lambda caller id to {}", id);
|
||||||
|
|
||||||
|
assert(id != 0);
|
||||||
|
|
||||||
|
current_lambda_caller_id_.push(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
void call_expression_context::leave_lambda_expression()
|
||||||
|
{
|
||||||
|
assert(!current_lambda_caller_id_.empty());
|
||||||
|
|
||||||
|
LOG_DBG("Leaving current lambda expression id to {}",
|
||||||
|
current_lambda_caller_id_.top());
|
||||||
|
|
||||||
|
current_lambda_caller_id_.pop();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
86
src/sequence_diagram/visitor/call_expression_context.h
Normal file
86
src/sequence_diagram/visitor/call_expression_context.h
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
/**
|
||||||
|
* src/sequence_diagram/visitor/call_expression_context.h
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021-2022 Bartek Kryza <bkryza@gmail.com>
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
//
|
||||||
|
//#include "common/visitor/translation_unit_visitor.h"
|
||||||
|
//#include "config/config.h"
|
||||||
|
//#include "sequence_diagram/model/diagram.h"
|
||||||
|
|
||||||
|
#include "util/util.h"
|
||||||
|
|
||||||
|
#include <clang/AST/Expr.h>
|
||||||
|
#include <clang/AST/RecursiveASTVisitor.h>
|
||||||
|
#include <clang/Basic/SourceManager.h>
|
||||||
|
|
||||||
|
#include <stack>
|
||||||
|
|
||||||
|
namespace clanguml::sequence_diagram::visitor {
|
||||||
|
|
||||||
|
struct call_expression_context {
|
||||||
|
call_expression_context();
|
||||||
|
|
||||||
|
void reset();
|
||||||
|
|
||||||
|
void dump();
|
||||||
|
|
||||||
|
bool valid() const;
|
||||||
|
|
||||||
|
clang::ASTContext *get_ast_context();
|
||||||
|
|
||||||
|
void update(clang::CXXRecordDecl *cls);
|
||||||
|
|
||||||
|
void update(clang::ClassTemplateSpecializationDecl *clst);
|
||||||
|
|
||||||
|
void update(clang::ClassTemplateDecl *clst);
|
||||||
|
|
||||||
|
void update(clang::CXXMethodDecl *method);
|
||||||
|
|
||||||
|
void update(clang::FunctionDecl *function);
|
||||||
|
|
||||||
|
void update(clang::FunctionTemplateDecl *function_template);
|
||||||
|
|
||||||
|
bool in_class_method() const;
|
||||||
|
|
||||||
|
bool in_function() const;
|
||||||
|
|
||||||
|
bool in_function_template() const;
|
||||||
|
|
||||||
|
std::int64_t caller_id() const;
|
||||||
|
|
||||||
|
std::int64_t lambda_caller_id() const;
|
||||||
|
|
||||||
|
void set_caller_id(std::int64_t id);
|
||||||
|
|
||||||
|
void enter_lambda_expression(std::int64_t id);
|
||||||
|
|
||||||
|
void leave_lambda_expression();
|
||||||
|
|
||||||
|
clang::CXXRecordDecl *current_class_decl_;
|
||||||
|
clang::ClassTemplateDecl *current_class_template_decl_;
|
||||||
|
clang::ClassTemplateSpecializationDecl
|
||||||
|
*current_class_template_specialization_decl_;
|
||||||
|
clang::CXXMethodDecl *current_method_decl_;
|
||||||
|
clang::FunctionDecl *current_function_decl_;
|
||||||
|
clang::FunctionTemplateDecl *current_function_template_decl_;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::int64_t current_caller_id_;
|
||||||
|
std::stack<std::int64_t> current_lambda_caller_id_;
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
||||||
@@ -1457,41 +1457,6 @@ void translation_unit_visitor::process_template_specialization_argument(
|
|||||||
|
|
||||||
cls->getLocation().dump(source_manager());
|
cls->getLocation().dump(source_manager());
|
||||||
}
|
}
|
||||||
// else if (arg.getKind() == clang::TemplateArgument::Expression) {
|
|
||||||
// if (clang::dyn_cast<clang::LambdaExpr>(arg.getAsExpr()) !=
|
|
||||||
// nullptr) {
|
|
||||||
// class_diagram::model::template_parameter argument;
|
|
||||||
//// const auto location =
|
|
||||||
//// arg.getAsType()->getAsCXXRecordDecl()->getLocation();
|
|
||||||
////
|
|
||||||
//// auto type_name = fmt::format("(lambda {}:{}:{})",
|
|
||||||
//// source_manager().getFilename(location).str(),
|
|
||||||
//// source_manager().getSpellingLineNumber(location),
|
|
||||||
//// source_manager().getSpellingColumnNumber(location));
|
|
||||||
////
|
|
||||||
//// argument.set_name(type_name);
|
|
||||||
//
|
|
||||||
// if (get_unique_id(
|
|
||||||
// arg.getAsType()->getAsCXXRecordDecl()->getID())
|
|
||||||
// .has_value()) {
|
|
||||||
// argument.set_name(get_participant(
|
|
||||||
// get_unique_id(
|
|
||||||
// arg.getAsType()->getAsCXXRecordDecl()->getID())
|
|
||||||
// .value())
|
|
||||||
// .value()
|
|
||||||
// .full_name(false));
|
|
||||||
// }
|
|
||||||
// else {
|
|
||||||
// const auto location =
|
|
||||||
// arg.getAsType()->getAsCXXRecordDecl()->getLocation();
|
|
||||||
// auto type_name = fmt::format("(lambda {}:{}:{})",
|
|
||||||
// source_manager().getFilename(location).str(),
|
|
||||||
// source_manager().getSpellingLineNumber(location),
|
|
||||||
// source_manager().getSpellingColumnNumber(location));
|
|
||||||
// argument.set_name(type_name);
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
else if (argument_kind == clang::TemplateArgument::Pack) {
|
else if (argument_kind == clang::TemplateArgument::Pack) {
|
||||||
// This will only work for now if pack is at the end
|
// This will only work for now if pack is at the end
|
||||||
size_t argument_pack_index{argument_index};
|
size_t argument_pack_index{argument_index};
|
||||||
|
|||||||
@@ -20,6 +20,7 @@
|
|||||||
#include "common/visitor/translation_unit_visitor.h"
|
#include "common/visitor/translation_unit_visitor.h"
|
||||||
#include "config/config.h"
|
#include "config/config.h"
|
||||||
#include "sequence_diagram/model/diagram.h"
|
#include "sequence_diagram/model/diagram.h"
|
||||||
|
#include "call_expression_context.h"
|
||||||
|
|
||||||
#include <clang/AST/Expr.h>
|
#include <clang/AST/Expr.h>
|
||||||
#include <clang/AST/RecursiveASTVisitor.h>
|
#include <clang/AST/RecursiveASTVisitor.h>
|
||||||
@@ -31,173 +32,6 @@ namespace clanguml::sequence_diagram::visitor {
|
|||||||
|
|
||||||
std::string to_string(const clang::FunctionTemplateDecl *decl);
|
std::string to_string(const clang::FunctionTemplateDecl *decl);
|
||||||
|
|
||||||
struct call_expression_context {
|
|
||||||
call_expression_context()
|
|
||||||
: current_class_decl_{nullptr}
|
|
||||||
, current_class_template_decl_{nullptr}
|
|
||||||
, current_class_template_specialization_decl_{nullptr}
|
|
||||||
, current_method_decl_{nullptr}
|
|
||||||
, current_function_decl_{nullptr}
|
|
||||||
, current_function_template_decl_{nullptr}
|
|
||||||
, current_caller_id_{0}
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
void reset()
|
|
||||||
{
|
|
||||||
current_caller_id_ = 0;
|
|
||||||
current_class_decl_ = nullptr;
|
|
||||||
current_class_template_decl_ = nullptr;
|
|
||||||
current_class_template_specialization_decl_ = nullptr;
|
|
||||||
current_method_decl_ = nullptr;
|
|
||||||
current_function_decl_ = nullptr;
|
|
||||||
current_function_template_decl_ = nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
void dump()
|
|
||||||
{
|
|
||||||
LOG_DBG("current_caller_id_ = {}", current_caller_id_);
|
|
||||||
LOG_DBG("current_class_decl_ = {}", (void *)current_class_decl_);
|
|
||||||
LOG_DBG("current_class_template_decl_ = {}",
|
|
||||||
(void *)current_class_template_decl_);
|
|
||||||
LOG_DBG("current_class_template_specialization_decl_ = {}",
|
|
||||||
(void *)current_class_template_specialization_decl_);
|
|
||||||
LOG_DBG("current_method_decl_ = {}", (void *)current_method_decl_);
|
|
||||||
LOG_DBG("current_function_decl_ = {}", (void *)current_function_decl_);
|
|
||||||
LOG_DBG("current_function_template_decl_ = {}",
|
|
||||||
(void *)current_function_template_decl_);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool valid() const
|
|
||||||
{
|
|
||||||
return (current_class_decl_ != nullptr) ||
|
|
||||||
(current_class_template_decl_ != nullptr) ||
|
|
||||||
(current_class_template_specialization_decl_ != nullptr) ||
|
|
||||||
(current_method_decl_ != nullptr) ||
|
|
||||||
(current_function_decl_ != nullptr) ||
|
|
||||||
(current_function_template_decl_ != nullptr);
|
|
||||||
}
|
|
||||||
|
|
||||||
clang::ASTContext *get_ast_context()
|
|
||||||
{
|
|
||||||
if (current_class_template_specialization_decl_)
|
|
||||||
return ¤t_class_template_specialization_decl_
|
|
||||||
->getASTContext();
|
|
||||||
|
|
||||||
if (current_class_template_decl_)
|
|
||||||
return ¤t_class_template_decl_->getASTContext();
|
|
||||||
|
|
||||||
if (current_class_decl_)
|
|
||||||
return ¤t_class_decl_->getASTContext();
|
|
||||||
|
|
||||||
if (current_function_template_decl_)
|
|
||||||
return ¤t_function_template_decl_->getASTContext();
|
|
||||||
|
|
||||||
return ¤t_function_decl_->getASTContext();
|
|
||||||
}
|
|
||||||
|
|
||||||
void update(clang::CXXRecordDecl *cls) { current_class_decl_ = cls; }
|
|
||||||
|
|
||||||
void update(clang::ClassTemplateSpecializationDecl *clst)
|
|
||||||
{
|
|
||||||
current_class_template_specialization_decl_ = clst;
|
|
||||||
}
|
|
||||||
|
|
||||||
void update(clang::ClassTemplateDecl *clst)
|
|
||||||
{
|
|
||||||
current_class_template_decl_ = clst;
|
|
||||||
}
|
|
||||||
|
|
||||||
void update(clang::CXXMethodDecl *method) { current_method_decl_ = method; }
|
|
||||||
|
|
||||||
void update(clang::FunctionDecl *function)
|
|
||||||
{
|
|
||||||
if (!function->isCXXClassMember())
|
|
||||||
reset();
|
|
||||||
|
|
||||||
current_function_decl_ = function;
|
|
||||||
|
|
||||||
// Check if this function is a part of template function declaration,
|
|
||||||
// If no - reset the current_function_template_decl_
|
|
||||||
if (current_function_template_decl_ &&
|
|
||||||
current_function_template_decl_->getQualifiedNameAsString() !=
|
|
||||||
function->getQualifiedNameAsString()) {
|
|
||||||
current_function_template_decl_ = nullptr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void update(clang::FunctionTemplateDecl *function_template)
|
|
||||||
{
|
|
||||||
current_function_template_decl_ = function_template;
|
|
||||||
|
|
||||||
if (!function_template->isCXXClassMember())
|
|
||||||
current_class_decl_ = nullptr;
|
|
||||||
|
|
||||||
current_function_template_decl_ = function_template;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool in_class_method() const { return current_class_decl_ != nullptr; }
|
|
||||||
|
|
||||||
bool in_function() const
|
|
||||||
{
|
|
||||||
return current_class_decl_ == nullptr &&
|
|
||||||
current_function_decl_ != nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool in_function_template() const
|
|
||||||
{
|
|
||||||
return current_function_decl_ != nullptr &&
|
|
||||||
current_function_template_decl_ != nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::int64_t caller_id() const { return current_caller_id_; }
|
|
||||||
|
|
||||||
std::int64_t lambda_caller_id() const
|
|
||||||
{
|
|
||||||
if(current_lambda_caller_id_.empty())
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
return current_lambda_caller_id_.top();
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_caller_id(std::int64_t id)
|
|
||||||
{
|
|
||||||
LOG_DBG("Setting current caller id to {}", id);
|
|
||||||
current_caller_id_ = id;
|
|
||||||
}
|
|
||||||
|
|
||||||
void enter_lambda_expression(std::int64_t id)
|
|
||||||
{
|
|
||||||
LOG_DBG("Setting current lambda caller id to {}", id);
|
|
||||||
|
|
||||||
assert(id != 0);
|
|
||||||
|
|
||||||
current_lambda_caller_id_.push(id);
|
|
||||||
}
|
|
||||||
|
|
||||||
void leave_lambda_expression()
|
|
||||||
{
|
|
||||||
assert(!current_lambda_caller_id_.empty());
|
|
||||||
|
|
||||||
LOG_DBG("Leaving current lambda expression id to {}",
|
|
||||||
current_lambda_caller_id_.top());
|
|
||||||
|
|
||||||
current_lambda_caller_id_.pop();
|
|
||||||
}
|
|
||||||
|
|
||||||
clang::CXXRecordDecl *current_class_decl_;
|
|
||||||
clang::ClassTemplateDecl *current_class_template_decl_;
|
|
||||||
clang::ClassTemplateSpecializationDecl
|
|
||||||
*current_class_template_specialization_decl_;
|
|
||||||
clang::CXXMethodDecl *current_method_decl_;
|
|
||||||
clang::FunctionDecl *current_function_decl_;
|
|
||||||
clang::FunctionTemplateDecl *current_function_template_decl_;
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::int64_t current_caller_id_;
|
|
||||||
std::stack<std::int64_t> current_lambda_caller_id_;
|
|
||||||
};
|
|
||||||
|
|
||||||
class translation_unit_visitor
|
class translation_unit_visitor
|
||||||
: public clang::RecursiveASTVisitor<translation_unit_visitor>,
|
: public clang::RecursiveASTVisitor<translation_unit_visitor>,
|
||||||
public common::visitor::translation_unit_visitor {
|
public common::visitor::translation_unit_visitor {
|
||||||
|
|||||||
Reference in New Issue
Block a user