1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
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 "idlc/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"
35 #ifndef WIN32_LEAN_AND_MEAN
36 # define WIN32_LEAN_AND_MEAN
38 # pragma warning(push,1)
45 # undef WIN32_LEAN_AND_MEAN
53 Options::Options(char const * progname
)
54 : m_program(progname
), m_stdin(false), m_verbose(false), m_quiet(false)
63 bool Options::checkArgument (std::vector
< std::string
> & rArgs
, char const * arg
, size_t len
)
65 bool result
= ((arg
!= 0) && (len
> 0));
66 OSL_PRECOND(result
, "idlc::Options::checkArgument(): invalid arguments");
72 if ((result
= (len
> 1)))
75 result
= Options::checkCommandFile (rArgs
, &(arg
[1]));
79 if ((result
= (len
> 1)))
89 // "-<option>[<param>]
90 std::string
option(&(arg
[0]), 2);
91 rArgs
.push_back(option
);
95 std::string
param(&(arg
[2]), len
- 2);
96 rArgs
.push_back(param
);
101 // "-<option>" ([long] option, w/o param)
102 rArgs
.push_back(std::string(arg
, len
));
109 rArgs
.push_back(std::string(arg
, len
));
117 bool Options::checkCommandFile (std::vector
< std::string
> & rArgs
, char const * filename
)
119 FILE * fp
= fopen(filename
, "r");
122 fprintf(stderr
, "ERROR: can't open command file \"%s\"\n", filename
);
131 while ((c
= fgetc(fp
)) != EOF
)
146 // append current argument.
147 if (!Options::checkArgument(rArgs
, buffer
.c_str(), buffer
.size()))
157 // quoted white-space fall through
158 buffer
.push_back(sal::static_int_cast
<char>(c
));
164 // append unterminated argument.
165 if (!Options::checkArgument(rArgs
, buffer
.c_str(), buffer
.size()))
172 return (fclose(fp
) == 0);
175 bool Options::badOption(char const * reason
, std::string
const & rArg
) throw(IllegalArgument
)
177 OStringBuffer message
;
180 message
.append(reason
); message
.append(" option '"); message
.append(rArg
.c_str()); message
.append("'");
181 throw IllegalArgument(message
.makeStringAndClear());
186 bool Options::setOption(char const * option
, std::string
const & rArg
)
188 bool result
= (0 == strcmp(option
, rArg
.c_str()));
190 m_options
[rArg
.c_str()] = OString(rArg
.c_str(), rArg
.size());
195 /* Helper functiopn to convert windows paths including spaces, brackets etc. into
196 a windows short Url. The ucpp preprocessor has problems with such paths and returns
199 OString
convertIncPathtoShortWindowsPath(const OString
& incPath
) {
200 OUString path
= OStringToOUString(incPath
, RTL_TEXTENCODING_UTF8
);
202 std::vector
<sal_Unicode
> vec(path
.getLength() + 1);
203 //GetShortPathNameW only works if the file can be found!
204 const DWORD len
= GetShortPathNameW(
205 reinterpret_cast<LPCWSTR
>(path
.getStr()), reinterpret_cast<LPWSTR
>(&vec
[0]), path
.getLength() + 1);
209 OUString
ret(&vec
[0], len
);
210 return OUStringToOString(ret
, RTL_TEXTENCODING_UTF8
);
217 bool Options::initOptions(std::vector
< std::string
> & rArgs
) throw(IllegalArgument
)
219 std::vector
< std::string
>::const_iterator first
= rArgs
.begin(), last
= rArgs
.end();
220 for (; first
!= last
; ++first
)
222 if ((*first
)[0] != '-')
224 OString
filename((*first
).c_str(), (*first
).size());
225 OString
tmp(filename
.toAsciiLowerCase());
226 if (tmp
.lastIndexOf(".idl") != (tmp
.getLength() - 4))
228 throw IllegalArgument("'" + filename
+ "' is not a valid input file, only '*.idl' files will be accepted");
230 m_inputFiles
.push_back(filename
);
234 std::string
const option(*first
);
239 if (!((++first
!= last
) && ((*first
)[0] != '-')))
241 return badOption("invalid", option
);
243 OString
param((*first
).c_str(), (*first
).size());
244 m_options
["-O"] = param
;
249 if (!((++first
!= last
) && ((*first
)[0] != '-')))
251 return badOption("invalid", option
);
253 OString
param((*first
).c_str(), (*first
).size());
254 m_options
["-M"] = param
;
259 if (!((++first
!= last
) && ((*first
)[0] != '-')))
261 return badOption("invalid", option
);
263 OString
param((*first
).c_str(), (*first
).size());
265 // quote param token(s).
266 OStringBuffer buffer
;
270 if (!buffer
.isEmpty())
272 // buffer.append("-I\"");
274 OString incpath
= convertIncPathtoShortWindowsPath(param
.getToken(0, ';', k
));
276 OString incpath
= param
.getToken(0, ';', k
);
278 buffer
.append(incpath
);
279 // buffer.append("\"");
281 param
= buffer
.makeStringAndClear();
283 if (m_options
.count("-I") > 0)
286 OStringBuffer
buffer(m_options
["-I"]);
287 buffer
.append(' '); buffer
.append(param
);
288 param
= buffer
.makeStringAndClear();
290 m_options
["-I"] = param
;
295 if (!((++first
!= last
) && ((*first
)[0] != '-')))
297 return badOption("invalid", option
);
299 OString
param("-D"); param
+= OString((*first
).c_str(), (*first
).size());
300 if (m_options
.count("-D") > 0)
302 OStringBuffer
buffer(m_options
["-D"]);
303 buffer
.append(' '); buffer
.append(param
);
304 param
= buffer
.makeStringAndClear();
306 m_options
["-D"] = param
;
311 if (!setOption("-C", option
))
313 return badOption("invalid", option
);
319 if (!setOption("-cid", option
))
321 return badOption("invalid", option
);
327 if (!setOption("-quiet", option
))
329 return badOption("invalid", option
);
336 if (!setOption("-verbose", option
))
338 return badOption("invalid", option
);
345 if (!(setOption("-w", option
) || setOption("-we", option
)))
347 return badOption("invalid", option
);
354 if (!(setOption("-h", option
) || setOption("-?", option
)))
356 return badOption("invalid", option
);
359 (void) fprintf(stdout
, "%s", prepareHelp().getStr());
362 // break; // Unreachable
366 if (!setOption("-stdin", option
))
368 return badOption("invalid", option
);
374 return badOption("unknown", option
);
380 OString
Options::prepareHelp()
382 OString
help("\nusing: ");
383 help
+= m_program
+ " [-options] <file_1> ... <file_n> | @<filename> | -stdin\n";
384 help
+= " <file_n> = file_n specifies one or more idl files.\n";
385 help
+= " Only files with the extension '.idl' are valid.\n";
386 help
+= " @<filename> = filename specifies the name of a command file.\n";
387 help
+= " -stdin = read idl file from standard input.\n";
388 help
+= " Options:\n";
389 help
+= " -O<path> = path specifies the output directory.\n";
390 help
+= " The generated output is a registry file with\n";
391 help
+= " the same name as the idl input file (or 'stdin'\n";
392 help
+= " for -stdin).\n";
393 help
+= " -M<path> = path specifies the output directory for deps.\n";
394 help
+= " Generate GNU make dependency files with the\n";
395 help
+= " same name as the idl input file.\n";
396 help
+= " -I<path> = path specifies a directory where include\n";
397 help
+= " files will be searched by the preprocessor.\n";
398 help
+= " Multiple directories can be combined with ';'.\n";
399 help
+= " -D<name> = name defines a macro for the preprocessor.\n";
400 help
+= " -C = generate complete type information, including\n";
401 help
+= " documentation.\n";
402 help
+= " -cid = check if identifiers fulfill the UNO naming\n";
403 help
+= " requirements.\n";
404 help
+= " -quiet = no output.\n";
405 help
+= " -verbose = verbose output.\n";
406 help
+= " -w = display warning messages.\n";
407 help
+= " -we = treat warnings as errors.\n";
408 help
+= " -h|-? = print this help message and exit.\n\n";
409 help
+= prepareVersion();
414 OString
Options::prepareVersion()
416 OString
version(m_program
);
417 version
+= " Version 1.1\n\n";
422 bool Options::isValid(const OString
& option
)
424 return (m_options
.count(option
) > 0);
427 const OString
& Options::getOption(const OString
& option
)
428 throw( IllegalArgument
)
430 if (!isValid(option
))
432 throw IllegalArgument("Option is not valid or currently not set.");
434 return m_options
[option
];
437 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */