1c243e490SMarcel Moolenaar // 2c243e490SMarcel Moolenaar // Automated Testing Framework (atf) 3c243e490SMarcel Moolenaar // 4c243e490SMarcel Moolenaar // Copyright (c) 2007 The NetBSD Foundation, Inc. 5c243e490SMarcel Moolenaar // All rights reserved. 6c243e490SMarcel Moolenaar // 7c243e490SMarcel Moolenaar // Redistribution and use in source and binary forms, with or without 8c243e490SMarcel Moolenaar // modification, are permitted provided that the following conditions 9c243e490SMarcel Moolenaar // are met: 10c243e490SMarcel Moolenaar // 1. Redistributions of source code must retain the above copyright 11c243e490SMarcel Moolenaar // notice, this list of conditions and the following disclaimer. 12c243e490SMarcel Moolenaar // 2. Redistributions in binary form must reproduce the above copyright 13c243e490SMarcel Moolenaar // notice, this list of conditions and the following disclaimer in the 14c243e490SMarcel Moolenaar // documentation and/or other materials provided with the distribution. 15c243e490SMarcel Moolenaar // 16c243e490SMarcel Moolenaar // THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND 17c243e490SMarcel Moolenaar // CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, 18c243e490SMarcel Moolenaar // INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF 19c243e490SMarcel Moolenaar // MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 20c243e490SMarcel Moolenaar // IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY 21c243e490SMarcel Moolenaar // DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 22c243e490SMarcel Moolenaar // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE 23c243e490SMarcel Moolenaar // GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 24c243e490SMarcel Moolenaar // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER 25c243e490SMarcel Moolenaar // IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR 26c243e490SMarcel Moolenaar // OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN 27c243e490SMarcel Moolenaar // IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 28c243e490SMarcel Moolenaar // 29c243e490SMarcel Moolenaar 30c243e490SMarcel Moolenaar #if defined(HAVE_CONFIG_H) 31c243e490SMarcel Moolenaar #include "bconfig.h" 32c243e490SMarcel Moolenaar #endif 33c243e490SMarcel Moolenaar 34c243e490SMarcel Moolenaar extern "C" { 35c243e490SMarcel Moolenaar #include <unistd.h> 36c243e490SMarcel Moolenaar } 37c243e490SMarcel Moolenaar 38c243e490SMarcel Moolenaar #include <cstdarg> 39c243e490SMarcel Moolenaar #include <cstdio> 40c243e490SMarcel Moolenaar #include <cstdlib> 41c243e490SMarcel Moolenaar #include <cstring> 42c243e490SMarcel Moolenaar #include <iostream> 43c243e490SMarcel Moolenaar 44c243e490SMarcel Moolenaar extern "C" { 45c243e490SMarcel Moolenaar #include "atf-c/defs.h" 46c243e490SMarcel Moolenaar } 47c243e490SMarcel Moolenaar 48c243e490SMarcel Moolenaar #include "application.hpp" 49c243e490SMarcel Moolenaar #include "sanity.hpp" 50c243e490SMarcel Moolenaar 51c243e490SMarcel Moolenaar #if !defined(HAVE_VSNPRINTF_IN_STD) 52c243e490SMarcel Moolenaar namespace std { 53c243e490SMarcel Moolenaar using ::vsnprintf; 54c243e490SMarcel Moolenaar } 55c243e490SMarcel Moolenaar #endif // !defined(HAVE_VSNPRINTF_IN_STD) 56c243e490SMarcel Moolenaar 57c243e490SMarcel Moolenaar namespace impl = atf::application; 58c243e490SMarcel Moolenaar #define IMPL_NAME "atf::application" 59c243e490SMarcel Moolenaar 60c243e490SMarcel Moolenaar // ------------------------------------------------------------------------ 61c243e490SMarcel Moolenaar // The "usage_error" class. 62c243e490SMarcel Moolenaar // ------------------------------------------------------------------------ 63c243e490SMarcel Moolenaar 64c243e490SMarcel Moolenaar impl::usage_error::usage_error(const char *fmt, ...) 65c243e490SMarcel Moolenaar throw() : 66c243e490SMarcel Moolenaar std::runtime_error("usage_error; message unformatted") 67c243e490SMarcel Moolenaar { 68c243e490SMarcel Moolenaar va_list ap; 69c243e490SMarcel Moolenaar 70c243e490SMarcel Moolenaar va_start(ap, fmt); 71c243e490SMarcel Moolenaar std::vsnprintf(m_text, sizeof(m_text), fmt, ap); 72c243e490SMarcel Moolenaar va_end(ap); 73c243e490SMarcel Moolenaar } 74c243e490SMarcel Moolenaar 75c243e490SMarcel Moolenaar impl::usage_error::~usage_error(void) 76c243e490SMarcel Moolenaar throw() 77c243e490SMarcel Moolenaar { 78c243e490SMarcel Moolenaar } 79c243e490SMarcel Moolenaar 80c243e490SMarcel Moolenaar const char* 81c243e490SMarcel Moolenaar impl::usage_error::what(void) 82c243e490SMarcel Moolenaar const throw() 83c243e490SMarcel Moolenaar { 84c243e490SMarcel Moolenaar return m_text; 85c243e490SMarcel Moolenaar } 86c243e490SMarcel Moolenaar 87c243e490SMarcel Moolenaar // ------------------------------------------------------------------------ 88c243e490SMarcel Moolenaar // The "application" class. 89c243e490SMarcel Moolenaar // ------------------------------------------------------------------------ 90c243e490SMarcel Moolenaar 91c243e490SMarcel Moolenaar impl::option::option(char ch, 92c243e490SMarcel Moolenaar const std::string& a, 93c243e490SMarcel Moolenaar const std::string& desc) : 94c243e490SMarcel Moolenaar m_character(ch), 95c243e490SMarcel Moolenaar m_argument(a), 96c243e490SMarcel Moolenaar m_description(desc) 97c243e490SMarcel Moolenaar { 98c243e490SMarcel Moolenaar } 99c243e490SMarcel Moolenaar 100c243e490SMarcel Moolenaar bool 101c243e490SMarcel Moolenaar impl::option::operator<(const impl::option& o) 102c243e490SMarcel Moolenaar const 103c243e490SMarcel Moolenaar { 104c243e490SMarcel Moolenaar return m_character < o.m_character; 105c243e490SMarcel Moolenaar } 106c243e490SMarcel Moolenaar 107c243e490SMarcel Moolenaar impl::app::app(const std::string& description, 108*1a61beb0SJulio Merino const std::string& manpage) : 109c243e490SMarcel Moolenaar m_argc(-1), 110c243e490SMarcel Moolenaar m_argv(NULL), 111c243e490SMarcel Moolenaar m_prog_name(NULL), 112c243e490SMarcel Moolenaar m_description(description), 113*1a61beb0SJulio Merino m_manpage(manpage) 114c243e490SMarcel Moolenaar { 115c243e490SMarcel Moolenaar } 116c243e490SMarcel Moolenaar 117c243e490SMarcel Moolenaar impl::app::~app(void) 118c243e490SMarcel Moolenaar { 119c243e490SMarcel Moolenaar } 120c243e490SMarcel Moolenaar 121c243e490SMarcel Moolenaar bool 122c243e490SMarcel Moolenaar impl::app::inited(void) 123c243e490SMarcel Moolenaar { 124c243e490SMarcel Moolenaar return m_argc != -1; 125c243e490SMarcel Moolenaar } 126c243e490SMarcel Moolenaar 127c243e490SMarcel Moolenaar impl::app::options_set 128c243e490SMarcel Moolenaar impl::app::options(void) 129c243e490SMarcel Moolenaar { 130*1a61beb0SJulio Merino return specific_options(); 131c243e490SMarcel Moolenaar } 132c243e490SMarcel Moolenaar 133c243e490SMarcel Moolenaar std::string 134c243e490SMarcel Moolenaar impl::app::specific_args(void) 135c243e490SMarcel Moolenaar const 136c243e490SMarcel Moolenaar { 137c243e490SMarcel Moolenaar return ""; 138c243e490SMarcel Moolenaar } 139c243e490SMarcel Moolenaar 140c243e490SMarcel Moolenaar impl::app::options_set 141c243e490SMarcel Moolenaar impl::app::specific_options(void) 142c243e490SMarcel Moolenaar const 143c243e490SMarcel Moolenaar { 144c243e490SMarcel Moolenaar return options_set(); 145c243e490SMarcel Moolenaar } 146c243e490SMarcel Moolenaar 147c243e490SMarcel Moolenaar void 148c243e490SMarcel Moolenaar impl::app::process_option(int ch ATF_DEFS_ATTRIBUTE_UNUSED, 149c243e490SMarcel Moolenaar const char* arg ATF_DEFS_ATTRIBUTE_UNUSED) 150c243e490SMarcel Moolenaar { 151c243e490SMarcel Moolenaar } 152c243e490SMarcel Moolenaar 153c243e490SMarcel Moolenaar void 154c243e490SMarcel Moolenaar impl::app::process_options(void) 155c243e490SMarcel Moolenaar { 156c243e490SMarcel Moolenaar PRE(inited()); 157c243e490SMarcel Moolenaar 158c243e490SMarcel Moolenaar std::string optstr; 159c243e490SMarcel Moolenaar #if defined(HAVE_GNU_GETOPT) 160c243e490SMarcel Moolenaar optstr += '+'; // Turn on POSIX behavior. 161c243e490SMarcel Moolenaar #endif 162c243e490SMarcel Moolenaar optstr += ':'; 163c243e490SMarcel Moolenaar { 164c243e490SMarcel Moolenaar options_set opts = options(); 165c243e490SMarcel Moolenaar for (options_set::const_iterator iter = opts.begin(); 166c243e490SMarcel Moolenaar iter != opts.end(); iter++) { 167c243e490SMarcel Moolenaar const option& opt = (*iter); 168c243e490SMarcel Moolenaar 169c243e490SMarcel Moolenaar optstr += opt.m_character; 170c243e490SMarcel Moolenaar if (!opt.m_argument.empty()) 171c243e490SMarcel Moolenaar optstr += ':'; 172c243e490SMarcel Moolenaar } 173c243e490SMarcel Moolenaar } 174c243e490SMarcel Moolenaar 175c243e490SMarcel Moolenaar int ch; 176c243e490SMarcel Moolenaar const int old_opterr = ::opterr; 177c243e490SMarcel Moolenaar ::opterr = 0; 178c243e490SMarcel Moolenaar while ((ch = ::getopt(m_argc, m_argv, optstr.c_str())) != -1) { 179c243e490SMarcel Moolenaar switch (ch) { 180c243e490SMarcel Moolenaar case ':': 181c243e490SMarcel Moolenaar throw usage_error("Option -%c requires an argument.", 182c243e490SMarcel Moolenaar ::optopt); 183c243e490SMarcel Moolenaar 184c243e490SMarcel Moolenaar case '?': 185c243e490SMarcel Moolenaar throw usage_error("Unknown option -%c.", ::optopt); 186c243e490SMarcel Moolenaar 187c243e490SMarcel Moolenaar default: 188c243e490SMarcel Moolenaar process_option(ch, ::optarg); 189c243e490SMarcel Moolenaar } 190c243e490SMarcel Moolenaar } 191c243e490SMarcel Moolenaar m_argc -= ::optind; 192c243e490SMarcel Moolenaar m_argv += ::optind; 193c243e490SMarcel Moolenaar 194c243e490SMarcel Moolenaar // Clear getopt state just in case the test wants to use it. 195c243e490SMarcel Moolenaar opterr = old_opterr; 196c243e490SMarcel Moolenaar optind = 1; 197c243e490SMarcel Moolenaar #if defined(HAVE_OPTRESET) 198c243e490SMarcel Moolenaar optreset = 1; 199c243e490SMarcel Moolenaar #endif 200c243e490SMarcel Moolenaar } 201c243e490SMarcel Moolenaar 202c243e490SMarcel Moolenaar int 203c243e490SMarcel Moolenaar impl::app::run(int argc, char* const* argv) 204c243e490SMarcel Moolenaar { 205c243e490SMarcel Moolenaar PRE(argc > 0); 206c243e490SMarcel Moolenaar PRE(argv != NULL); 207c243e490SMarcel Moolenaar 208c243e490SMarcel Moolenaar m_argc = argc; 209c243e490SMarcel Moolenaar m_argv = argv; 210c243e490SMarcel Moolenaar 211c243e490SMarcel Moolenaar m_argv0 = m_argv[0]; 212c243e490SMarcel Moolenaar 213c243e490SMarcel Moolenaar m_prog_name = std::strrchr(m_argv[0], '/'); 214c243e490SMarcel Moolenaar if (m_prog_name == NULL) 215c243e490SMarcel Moolenaar m_prog_name = m_argv[0]; 216c243e490SMarcel Moolenaar else 217c243e490SMarcel Moolenaar m_prog_name++; 218c243e490SMarcel Moolenaar 219c243e490SMarcel Moolenaar // Libtool workaround: if running from within the source tree (binaries 220c243e490SMarcel Moolenaar // that are not installed yet), skip the "lt-" prefix added to files in 221c243e490SMarcel Moolenaar // the ".libs" directory to show the real (not temporary) name. 222c243e490SMarcel Moolenaar if (std::strncmp(m_prog_name, "lt-", 3) == 0) 223c243e490SMarcel Moolenaar m_prog_name += 3; 224c243e490SMarcel Moolenaar 225c243e490SMarcel Moolenaar const std::string bug = 226c243e490SMarcel Moolenaar std::string("This is probably a bug in ") + m_prog_name + 227c243e490SMarcel Moolenaar " or one of the libraries it uses. Please report this problem to " 228c243e490SMarcel Moolenaar PACKAGE_BUGREPORT " and provide as many details as possible " 229c243e490SMarcel Moolenaar "describing how you got to this condition."; 230c243e490SMarcel Moolenaar 231c243e490SMarcel Moolenaar int errcode; 232c243e490SMarcel Moolenaar try { 233c243e490SMarcel Moolenaar process_options(); 234c243e490SMarcel Moolenaar errcode = main(); 235c243e490SMarcel Moolenaar } catch (const usage_error& e) { 236c243e490SMarcel Moolenaar std::cerr << m_prog_name << ": ERROR: " << e.what() << "\n"; 237c243e490SMarcel Moolenaar std::cerr << m_prog_name << ": See " << m_manpage << " for usage " 238c243e490SMarcel Moolenaar "details.\n"; 239c243e490SMarcel Moolenaar errcode = EXIT_FAILURE; 240c243e490SMarcel Moolenaar } catch (const std::runtime_error& e) { 241c243e490SMarcel Moolenaar std::cerr << m_prog_name << ": ERROR: " << e.what() << "\n"; 242c243e490SMarcel Moolenaar errcode = EXIT_FAILURE; 243c243e490SMarcel Moolenaar } catch (const std::exception& e) { 244c243e490SMarcel Moolenaar std::cerr << m_prog_name << ": ERROR: Caught unexpected error: " 245c243e490SMarcel Moolenaar << e.what() << "\n"; 246c243e490SMarcel Moolenaar errcode = EXIT_FAILURE; 247c243e490SMarcel Moolenaar } catch (...) { 248c243e490SMarcel Moolenaar std::cerr << m_prog_name << ": ERROR: Caught unknown error\n"; 249c243e490SMarcel Moolenaar errcode = EXIT_FAILURE; 250c243e490SMarcel Moolenaar } 251c243e490SMarcel Moolenaar return errcode; 252c243e490SMarcel Moolenaar } 253