xref: /freebsd/contrib/atf/atf-c++/detail/application.cpp (revision 1a61beb0549e05b33df31380e427d90f6e46ff7e)
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