mirror of
https://github.com/pocoproject/poco.git
synced 2024-12-12 18:20:26 +01:00
256 lines
4.4 KiB
C++
256 lines
4.4 KiB
C++
//
|
|
// Function.cpp
|
|
//
|
|
// Library: CppParser
|
|
// Package: SymbolTable
|
|
// Module: Function
|
|
//
|
|
// Copyright (c) 2006, Applied Informatics Software Engineering GmbH.
|
|
// and Contributors.
|
|
//
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
//
|
|
|
|
|
|
#include "Poco/CppParser/Function.h"
|
|
#include "Poco/CppParser/Parameter.h"
|
|
#include "Poco/CppParser/NameSpace.h"
|
|
#include "Poco/CppParser/Struct.h"
|
|
#include "Poco/CppParser/Utility.h"
|
|
#include "Poco/String.h"
|
|
#include <sstream>
|
|
#include <cctype>
|
|
#include <cstddef>
|
|
|
|
|
|
namespace Poco {
|
|
namespace CppParser {
|
|
|
|
|
|
Function::Function(const std::string& decl, NameSpace* pNameSpace):
|
|
Decl(decl, pNameSpace),
|
|
_flags(0),
|
|
_retParam()
|
|
{
|
|
if (hasAttr(decl, "static"))
|
|
_flags |= FN_STATIC;
|
|
if (hasAttr(decl, "virtual"))
|
|
_flags |= FN_VIRTUAL;
|
|
if (hasAttr(decl, "inline"))
|
|
_flags |= FN_INLINE;
|
|
if (hasAttr(decl, "template"))
|
|
_flags |= FN_TEMPLATE;
|
|
|
|
if (isMethod() || isFunction())
|
|
{
|
|
// parse the decl
|
|
std::size_t pos = decl.rfind(name());
|
|
_retParam = decl.substr(0, pos-1);
|
|
// eliminate static, virtual, inline, template
|
|
_retParam = replace(_retParam, "static ", "");
|
|
_retParam = replace(_retParam, "virtual ", "");
|
|
_retParam = replace(_retParam, "inline ", "");
|
|
if (_flags & FN_TEMPLATE)
|
|
{
|
|
std::size_t pos2 = _retParam.find(">");
|
|
poco_assert (pos2 != std::string::npos);
|
|
_retParam = _retParam.substr(pos2+1);
|
|
}
|
|
Poco::trimInPlace(_retParam);
|
|
}
|
|
}
|
|
|
|
|
|
Function::~Function()
|
|
{
|
|
for (Parameters::iterator it = _params.begin(); it != _params.end(); ++it)
|
|
{
|
|
delete *it;
|
|
}
|
|
}
|
|
|
|
|
|
void Function::addParameter(Parameter* pParam)
|
|
{
|
|
_params.push_back(pParam);
|
|
}
|
|
|
|
|
|
Function::Iterator Function::begin() const
|
|
{
|
|
return _params.begin();
|
|
}
|
|
|
|
|
|
Function::Iterator Function::end() const
|
|
{
|
|
return _params.end();
|
|
}
|
|
|
|
|
|
void Function::makeInline()
|
|
{
|
|
_flags |= FN_INLINE;
|
|
}
|
|
|
|
|
|
void Function::makeConst()
|
|
{
|
|
_flags |= FN_CONST;
|
|
}
|
|
|
|
|
|
void Function::makePureVirtual()
|
|
{
|
|
_flags |= FN_PURE_VIRTUAL;
|
|
}
|
|
|
|
|
|
void Function::makeFinal()
|
|
{
|
|
_flags |= FN_FINAL;
|
|
}
|
|
|
|
|
|
void Function::makeOverride()
|
|
{
|
|
_flags |= FN_OVERRIDE;
|
|
}
|
|
|
|
|
|
void Function::makeNoexcept()
|
|
{
|
|
_flags |= FN_NOEXCEPT;
|
|
}
|
|
|
|
|
|
void Function::makeDefault()
|
|
{
|
|
_flags |= FN_DEFAULT;
|
|
}
|
|
|
|
|
|
void Function::makeDelete()
|
|
{
|
|
_flags |= FN_DELETE;
|
|
}
|
|
|
|
|
|
bool Function::isConstructor() const
|
|
{
|
|
return name() == nameSpace()->name();
|
|
}
|
|
|
|
|
|
bool Function::isDestructor() const
|
|
{
|
|
return name()[0] == '~';
|
|
}
|
|
|
|
|
|
bool Function::isMethod() const
|
|
{
|
|
return !isConstructor() && !isDestructor() && nameSpace()->kind() == Symbol::SYM_STRUCT;
|
|
}
|
|
|
|
|
|
bool Function::isFunction() const
|
|
{
|
|
return nameSpace()->kind() == Symbol::SYM_NAMESPACE;
|
|
}
|
|
|
|
|
|
int Function::countParameters() const
|
|
{
|
|
return (int) _params.size();
|
|
}
|
|
|
|
|
|
Symbol::Kind Function::kind() const
|
|
{
|
|
return Symbol::SYM_FUNCTION;
|
|
}
|
|
|
|
|
|
std::string Function::signature() const
|
|
{
|
|
std::string signature(declaration());
|
|
if (signature.compare(0, 8, "virtual ") == 0)
|
|
signature.erase(0, 8);
|
|
else if (signature.compare(0, 7, "static ") == 0)
|
|
signature.erase(0, 8);
|
|
if (signature.compare(0, 7, "inline ") == 0)
|
|
signature.erase(0, 7);
|
|
signature += "(";
|
|
bool isFirst = true;
|
|
for (Iterator it = begin(); it != end(); ++it)
|
|
{
|
|
if (isFirst)
|
|
isFirst = false;
|
|
else
|
|
signature += ", ";
|
|
std::string arg = (*it)->declaration();
|
|
std::string::size_type pos = arg.size() - 1;
|
|
while (pos > 0 && !std::isspace(arg[pos])) --pos;
|
|
while (pos > 0 && std::isspace(arg[pos])) --pos;
|
|
signature.append(arg, 0, pos + 1);
|
|
}
|
|
signature += ")";
|
|
if (_flags & FN_CONST)
|
|
signature += " const";
|
|
return signature;
|
|
}
|
|
|
|
|
|
bool Function::isVirtual() const
|
|
{
|
|
if (_flags & FN_VIRTUAL)
|
|
{
|
|
return true;
|
|
}
|
|
else if (isDestructor())
|
|
{
|
|
Struct* pClass = dynamic_cast<Struct*>(nameSpace());
|
|
return pClass && pClass->hasVirtualDestructor();
|
|
}
|
|
else return getOverridden() != 0;
|
|
}
|
|
|
|
|
|
Function* Function::getOverridden() const
|
|
{
|
|
if (isMethod() && !(_flags & FN_STATIC))
|
|
{
|
|
Struct* pClass = dynamic_cast<Struct*>(nameSpace());
|
|
if (pClass)
|
|
{
|
|
for (Struct::BaseIterator it = pClass->baseBegin(); it != pClass->baseEnd(); ++it)
|
|
{
|
|
if (it->pClass)
|
|
{
|
|
Function* pOverridden = it->pClass->findFunction(signature());
|
|
if (pOverridden && pOverridden->isVirtual())
|
|
return pOverridden;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
std::string Function::toString() const
|
|
{
|
|
std::ostringstream ostr;
|
|
ostr << Decl::toString() << "(\n";
|
|
for (Iterator it = begin(); it != end(); ++it)
|
|
{
|
|
ostr << "\t" << (*it)->toString() << "\n";
|
|
}
|
|
ostr << ");";
|
|
return ostr.str();
|
|
}
|
|
|
|
|
|
} } // namespace Poco::CppParser
|