# Checking the C++ Features. -*- Autotest -*- # Copyright (C) 2004-2005, 2007, 2009-2012 Free Software Foundation, # Inc. # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . AT_BANNER([[C++ Features.]]) ## ----------------------- ## ## Doxygen Documentation. ## ## ----------------------- ## m4_define([AT_CHECK_DOXYGEN], [m4_case([$1], [Public], [m4_pushdef([AT_DOXYGEN_PRIVATE], [NO])], [Private], [m4_pushdef([AT_DOXYGEN_PRIVATE], [YES])], [m4_fatal([invalid argument: $1])]) AT_SETUP([Doxygen $1 Documentation]) AT_BISON_OPTION_PUSHDEFS([%skeleton "lalr1.cc"]) AT_DATA([input.yy], [[%skeleton "lalr1.cc" %locations %debug %defines %% exp:; %% ]AT_YYERROR_DEFINE[ ]]) AT_BISON_CHECK([-o input.cc input.yy], 0) AT_DATA([Doxyfile], [# The PROJECT_NAME tag is a single word (or a sequence of words # surrounded by quotes) that should identify the project. PROJECT_NAME = "Bison C++ Parser" # The QUIET tag can be used to turn on/off the messages that are # generated by doxygen. Possible values are YES and NO. If left blank # NO is used. QUIET = YES # The WARNINGS tag can be used to turn on/off the warning messages # that are generated by doxygen. Possible values are YES and NO. If # left blank NO is used. WARNINGS = YES # If WARN_IF_UNDOCUMENTED is set to YES, then doxygen will generate # warnings for undocumented members. If EXTRACT_ALL is set to YES then # this flag will automatically be disabled. WARN_IF_UNDOCUMENTED = YES # If WARN_IF_DOC_ERROR is set to YES, doxygen will generate warnings # for potential errors in the documentation, such as not documenting # some parameters in a documented function, or documenting parameters # that don't exist or using markup commands wrongly. WARN_IF_DOC_ERROR = YES # The WARN_FORMAT tag determines the format of the warning messages # that doxygen can produce. The string should contain the $file, # $line, and $text tags, which will be replaced by the file and line # number from which the warning originated and the warning text. WARN_FORMAT = "$file:$line: $text" # If the EXTRACT_ALL tag is set to YES doxygen will assume all # entities in documentation are documented, even if no documentation # was available. Private class members and static file members will # be hidden unless the EXTRACT_PRIVATE and EXTRACT_STATIC tags are set # to YES EXTRACT_ALL = YES # If the EXTRACT_PRIVATE tag is set to YES all private members of a # class will be included in the documentation. EXTRACT_PRIVATE = AT_DOXYGEN_PRIVATE # If the EXTRACT_STATIC tag is set to YES all static members of a file # will be included in the documentation. EXTRACT_STATIC = AT_DOXYGEN_PRIVATE ]) AT_CHECK([doxygen --version || exit 77], 0, ignore) AT_CHECK([doxygen], 0, [], [ignore]) AT_BISON_OPTION_POPDEFS AT_CLEANUP m4_popdef([AT_DOXYGEN_PRIVATE]) ])# AT_CHECK_DOXYGEN AT_CHECK_DOXYGEN([Public]) AT_CHECK_DOXYGEN([Private]) ## ------------ ## ## Namespaces. ## ## ------------ ## # AT_CHECK_NAMESPACE(NAMESPACE-DECL, [COMPILE-ERROR]) # --------------------------------------------------- # See if Bison can handle %define namespace "NAMESPACE-DECL". If COMPILE-ERROR # is specified, then Bison should accept the input, but compilation will fail, # so don't check compilation. m4_define([AT_CHECK_NAMESPACE], [ AT_DATA_GRAMMAR([[input.y]], [[%language "C++" %defines %define namespace "]$1[" %union { int i; } %define global_tokens_and_yystype %code { // YYSTYPE contains a namespace reference. int yylex (YYSTYPE *lval) { lval->i = 3; return 0; } } %% start: ; %% void ]$1[::parser::error (const ]$1[::parser::location_type &loc, const std::string &msg) { std::cerr << "At " << loc << ": " << msg << std::endl; } int main (void) { ]$1[::parser p; return p.parse (); } ]]) AT_BISON_CHECK([[-o input.cc input.y]]) m4_if([$#], [1], [AT_COMPILE_CXX([[input]], [[input.cc]]) AT_PARSER_CHECK([[./input]])]) ]) AT_SETUP([[Relative namespace references]]) AT_CHECK_NAMESPACE([[foo]]) AT_CHECK_NAMESPACE([[foo::bar]]) AT_CHECK_NAMESPACE([[foo::bar::baz]]) AT_CLEANUP AT_SETUP([[Absolute namespace references]]) AT_CHECK_NAMESPACE([[::foo]]) AT_CHECK_NAMESPACE([[::foo::bar]]) AT_CHECK_NAMESPACE([[::foo::bar::baz]]) AT_CHECK_NAMESPACE([[ ::foo]]) AT_CHECK_NAMESPACE([[ ::foo::bar]]) AT_CHECK_NAMESPACE([[ ::foo::bar::baz]]) AT_CLEANUP AT_SETUP([[Syntactically invalid namespace references]]) AT_CHECK_NAMESPACE([[:foo:bar]], [[-]]) AT_CHECK_NAMESPACE([[foo: :bar]], [[-]]) # This one is interesting because `[3]' is encoded as `@<:@3@:>@', which # contains single occurrences of `:'. AT_CHECK_NAMESPACE([[foo[3]::bar::baz]], [[-]]) AT_CHECK_NAMESPACE([[foo::bar,baz]], [[-]]) AT_CHECK_NAMESPACE([[foo::bar::(baz]], [[-]]) AT_CLEANUP ## ------------------ ## ## Exception safety. ## ## ------------------ ## AT_SETUP([[Exception safety]]) AT_BISON_OPTION_PUSHDEFS([%skeleton "lalr1.cc"]) AT_DATA_GRAMMAR([[input.yy]], [[%skeleton "lalr1.cc" %defines // FIXME: Mandated in 2.6. %debug %code requires { #include // size_t and getenv. #include int debug = 0; struct Object { static size_t counter; Object () { ++counter; if (debug) std::cerr << "Object::Object() => counter == " << counter << std::endl; } ~Object () { --counter; if (debug) std::cerr << "Object::~Object() => counter == " << counter << std::endl; } }; } %code { #include #include int yylex (yy::parser::semantic_type *); size_t Object::counter = 0; static char const *input; } %union { Object* obj; } %destructor { delete $$; } ; %printer { yyo << "counter == " << $$->counter; } ; %token 'a' 's' %type list item %% start: list { delete $1; }; list: item { $$ = $1; } | item list { $$ = $1; delete $2; } /* Right recursion to load the stack. */ ; item: 'a' { std::swap ($$, $1); } | 's' { std::swap ($$, $1); throw std::runtime_error ("invalid expression"); } %% int yylex (yy::parser::semantic_type *lvalp) { // 'l': lexical exception, 's': syntactic exception. switch (int res = *input++) { case 'l': throw std::runtime_error ("invalid character"); default: lvalp->obj = new Object; // Fall through. case 0: return res; } } ]AT_YYERROR_DEFINE[ int main (int argc, const char *argv[]) { assert (argc == 2); input = argv[1]; yy::parser parser; debug = !!getenv ("YYDEBUG"); parser.set_debug_level (debug); int res = 2; try { res = parser.parse (); } catch (const std::exception& e) { std::cerr << "exception caught: " << e.what () << std::endl; } catch (...) { std::cerr << "unknown exception caught" << std::endl; } assert (Object::counter == 0); return res; } ]]) AT_BISON_CHECK([[-o input.cc input.yy]]) AT_COMPILE_CXX([[input]]) AT_PARSER_CHECK([[./input aaaas]], [[2]], [[]], [[exception caught: invalid expression ]]) AT_PARSER_CHECK([[./input aaaal]], [[2]], [[]], [[exception caught: invalid character ]]) AT_BISON_OPTION_POPDEFS AT_CLEANUP