efl/src/lib/eolian_cxx/grammar/function_declaration.hpp

134 lines
4.2 KiB
C++

/*
* Copyright 2019 by its authors. See AUTHORS.
*
* 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.
*/
#ifndef EOLIAN_CXX_FUNCTION_DECLARATION_HH
#define EOLIAN_CXX_FUNCTION_DECLARATION_HH
#include "grammar/generator.hpp"
#include "grammar/klass_def.hpp"
#include "grammar/string.hpp"
#include "grammar/indentation.hpp"
#include "grammar/list.hpp"
#include "grammar/alternative.hpp"
#include "grammar/type.hpp"
#include "grammar/parameter.hpp"
#include "grammar/keyword.hpp"
#include "grammar/blacklist.hpp"
namespace efl { namespace eolian { namespace grammar {
struct function_declaration_generator
{
function_declaration_generator(attributes::klass_name const& name)
: _klass_name(name)
{}
template <typename OutputIterator, typename Context>
bool generate(OutputIterator sink, attributes::function_def const& f, Context const& ctx) const
{
if (blacklist::is_blacklisted(f))
return true;
std::string suffix, static_flag, const_flag;
switch(_klass_name.type)
{
case attributes::class_type::regular:
case attributes::class_type::abstract_:
suffix = "CLASS";
break;
case attributes::class_type::mixin:
suffix = "MIXIN";
break;
case attributes::class_type::interface_:
suffix = "INTERFACE";
break;
}
if(f.is_beta &&
!as_generator(lit("#ifdef EFL_BETA_API_SUPPORT\n"))
.generate(sink, attributes::unused, ctx))
return false;
if(f.is_protected &&
!as_generator("#ifdef " << *(string << "_") << string << "_PROTECTED\n")
.generate(sink, std::make_tuple(_klass_name.namespaces, _klass_name.eolian_name), add_upper_case_context(ctx)))
return false;
#ifdef USE_EOCXX_MANUAL_OVERRIDES
if(!as_generator
("#ifndef EOLIAN_CXX_" << string << "_DECLARATION\n")
.generate(sink, f.c_name, add_upper_case_context(ctx)))
return false;
#endif
std::string template_statement(f.template_statement());
if (!template_statement.empty() &&
!as_generator(template_statement << " ")
.generate(sink, attributes::unused, ctx))
return false;
if (f.is_static) static_flag = "static ";
else const_flag = " const";
if(!as_generator
(scope_tab << static_flag << "::efl::eolian::return_traits<" << grammar::type(true) << ">::type "
<< string << "(" << (parameter % ", ") << ")" << const_flag << ";\n")
.generate(sink, std::make_tuple(f.return_type, escape_keyword(f.name), f.parameters), ctx))
return false;
#ifdef USE_EOCXX_MANUAL_OVERRIDES
if(!as_generator
("#else\n" << scope_tab << "EOLIAN_CXX_" << string << "_DECLARATION\n"
"#endif\n")
.generate(sink, f.c_name, add_upper_case_context(ctx)))
return false;
#endif
if(f.is_protected &&
!as_generator("#endif\n").generate(sink, attributes::unused, ctx))
return false;
if(f.is_beta &&
!as_generator("#endif\n").generate(sink, attributes::unused, ctx))
return false;
return true;
}
attributes::klass_name _klass_name;
};
template <>
struct is_eager_generator<function_declaration_generator> : std::true_type {};
template <>
struct is_generator<function_declaration_generator> : std::true_type {};
namespace type_traits {
template <>
struct attributes_needed<function_declaration_generator> : std::integral_constant<int, 1> {};
}
struct function_declaration_terminal
{
function_declaration_generator operator()(attributes::klass_name name) const
{
return function_declaration_generator{name};
}
} const function_declaration = {};
} } }
#endif