bump product version to 6.3.0.0.beta1
[LibreOffice.git] / idlc / source / options.cxx
blob4f49cd202de4d80c05b4c114b1d5aa671b9bf40f
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
21 #include <options.hxx>
23 #include <osl/diagnose.h>
24 #include <rtl/string.hxx>
25 #include <rtl/strbuf.hxx>
27 #include <rtl/ustring.hxx>
28 #include <osl/file.hxx>
29 #include <o3tl/char16_t2wchar_t.hxx>
31 #ifdef _WIN32
32 # if !defined WIN32_LEAN_AND_MEAN
33 # define WIN32_LEAN_AND_MEAN
34 # endif
35 # include <windows.h>
36 #endif
38 #include <stdio.h>
39 #include <string.h>
42 Options::Options(char const * progname)
43 : m_program(progname), m_stdin(false), m_verbose(false), m_quiet(false)
47 Options::~Options()
51 // static
52 bool Options::checkArgument (std::vector< std::string > & rArgs, char const * arg, size_t len)
54 bool result = ((arg != nullptr) && (len > 0));
55 OSL_PRECOND(result, "idlc::Options::checkArgument(): invalid arguments");
56 if (result)
58 switch(arg[0])
60 case '@':
61 result = len > 1;
62 if (result)
64 // "@<cmdfile>"
65 result = Options::checkCommandFile (rArgs, &(arg[1]));
67 break;
68 case '-':
69 result = len > 1;
70 if (result)
72 // "-<option>"
73 switch (arg[1])
75 case 'O':
76 case 'M':
77 case 'I':
78 case 'D':
80 // "-<option>[<param>]
81 std::string option(&(arg[0]), 2);
82 rArgs.push_back(option);
83 if (len > 2)
85 // "-<option><param>"
86 std::string param(&(arg[2]), len - 2);
87 rArgs.push_back(param);
89 break;
91 default:
92 // "-<option>" ([long] option, w/o param)
93 rArgs.emplace_back(arg, len);
94 break;
97 break;
98 default:
99 // "<param>"
100 rArgs.emplace_back(arg, len);
101 break;
104 return result;
107 // static
108 bool Options::checkCommandFile (std::vector< std::string > & rArgs, char const * filename)
110 FILE * fp = fopen(filename, "r");
111 if (fp == nullptr)
113 fprintf(stderr, "ERROR: can't open command file \"%s\"\n", filename);
114 return false;
117 std::string buffer;
118 buffer.reserve(256);
120 bool quoted = false;
121 int c = EOF;
122 while ((c = fgetc(fp)) != EOF)
124 switch(c)
126 case '\"':
127 quoted = !quoted;
128 break;
129 case ' ':
130 case '\t':
131 case '\r':
132 case '\n':
133 if (!quoted)
135 if (!buffer.empty())
137 // append current argument.
138 if (!Options::checkArgument(rArgs, buffer.c_str(), buffer.size()))
140 (void) fclose(fp);
141 return false;
143 buffer.clear();
145 break;
147 [[fallthrough]];
148 default:
149 buffer.push_back(sal::static_int_cast<char>(c));
150 break;
153 if (!buffer.empty())
155 // append unterminated argument.
156 if (!Options::checkArgument(rArgs, buffer.c_str(), buffer.size()))
158 (void) fclose(fp);
159 return false;
161 buffer.clear();
163 return (fclose(fp) == 0);
166 bool Options::badOption(char const * reason, std::string const & rArg)
168 OStringBuffer message;
169 if (reason != nullptr)
171 message.append(reason); message.append(" option '"); message.append(rArg.c_str()); message.append("'");
172 throw IllegalArgument(message.makeStringAndClear());
174 return false;
177 bool Options::setOption(char const * option, std::string const & rArg)
179 bool result = (0 == strcmp(option, rArg.c_str()));
180 if (result)
181 m_options[rArg.c_str()] = OString(rArg.c_str(), rArg.size());
182 return result;
185 #ifdef _WIN32
186 /* Helper function to convert windows paths including spaces, brackets etc. into
187 a windows short Url. The ucpp preprocessor has problems with such paths and returns
188 with error.
190 static OString convertIncPathtoShortWindowsPath(const OString& incPath) {
191 OUString path = OStringToOUString(incPath, RTL_TEXTENCODING_UTF8);
193 std::vector<sal_Unicode> vec(path.getLength() + 1);
194 //GetShortPathNameW only works if the file can be found!
195 const DWORD len = GetShortPathNameW(
196 o3tl::toW(path.getStr()), o3tl::toW(&vec[0]), path.getLength() + 1);
198 if (len > 0)
200 OUString ret(&vec[0], len);
201 return OUStringToOString(ret, RTL_TEXTENCODING_UTF8);
204 return incPath;
206 #endif
208 bool Options::initOptions(std::vector< std::string > & rArgs)
210 std::vector< std::string >::const_iterator first = rArgs.begin(), last = rArgs.end();
211 for (; first != last; ++first)
213 if ((*first)[0] != '-')
215 OString filename((*first).c_str(), (*first).size());
216 OString tmp(filename.toAsciiLowerCase());
217 if (tmp.lastIndexOf(".idl") != (tmp.getLength() - 4))
219 throw IllegalArgument("'" + filename + "' is not a valid input file, only '*.idl' files will be accepted");
221 m_inputFiles.push_back(filename);
222 continue;
225 std::string const option(*first);
226 switch((*first)[1])
228 case 'O':
230 if ((++first == last) || ((*first)[0] == '-'))
232 return badOption("invalid", option);
234 OString param((*first).c_str(), (*first).size());
235 m_options["-O"] = param;
236 break;
238 case 'M':
240 if ((++first == last) || ((*first)[0] == '-'))
242 return badOption("invalid", option);
244 OString param((*first).c_str(), (*first).size());
245 m_options["-M"] = param;
246 break;
248 case 'I':
250 if ((++first == last) || ((*first)[0] == '-'))
252 return badOption("invalid", option);
254 OString param((*first).c_str(), (*first).size());
256 // quote param token(s).
257 OStringBuffer buffer;
258 sal_Int32 k = 0;
261 if (!buffer.isEmpty())
262 buffer.append(' ');
263 // buffer.append("-I\"");
264 #ifdef _WIN32
265 OString incpath = convertIncPathtoShortWindowsPath(param.getToken(0, ';', k));
266 #else
267 OString incpath = param.getToken(0, ';', k);
268 #endif
269 buffer.append(incpath);
270 // buffer.append("\"");
271 } while (k != -1);
272 param = buffer.makeStringAndClear();
274 if (m_options.count("-I") > 0)
276 // append param.
277 OStringBuffer buffer(m_options["-I"]);
278 buffer.append(' '); buffer.append(param);
279 param = buffer.makeStringAndClear();
281 m_options["-I"] = param;
282 break;
284 case 'D':
286 if ((++first == last) || ((*first)[0] == '-'))
288 return badOption("invalid", option);
290 OString param("-D");
291 param += OString((*first).c_str(), (*first).size());
292 if (m_options.count("-D") > 0)
294 OStringBuffer buffer(m_options["-D"]);
295 buffer.append(' '); buffer.append(param);
296 param = buffer.makeStringAndClear();
298 m_options["-D"] = param;
299 break;
301 case 'C':
303 if (!setOption("-C", option))
305 return badOption("invalid", option);
307 break;
309 case 'c':
311 if (!setOption("-cid", option))
313 return badOption("invalid", option);
315 break;
317 case 'q':
319 if (!setOption("-quiet", option))
321 return badOption("invalid", option);
323 m_quiet = true;
324 break;
326 case 'v':
328 if (!setOption("-verbose", option))
330 return badOption("invalid", option);
332 m_verbose = true;
333 break;
335 case 'w':
337 if (!(setOption("-w", option) || setOption("-we", option)))
339 return badOption("invalid", option);
341 break;
343 case 'h':
344 case '?':
346 if (!(setOption("-h", option) || setOption("-?", option)))
348 return badOption("invalid", option);
351 (void) fprintf(stdout, "%s", prepareHelp().getStr());
352 return false;
354 // break; // Unreachable
356 case 's':
358 if (!setOption("-stdin", option))
360 return badOption("invalid", option);
362 m_stdin = true;
363 break;
365 default:
366 return badOption("unknown", option);
369 return true;
372 OString Options::prepareHelp() const
374 OString help("\nusing: ");
375 help += m_program + " [-options] <file_1> ... <file_n> | @<filename> | -stdin\n";
376 help += " <file_n> = file_n specifies one or more idl files.\n";
377 help += " Only files with the extension '.idl' are valid.\n";
378 help += " @<filename> = filename specifies the name of a command file.\n";
379 help += " -stdin = read idl file from standard input.\n";
380 help += " Options:\n";
381 help += " -O<path> = path specifies the output directory.\n";
382 help += " The generated output is a registry file with\n";
383 help += " the same name as the idl input file (or 'stdin'\n";
384 help += " for -stdin).\n";
385 help += " -M<path> = path specifies the output directory for deps.\n";
386 help += " Generate GNU make dependency files with the\n";
387 help += " same name as the idl input file.\n";
388 help += " -I<path> = path specifies a directory where include\n";
389 help += " files will be searched by the preprocessor.\n";
390 help += " Multiple directories can be combined with ';'.\n";
391 help += " -D<name> = name defines a macro for the preprocessor.\n";
392 help += " -C = generate complete type information, including\n";
393 help += " documentation.\n";
394 help += " -cid = check if identifiers fulfill the UNO naming\n";
395 help += " requirements.\n";
396 help += " -quiet = no output.\n";
397 help += " -verbose = verbose output.\n";
398 help += " -w = display warning messages.\n";
399 help += " -we = treat warnings as errors.\n";
400 help += " -h|-? = print this help message and exit.\n\n";
401 help += prepareVersion();
403 return help;
406 OString Options::prepareVersion() const
408 OString version(m_program);
409 version += " Version 1.1\n\n";
410 return version;
414 bool Options::isValid(const OString& option) const
416 return (m_options.count(option) > 0);
419 const OString& Options::getOption(const OString& option)
421 if (!isValid(option))
423 throw IllegalArgument("Option is not valid or currently not set.");
425 return m_options[option];
428 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */