VirtualBox

source: vbox/trunk/src/bldprogs/scm.cpp@ 69753

Last change on this file since 69753 was 69753, checked in by vboxsync, 7 years ago

iprt/dir: Morphing PRTDIR into a handle named RTDIR. (Been wanting to correct this for years. Don't know why I makde it a pointer rather than an abstrct handle like everything else.)

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 106.7 KB
Line 
1/* $Id: scm.cpp 69753 2017-11-19 14:27:58Z vboxsync $ */
2/** @file
3 * IPRT Testcase / Tool - Source Code Massager.
4 */
5
6/*
7 * Copyright (C) 2010-2017 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#include <iprt/assert.h>
23#include <iprt/ctype.h>
24#include <iprt/dir.h>
25#include <iprt/env.h>
26#include <iprt/file.h>
27#include <iprt/err.h>
28#include <iprt/getopt.h>
29#include <iprt/initterm.h>
30#include <iprt/mem.h>
31#include <iprt/message.h>
32#include <iprt/param.h>
33#include <iprt/path.h>
34#include <iprt/process.h>
35#include <iprt/stream.h>
36#include <iprt/string.h>
37
38#include "scm.h"
39#include "scmdiff.h"
40
41
42/*********************************************************************************************************************************
43* Defined Constants And Macros *
44*********************************************************************************************************************************/
45/** The name of the settings files. */
46#define SCM_SETTINGS_FILENAME ".scm-settings"
47
48
49/*********************************************************************************************************************************
50* Structures and Typedefs *
51*********************************************************************************************************************************/
52
53/**
54 * Option identifiers.
55 *
56 * @note The first chunk, down to SCMOPT_TAB_SIZE, are alternately set &
57 * clear. So, the option setting a flag (boolean) will have an even
58 * number and the one clearing it will have an odd number.
59 * @note Down to SCMOPT_LAST_SETTINGS corresponds exactly to SCMSETTINGSBASE.
60 */
61typedef enum SCMOPT
62{
63 SCMOPT_CONVERT_EOL = 10000,
64 SCMOPT_NO_CONVERT_EOL,
65 SCMOPT_CONVERT_TABS,
66 SCMOPT_NO_CONVERT_TABS,
67 SCMOPT_FORCE_FINAL_EOL,
68 SCMOPT_NO_FORCE_FINAL_EOL,
69 SCMOPT_FORCE_TRAILING_LINE,
70 SCMOPT_NO_FORCE_TRAILING_LINE,
71 SCMOPT_STRIP_TRAILING_BLANKS,
72 SCMOPT_NO_STRIP_TRAILING_BLANKS,
73 SCMOPT_STRIP_TRAILING_LINES,
74 SCMOPT_NO_STRIP_TRAILING_LINES,
75 SCMOPT_FIX_FLOWER_BOX_MARKERS,
76 SCMOPT_NO_FIX_FLOWER_BOX_MARKERS,
77 SCMOPT_FIX_TODOS,
78 SCMOPT_NO_FIX_TODOS,
79 SCMOPT_UPDATE_COPYRIGHT_YEAR,
80 SCMOPT_NO_UPDATE_COPYRIGHT_YEAR,
81 SCMOPT_EXTERNAL_COPYRIGHT,
82 SCMOPT_NO_EXTERNAL_COPYRIGHT,
83 SCMOPT_NO_UPDATE_LICENSE,
84 SCMOPT_LICENSE_OSE_GPL,
85 SCMOPT_LICENSE_OSE_DUAL_GPL_CDDL,
86 SCMOPT_LICENSE_OSE_CDDL,
87 SCMOPT_LICENSE_LGPL,
88 SCMOPT_LICENSE_MIT,
89 SCMOPT_LICENSE_BASED_ON_MIT,
90 SCMOPT_LGPL_DISCLAIMER,
91 SCMOPT_NO_LGPL_DISCLAIMER,
92 SCMOPT_MIN_BLANK_LINES_BEFORE_FLOWER_BOX_MARKERS,
93 SCMOPT_ONLY_SVN_DIRS,
94 SCMOPT_NOT_ONLY_SVN_DIRS,
95 SCMOPT_ONLY_SVN_FILES,
96 SCMOPT_NOT_ONLY_SVN_FILES,
97 SCMOPT_SET_SVN_EOL,
98 SCMOPT_DONT_SET_SVN_EOL,
99 SCMOPT_SET_SVN_EXECUTABLE,
100 SCMOPT_DONT_SET_SVN_EXECUTABLE,
101 SCMOPT_SET_SVN_KEYWORDS,
102 SCMOPT_DONT_SET_SVN_KEYWORDS,
103 SCMOPT_TAB_SIZE,
104 SCMOPT_WIDTH,
105 SCMOPT_FILTER_OUT_DIRS,
106 SCMOPT_FILTER_FILES,
107 SCMOPT_FILTER_OUT_FILES,
108 SCMOPT_TREAT_AS,
109 SCMOPT_ADD_ACTION,
110 SCMOPT_DEL_ACTION,
111 SCMOPT_LAST_SETTINGS = SCMOPT_DEL_ACTION,
112 //
113 SCMOPT_CHECK_RUN,
114 SCMOPT_DIFF_IGNORE_EOL,
115 SCMOPT_DIFF_NO_IGNORE_EOL,
116 SCMOPT_DIFF_IGNORE_SPACE,
117 SCMOPT_DIFF_NO_IGNORE_SPACE,
118 SCMOPT_DIFF_IGNORE_LEADING_SPACE,
119 SCMOPT_DIFF_NO_IGNORE_LEADING_SPACE,
120 SCMOPT_DIFF_IGNORE_TRAILING_SPACE,
121 SCMOPT_DIFF_NO_IGNORE_TRAILING_SPACE,
122 SCMOPT_DIFF_SPECIAL_CHARS,
123 SCMOPT_DIFF_NO_SPECIAL_CHARS,
124 SCMOPT_HELP_CONFIG,
125 SCMOPT_HELP_ACTIONS,
126 SCMOPT_END
127} SCMOPT;
128
129
130/*********************************************************************************************************************************
131* Global Variables *
132*********************************************************************************************************************************/
133const char g_szTabSpaces[16+1] = " ";
134const char g_szAsterisks[255+1] =
135"****************************************************************************************************"
136"****************************************************************************************************"
137"*******************************************************";
138const char g_szSpaces[255+1] =
139" "
140" "
141" ";
142static const char g_szProgName[] = "scm";
143static const char *g_pszChangedSuff = "";
144static bool g_fDryRun = true;
145static bool g_fDiffSpecialChars = true;
146static bool g_fDiffIgnoreEol = false;
147static bool g_fDiffIgnoreLeadingWS = false;
148static bool g_fDiffIgnoreTrailingWS = false;
149static int g_iVerbosity = 2;//99; //0;
150uint32_t g_uYear = 0; /**< The current year. */
151/** @name Statistics
152 * @{ */
153static uint32_t g_cDirsProcessed = 0;
154static uint32_t g_cFilesProcessed = 0;
155static uint32_t g_cFilesModified = 0;
156static uint32_t g_cFilesSkipped = 0;
157static uint32_t g_cFilesNotInSvn = 0;
158static uint32_t g_cFilesNoRewriters = 0;
159static uint32_t g_cFilesBinaries = 0;
160/** @} */
161
162/** The global settings. */
163static SCMSETTINGSBASE const g_Defaults =
164{
165 /* .fConvertEol = */ true,
166 /* .fConvertTabs = */ true,
167 /* .fForceFinalEol = */ true,
168 /* .fForceTrailingLine = */ false,
169 /* .fStripTrailingBlanks = */ true,
170 /* .fStripTrailingLines = */ true,
171 /* .fFixFlowerBoxMarkers = */ true,
172 /* .cMinBlankLinesBeforeFlowerBoxMakers = */ 2,
173 /* .fFixTodos = */ true,
174 /* .fUpdateCopyrightYear = */ false,
175 /* .fExternalCopyright = */ false,
176 /* .fLgplDisclaimer = */ false,
177 /* .enmUpdateLicense = */ kScmLicense_OseGpl,
178 /* .fOnlySvnFiles = */ false,
179 /* .fOnlySvnDirs = */ false,
180 /* .fSetSvnEol = */ false,
181 /* .fSetSvnExecutable = */ false,
182 /* .fSetSvnKeywords = */ false,
183 /* .cchTab = */ 8,
184 /* .cchWidth = */ 130,
185 /* .fFreeTreatAs = */ false,
186 /* .pTreatAs = */ NULL,
187 /* .pszFilterFiles = */ (char *)"",
188 /* .pszFilterOutFiles = */ (char *)"*.exe|*.com|20*-*-*.log",
189 /* .pszFilterOutDirs = */ (char *)".svn|.hg|.git|CVS",
190};
191
192/** Option definitions for the base settings. */
193static RTGETOPTDEF g_aScmOpts[] =
194{
195 /* rewriters */
196 { "--convert-eol", SCMOPT_CONVERT_EOL, RTGETOPT_REQ_NOTHING },
197 { "--no-convert-eol", SCMOPT_NO_CONVERT_EOL, RTGETOPT_REQ_NOTHING },
198 { "--convert-tabs", SCMOPT_CONVERT_TABS, RTGETOPT_REQ_NOTHING },
199 { "--no-convert-tabs", SCMOPT_NO_CONVERT_TABS, RTGETOPT_REQ_NOTHING },
200 { "--force-final-eol", SCMOPT_FORCE_FINAL_EOL, RTGETOPT_REQ_NOTHING },
201 { "--no-force-final-eol", SCMOPT_NO_FORCE_FINAL_EOL, RTGETOPT_REQ_NOTHING },
202 { "--force-trailing-line", SCMOPT_FORCE_TRAILING_LINE, RTGETOPT_REQ_NOTHING },
203 { "--no-force-trailing-line", SCMOPT_NO_FORCE_TRAILING_LINE, RTGETOPT_REQ_NOTHING },
204 { "--strip-trailing-blanks", SCMOPT_STRIP_TRAILING_BLANKS, RTGETOPT_REQ_NOTHING },
205 { "--no-strip-trailing-blanks", SCMOPT_NO_STRIP_TRAILING_BLANKS, RTGETOPT_REQ_NOTHING },
206 { "--strip-trailing-lines", SCMOPT_STRIP_TRAILING_LINES, RTGETOPT_REQ_NOTHING },
207 { "--strip-no-trailing-lines", SCMOPT_NO_STRIP_TRAILING_LINES, RTGETOPT_REQ_NOTHING },
208 { "--min-blank-lines-before-flower-box-makers", SCMOPT_MIN_BLANK_LINES_BEFORE_FLOWER_BOX_MARKERS, RTGETOPT_REQ_UINT8 },
209 { "--fix-flower-box-markers", SCMOPT_FIX_FLOWER_BOX_MARKERS, RTGETOPT_REQ_NOTHING },
210 { "--no-fix-flower-box-markers", SCMOPT_NO_FIX_FLOWER_BOX_MARKERS, RTGETOPT_REQ_NOTHING },
211 { "--fix-todos", SCMOPT_FIX_TODOS, RTGETOPT_REQ_NOTHING },
212 { "--no-fix-todos", SCMOPT_NO_FIX_TODOS, RTGETOPT_REQ_NOTHING },
213 { "--update-copyright-year", SCMOPT_UPDATE_COPYRIGHT_YEAR, RTGETOPT_REQ_NOTHING },
214 { "--no-update-copyright-year", SCMOPT_NO_UPDATE_COPYRIGHT_YEAR, RTGETOPT_REQ_NOTHING },
215 { "--external-copyright", SCMOPT_EXTERNAL_COPYRIGHT, RTGETOPT_REQ_NOTHING },
216 { "--no-external-copyright", SCMOPT_NO_EXTERNAL_COPYRIGHT, RTGETOPT_REQ_NOTHING },
217 { "--no-update-license", SCMOPT_NO_UPDATE_LICENSE, RTGETOPT_REQ_NOTHING },
218 { "--license-ose-gpl", SCMOPT_LICENSE_OSE_GPL, RTGETOPT_REQ_NOTHING },
219 { "--license-ose-dual", SCMOPT_LICENSE_OSE_DUAL_GPL_CDDL, RTGETOPT_REQ_NOTHING },
220 { "--license-ose-cddl", SCMOPT_LICENSE_OSE_CDDL, RTGETOPT_REQ_NOTHING },
221 { "--license-lgpl", SCMOPT_LICENSE_LGPL, RTGETOPT_REQ_NOTHING },
222 { "--license-mit", SCMOPT_LICENSE_MIT, RTGETOPT_REQ_NOTHING },
223 { "--license-based-on-mit", SCMOPT_LICENSE_BASED_ON_MIT, RTGETOPT_REQ_NOTHING },
224 { "--lgpl-disclaimer", SCMOPT_LGPL_DISCLAIMER, RTGETOPT_REQ_NOTHING },
225 { "--no-lgpl-disclaimer", SCMOPT_NO_LGPL_DISCLAIMER, RTGETOPT_REQ_NOTHING },
226 { "--set-svn-eol", SCMOPT_SET_SVN_EOL, RTGETOPT_REQ_NOTHING },
227 { "--dont-set-svn-eol", SCMOPT_DONT_SET_SVN_EOL, RTGETOPT_REQ_NOTHING },
228 { "--set-svn-executable", SCMOPT_SET_SVN_EXECUTABLE, RTGETOPT_REQ_NOTHING },
229 { "--dont-set-svn-executable", SCMOPT_DONT_SET_SVN_EXECUTABLE, RTGETOPT_REQ_NOTHING },
230 { "--set-svn-keywords", SCMOPT_SET_SVN_KEYWORDS, RTGETOPT_REQ_NOTHING },
231 { "--dont-set-svn-keywords", SCMOPT_DONT_SET_SVN_KEYWORDS, RTGETOPT_REQ_NOTHING },
232 { "--tab-size", SCMOPT_TAB_SIZE, RTGETOPT_REQ_UINT8 },
233 { "--width", SCMOPT_WIDTH, RTGETOPT_REQ_UINT8 },
234
235 /* input selection */
236 { "--only-svn-dirs", SCMOPT_ONLY_SVN_DIRS, RTGETOPT_REQ_NOTHING },
237 { "--not-only-svn-dirs", SCMOPT_NOT_ONLY_SVN_DIRS, RTGETOPT_REQ_NOTHING },
238 { "--only-svn-files", SCMOPT_ONLY_SVN_FILES, RTGETOPT_REQ_NOTHING },
239 { "--not-only-svn-files", SCMOPT_NOT_ONLY_SVN_FILES, RTGETOPT_REQ_NOTHING },
240 { "--filter-out-dirs", SCMOPT_FILTER_OUT_DIRS, RTGETOPT_REQ_STRING },
241 { "--filter-files", SCMOPT_FILTER_FILES, RTGETOPT_REQ_STRING },
242 { "--filter-out-files", SCMOPT_FILTER_OUT_FILES, RTGETOPT_REQ_STRING },
243
244 /* rewriter selection */
245 { "--treat-as", SCMOPT_TREAT_AS, RTGETOPT_REQ_STRING },
246 { "--add-action", SCMOPT_ADD_ACTION, RTGETOPT_REQ_STRING },
247 { "--del-action", SCMOPT_DEL_ACTION, RTGETOPT_REQ_STRING },
248
249 /* Additional help */
250 { "--help-config", SCMOPT_HELP_CONFIG, RTGETOPT_REQ_NOTHING },
251 { "--help-actions", SCMOPT_HELP_ACTIONS, RTGETOPT_REQ_NOTHING },
252};
253
254/** Consider files matching the following patterns (base names only). */
255static const char *g_pszFileFilter = NULL;
256
257/* The rewriter configuration. */
258#define SCM_REWRITER_CFG(a_Global, a_szName, fnRewriter) static const SCMREWRITERCFG a_Global = { &fnRewriter, a_szName }
259SCM_REWRITER_CFG(g_StripTrailingBlanks, "strip-trailing-blanks", rewrite_StripTrailingBlanks);
260SCM_REWRITER_CFG(g_ExpandTabs, "expand-tabs", rewrite_ExpandTabs);
261SCM_REWRITER_CFG(g_ForceNativeEol, "force-native-eol", rewrite_ForceNativeEol);
262SCM_REWRITER_CFG(g_ForceLF, "force-lf", rewrite_ForceLF);
263SCM_REWRITER_CFG(g_ForceCRLF, "force-crlf", rewrite_ForceCRLF);
264SCM_REWRITER_CFG(g_AdjustTrailingLines, "adjust-trailing-lines", rewrite_AdjustTrailingLines);
265SCM_REWRITER_CFG(g_SvnNoExecutable, "svn-no-executable", rewrite_SvnNoExecutable);
266SCM_REWRITER_CFG(g_SvnNoKeywords, "svn-no-keywords", rewrite_SvnNoKeywords);
267SCM_REWRITER_CFG(g_SvnNoEolStyle, "svn-no-eol-style", rewrite_SvnNoEolStyle);
268SCM_REWRITER_CFG(g_SvnBinary, "svn-binary", rewrite_SvnBinary);
269SCM_REWRITER_CFG(g_SvnKeywords, "svn-keywords", rewrite_SvnKeywords);
270SCM_REWRITER_CFG(g_Copyright_CstyleComment, "copyright-c-style", rewrite_Copyright_CstyleComment);
271SCM_REWRITER_CFG(g_Copyright_HashComment, "copyright-hash-style", rewrite_Copyright_HashComment);
272SCM_REWRITER_CFG(g_Copyright_PythonComment, "copyright-python-style", rewrite_Copyright_PythonComment);
273SCM_REWRITER_CFG(g_Copyright_RemComment, "copyright-rem-style", rewrite_Copyright_RemComment);
274SCM_REWRITER_CFG(g_Copyright_SemicolonComment, "copyright-semicolon-style", rewrite_Copyright_SemicolonComment);
275SCM_REWRITER_CFG(g_Copyright_SqlComment, "copyright-sql-style", rewrite_Copyright_SqlComment);
276SCM_REWRITER_CFG(g_Copyright_TickComment, "copyright-tick-style", rewrite_Copyright_TickComment);
277SCM_REWRITER_CFG(g_Makefile_kup, "makefile-kup", rewrite_Makefile_kup);
278SCM_REWRITER_CFG(g_Makefile_kmk, "makefile-kmk", rewrite_Makefile_kmk);
279SCM_REWRITER_CFG(g_FixFlowerBoxMarkers, "fix-flower-boxes", rewrite_FixFlowerBoxMarkers);
280SCM_REWRITER_CFG(g_Fix_C_and_CPP_Todos, "fix-c-todos", rewrite_Fix_C_and_CPP_Todos);
281SCM_REWRITER_CFG(g_C_and_CPP, "c-and-cpp", rewrite_C_and_CPP);
282
283/** The rewriter actions. */
284static PCSCMREWRITERCFG const g_papRewriterActions[] =
285{
286 &g_StripTrailingBlanks,
287 &g_ExpandTabs,
288 &g_ForceNativeEol,
289 &g_ForceLF,
290 &g_ForceCRLF,
291 &g_AdjustTrailingLines,
292 &g_SvnNoExecutable,
293 &g_SvnNoKeywords,
294 &g_SvnNoEolStyle,
295 &g_SvnBinary,
296 &g_SvnKeywords,
297 &g_Copyright_CstyleComment,
298 &g_Copyright_HashComment,
299 &g_Copyright_PythonComment,
300 &g_Copyright_RemComment,
301 &g_Copyright_SemicolonComment,
302 &g_Copyright_SqlComment,
303 &g_Copyright_TickComment,
304 &g_Makefile_kup,
305 &g_Makefile_kmk,
306 &g_FixFlowerBoxMarkers,
307 &g_Fix_C_and_CPP_Todos,
308 &g_C_and_CPP,
309};
310
311
312static PCSCMREWRITERCFG const g_apRewritersFor_Makefile_kup[] =
313{
314 &g_SvnNoExecutable,
315 &g_Makefile_kup
316};
317
318static PCSCMREWRITERCFG const g_apRewritersFor_Makefile_kmk[] =
319{
320 &g_ForceNativeEol,
321 &g_StripTrailingBlanks,
322 &g_AdjustTrailingLines,
323 &g_SvnNoExecutable,
324 &g_SvnKeywords,
325 &g_Copyright_HashComment,
326 &g_Makefile_kmk
327};
328
329static PCSCMREWRITERCFG const g_apRewritersFor_OtherMakefiles[] =
330{
331 &g_ForceNativeEol,
332 &g_StripTrailingBlanks,
333 &g_AdjustTrailingLines,
334 &g_SvnNoExecutable,
335 &g_SvnKeywords,
336 &g_Copyright_HashComment,
337};
338
339static PCSCMREWRITERCFG const g_apRewritersFor_C_and_CPP[] =
340{
341 &g_ForceNativeEol,
342 &g_ExpandTabs,
343 &g_StripTrailingBlanks,
344 &g_AdjustTrailingLines,
345 &g_SvnNoExecutable,
346 &g_SvnKeywords,
347 &g_Copyright_CstyleComment,
348 &g_FixFlowerBoxMarkers,
349 &g_Fix_C_and_CPP_Todos,
350 &g_C_and_CPP
351};
352
353static PCSCMREWRITERCFG const g_apRewritersFor_H_and_HPP[] =
354{
355 &g_ForceNativeEol,
356 &g_ExpandTabs,
357 &g_StripTrailingBlanks,
358 &g_AdjustTrailingLines,
359 &g_SvnNoExecutable,
360 &g_SvnKeywords,
361 &g_Copyright_CstyleComment,
362 &g_C_and_CPP
363};
364
365static PCSCMREWRITERCFG const g_apRewritersFor_RC[] =
366{
367 &g_ForceNativeEol,
368 &g_ExpandTabs,
369 &g_StripTrailingBlanks,
370 &g_AdjustTrailingLines,
371 &g_SvnNoExecutable,
372 &g_SvnKeywords,
373 &g_Copyright_CstyleComment,
374};
375
376static PCSCMREWRITERCFG const g_apRewritersFor_DTrace[] =
377{
378 &g_ForceNativeEol,
379 &g_ExpandTabs,
380 &g_StripTrailingBlanks,
381 &g_AdjustTrailingLines,
382 &g_SvnKeywords,
383 &g_Copyright_CstyleComment,
384};
385
386static PCSCMREWRITERCFG const g_apRewritersFor_DSL[] =
387{
388 &g_ForceNativeEol,
389 &g_ExpandTabs,
390 &g_StripTrailingBlanks,
391 &g_AdjustTrailingLines,
392 &g_SvnNoExecutable,
393 &g_SvnKeywords,
394 &g_Copyright_CstyleComment,
395};
396
397static PCSCMREWRITERCFG const g_apRewritersFor_ASM[] =
398{
399 &g_ForceNativeEol,
400 &g_ExpandTabs,
401 &g_StripTrailingBlanks,
402 &g_AdjustTrailingLines,
403 &g_SvnNoExecutable,
404 &g_SvnKeywords,
405 &g_Copyright_SemicolonComment,
406};
407
408static PCSCMREWRITERCFG const g_apRewritersFor_DEF[] =
409{
410 &g_ForceNativeEol,
411 &g_ExpandTabs,
412 &g_StripTrailingBlanks,
413 &g_AdjustTrailingLines,
414 &g_SvnNoExecutable,
415 &g_SvnKeywords,
416 &g_Copyright_SemicolonComment,
417};
418
419static PCSCMREWRITERCFG const g_apRewritersFor_ShellScripts[] =
420{
421 &g_ForceLF,
422 &g_ExpandTabs,
423 &g_StripTrailingBlanks,
424 &g_Copyright_HashComment,
425};
426
427static PCSCMREWRITERCFG const g_apRewritersFor_BatchFiles[] =
428{
429 &g_ForceCRLF,
430 &g_ExpandTabs,
431 &g_StripTrailingBlanks,
432 &g_Copyright_RemComment,
433};
434
435static PCSCMREWRITERCFG const g_apRewritersFor_BasicScripts[] =
436{
437 &g_ForceCRLF,
438 &g_ExpandTabs,
439 &g_StripTrailingBlanks,
440 &g_Copyright_TickComment,
441};
442
443static PCSCMREWRITERCFG const g_apRewritersFor_SedScripts[] =
444{
445 &g_ForceLF,
446 &g_ExpandTabs,
447 &g_StripTrailingBlanks,
448 &g_Copyright_HashComment,
449};
450
451static PCSCMREWRITERCFG const g_apRewritersFor_Python[] =
452{
453 /** @todo &g_ForceLFIfExecutable */
454 &g_ExpandTabs,
455 &g_StripTrailingBlanks,
456 &g_AdjustTrailingLines,
457 &g_SvnKeywords,
458 &g_Copyright_PythonComment,
459};
460
461static PCSCMREWRITERCFG const g_apRewritersFor_Perl[] =
462{
463 /** @todo &g_ForceLFIfExecutable */
464 &g_ExpandTabs,
465 &g_StripTrailingBlanks,
466 &g_AdjustTrailingLines,
467 &g_SvnKeywords,
468 &g_Copyright_HashComment,
469};
470
471static PCSCMREWRITERCFG const g_apRewritersFor_DriverInfFiles[] =
472{
473 &g_ForceNativeEol,
474 &g_ExpandTabs,
475 &g_StripTrailingBlanks,
476 &g_AdjustTrailingLines,
477 &g_SvnKeywords,
478 &g_SvnNoExecutable,
479 &g_Copyright_SemicolonComment,
480};
481
482static PCSCMREWRITERCFG const g_apRewritersFor_NsisFiles[] =
483{
484 &g_ForceNativeEol,
485 &g_ExpandTabs,
486 &g_StripTrailingBlanks,
487 &g_AdjustTrailingLines,
488 &g_SvnKeywords,
489 &g_SvnNoExecutable,
490 &g_Copyright_SemicolonComment,
491};
492
493static PCSCMREWRITERCFG const g_apRewritersFor_Java[] =
494{
495 &g_ForceNativeEol,
496 &g_ExpandTabs,
497 &g_StripTrailingBlanks,
498 &g_AdjustTrailingLines,
499 &g_SvnNoExecutable,
500 &g_SvnKeywords,
501 &g_Copyright_CstyleComment,
502 &g_FixFlowerBoxMarkers,
503 &g_Fix_C_and_CPP_Todos,
504};
505
506static PCSCMREWRITERCFG const g_apRewritersFor_ScmSettings[] =
507{
508 &g_ForceNativeEol,
509 &g_ExpandTabs,
510 &g_StripTrailingBlanks,
511 &g_AdjustTrailingLines,
512 &g_SvnNoExecutable,
513 &g_SvnKeywords,
514 &g_Copyright_HashComment,
515};
516
517static PCSCMREWRITERCFG const g_apRewritersFor_Images[] =
518{
519 &g_SvnNoExecutable,
520 &g_SvnBinary,
521};
522
523static PCSCMREWRITERCFG const g_apRewritersFor_Xslt[] =
524{
525 &g_ForceNativeEol,
526 &g_ExpandTabs,
527 &g_StripTrailingBlanks,
528 &g_AdjustTrailingLines,
529 &g_SvnNoExecutable,
530 &g_SvnKeywords,
531 /** @todo copyright is in an XML comment. */
532};
533
534static PCSCMREWRITERCFG const g_apRewritersFor_Xml[] =
535{
536 &g_ForceNativeEol,
537 &g_ExpandTabs,
538 &g_StripTrailingBlanks,
539 &g_AdjustTrailingLines,
540 &g_SvnNoExecutable,
541 &g_SvnKeywords,
542 /** @todo copyright is in an XML comment. */
543};
544
545static PCSCMREWRITERCFG const g_apRewritersFor_Wix[] =
546{
547 &g_ForceNativeEol,
548 &g_ExpandTabs,
549 &g_StripTrailingBlanks,
550 &g_AdjustTrailingLines,
551 &g_SvnNoExecutable,
552 &g_SvnKeywords,
553 /** @todo copyright is in an XML comment. */
554};
555
556static PCSCMREWRITERCFG const g_apRewritersFor_QtProject[] =
557{
558 &g_ForceNativeEol,
559 &g_StripTrailingBlanks,
560 &g_AdjustTrailingLines,
561 &g_SvnNoExecutable,
562 &g_SvnKeywords,
563 &g_Copyright_HashComment,
564};
565
566static PCSCMREWRITERCFG const g_apRewritersFor_QtResourceFiles[] =
567{
568 &g_ForceNativeEol,
569 &g_SvnNoExecutable,
570 &g_SvnKeywords,
571 /** @todo figure out copyright for Qt resource XML files. */
572};
573
574static PCSCMREWRITERCFG const g_apRewritersFor_QtTranslations[] =
575{
576 &g_ForceNativeEol,
577 &g_SvnNoExecutable,
578};
579
580static PCSCMREWRITERCFG const g_apRewritersFor_QtUiFiles[] =
581{
582 &g_ForceNativeEol,
583 &g_SvnNoExecutable,
584 &g_SvnKeywords,
585 /** @todo copyright is in an XML 'comment' element. */
586};
587
588static PCSCMREWRITERCFG const g_apRewritersFor_SifFiles[] =
589{
590 &g_ForceCRLF,
591 &g_ExpandTabs,
592 &g_StripTrailingBlanks,
593 &g_AdjustTrailingLines,
594 &g_SvnKeywords,
595 &g_SvnNoExecutable,
596 &g_Copyright_SemicolonComment,
597};
598
599static PCSCMREWRITERCFG const g_apRewritersFor_SqlFiles[] =
600{
601 &g_ForceNativeEol,
602 &g_ExpandTabs,
603 &g_StripTrailingBlanks,
604 &g_AdjustTrailingLines,
605 &g_SvnKeywords,
606 &g_SvnNoExecutable,
607 &g_Copyright_SqlComment,
608};
609
610static PCSCMREWRITERCFG const g_apRewritersFor_GnuAsm[] =
611{
612 &g_ForceNativeEol,
613 &g_ExpandTabs,
614 &g_StripTrailingBlanks,
615 &g_AdjustTrailingLines,
616 &g_SvnKeywords,
617 &g_SvnNoExecutable,
618 &g_Copyright_CstyleComment,
619};
620
621static PCSCMREWRITERCFG const g_apRewritersFor_TextFiles[] =
622{
623 &g_ForceNativeEol,
624 &g_StripTrailingBlanks,
625 &g_SvnKeywords,
626 &g_SvnNoExecutable,
627 /** @todo check for plain copyright + license in text files. */
628};
629
630static PCSCMREWRITERCFG const g_apRewritersFor_PlainTextFiles[] =
631{
632 &g_ForceNativeEol,
633 &g_StripTrailingBlanks,
634 &g_SvnKeywords,
635 &g_SvnNoExecutable,
636};
637
638static PCSCMREWRITERCFG const g_apRewritersFor_BinaryFiles[] =
639{
640 &g_SvnBinary,
641};
642
643static PCSCMREWRITERCFG const g_apRewritersFor_FileLists[] = /* both makefile and shell script */
644{
645 &g_ForceLF,
646 &g_ExpandTabs,
647 &g_StripTrailingBlanks,
648 &g_AdjustTrailingLines,
649 &g_Copyright_HashComment,
650};
651
652
653/**
654 * Array of standard rewriter configurations.
655 */
656static SCMCFGENTRY const g_aConfigs[] =
657{
658#define SCM_CFG_ENTRY(a_szName, a_aRewriters, a_fBinary, a_szFilePatterns) \
659 { RT_ELEMENTS(a_aRewriters), &a_aRewriters[0], a_fBinary, a_szFilePatterns, a_szName }
660 SCM_CFG_ENTRY("kup", g_apRewritersFor_Makefile_kup, false, "Makefile.kup" ),
661 SCM_CFG_ENTRY("kmk", g_apRewritersFor_Makefile_kmk, false, "*.kmk" ),
662 SCM_CFG_ENTRY("c", g_apRewritersFor_C_and_CPP, false, "*.c|*.cpp|*.C|*.CPP|*.cxx|*.cc|*.m|*.mm" ),
663 SCM_CFG_ENTRY("h", g_apRewritersFor_H_and_HPP, false, "*.h|*.hpp" ),
664 SCM_CFG_ENTRY("rc", g_apRewritersFor_RC, false, "*.rc" ),
665 SCM_CFG_ENTRY("asm", g_apRewritersFor_ASM, false, "*.asm|*.mac|*.inc" ),
666 SCM_CFG_ENTRY("dtrace", g_apRewritersFor_DTrace, false, "*.d" ),
667 SCM_CFG_ENTRY("def", g_apRewritersFor_DEF, false, "*.def" ),
668 SCM_CFG_ENTRY("iasl", g_apRewritersFor_DSL, false, "*.dsl" ),
669 SCM_CFG_ENTRY("shell", g_apRewritersFor_ShellScripts, false, "*.sh|configure" ),
670 SCM_CFG_ENTRY("batch", g_apRewritersFor_BatchFiles, false, "*.bat|*.cmd|*.btm" ),
671 SCM_CFG_ENTRY("vbs", g_apRewritersFor_BasicScripts, false, "*.vbs|*.vb" ),
672 SCM_CFG_ENTRY("sed", g_apRewritersFor_SedScripts, false, "*.sed" ),
673 SCM_CFG_ENTRY("python", g_apRewritersFor_Python, false, "*.py" ),
674 SCM_CFG_ENTRY("perl", g_apRewritersFor_Perl, false, "*.pl|*.pm" ),
675 SCM_CFG_ENTRY("drvinf", g_apRewritersFor_DriverInfFiles, false, "*.inf" ),
676 SCM_CFG_ENTRY("nsis", g_apRewritersFor_NsisFiles, false, "*.nsh|*.nsi|*.nsis" ),
677 SCM_CFG_ENTRY("java", g_apRewritersFor_Java, false, "*.java" ),
678 SCM_CFG_ENTRY("scm", g_apRewritersFor_ScmSettings, false, "*.scm-settings" ),
679 SCM_CFG_ENTRY("image", g_apRewritersFor_Images, true, "*.png|*.bmp|*.jpg|*.pnm|*.ico|*.icns|*.tiff|*.tif|*.xcf|*.gif" ),
680 SCM_CFG_ENTRY("xslt", g_apRewritersFor_Xslt, false, "*.xsl" ),
681 SCM_CFG_ENTRY("xml", g_apRewritersFor_Xml, false, "*.xml" ),
682 SCM_CFG_ENTRY("wix", g_apRewritersFor_Wix, false, "*.wxi|*.wxs|*.wxl" ),
683 SCM_CFG_ENTRY("qt-pro", g_apRewritersFor_QtProject, false, "*.pro" ),
684 SCM_CFG_ENTRY("qt-rc", g_apRewritersFor_QtResourceFiles, false, "*.qrc" ),
685 SCM_CFG_ENTRY("qt-ts", g_apRewritersFor_QtTranslations, false, "*.ts" ),
686 SCM_CFG_ENTRY("qt-ui", g_apRewritersFor_QtUiFiles, false, "*.ui" ),
687 SCM_CFG_ENTRY("sif", g_apRewritersFor_SifFiles, false, "*.sif" ),
688 SCM_CFG_ENTRY("sql", g_apRewritersFor_SqlFiles, false, "*.pgsql|*.sql" ),
689 SCM_CFG_ENTRY("gas", g_apRewritersFor_GnuAsm, false, "*.S" ),
690 SCM_CFG_ENTRY("binary", g_apRewritersFor_BinaryFiles, true, "*.bin|*.pdf|*.zip|*.bz2|*.gz" ),
691 /* These should be be last: */
692 SCM_CFG_ENTRY("make", g_apRewritersFor_OtherMakefiles, false, "Makefile|makefile|GNUmakefile|SMakefile|Makefile.am|Makefile.in|*.cmake" ),
693 SCM_CFG_ENTRY("text", g_apRewritersFor_TextFiles, false, "*.txt|README*|readme*|ReadMe*|NOTE*|TODO*" ),
694 SCM_CFG_ENTRY("plaintext", g_apRewritersFor_PlainTextFiles, false, "LICENSE|ChangeLog|FAQ|AUTHORS|INSTALL|NEWS" ),
695 SCM_CFG_ENTRY("file-list", g_apRewritersFor_FileLists, false, "files_*" ),
696};
697
698
699
700/* -=-=-=-=-=- settings -=-=-=-=-=- */
701
702/**
703 * Delete the given config entry.
704 *
705 * @param pEntry The configuration entry to delete.
706 */
707static void scmCfgEntryDelete(PSCMCFGENTRY pEntry)
708{
709 RTMemFree((void *)pEntry->paRewriters);
710 pEntry->paRewriters = NULL;
711 RTMemFree(pEntry);
712}
713
714/**
715 * Create a new configuration entry.
716 *
717 * @returns The new entry. NULL if out of memory.
718 * @param pEntry The configuration entry to duplicate.
719 */
720static PSCMCFGENTRY scmCfgEntryNew(void)
721{
722 PSCMCFGENTRY pNew = (PSCMCFGENTRY)RTMemAlloc(sizeof(*pNew));
723 if (pNew)
724 {
725 pNew->pszName = "custom";
726 pNew->pszFilePattern = "custom";
727 pNew->cRewriters = 0;
728 pNew->paRewriters = NULL;
729 pNew->fBinary = false;
730 }
731 return pNew;
732}
733
734/**
735 * Duplicate the given config entry.
736 *
737 * @returns The duplicate. NULL if out of memory.
738 * @param pEntry The configuration entry to duplicate.
739 */
740static PSCMCFGENTRY scmCfgEntryDup(PCSCMCFGENTRY pEntry)
741{
742 if (pEntry)
743 {
744 PSCMCFGENTRY pDup = (PSCMCFGENTRY)RTMemDup(pEntry, sizeof(*pEntry));
745 if (pDup)
746 {
747 size_t cbRewriters = sizeof(pEntry->paRewriters[0]) * RT_ALIGN_Z(pEntry->cRewriters, 8);
748 pDup->paRewriters = (PCSCMREWRITERCFG const *)RTMemDup(pEntry->paRewriters, cbRewriters);
749 if (pDup->paRewriters)
750 return pDup;
751
752 RTMemFree(pDup);
753 }
754 return NULL;
755 }
756 return scmCfgEntryNew();
757}
758
759/**
760 * Adds a rewriter action to the given config entry (--add-action).
761 *
762 * @returns VINF_SUCCESS.
763 * @param pEntry The configuration entry.
764 * @param pAction The rewriter action to add.
765 */
766static int scmCfgEntryAddAction(PSCMCFGENTRY pEntry, PCSCMREWRITERCFG pAction)
767{
768 PCSCMREWRITERCFG *paRewriters = (PCSCMREWRITERCFG *)pEntry->paRewriters;
769 if (pEntry->cRewriters % 8 == 0)
770 {
771 size_t cbRewriters = sizeof(pEntry->paRewriters[0]) * RT_ALIGN_Z((pEntry->cRewriters + 1), 8);
772 void *pvNew = RTMemRealloc(paRewriters, cbRewriters);
773 if (pvNew)
774 pEntry->paRewriters = paRewriters = (PCSCMREWRITERCFG *)pvNew;
775 else
776 return VERR_NO_MEMORY;
777 }
778
779 paRewriters[pEntry->cRewriters++] = pAction;
780 return VINF_SUCCESS;
781}
782
783/**
784 * Delets an rewriter action from the given config entry (--del-action).
785 *
786 * @param pEntry The configuration entry.
787 * @param pAction The rewriter action to remove.
788 */
789static void scmCfgEntryDelAction(PSCMCFGENTRY pEntry, PCSCMREWRITERCFG pAction)
790{
791 PCSCMREWRITERCFG *paRewriters = (PCSCMREWRITERCFG *)pEntry->paRewriters;
792 size_t const cEntries = pEntry->cRewriters;
793 size_t iDst = 0;
794 for (size_t iSrc = 0; iSrc < cEntries; iSrc++)
795 {
796 PCSCMREWRITERCFG pCurAction = paRewriters[iSrc];
797 if (pCurAction != pAction)
798 paRewriters[iDst++] = pCurAction;
799 }
800 pEntry->cRewriters = iDst;
801}
802
803/**
804 * Init a settings structure with settings from @a pSrc.
805 *
806 * @returns IPRT status code
807 * @param pSettings The settings.
808 * @param pSrc The source settings.
809 */
810static int scmSettingsBaseInitAndCopy(PSCMSETTINGSBASE pSettings, PCSCMSETTINGSBASE pSrc)
811{
812 *pSettings = *pSrc;
813
814 int rc = RTStrDupEx(&pSettings->pszFilterFiles, pSrc->pszFilterFiles);
815 if (RT_SUCCESS(rc))
816 {
817 rc = RTStrDupEx(&pSettings->pszFilterOutFiles, pSrc->pszFilterOutFiles);
818 if (RT_SUCCESS(rc))
819 {
820 rc = RTStrDupEx(&pSettings->pszFilterOutDirs, pSrc->pszFilterOutDirs);
821 if (RT_SUCCESS(rc))
822 {
823 if (!pSrc->fFreeTreatAs)
824 return VINF_SUCCESS;
825
826 pSettings->pTreatAs = scmCfgEntryDup(pSrc->pTreatAs);
827 if (pSettings->pTreatAs)
828 return VINF_SUCCESS;
829
830 RTStrFree(pSettings->pszFilterOutDirs);
831 }
832 RTStrFree(pSettings->pszFilterOutFiles);
833 }
834 RTStrFree(pSettings->pszFilterFiles);
835 }
836
837 pSettings->pszFilterFiles = NULL;
838 pSettings->pszFilterOutFiles = NULL;
839 pSettings->pszFilterOutDirs = NULL;
840 pSettings->pTreatAs = NULL;
841 return rc;
842}
843
844/**
845 * Init a settings structure.
846 *
847 * @returns IPRT status code
848 * @param pSettings The settings.
849 */
850static int scmSettingsBaseInit(PSCMSETTINGSBASE pSettings)
851{
852 return scmSettingsBaseInitAndCopy(pSettings, &g_Defaults);
853}
854
855/**
856 * Deletes the settings, i.e. free any dynamically allocated content.
857 *
858 * @param pSettings The settings.
859 */
860static void scmSettingsBaseDelete(PSCMSETTINGSBASE pSettings)
861{
862 if (pSettings)
863 {
864 Assert(pSettings->cchTab != UINT8_MAX);
865 pSettings->cchTab = UINT8_MAX;
866
867 RTStrFree(pSettings->pszFilterFiles);
868 RTStrFree(pSettings->pszFilterOutFiles);
869 RTStrFree(pSettings->pszFilterOutDirs);
870 if (pSettings->fFreeTreatAs)
871 scmCfgEntryDelete((PSCMCFGENTRY)pSettings->pTreatAs);
872
873 pSettings->pszFilterOutDirs = NULL;
874 pSettings->pszFilterOutFiles = NULL;
875 pSettings->pszFilterFiles = NULL;
876 pSettings->pTreatAs = NULL;
877 pSettings->fFreeTreatAs = false;
878 }
879}
880
881/**
882 * Processes a RTGetOpt result.
883 *
884 * @retval VINF_SUCCESS if handled.
885 * @retval VERR_OUT_OF_RANGE if the option value was out of range.
886 * @retval VERR_GETOPT_UNKNOWN_OPTION if the option was not recognized.
887 *
888 * @param pSettings The settings to change.
889 * @param rc The RTGetOpt return value.
890 * @param pValueUnion The RTGetOpt value union.
891 * @param pchDir The absolute path to the directory relative
892 * components in pchLine should be relative to.
893 * @param cchDir The length of the @a pchDir string.
894 */
895static int scmSettingsBaseHandleOpt(PSCMSETTINGSBASE pSettings, int rc, PRTGETOPTUNION pValueUnion,
896 const char *pchDir, size_t cchDir)
897{
898 Assert(pchDir[cchDir - 1] == '/');
899
900 switch (rc)
901 {
902 case SCMOPT_CONVERT_EOL:
903 pSettings->fConvertEol = true;
904 return VINF_SUCCESS;
905 case SCMOPT_NO_CONVERT_EOL:
906 pSettings->fConvertEol = false;
907 return VINF_SUCCESS;
908
909 case SCMOPT_CONVERT_TABS:
910 pSettings->fConvertTabs = true;
911 return VINF_SUCCESS;
912 case SCMOPT_NO_CONVERT_TABS:
913 pSettings->fConvertTabs = false;
914 return VINF_SUCCESS;
915
916 case SCMOPT_FORCE_FINAL_EOL:
917 pSettings->fForceFinalEol = true;
918 return VINF_SUCCESS;
919 case SCMOPT_NO_FORCE_FINAL_EOL:
920 pSettings->fForceFinalEol = false;
921 return VINF_SUCCESS;
922
923 case SCMOPT_FORCE_TRAILING_LINE:
924 pSettings->fForceTrailingLine = true;
925 return VINF_SUCCESS;
926 case SCMOPT_NO_FORCE_TRAILING_LINE:
927 pSettings->fForceTrailingLine = false;
928 return VINF_SUCCESS;
929
930
931 case SCMOPT_STRIP_TRAILING_BLANKS:
932 pSettings->fStripTrailingBlanks = true;
933 return VINF_SUCCESS;
934 case SCMOPT_NO_STRIP_TRAILING_BLANKS:
935 pSettings->fStripTrailingBlanks = false;
936 return VINF_SUCCESS;
937
938 case SCMOPT_MIN_BLANK_LINES_BEFORE_FLOWER_BOX_MARKERS:
939 pSettings->cMinBlankLinesBeforeFlowerBoxMakers = pValueUnion->u8;
940 return VINF_SUCCESS;
941
942
943 case SCMOPT_STRIP_TRAILING_LINES:
944 pSettings->fStripTrailingLines = true;
945 return VINF_SUCCESS;
946 case SCMOPT_NO_STRIP_TRAILING_LINES:
947 pSettings->fStripTrailingLines = false;
948 return VINF_SUCCESS;
949
950 case SCMOPT_FIX_FLOWER_BOX_MARKERS:
951 pSettings->fFixFlowerBoxMarkers = true;
952 return VINF_SUCCESS;
953 case SCMOPT_NO_FIX_FLOWER_BOX_MARKERS:
954 pSettings->fFixFlowerBoxMarkers = false;
955 return VINF_SUCCESS;
956
957 case SCMOPT_FIX_TODOS:
958 pSettings->fFixTodos = true;
959 return VINF_SUCCESS;
960 case SCMOPT_NO_FIX_TODOS:
961 pSettings->fFixTodos = false;
962 return VINF_SUCCESS;
963
964 case SCMOPT_UPDATE_COPYRIGHT_YEAR:
965 pSettings->fUpdateCopyrightYear = true;
966 return VINF_SUCCESS;
967 case SCMOPT_NO_UPDATE_COPYRIGHT_YEAR:
968 pSettings->fUpdateCopyrightYear = false;
969 return VINF_SUCCESS;
970
971 case SCMOPT_EXTERNAL_COPYRIGHT:
972 pSettings->fExternalCopyright = true;
973 return VINF_SUCCESS;
974 case SCMOPT_NO_EXTERNAL_COPYRIGHT:
975 pSettings->fExternalCopyright = false;
976 return VINF_SUCCESS;
977
978 case SCMOPT_NO_UPDATE_LICENSE:
979 pSettings->enmUpdateLicense = kScmLicense_LeaveAlone;
980 return VINF_SUCCESS;
981 case SCMOPT_LICENSE_OSE_GPL:
982 pSettings->enmUpdateLicense = kScmLicense_OseGpl;
983 return VINF_SUCCESS;
984 case SCMOPT_LICENSE_OSE_DUAL_GPL_CDDL:
985 pSettings->enmUpdateLicense = kScmLicense_OseDualGplCddl;
986 return VINF_SUCCESS;
987 case SCMOPT_LICENSE_OSE_CDDL:
988 pSettings->enmUpdateLicense = kScmLicense_OseCddl;
989 return VINF_SUCCESS;
990 case SCMOPT_LICENSE_LGPL:
991 pSettings->enmUpdateLicense = kScmLicense_Lgpl;
992 return VINF_SUCCESS;
993 case SCMOPT_LICENSE_MIT:
994 pSettings->enmUpdateLicense = kScmLicense_Mit;
995 return VINF_SUCCESS;
996 case SCMOPT_LICENSE_BASED_ON_MIT:
997 pSettings->enmUpdateLicense = kScmLicense_BasedOnMit;
998 return VINF_SUCCESS;
999
1000 case SCMOPT_LGPL_DISCLAIMER:
1001 pSettings->fLgplDisclaimer = true;
1002 return VINF_SUCCESS;
1003 case SCMOPT_NO_LGPL_DISCLAIMER:
1004 pSettings->fLgplDisclaimer = false;
1005 return VINF_SUCCESS;
1006
1007 case SCMOPT_ONLY_SVN_DIRS:
1008 pSettings->fOnlySvnDirs = true;
1009 return VINF_SUCCESS;
1010 case SCMOPT_NOT_ONLY_SVN_DIRS:
1011 pSettings->fOnlySvnDirs = false;
1012 return VINF_SUCCESS;
1013
1014 case SCMOPT_ONLY_SVN_FILES:
1015 pSettings->fOnlySvnFiles = true;
1016 return VINF_SUCCESS;
1017 case SCMOPT_NOT_ONLY_SVN_FILES:
1018 pSettings->fOnlySvnFiles = false;
1019 return VINF_SUCCESS;
1020
1021 case SCMOPT_SET_SVN_EOL:
1022 pSettings->fSetSvnEol = true;
1023 return VINF_SUCCESS;
1024 case SCMOPT_DONT_SET_SVN_EOL:
1025 pSettings->fSetSvnEol = false;
1026 return VINF_SUCCESS;
1027
1028 case SCMOPT_SET_SVN_EXECUTABLE:
1029 pSettings->fSetSvnExecutable = true;
1030 return VINF_SUCCESS;
1031 case SCMOPT_DONT_SET_SVN_EXECUTABLE:
1032 pSettings->fSetSvnExecutable = false;
1033 return VINF_SUCCESS;
1034
1035 case SCMOPT_SET_SVN_KEYWORDS:
1036 pSettings->fSetSvnKeywords = true;
1037 return VINF_SUCCESS;
1038 case SCMOPT_DONT_SET_SVN_KEYWORDS:
1039 pSettings->fSetSvnKeywords = false;
1040 return VINF_SUCCESS;
1041
1042 case SCMOPT_TAB_SIZE:
1043 if ( pValueUnion->u8 < 1
1044 || pValueUnion->u8 >= RT_ELEMENTS(g_szTabSpaces))
1045 {
1046 RTMsgError("Invalid tab size: %u - must be in {1..%u}\n",
1047 pValueUnion->u8, RT_ELEMENTS(g_szTabSpaces) - 1);
1048 return VERR_OUT_OF_RANGE;
1049 }
1050 pSettings->cchTab = pValueUnion->u8;
1051 return VINF_SUCCESS;
1052
1053 case SCMOPT_WIDTH:
1054 if (pValueUnion->u8 < 20 || pValueUnion->u8 > 200)
1055 {
1056 RTMsgError("Invalid width size: %u - must be in {20..200} range\n", pValueUnion->u8);
1057 return VERR_OUT_OF_RANGE;
1058 }
1059 pSettings->cchWidth = pValueUnion->u8;
1060 return VINF_SUCCESS;
1061
1062 case SCMOPT_FILTER_OUT_DIRS:
1063 case SCMOPT_FILTER_FILES:
1064 case SCMOPT_FILTER_OUT_FILES:
1065 {
1066 char **ppsz = NULL;
1067 switch (rc)
1068 {
1069 case SCMOPT_FILTER_OUT_DIRS: ppsz = &pSettings->pszFilterOutDirs; break;
1070 case SCMOPT_FILTER_FILES: ppsz = &pSettings->pszFilterFiles; break;
1071 case SCMOPT_FILTER_OUT_FILES: ppsz = &pSettings->pszFilterOutFiles; break;
1072 }
1073
1074 /*
1075 * An empty string zaps the current list.
1076 */
1077 if (!*pValueUnion->psz)
1078 return RTStrATruncate(ppsz, 0);
1079
1080 /*
1081 * Non-empty strings are appended to the pattern list.
1082 *
1083 * Strip leading and trailing pattern separators before attempting
1084 * to append it. If it's just separators, don't do anything.
1085 */
1086 const char *pszSrc = pValueUnion->psz;
1087 while (*pszSrc == '|')
1088 pszSrc++;
1089 size_t cchSrc = strlen(pszSrc);
1090 while (cchSrc > 0 && pszSrc[cchSrc - 1] == '|')
1091 cchSrc--;
1092 if (!cchSrc)
1093 return VINF_SUCCESS;
1094
1095 /* Append it pattern by pattern, turning settings-relative paths into absolute ones. */
1096 for (;;)
1097 {
1098 const char *pszEnd = (const char *)memchr(pszSrc, '|', cchSrc);
1099 size_t cchPattern = pszEnd ? pszEnd - pszSrc : cchSrc;
1100 int rc2;
1101 if (*pszSrc == '/')
1102 rc2 = RTStrAAppendExN(ppsz, 3,
1103 "|", *ppsz && **ppsz != '\0' ? (size_t)1 : (size_t)0,
1104 pchDir, cchDir - 1,
1105 pszSrc, cchPattern);
1106 else
1107 rc2 = RTStrAAppendExN(ppsz, 2,
1108 "|", *ppsz && **ppsz != '\0' ? (size_t)1 : (size_t)0,
1109 pszSrc, cchPattern);
1110 if (RT_FAILURE(rc2))
1111 return rc2;
1112
1113 /* next */
1114 cchSrc -= cchPattern;
1115 if (!cchSrc)
1116 return VINF_SUCCESS;
1117 cchSrc -= 1;
1118 pszSrc += cchPattern + 1;
1119 }
1120 /* not reached */
1121 }
1122
1123 case SCMOPT_TREAT_AS:
1124 if (pSettings->fFreeTreatAs)
1125 {
1126 scmCfgEntryDelete((PSCMCFGENTRY)pSettings->pTreatAs);
1127 pSettings->pTreatAs = NULL;
1128 pSettings->fFreeTreatAs = false;
1129 }
1130
1131 if (*pValueUnion->psz)
1132 {
1133 /* first check the names, then patterns (legacy). */
1134 for (size_t iCfg = 0; iCfg < RT_ELEMENTS(g_aConfigs); iCfg++)
1135 if (strcmp(g_aConfigs[iCfg].pszName, pValueUnion->psz) == 0)
1136 {
1137 pSettings->pTreatAs = &g_aConfigs[iCfg];
1138 return VINF_SUCCESS;
1139 }
1140 for (size_t iCfg = 0; iCfg < RT_ELEMENTS(g_aConfigs); iCfg++)
1141 if (RTStrSimplePatternMultiMatch(g_aConfigs[iCfg].pszFilePattern, RTSTR_MAX,
1142 pValueUnion->psz, RTSTR_MAX, NULL))
1143 {
1144 pSettings->pTreatAs = &g_aConfigs[iCfg];
1145 return VINF_SUCCESS;
1146 }
1147 /* Special help for listing the possibilities? */
1148 if (strcmp(pValueUnion->psz, "help") == 0)
1149 {
1150 RTPrintf("Possible --treat-as values:\n");
1151 for (size_t iCfg = 0; iCfg < RT_ELEMENTS(g_aConfigs); iCfg++)
1152 RTPrintf(" %s (%s)\n", g_aConfigs[iCfg].pszName, g_aConfigs[iCfg].pszFilePattern);
1153 }
1154 return VERR_NOT_FOUND;
1155 }
1156
1157 pSettings->pTreatAs = NULL;
1158 return VINF_SUCCESS;
1159
1160 case SCMOPT_ADD_ACTION:
1161 for (uint32_t iAction = 0; iAction < RT_ELEMENTS(g_papRewriterActions); iAction++)
1162 if (strcmp(g_papRewriterActions[iAction]->pszName, pValueUnion->psz) == 0)
1163 {
1164 PSCMCFGENTRY pEntry = (PSCMCFGENTRY)pSettings->pTreatAs;
1165 if (!pSettings->fFreeTreatAs)
1166 {
1167 pEntry = scmCfgEntryDup(pEntry);
1168 if (!pEntry)
1169 return VERR_NO_MEMORY;
1170 pSettings->pTreatAs = pEntry;
1171 pSettings->fFreeTreatAs = true;
1172 }
1173 return scmCfgEntryAddAction(pEntry, g_papRewriterActions[iAction]);
1174 }
1175 RTMsgError("Unknown --add-action value '%s'. Try --help-actions for a list.", pValueUnion->psz);
1176 return VERR_NOT_FOUND;
1177
1178 case SCMOPT_DEL_ACTION:
1179 {
1180 uint32_t cActions = 0;
1181 for (uint32_t iAction = 0; iAction < RT_ELEMENTS(g_papRewriterActions); iAction++)
1182 if (RTStrSimplePatternMatch(pValueUnion->psz, g_papRewriterActions[iAction]->pszName))
1183 {
1184 cActions++;
1185 PSCMCFGENTRY pEntry = (PSCMCFGENTRY)pSettings->pTreatAs;
1186 if (!pSettings->fFreeTreatAs)
1187 {
1188 pEntry = scmCfgEntryDup(pEntry);
1189 if (!pEntry)
1190 return VERR_NO_MEMORY;
1191 pSettings->pTreatAs = pEntry;
1192 pSettings->fFreeTreatAs = true;
1193 }
1194 scmCfgEntryDelAction(pEntry, g_papRewriterActions[iAction]);
1195 if (!strchr(pValueUnion->psz, '*'))
1196 return VINF_SUCCESS;
1197 }
1198 if (cActions > 0)
1199 return VINF_SUCCESS;
1200 RTMsgError("Unknown --del-action value '%s'. Try --help-actions for a list.", pValueUnion->psz);
1201 return VERR_NOT_FOUND;
1202 }
1203
1204 default:
1205 return VERR_GETOPT_UNKNOWN_OPTION;
1206 }
1207}
1208
1209/**
1210 * Parses an option string.
1211 *
1212 * @returns IPRT status code.
1213 * @param pBase The base settings structure to apply the options
1214 * to.
1215 * @param pszOptions The options to parse.
1216 * @param pchDir The absolute path to the directory relative
1217 * components in pchLine should be relative to.
1218 * @param cchDir The length of the @a pchDir string.
1219 */
1220static int scmSettingsBaseParseString(PSCMSETTINGSBASE pBase, const char *pszLine, const char *pchDir, size_t cchDir)
1221{
1222 int cArgs;
1223 char **papszArgs;
1224 int rc = RTGetOptArgvFromString(&papszArgs, &cArgs, pszLine, RTGETOPTARGV_CNV_QUOTE_BOURNE_SH, NULL);
1225 if (RT_SUCCESS(rc))
1226 {
1227 RTGETOPTUNION ValueUnion;
1228 RTGETOPTSTATE GetOptState;
1229 rc = RTGetOptInit(&GetOptState, cArgs, papszArgs, &g_aScmOpts[0], RT_ELEMENTS(g_aScmOpts), 0, 0 /*fFlags*/);
1230 if (RT_SUCCESS(rc))
1231 {
1232 while ((rc = RTGetOpt(&GetOptState, &ValueUnion)) != 0)
1233 {
1234 rc = scmSettingsBaseHandleOpt(pBase, rc, &ValueUnion, pchDir, cchDir);
1235 if (RT_FAILURE(rc))
1236 break;
1237 }
1238 }
1239 RTGetOptArgvFree(papszArgs);
1240 }
1241
1242 return rc;
1243}
1244
1245/**
1246 * Parses an unterminated option string.
1247 *
1248 * @returns IPRT status code.
1249 * @param pBase The base settings structure to apply the options
1250 * to.
1251 * @param pchLine The line.
1252 * @param cchLine The line length.
1253 * @param pchDir The absolute path to the directory relative
1254 * components in pchLine should be relative to.
1255 * @param cchDir The length of the @a pchDir string.
1256 */
1257static int scmSettingsBaseParseStringN(PSCMSETTINGSBASE pBase, const char *pchLine, size_t cchLine,
1258 const char *pchDir, size_t cchDir)
1259{
1260 char *pszLine = RTStrDupN(pchLine, cchLine);
1261 if (!pszLine)
1262 return VERR_NO_MEMORY;
1263 int rc = scmSettingsBaseParseString(pBase, pszLine, pchDir, cchDir);
1264 RTStrFree(pszLine);
1265 return rc;
1266}
1267
1268/**
1269 * Verifies the options string.
1270 *
1271 * @returns IPRT status code.
1272 * @param pszOptions The options to verify .
1273 */
1274static int scmSettingsBaseVerifyString(const char *pszOptions)
1275{
1276 SCMSETTINGSBASE Base;
1277 int rc = scmSettingsBaseInit(&Base);
1278 if (RT_SUCCESS(rc))
1279 {
1280 rc = scmSettingsBaseParseString(&Base, pszOptions, "/", 1);
1281 scmSettingsBaseDelete(&Base);
1282 }
1283 return rc;
1284}
1285
1286/**
1287 * Loads settings found in editor and SCM settings directives within the
1288 * document (@a pStream).
1289 *
1290 * @returns IPRT status code.
1291 * @param pBase The settings base to load settings into.
1292 * @param pStream The stream to scan for settings directives.
1293 */
1294static int scmSettingsBaseLoadFromDocument(PSCMSETTINGSBASE pBase, PSCMSTREAM pStream)
1295{
1296 /** @todo Editor and SCM settings directives in documents. */
1297 RT_NOREF2(pBase, pStream);
1298 return VINF_SUCCESS;
1299}
1300
1301/**
1302 * Creates a new settings file struct, cloning @a pSettings.
1303 *
1304 * @returns IPRT status code.
1305 * @param ppSettings Where to return the new struct.
1306 * @param pSettingsBase The settings to inherit from.
1307 */
1308static int scmSettingsCreate(PSCMSETTINGS *ppSettings, PCSCMSETTINGSBASE pSettingsBase)
1309{
1310 PSCMSETTINGS pSettings = (PSCMSETTINGS)RTMemAlloc(sizeof(*pSettings));
1311 if (!pSettings)
1312 return VERR_NO_MEMORY;
1313 int rc = scmSettingsBaseInitAndCopy(&pSettings->Base, pSettingsBase);
1314 if (RT_SUCCESS(rc))
1315 {
1316 pSettings->pDown = NULL;
1317 pSettings->pUp = NULL;
1318 pSettings->paPairs = NULL;
1319 pSettings->cPairs = 0;
1320 *ppSettings = pSettings;
1321 return VINF_SUCCESS;
1322 }
1323 RTMemFree(pSettings);
1324 return rc;
1325}
1326
1327/**
1328 * Destroys a settings structure.
1329 *
1330 * @param pSettings The settings structure to destroy. NULL is OK.
1331 */
1332static void scmSettingsDestroy(PSCMSETTINGS pSettings)
1333{
1334 if (pSettings)
1335 {
1336 scmSettingsBaseDelete(&pSettings->Base);
1337 for (size_t i = 0; i < pSettings->cPairs; i++)
1338 {
1339 RTStrFree(pSettings->paPairs[i].pszPattern);
1340 RTStrFree(pSettings->paPairs[i].pszOptions);
1341 RTStrFree(pSettings->paPairs[i].pszRelativeTo);
1342 pSettings->paPairs[i].pszPattern = NULL;
1343 pSettings->paPairs[i].pszOptions = NULL;
1344 pSettings->paPairs[i].pszRelativeTo = NULL;
1345 }
1346 RTMemFree(pSettings->paPairs);
1347 pSettings->paPairs = NULL;
1348 RTMemFree(pSettings);
1349 }
1350}
1351
1352/**
1353 * Adds a pattern/options pair to the settings structure.
1354 *
1355 * @returns IPRT status code.
1356 * @param pSettings The settings.
1357 * @param pchLine The line containing the unparsed pair.
1358 * @param cchLine The length of the line.
1359 * @param offColon The offset of the colon into the line.
1360 * @param pchDir The absolute path to the directory relative
1361 * components in pchLine should be relative to.
1362 * @param cchDir The length of the @a pchDir string.
1363 */
1364static int scmSettingsAddPair(PSCMSETTINGS pSettings, const char *pchLine, size_t cchLine, size_t offColon,
1365 const char *pchDir, size_t cchDir)
1366{
1367 Assert(pchLine[offColon] == ':' && offColon < cchLine);
1368 Assert(pchDir[cchDir - 1] == '/');
1369
1370 /*
1371 * Split the string.
1372 */
1373 size_t cchPattern = offColon;
1374 size_t cchOptions = cchLine - cchPattern - 1;
1375
1376 /* strip spaces everywhere */
1377 while (cchPattern > 0 && RT_C_IS_SPACE(pchLine[cchPattern - 1]))
1378 cchPattern--;
1379 while (cchPattern > 0 && RT_C_IS_SPACE(*pchLine))
1380 cchPattern--, pchLine++;
1381
1382 const char *pchOptions = &pchLine[offColon + 1];
1383 while (cchOptions > 0 && RT_C_IS_SPACE(pchOptions[cchOptions - 1]))
1384 cchOptions--;
1385 while (cchOptions > 0 && RT_C_IS_SPACE(*pchOptions))
1386 cchOptions--, pchOptions++;
1387
1388 /* Quietly ignore empty patterns and empty options. */
1389 if (!cchOptions || !cchPattern)
1390 return VINF_SUCCESS;
1391
1392 /*
1393 * Prepair the pair and verify the option string.
1394 */
1395 uint32_t iPair = pSettings->cPairs;
1396 if ((iPair % 32) == 0)
1397 {
1398 void *pvNew = RTMemRealloc(pSettings->paPairs, (iPair + 32) * sizeof(pSettings->paPairs[0]));
1399 if (!pvNew)
1400 return VERR_NO_MEMORY;
1401 pSettings->paPairs = (PSCMPATRNOPTPAIR)pvNew;
1402 }
1403
1404 pSettings->paPairs[iPair].pszPattern = RTStrDupN(pchLine, cchPattern);
1405 pSettings->paPairs[iPair].pszOptions = RTStrDupN(pchOptions, cchOptions);
1406 pSettings->paPairs[iPair].pszRelativeTo = RTStrDupN(pchDir, cchDir);
1407 int rc;
1408 if ( pSettings->paPairs[iPair].pszPattern
1409 && pSettings->paPairs[iPair].pszOptions
1410 && pSettings->paPairs[iPair].pszRelativeTo)
1411 rc = scmSettingsBaseVerifyString(pSettings->paPairs[iPair].pszOptions);
1412 else
1413 rc = VERR_NO_MEMORY;
1414
1415 /*
1416 * If it checked out fine, expand any relative paths in the pattern.
1417 */
1418 if (RT_SUCCESS(rc))
1419 {
1420 size_t cPattern = 1;
1421 size_t cRelativePaths = 0;
1422 const char *pszSrc = pSettings->paPairs[iPair].pszPattern;
1423 for (;;)
1424 {
1425 if (*pszSrc == '/')
1426 cRelativePaths++;
1427 pszSrc = strchr(pszSrc, '|');
1428 if (!pszSrc)
1429 break;
1430 pszSrc++;
1431 cPattern++;
1432 }
1433 pSettings->paPairs[iPair].fMultiPattern = cPattern > 1;
1434 if (cRelativePaths > 0)
1435 {
1436 char *pszNewPattern = RTStrAlloc(cchPattern + cRelativePaths * (cchDir - 1) + 1);
1437 if (pszNewPattern)
1438 {
1439 char *pszDst = pszNewPattern;
1440 pszSrc = pSettings->paPairs[iPair].pszPattern;
1441 for (;;)
1442 {
1443 if (*pszSrc == '/')
1444 {
1445 memcpy(pszDst, pchDir, cchDir);
1446 pszDst += cchDir;
1447 pszSrc += 1;
1448 }
1449
1450 /* Look for the next relative path. */
1451 const char *pszSrcNext = strchr(pszSrc, '|');
1452 while (pszSrcNext && pszSrcNext[1] != '/')
1453 pszSrcNext = strchr(pszSrcNext, '|');
1454 if (!pszSrcNext)
1455 break;
1456
1457 /* Copy stuff between current and the next path. */
1458 pszSrcNext++;
1459 memcpy(pszDst, pszSrc, pszSrcNext - pszSrc);
1460 pszDst += pszSrcNext - pszSrc;
1461 pszSrc = pszSrcNext;
1462 }
1463
1464 /* Copy the final portion and replace the pattern. */
1465 strcpy(pszDst, pszSrc);
1466
1467 RTStrFree(pSettings->paPairs[iPair].pszPattern);
1468 pSettings->paPairs[iPair].pszPattern = pszNewPattern;
1469 }
1470 else
1471 rc = VERR_NO_MEMORY;
1472 }
1473 }
1474 if (RT_SUCCESS(rc))
1475 /*
1476 * Commit the pair.
1477 */
1478 pSettings->cPairs = iPair + 1;
1479 else
1480 {
1481 RTStrFree(pSettings->paPairs[iPair].pszPattern);
1482 RTStrFree(pSettings->paPairs[iPair].pszOptions);
1483 RTStrFree(pSettings->paPairs[iPair].pszRelativeTo);
1484 }
1485 return rc;
1486}
1487
1488/**
1489 * Loads in the settings from @a pszFilename.
1490 *
1491 * @returns IPRT status code.
1492 * @param pSettings Where to load the settings file.
1493 * @param pszFilename The file to load.
1494 */
1495static int scmSettingsLoadFile(PSCMSETTINGS pSettings, const char *pszFilename)
1496{
1497 ScmVerbose(NULL, 3, "Loading settings file '%s'...\n", pszFilename);
1498
1499 /* Turn filename into an absolute path and drop the filename. */
1500 char szAbsPath[RTPATH_MAX];
1501 int rc = RTPathAbs(pszFilename, szAbsPath, sizeof(szAbsPath));
1502 if (RT_FAILURE(rc))
1503 {
1504 RTMsgError("%s: RTPathAbs -> %Rrc\n", pszFilename, rc);
1505 return rc;
1506 }
1507 RTPathChangeToUnixSlashes(szAbsPath, true);
1508 size_t cchDir = RTPathFilename(szAbsPath) - &szAbsPath[0];
1509
1510 /* Try open it.*/
1511 SCMSTREAM Stream;
1512 rc = ScmStreamInitForReading(&Stream, pszFilename);
1513 if (RT_SUCCESS(rc))
1514 {
1515 SCMEOL enmEol;
1516 const char *pchLine;
1517 size_t cchLine;
1518 while ((pchLine = ScmStreamGetLine(&Stream, &cchLine, &enmEol)) != NULL)
1519 {
1520 /* Ignore leading spaces. */
1521 while (cchLine > 0 && RT_C_IS_SPACE(*pchLine))
1522 pchLine++, cchLine--;
1523
1524 /* Ignore empty lines and comment lines. */
1525 if (cchLine < 1 || *pchLine == '#')
1526 continue;
1527
1528 /* Deal with escaped newlines. */
1529 size_t iFirstLine = ~(size_t)0;
1530 char *pszFreeLine = NULL;
1531 if ( pchLine[cchLine - 1] == '\\'
1532 && ( cchLine < 2
1533 || pchLine[cchLine - 2] != '\\') )
1534 {
1535 iFirstLine = ScmStreamTellLine(&Stream);
1536
1537 cchLine--;
1538 while (cchLine > 0 && RT_C_IS_SPACE(pchLine[cchLine - 1]))
1539 cchLine--;
1540
1541 size_t cchTotal = cchLine;
1542 pszFreeLine = RTStrDupN(pchLine, cchLine);
1543 if (pszFreeLine)
1544 {
1545 /* Append following lines. */
1546 while ((pchLine = ScmStreamGetLine(&Stream, &cchLine, &enmEol)) != NULL)
1547 {
1548 while (cchLine > 0 && RT_C_IS_SPACE(*pchLine))
1549 pchLine++, cchLine--;
1550
1551 bool const fDone = cchLine == 0
1552 || pchLine[cchLine - 1] != '\\'
1553 || (cchLine >= 2 && pchLine[cchLine - 2] == '\\');
1554 if (!fDone)
1555 {
1556 cchLine--;
1557 while (cchLine > 0 && RT_C_IS_SPACE(pchLine[cchLine - 1]))
1558 cchLine--;
1559 }
1560
1561 rc = RTStrRealloc(&pszFreeLine, cchTotal + 1 + cchLine + 1);
1562 if (RT_FAILURE(rc))
1563 break;
1564 pszFreeLine[cchTotal++] = ' ';
1565 memcpy(&pszFreeLine[cchTotal], pchLine, cchLine);
1566 cchTotal += cchLine;
1567 pszFreeLine[cchTotal] = '\0';
1568
1569 if (fDone)
1570 break;
1571 }
1572 }
1573 else
1574 rc = VERR_NO_STR_MEMORY;
1575
1576 if (RT_FAILURE(rc))
1577 {
1578 RTStrFree(pszFreeLine);
1579 rc = RTMsgErrorRc(VERR_NO_MEMORY, "%s: Ran out of memory deal with escaped newlines", pszFilename);
1580 break;
1581 }
1582
1583 pchLine = pszFreeLine;
1584 cchLine = cchTotal;
1585 }
1586
1587 /* What kind of line is it? */
1588 const char *pchColon = (const char *)memchr(pchLine, ':', cchLine);
1589 if (pchColon)
1590 rc = scmSettingsAddPair(pSettings, pchLine, cchLine, pchColon - pchLine, szAbsPath, cchDir);
1591 else
1592 rc = scmSettingsBaseParseStringN(&pSettings->Base, pchLine, cchLine, szAbsPath, cchDir);
1593 if (pszFreeLine)
1594 RTStrFree(pszFreeLine);
1595 if (RT_FAILURE(rc))
1596 {
1597 RTMsgError("%s:%d: %Rrc\n",
1598 pszFilename, iFirstLine == ~(size_t)0 ? ScmStreamTellLine(&Stream) : iFirstLine, rc);
1599 break;
1600 }
1601 }
1602
1603 if (RT_SUCCESS(rc))
1604 {
1605 rc = ScmStreamGetStatus(&Stream);
1606 if (RT_FAILURE(rc))
1607 RTMsgError("%s: ScmStreamGetStatus- > %Rrc\n", pszFilename, rc);
1608 }
1609 ScmStreamDelete(&Stream);
1610 }
1611 else
1612 RTMsgError("%s: ScmStreamInitForReading -> %Rrc\n", pszFilename, rc);
1613 return rc;
1614}
1615
1616#if 0 /* unused */
1617/**
1618 * Parse the specified settings file creating a new settings struct from it.
1619 *
1620 * @returns IPRT status code
1621 * @param ppSettings Where to return the new settings.
1622 * @param pszFilename The file to parse.
1623 * @param pSettingsBase The base settings we inherit from.
1624 */
1625static int scmSettingsCreateFromFile(PSCMSETTINGS *ppSettings, const char *pszFilename, PCSCMSETTINGSBASE pSettingsBase)
1626{
1627 PSCMSETTINGS pSettings;
1628 int rc = scmSettingsCreate(&pSettings, pSettingsBase);
1629 if (RT_SUCCESS(rc))
1630 {
1631 rc = scmSettingsLoadFile(pSettings, pszFilename, RTPathFilename(pszFilename) - pszFilename);
1632 if (RT_SUCCESS(rc))
1633 {
1634 *ppSettings = pSettings;
1635 return VINF_SUCCESS;
1636 }
1637
1638 scmSettingsDestroy(pSettings);
1639 }
1640 *ppSettings = NULL;
1641 return rc;
1642}
1643#endif
1644
1645
1646/**
1647 * Create an initial settings structure when starting processing a new file or
1648 * directory.
1649 *
1650 * This will look for .scm-settings files from the root and down to the
1651 * specified directory, combining them into the returned settings structure.
1652 *
1653 * @returns IPRT status code.
1654 * @param ppSettings Where to return the pointer to the top stack
1655 * object.
1656 * @param pBaseSettings The base settings we inherit from (globals
1657 * typically).
1658 * @param pszPath The absolute path to the new directory or file.
1659 */
1660static int scmSettingsCreateForPath(PSCMSETTINGS *ppSettings, PCSCMSETTINGSBASE pBaseSettings, const char *pszPath)
1661{
1662 *ppSettings = NULL; /* try shut up gcc. */
1663
1664 /*
1665 * We'll be working with a stack copy of the path.
1666 */
1667 char szFile[RTPATH_MAX];
1668 size_t cchDir = strlen(pszPath);
1669 if (cchDir >= sizeof(szFile) - sizeof(SCM_SETTINGS_FILENAME))
1670 return VERR_FILENAME_TOO_LONG;
1671
1672 /*
1673 * Create the bottom-most settings.
1674 */
1675 PSCMSETTINGS pSettings;
1676 int rc = scmSettingsCreate(&pSettings, pBaseSettings);
1677 if (RT_FAILURE(rc))
1678 return rc;
1679
1680 /*
1681 * Enumerate the path components from the root and down. Load any setting
1682 * files we find.
1683 */
1684 size_t cComponents = RTPathCountComponents(pszPath);
1685 for (size_t i = 1; i <= cComponents; i++)
1686 {
1687 rc = RTPathCopyComponents(szFile, sizeof(szFile), pszPath, i);
1688 if (RT_SUCCESS(rc))
1689 rc = RTPathAppend(szFile, sizeof(szFile), SCM_SETTINGS_FILENAME);
1690 if (RT_FAILURE(rc))
1691 break;
1692 RTPathChangeToUnixSlashes(szFile, true);
1693
1694 if (RTFileExists(szFile))
1695 {
1696 rc = scmSettingsLoadFile(pSettings, szFile);
1697 if (RT_FAILURE(rc))
1698 break;
1699 }
1700 }
1701
1702 if (RT_SUCCESS(rc))
1703 *ppSettings = pSettings;
1704 else
1705 scmSettingsDestroy(pSettings);
1706 return rc;
1707}
1708
1709/**
1710 * Pushes a new settings set onto the stack.
1711 *
1712 * @param ppSettingsStack The pointer to the pointer to the top stack
1713 * element. This will be used as input and output.
1714 * @param pSettings The settings to push onto the stack.
1715 */
1716static void scmSettingsStackPush(PSCMSETTINGS *ppSettingsStack, PSCMSETTINGS pSettings)
1717{
1718 PSCMSETTINGS pOld = *ppSettingsStack;
1719 pSettings->pDown = pOld;
1720 pSettings->pUp = NULL;
1721 if (pOld)
1722 pOld->pUp = pSettings;
1723 *ppSettingsStack = pSettings;
1724}
1725
1726/**
1727 * Pushes the settings of the specified directory onto the stack.
1728 *
1729 * We will load any .scm-settings in the directory. A stack entry is added even
1730 * if no settings file was found.
1731 *
1732 * @returns IPRT status code.
1733 * @param ppSettingsStack The pointer to the pointer to the top stack
1734 * element. This will be used as input and output.
1735 * @param pszDir The directory to do this for.
1736 */
1737static int scmSettingsStackPushDir(PSCMSETTINGS *ppSettingsStack, const char *pszDir)
1738{
1739 char szFile[RTPATH_MAX];
1740 int rc = RTPathJoin(szFile, sizeof(szFile), pszDir, SCM_SETTINGS_FILENAME);
1741 if (RT_SUCCESS(rc))
1742 {
1743 RTPathChangeToUnixSlashes(szFile, true);
1744
1745 PSCMSETTINGS pSettings;
1746 rc = scmSettingsCreate(&pSettings, &(*ppSettingsStack)->Base);
1747 if (RT_SUCCESS(rc))
1748 {
1749 if (RTFileExists(szFile))
1750 rc = scmSettingsLoadFile(pSettings, szFile);
1751 if (RT_SUCCESS(rc))
1752 {
1753 scmSettingsStackPush(ppSettingsStack, pSettings);
1754 return VINF_SUCCESS;
1755 }
1756
1757 scmSettingsDestroy(pSettings);
1758 }
1759 }
1760 return rc;
1761}
1762
1763
1764/**
1765 * Pops a settings set off the stack.
1766 *
1767 * @returns The popped setttings.
1768 * @param ppSettingsStack The pointer to the pointer to the top stack
1769 * element. This will be used as input and output.
1770 */
1771static PSCMSETTINGS scmSettingsStackPop(PSCMSETTINGS *ppSettingsStack)
1772{
1773 PSCMSETTINGS pRet = *ppSettingsStack;
1774 PSCMSETTINGS pNew = pRet ? pRet->pDown : NULL;
1775 *ppSettingsStack = pNew;
1776 if (pNew)
1777 pNew->pUp = NULL;
1778 if (pRet)
1779 {
1780 pRet->pUp = NULL;
1781 pRet->pDown = NULL;
1782 }
1783 return pRet;
1784}
1785
1786/**
1787 * Pops and destroys the top entry of the stack.
1788 *
1789 * @param ppSettingsStack The pointer to the pointer to the top stack
1790 * element. This will be used as input and output.
1791 */
1792static void scmSettingsStackPopAndDestroy(PSCMSETTINGS *ppSettingsStack)
1793{
1794 scmSettingsDestroy(scmSettingsStackPop(ppSettingsStack));
1795}
1796
1797/**
1798 * Constructs the base settings for the specified file name.
1799 *
1800 * @returns IPRT status code.
1801 * @param pSettingsStack The top element on the settings stack.
1802 * @param pszFilename The file name.
1803 * @param pszBasename The base name (pointer within @a pszFilename).
1804 * @param cchBasename The length of the base name. (For passing to
1805 * RTStrSimplePatternMultiMatch.)
1806 * @param pBase Base settings to initialize.
1807 */
1808static int scmSettingsStackMakeFileBase(PCSCMSETTINGS pSettingsStack, const char *pszFilename,
1809 const char *pszBasename, size_t cchBasename, PSCMSETTINGSBASE pBase)
1810{
1811 ScmVerbose(NULL, 5, "scmSettingsStackMakeFileBase(%s, %.*s)\n", pszFilename, cchBasename, pszBasename);
1812
1813 int rc = scmSettingsBaseInitAndCopy(pBase, &pSettingsStack->Base);
1814 if (RT_SUCCESS(rc))
1815 {
1816 /* find the bottom entry in the stack. */
1817 PCSCMSETTINGS pCur = pSettingsStack;
1818 while (pCur->pDown)
1819 pCur = pCur->pDown;
1820
1821 /* Work our way up thru the stack and look for matching pairs. */
1822 while (pCur)
1823 {
1824 size_t const cPairs = pCur->cPairs;
1825 if (cPairs)
1826 {
1827 for (size_t i = 0; i < cPairs; i++)
1828 if ( !pCur->paPairs[i].fMultiPattern
1829 ? RTStrSimplePatternNMatch(pCur->paPairs[i].pszPattern, RTSTR_MAX,
1830 pszBasename, cchBasename)
1831 || RTStrSimplePatternMatch(pCur->paPairs[i].pszPattern, pszFilename)
1832 : RTStrSimplePatternMultiMatch(pCur->paPairs[i].pszPattern, RTSTR_MAX,
1833 pszBasename, cchBasename, NULL)
1834 || RTStrSimplePatternMultiMatch(pCur->paPairs[i].pszPattern, RTSTR_MAX,
1835 pszFilename, RTSTR_MAX, NULL))
1836 {
1837 ScmVerbose(NULL, 5, "scmSettingsStackMakeFileBase: Matched '%s' : '%s'\n",
1838 pCur->paPairs[i].pszPattern, pCur->paPairs[i].pszOptions);
1839 rc = scmSettingsBaseParseString(pBase, pCur->paPairs[i].pszOptions,
1840 pCur->paPairs[i].pszRelativeTo, strlen(pCur->paPairs[i].pszRelativeTo));
1841 if (RT_FAILURE(rc))
1842 break;
1843 }
1844 if (RT_FAILURE(rc))
1845 break;
1846 }
1847
1848 /* advance */
1849 pCur = pCur->pUp;
1850 }
1851 }
1852 if (RT_FAILURE(rc))
1853 scmSettingsBaseDelete(pBase);
1854 return rc;
1855}
1856
1857
1858/* -=-=-=-=-=- misc -=-=-=-=-=- */
1859
1860
1861/**
1862 * Prints the per file banner needed and the message level is high enough.
1863 *
1864 * @param pState The rewrite state.
1865 * @param iLevel The required verbosity level.
1866 */
1867void ScmVerboseBanner(PSCMRWSTATE pState, int iLevel)
1868{
1869 if (iLevel <= g_iVerbosity && !pState->fFirst)
1870 {
1871 RTPrintf("%s: info: --= Rewriting '%s' =--\n", g_szProgName, pState->pszFilename);
1872 pState->fFirst = true;
1873 }
1874}
1875
1876
1877/**
1878 * Prints a verbose message if the level is high enough.
1879 *
1880 * @param pState The rewrite state. Optional.
1881 * @param iLevel The required verbosity level.
1882 * @param pszFormat The message format string. Can be NULL if we
1883 * only want to trigger the per file message.
1884 * @param ... Format arguments.
1885 */
1886void ScmVerbose(PSCMRWSTATE pState, int iLevel, const char *pszFormat, ...)
1887{
1888 if (iLevel <= g_iVerbosity)
1889 {
1890 if (pState && !pState->fFirst)
1891 {
1892 RTPrintf("%s: info: --= Rewriting '%s' =--\n", g_szProgName, pState->pszFilename);
1893 pState->fFirst = true;
1894 }
1895 RTPrintf(pState
1896 ? "%s: info: "
1897 : "%s: info: ",
1898 g_szProgName);
1899 va_list va;
1900 va_start(va, pszFormat);
1901 RTPrintfV(pszFormat, va);
1902 va_end(va);
1903 }
1904}
1905
1906
1907/**
1908 * Prints an error message.
1909 *
1910 * @returns false
1911 * @param pState The rewrite state. Optional.
1912 * @param rc The error code.
1913 * @param pszFormat The message format string.
1914 * @param ... Format arguments.
1915 */
1916bool ScmError(PSCMRWSTATE pState, int rc, const char *pszFormat, ...)
1917{
1918 if (RT_SUCCESS(pState->rc))
1919 pState->rc = rc;
1920
1921 if (!pState->fFirst)
1922 {
1923 RTPrintf("%s: info: --= Rewriting '%s' =--\n", g_szProgName, pState->pszFilename);
1924 pState->fFirst = true;
1925 }
1926 va_list va;
1927 va_start(va, pszFormat);
1928 RTPrintf("%s: error: %s: %N", g_szProgName, pState->pszFilename, pszFormat, &va);
1929 va_end(va);
1930
1931 return false;
1932}
1933
1934
1935/* -=-=-=-=-=- file and directory processing -=-=-=-=-=- */
1936
1937
1938/**
1939 * Processes a file.
1940 *
1941 * @returns IPRT status code.
1942 * @param pState The rewriter state.
1943 * @param pszFilename The file name.
1944 * @param pszBasename The base name (pointer within @a pszFilename).
1945 * @param cchBasename The length of the base name. (For passing to
1946 * RTStrSimplePatternMultiMatch.)
1947 * @param pBaseSettings The base settings to use. It's OK to modify
1948 * these.
1949 */
1950static int scmProcessFileInner(PSCMRWSTATE pState, const char *pszFilename, const char *pszBasename, size_t cchBasename,
1951 PSCMSETTINGSBASE pBaseSettings)
1952{
1953 /*
1954 * Do the file level filtering.
1955 */
1956 if ( pBaseSettings->pszFilterFiles
1957 && *pBaseSettings->pszFilterFiles
1958 && !RTStrSimplePatternMultiMatch(pBaseSettings->pszFilterFiles, RTSTR_MAX, pszBasename, cchBasename, NULL))
1959 {
1960 ScmVerbose(NULL, 5, "skipping '%s': file filter mismatch\n", pszFilename);
1961 g_cFilesSkipped++;
1962 return VINF_SUCCESS;
1963 }
1964 if ( pBaseSettings->pszFilterOutFiles
1965 && *pBaseSettings->pszFilterOutFiles
1966 && ( RTStrSimplePatternMultiMatch(pBaseSettings->pszFilterOutFiles, RTSTR_MAX, pszBasename, cchBasename, NULL)
1967 || RTStrSimplePatternMultiMatch(pBaseSettings->pszFilterOutFiles, RTSTR_MAX, pszFilename, RTSTR_MAX, NULL)) )
1968 {
1969 ScmVerbose(NULL, 5, "skipping '%s': filterd out\n", pszFilename);
1970 g_cFilesSkipped++;
1971 return VINF_SUCCESS;
1972 }
1973 if ( pBaseSettings->fOnlySvnFiles
1974 && !ScmSvnIsInWorkingCopy(pState))
1975 {
1976 ScmVerbose(NULL, 5, "skipping '%s': not in SVN WC\n", pszFilename);
1977 g_cFilesNotInSvn++;
1978 return VINF_SUCCESS;
1979 }
1980
1981 /*
1982 * Create an input stream from the file and check that it's text.
1983 */
1984 SCMSTREAM Stream1;
1985 int rc = ScmStreamInitForReading(&Stream1, pszFilename);
1986 if (RT_FAILURE(rc))
1987 {
1988 RTMsgError("Failed to read '%s': %Rrc\n", pszFilename, rc);
1989 return rc;
1990 }
1991 bool const fIsText = ScmStreamIsText(&Stream1);
1992
1993 /*
1994 * Try find a matching rewrite config for this filename.
1995 */
1996 PCSCMCFGENTRY pCfg = pBaseSettings->pTreatAs;
1997 if (!pCfg)
1998 {
1999 for (size_t iCfg = 0; iCfg < RT_ELEMENTS(g_aConfigs); iCfg++)
2000 if (RTStrSimplePatternMultiMatch(g_aConfigs[iCfg].pszFilePattern, RTSTR_MAX, pszBasename, cchBasename, NULL))
2001 {
2002 pCfg = &g_aConfigs[iCfg];
2003 break;
2004 }
2005 if (!pCfg)
2006 {
2007 /* On failure try check for hash-bang stuff before giving up. */
2008 if (fIsText)
2009 {
2010 SCMEOL enmIgn;
2011 size_t cchFirst;
2012 const char *pchFirst = ScmStreamGetLine(&Stream1, &cchFirst, &enmIgn);
2013 if (cchFirst >= 9 && pchFirst && *pchFirst == '#')
2014 {
2015 do
2016 {
2017 pchFirst++;
2018 cchFirst--;
2019 } while (cchFirst > 0 && RT_C_IS_BLANK(*pchFirst));
2020 if (*pchFirst == '!')
2021 {
2022 do
2023 {
2024 pchFirst++;
2025 cchFirst--;
2026 } while (cchFirst > 0 && RT_C_IS_BLANK(*pchFirst));
2027 const char *pszTreatAs = NULL;
2028 if ( (cchFirst >= 7 && strncmp(pchFirst, "/bin/sh", 7) == 0)
2029 || (cchFirst >= 9 && strncmp(pchFirst, "/bin/bash", 9) == 0)
2030 || (cchFirst >= 4+9 && strncmp(pchFirst, "/usr/bin/bash", 4+9) == 0) )
2031 pszTreatAs = "shell";
2032 else if ( (cchFirst >= 15 && strncmp(pchFirst, "/usr/bin/python", 15) == 0)
2033 || (cchFirst >= 19 && strncmp(pchFirst, "/usr/bin/env python", 19) == 0) )
2034 pszTreatAs = "python";
2035 else if ( (cchFirst >= 13 && strncmp(pchFirst, "/usr/bin/perl", 13) == 0)
2036 || (cchFirst >= 17 && strncmp(pchFirst, "/usr/bin/env perl", 17) == 0) )
2037 pszTreatAs = "perl";
2038 if (pszTreatAs)
2039 {
2040 for (size_t iCfg = 0; iCfg < RT_ELEMENTS(g_aConfigs); iCfg++)
2041 if (strcmp(pszTreatAs, g_aConfigs[iCfg].pszName) == 0)
2042 {
2043 pCfg = &g_aConfigs[iCfg];
2044 break;
2045 }
2046 Assert(pCfg);
2047 }
2048 }
2049 }
2050 ScmStreamRewindForReading(&Stream1);
2051 }
2052 if (!pCfg)
2053 {
2054 ScmVerbose(NULL, 2, "skipping '%s': no rewriters configured\n", pszFilename);
2055 g_cFilesNoRewriters++;
2056 ScmStreamDelete(&Stream1);
2057 return VINF_SUCCESS;
2058 }
2059 }
2060 ScmVerbose(pState, 4, "matched \"%s\" (%s)\n", pCfg->pszFilePattern, pCfg->pszName);
2061 }
2062 else
2063 ScmVerbose(pState, 4, "treat-as \"%s\"\n", pCfg->pszName);
2064
2065 if (fIsText || pCfg->fBinary)
2066 {
2067 ScmVerboseBanner(pState, 3);
2068
2069 /*
2070 * Gather SCM and editor settings from the stream.
2071 */
2072 rc = scmSettingsBaseLoadFromDocument(pBaseSettings, &Stream1);
2073 if (RT_SUCCESS(rc))
2074 {
2075 ScmStreamRewindForReading(&Stream1);
2076
2077 /*
2078 * Create two more streams for output and push the text thru all the
2079 * rewriters, switching the two streams around when something is
2080 * actually rewritten. Stream1 remains unchanged.
2081 */
2082 SCMSTREAM Stream2;
2083 rc = ScmStreamInitForWriting(&Stream2, &Stream1);
2084 if (RT_SUCCESS(rc))
2085 {
2086 SCMSTREAM Stream3;
2087 rc = ScmStreamInitForWriting(&Stream3, &Stream1);
2088 if (RT_SUCCESS(rc))
2089 {
2090 bool fModified = false;
2091 PSCMSTREAM pIn = &Stream1;
2092 PSCMSTREAM pOut = &Stream2;
2093 for (size_t iRw = 0; iRw < pCfg->cRewriters; iRw++)
2094 {
2095 pState->rc = VINF_SUCCESS;
2096 bool fRc = pCfg->paRewriters[iRw]->pfnRewriter(pState, pIn, pOut, pBaseSettings);
2097 if (RT_FAILURE(pState->rc))
2098 break;
2099 if (fRc)
2100 {
2101 PSCMSTREAM pTmp = pOut;
2102 pOut = pIn == &Stream1 ? &Stream3 : pIn;
2103 pIn = pTmp;
2104 fModified = true;
2105 }
2106
2107 ScmStreamRewindForReading(pIn);
2108 ScmStreamRewindForWriting(pOut);
2109 }
2110
2111 rc = pState->rc;
2112 if (RT_SUCCESS(rc))
2113 {
2114 rc = ScmStreamGetStatus(&Stream1);
2115 if (RT_SUCCESS(rc))
2116 rc = ScmStreamGetStatus(&Stream2);
2117 if (RT_SUCCESS(rc))
2118 rc = ScmStreamGetStatus(&Stream3);
2119 if (RT_SUCCESS(rc))
2120 {
2121 /*
2122 * If rewritten, write it back to disk.
2123 */
2124 if (fModified && !pCfg->fBinary)
2125 {
2126 if (!g_fDryRun)
2127 {
2128 ScmVerbose(pState, 1, "writing modified file to \"%s%s\"\n", pszFilename, g_pszChangedSuff);
2129 rc = ScmStreamWriteToFile(pIn, "%s%s", pszFilename, g_pszChangedSuff);
2130 if (RT_FAILURE(rc))
2131 RTMsgError("Error writing '%s%s': %Rrc\n", pszFilename, g_pszChangedSuff, rc);
2132 }
2133 else
2134 {
2135 ScmVerboseBanner(pState, 1);
2136 ScmDiffStreams(pszFilename, &Stream1, pIn, g_fDiffIgnoreEol,
2137 g_fDiffIgnoreLeadingWS, g_fDiffIgnoreTrailingWS, g_fDiffSpecialChars,
2138 pBaseSettings->cchTab, g_pStdOut);
2139 ScmVerbose(pState, 2, "would have modified the file \"%s%s\"\n",
2140 pszFilename, g_pszChangedSuff);
2141 }
2142 g_cFilesModified++;
2143 }
2144 else if (fModified)
2145 rc = RTMsgErrorRc(VERR_INTERNAL_ERROR, "Rewriters modified binary file! Impossible!");
2146
2147 /*
2148 * If pending SVN property changes, apply them.
2149 */
2150 if (pState->cSvnPropChanges && RT_SUCCESS(rc))
2151 {
2152 if (!g_fDryRun)
2153 {
2154 rc = ScmSvnApplyChanges(pState);
2155 if (RT_FAILURE(rc))
2156 RTMsgError("%s: failed to apply SVN property changes (%Rrc)\n", pszFilename, rc);
2157 }
2158 else
2159 ScmSvnDisplayChanges(pState);
2160 if (!fModified)
2161 g_cFilesModified++;
2162 }
2163
2164 if (!fModified && !pState->cSvnPropChanges)
2165 ScmVerbose(pState, 3, "%s: no change\n", pszFilename);
2166 }
2167 else
2168 RTMsgError("%s: stream error %Rrc\n", pszFilename, rc);
2169 }
2170 ScmStreamDelete(&Stream3);
2171 }
2172 else
2173 RTMsgError("Failed to init stream for writing: %Rrc\n", rc);
2174 ScmStreamDelete(&Stream2);
2175 }
2176 else
2177 RTMsgError("Failed to init stream for writing: %Rrc\n", rc);
2178 }
2179 else
2180 RTMsgError("scmSettingsBaseLoadFromDocument: %Rrc\n", rc);
2181 }
2182 else
2183 {
2184 ScmVerbose(pState, 2, "not text file: \"%s\"\n", pszFilename);
2185 g_cFilesBinaries++;
2186 }
2187 ScmStreamDelete(&Stream1);
2188
2189 return rc;
2190}
2191
2192/**
2193 * Processes a file.
2194 *
2195 * This is just a wrapper for scmProcessFileInner for avoid wasting stack in the
2196 * directory recursion method.
2197 *
2198 * @returns IPRT status code.
2199 * @param pszFilename The file name.
2200 * @param pszBasename The base name (pointer within @a pszFilename).
2201 * @param cchBasename The length of the base name. (For passing to
2202 * RTStrSimplePatternMultiMatch.)
2203 * @param pSettingsStack The settings stack (pointer to the top element).
2204 */
2205static int scmProcessFile(const char *pszFilename, const char *pszBasename, size_t cchBasename,
2206 PSCMSETTINGS pSettingsStack)
2207{
2208 SCMSETTINGSBASE Base;
2209 int rc = scmSettingsStackMakeFileBase(pSettingsStack, pszFilename, pszBasename, cchBasename, &Base);
2210 if (RT_SUCCESS(rc))
2211 {
2212 SCMRWSTATE State;
2213 State.pszFilename = pszFilename;
2214 State.fFirst = false;
2215 State.fIsInSvnWorkingCopy = 0;
2216 State.cSvnPropChanges = 0;
2217 State.paSvnPropChanges = NULL;
2218 State.rc = VINF_SUCCESS;
2219
2220 rc = scmProcessFileInner(&State, pszFilename, pszBasename, cchBasename, &Base);
2221
2222 size_t i = State.cSvnPropChanges;
2223 while (i-- > 0)
2224 {
2225 RTStrFree(State.paSvnPropChanges[i].pszName);
2226 RTStrFree(State.paSvnPropChanges[i].pszValue);
2227 }
2228 RTMemFree(State.paSvnPropChanges);
2229
2230 scmSettingsBaseDelete(&Base);
2231
2232 g_cFilesProcessed++;
2233 }
2234 return rc;
2235}
2236
2237/**
2238 * Tries to correct RTDIRENTRY_UNKNOWN.
2239 *
2240 * @returns Corrected type.
2241 * @param pszPath The path to the object in question.
2242 */
2243static RTDIRENTRYTYPE scmFigureUnknownType(const char *pszPath)
2244{
2245 RTFSOBJINFO Info;
2246 int rc = RTPathQueryInfo(pszPath, &Info, RTFSOBJATTRADD_NOTHING);
2247 if (RT_FAILURE(rc))
2248 return RTDIRENTRYTYPE_UNKNOWN;
2249 if (RTFS_IS_DIRECTORY(Info.Attr.fMode))
2250 return RTDIRENTRYTYPE_DIRECTORY;
2251 if (RTFS_IS_FILE(Info.Attr.fMode))
2252 return RTDIRENTRYTYPE_FILE;
2253 return RTDIRENTRYTYPE_UNKNOWN;
2254}
2255
2256/**
2257 * Recurse into a sub-directory and process all the files and directories.
2258 *
2259 * @returns IPRT status code.
2260 * @param pszBuf Path buffer containing the directory path on
2261 * entry. This ends with a dot. This is passed
2262 * along when recursing in order to save stack space
2263 * and avoid needless copying.
2264 * @param cchDir Length of our path in pszbuf.
2265 * @param pEntry Directory entry buffer. This is also passed
2266 * along when recursing to save stack space.
2267 * @param pSettingsStack The settings stack (pointer to the top element).
2268 * @param iRecursion The recursion depth. This is used to restrict
2269 * the recursions.
2270 */
2271static int scmProcessDirTreeRecursion(char *pszBuf, size_t cchDir, PRTDIRENTRY pEntry,
2272 PSCMSETTINGS pSettingsStack, unsigned iRecursion)
2273{
2274 int rc;
2275 Assert(cchDir > 1 && pszBuf[cchDir - 1] == '.');
2276
2277 /*
2278 * Make sure we stop somewhere.
2279 */
2280 if (iRecursion > 128)
2281 {
2282 RTMsgError("recursion too deep: %d\n", iRecursion);
2283 return VINF_SUCCESS; /* ignore */
2284 }
2285
2286 /*
2287 * Check if it's excluded by --only-svn-dir.
2288 */
2289 if (pSettingsStack->Base.fOnlySvnDirs)
2290 {
2291 if (!ScmSvnIsDirInWorkingCopy(pszBuf))
2292 return VINF_SUCCESS;
2293 }
2294 g_cDirsProcessed++;
2295
2296 /*
2297 * Try open and read the directory.
2298 */
2299 RTDIR hDir;
2300 rc = RTDirOpenFiltered(&hDir, pszBuf, RTDIRFILTER_NONE, 0 /*fFlags*/);
2301 if (RT_FAILURE(rc))
2302 {
2303 RTMsgError("Failed to enumerate directory '%s': %Rrc", pszBuf, rc);
2304 return rc;
2305 }
2306 for (;;)
2307 {
2308 /* Read the next entry. */
2309 rc = RTDirRead(hDir, pEntry, NULL);
2310 if (RT_FAILURE(rc))
2311 {
2312 if (rc == VERR_NO_MORE_FILES)
2313 rc = VINF_SUCCESS;
2314 else
2315 RTMsgError("RTDirRead -> %Rrc\n", rc);
2316 break;
2317 }
2318
2319 /* Skip '.' and '..'. */
2320 if ( pEntry->szName[0] == '.'
2321 && ( pEntry->cbName == 1
2322 || ( pEntry->cbName == 2
2323 && pEntry->szName[1] == '.')))
2324 continue;
2325
2326 /* Enter it into the buffer so we've got a full name to work
2327 with when needed. */
2328 if (pEntry->cbName + cchDir >= RTPATH_MAX)
2329 {
2330 RTMsgError("Skipping too long entry: %s", pEntry->szName);
2331 continue;
2332 }
2333 memcpy(&pszBuf[cchDir - 1], pEntry->szName, pEntry->cbName + 1);
2334
2335 /* Figure the type. */
2336 RTDIRENTRYTYPE enmType = pEntry->enmType;
2337 if (enmType == RTDIRENTRYTYPE_UNKNOWN)
2338 enmType = scmFigureUnknownType(pszBuf);
2339
2340 /* Process the file or directory, skip the rest. */
2341 if (enmType == RTDIRENTRYTYPE_FILE)
2342 rc = scmProcessFile(pszBuf, pEntry->szName, pEntry->cbName, pSettingsStack);
2343 else if (enmType == RTDIRENTRYTYPE_DIRECTORY)
2344 {
2345 /* Append the dot for the benefit of the pattern matching. */
2346 if (pEntry->cbName + cchDir + 5 >= RTPATH_MAX)
2347 {
2348 RTMsgError("Skipping too deep dir entry: %s", pEntry->szName);
2349 continue;
2350 }
2351 memcpy(&pszBuf[cchDir - 1 + pEntry->cbName], "/.", sizeof("/."));
2352 size_t cchSubDir = cchDir - 1 + pEntry->cbName + sizeof("/.") - 1;
2353
2354 if ( !pSettingsStack->Base.pszFilterOutDirs
2355 || !*pSettingsStack->Base.pszFilterOutDirs
2356 || ( !RTStrSimplePatternMultiMatch(pSettingsStack->Base.pszFilterOutDirs, RTSTR_MAX,
2357 pEntry->szName, pEntry->cbName, NULL)
2358 && !RTStrSimplePatternMultiMatch(pSettingsStack->Base.pszFilterOutDirs, RTSTR_MAX,
2359 pszBuf, cchSubDir, NULL)
2360 )
2361 )
2362 {
2363 rc = scmSettingsStackPushDir(&pSettingsStack, pszBuf);
2364 if (RT_SUCCESS(rc))
2365 {
2366 rc = scmProcessDirTreeRecursion(pszBuf, cchSubDir, pEntry, pSettingsStack, iRecursion + 1);
2367 scmSettingsStackPopAndDestroy(&pSettingsStack);
2368 }
2369 }
2370 }
2371 if (RT_FAILURE(rc))
2372 break;
2373 }
2374 RTDirClose(hDir);
2375 return rc;
2376
2377}
2378
2379/**
2380 * Process a directory tree.
2381 *
2382 * @returns IPRT status code.
2383 * @param pszDir The directory to start with. This is pointer to
2384 * a RTPATH_MAX sized buffer.
2385 */
2386static int scmProcessDirTree(char *pszDir, PSCMSETTINGS pSettingsStack)
2387{
2388 /*
2389 * Setup the recursion.
2390 */
2391 int rc = RTPathAppend(pszDir, RTPATH_MAX, ".");
2392 if (RT_SUCCESS(rc))
2393 {
2394 RTPathChangeToUnixSlashes(pszDir, true);
2395
2396 RTDIRENTRY Entry;
2397 rc = scmProcessDirTreeRecursion(pszDir, strlen(pszDir), &Entry, pSettingsStack, 0);
2398 }
2399 else
2400 RTMsgError("RTPathAppend: %Rrc\n", rc);
2401 return rc;
2402}
2403
2404
2405/**
2406 * Processes a file or directory specified as an command line argument.
2407 *
2408 * @returns IPRT status code
2409 * @param pszSomething What we found in the command line arguments.
2410 * @param pSettingsStack The settings stack (pointer to the top element).
2411 */
2412static int scmProcessSomething(const char *pszSomething, PSCMSETTINGS pSettingsStack)
2413{
2414 char szBuf[RTPATH_MAX];
2415 int rc = RTPathAbs(pszSomething, szBuf, sizeof(szBuf));
2416 if (RT_SUCCESS(rc))
2417 {
2418 RTPathChangeToUnixSlashes(szBuf, false /*fForce*/);
2419
2420 PSCMSETTINGS pSettings;
2421 rc = scmSettingsCreateForPath(&pSettings, &pSettingsStack->Base, szBuf);
2422 if (RT_SUCCESS(rc))
2423 {
2424 scmSettingsStackPush(&pSettingsStack, pSettings);
2425
2426 if (RTFileExists(szBuf))
2427 {
2428 const char *pszBasename = RTPathFilename(szBuf);
2429 if (pszBasename)
2430 {
2431 size_t cchBasename = strlen(pszBasename);
2432 rc = scmProcessFile(szBuf, pszBasename, cchBasename, pSettingsStack);
2433 }
2434 else
2435 {
2436 RTMsgError("RTPathFilename: NULL\n");
2437 rc = VERR_IS_A_DIRECTORY;
2438 }
2439 }
2440 else
2441 rc = scmProcessDirTree(szBuf, pSettingsStack);
2442
2443 PSCMSETTINGS pPopped = scmSettingsStackPop(&pSettingsStack);
2444 Assert(pPopped == pSettings); RT_NOREF_PV(pPopped);
2445 scmSettingsDestroy(pSettings);
2446 }
2447 else
2448 RTMsgError("scmSettingsInitStack: %Rrc\n", rc);
2449 }
2450 else
2451 RTMsgError("RTPathAbs: %Rrc\n", rc);
2452 return rc;
2453}
2454
2455/**
2456 * Print some stats.
2457 */
2458static void scmPrintStats(void)
2459{
2460 ScmVerbose(NULL, 0,
2461 g_fDryRun
2462 ? "%u out of %u file%s in %u dir%s would be modified (%u without rewriter%s, %u binar%s, %u not in svn, %u skipped)\n"
2463 : "%u out of %u file%s in %u dir%s was modified (%u without rewriter%s, %u binar%s, %u not in svn, %u skipped)\n",
2464 g_cFilesModified,
2465 g_cFilesProcessed, g_cFilesProcessed == 1 ? "" : "s",
2466 g_cDirsProcessed, g_cDirsProcessed == 1 ? "" : "s",
2467 g_cFilesNoRewriters, g_cFilesNoRewriters == 1 ? "" : "s",
2468 g_cFilesBinaries, g_cFilesBinaries == 1 ? "y" : "ies",
2469 g_cFilesNotInSvn, g_cFilesSkipped);
2470}
2471
2472/**
2473 * Display the rewriter actions.
2474 *
2475 * @returns RTEXITCODE_SUCCESS.
2476 */
2477static int scmHelpActions(void)
2478{
2479 RTPrintf("Available rewriter actions:\n");
2480 for (uint32_t i = 0; i < RT_ELEMENTS(g_papRewriterActions); i++)
2481 RTPrintf(" %s\n", g_papRewriterActions[i]->pszName);
2482 return RTEXITCODE_SUCCESS;
2483}
2484
2485/**
2486 * Display the default configuration.
2487 *
2488 * @returns RTEXITCODE_SUCCESS.
2489 */
2490static int scmHelpConfig(void)
2491{
2492 RTPrintf("Rewriter configuration:\n");
2493 for (size_t iCfg = 0; iCfg < RT_ELEMENTS(g_aConfigs); iCfg++)
2494 {
2495 RTPrintf("\n %s%s - %s:\n",
2496 g_aConfigs[iCfg].pszName, g_aConfigs[iCfg].fBinary ? " (binary)" : "", g_aConfigs[iCfg].pszFilePattern);
2497 for (size_t i = 0; i < g_aConfigs[iCfg].cRewriters; i++)
2498 RTPrintf(" %s\n", g_aConfigs[iCfg].paRewriters[i]->pszName);
2499 }
2500 return RTEXITCODE_SUCCESS;
2501}
2502
2503/**
2504 * Display the primary help text.
2505 *
2506 * @returns RTEXITCODE_SUCCESS.
2507 * @param paOpts Options.
2508 * @param cOpts Number of options.
2509 */
2510static int scmHelp(PCRTGETOPTDEF paOpts, size_t cOpts)
2511{
2512 RTPrintf("VirtualBox Source Code Massager\n"
2513 "\n"
2514 "Usage: %s [options] <files & dirs>\n"
2515 "\n"
2516 "General options:\n", g_szProgName);
2517 for (size_t i = 0; i < cOpts; i++)
2518 {
2519 /* Grouping. */
2520 switch (paOpts[i].iShort)
2521 {
2522 case SCMOPT_DIFF_IGNORE_EOL:
2523 RTPrintf("\nDiff options (dry runs):\n");
2524 break;
2525 case SCMOPT_CONVERT_EOL:
2526 RTPrintf("\nRewriter action options:\n");
2527 break;
2528 case SCMOPT_ONLY_SVN_DIRS:
2529 RTPrintf("\nInput selection options:\n");
2530 break;
2531 case SCMOPT_TREAT_AS:
2532 RTPrintf("\nMisc options:\n");
2533 break;
2534 }
2535
2536 size_t cExtraAdvance = 0;
2537 if ((paOpts[i].fFlags & RTGETOPT_REQ_MASK) == RTGETOPT_REQ_NOTHING)
2538 {
2539 cExtraAdvance = i + 1 < cOpts
2540 && ( strstr(paOpts[i+1].pszLong, "-no-") != NULL
2541 || strstr(paOpts[i+1].pszLong, "-not-") != NULL
2542 || strstr(paOpts[i+1].pszLong, "-dont-") != NULL
2543 || (paOpts[i].iShort == 'q' && paOpts[i+1].iShort == 'v')
2544 || (paOpts[i].iShort == 'd' && paOpts[i+1].iShort == 'D')
2545 );
2546 if (cExtraAdvance)
2547 RTPrintf(" %s, %s\n", paOpts[i].pszLong, paOpts[i + 1].pszLong);
2548 else if (paOpts[i].iShort != SCMOPT_NO_UPDATE_LICENSE)
2549 RTPrintf(" %s\n", paOpts[i].pszLong);
2550 else
2551 {
2552 RTPrintf(" %s,\n"
2553 " %s,\n"
2554 " %s,\n"
2555 " %s,\n"
2556 " %s,\n"
2557 " %s,\n"
2558 " %s\n",
2559 paOpts[i].pszLong,
2560 paOpts[i + 1].pszLong,
2561 paOpts[i + 2].pszLong,
2562 paOpts[i + 3].pszLong,
2563 paOpts[i + 4].pszLong,
2564 paOpts[i + 5].pszLong,
2565 paOpts[i + 6].pszLong);
2566 cExtraAdvance = 6;
2567 }
2568 }
2569 else if ((paOpts[i].fFlags & RTGETOPT_REQ_MASK) == RTGETOPT_REQ_STRING)
2570 switch (paOpts[i].iShort)
2571 {
2572 case SCMOPT_DEL_ACTION:
2573 RTPrintf(" %s pattern\n", paOpts[i].pszLong);
2574 break;
2575 case SCMOPT_FILTER_OUT_DIRS:
2576 case SCMOPT_FILTER_FILES:
2577 case SCMOPT_FILTER_OUT_FILES:
2578 RTPrintf(" %s multi-pattern\n", paOpts[i].pszLong);
2579 break;
2580 default:
2581 RTPrintf(" %s string\n", paOpts[i].pszLong);
2582 }
2583 else
2584 RTPrintf(" %s value\n", paOpts[i].pszLong);
2585 switch (paOpts[i].iShort)
2586 {
2587 case 'd':
2588 case 'D': RTPrintf(" Default: --dry-run\n"); break;
2589 case SCMOPT_CHECK_RUN: RTPrintf(" Default: --dry-run\n"); break;
2590 case 'f': RTPrintf(" Default: none\n"); break;
2591 case 'q':
2592 case 'v': RTPrintf(" Default: -vv\n"); break;
2593 case SCMOPT_HELP_CONFIG: RTPrintf(" Shows the standard file rewriter configurations.\n"); break;
2594 case SCMOPT_HELP_ACTIONS: RTPrintf(" Shows the available rewriter actions.\n"); break;
2595
2596 case SCMOPT_DIFF_IGNORE_EOL: RTPrintf(" Default: false\n"); break;
2597 case SCMOPT_DIFF_IGNORE_SPACE: RTPrintf(" Default: false\n"); break;
2598 case SCMOPT_DIFF_IGNORE_LEADING_SPACE: RTPrintf(" Default: false\n"); break;
2599 case SCMOPT_DIFF_IGNORE_TRAILING_SPACE: RTPrintf(" Default: false\n"); break;
2600 case SCMOPT_DIFF_SPECIAL_CHARS: RTPrintf(" Default: true\n"); break;
2601
2602 case SCMOPT_CONVERT_EOL: RTPrintf(" Default: %RTbool\n", g_Defaults.fConvertEol); break;
2603 case SCMOPT_CONVERT_TABS: RTPrintf(" Default: %RTbool\n", g_Defaults.fConvertTabs); break;
2604 case SCMOPT_FORCE_FINAL_EOL: RTPrintf(" Default: %RTbool\n", g_Defaults.fForceFinalEol); break;
2605 case SCMOPT_FORCE_TRAILING_LINE: RTPrintf(" Default: %RTbool\n", g_Defaults.fForceTrailingLine); break;
2606 case SCMOPT_STRIP_TRAILING_BLANKS: RTPrintf(" Default: %RTbool\n", g_Defaults.fStripTrailingBlanks); break;
2607 case SCMOPT_STRIP_TRAILING_LINES: RTPrintf(" Default: %RTbool\n", g_Defaults.fStripTrailingLines); break;
2608 case SCMOPT_FIX_FLOWER_BOX_MARKERS: RTPrintf(" Default: %RTbool\n", g_Defaults.fFixFlowerBoxMarkers); break;
2609 case SCMOPT_MIN_BLANK_LINES_BEFORE_FLOWER_BOX_MARKERS: RTPrintf(" Default: %u\n", g_Defaults.cMinBlankLinesBeforeFlowerBoxMakers); break;
2610
2611 case SCMOPT_FIX_TODOS:
2612 RTPrintf(" Fix @todo statements so doxygen sees them. Default: %RTbool\n", g_Defaults.fFixTodos);
2613 break;
2614 case SCMOPT_UPDATE_COPYRIGHT_YEAR:
2615 RTPrintf(" Update the copyright year. Default: %RTbool\n", g_Defaults.fUpdateCopyrightYear);
2616 break;
2617 case SCMOPT_EXTERNAL_COPYRIGHT:
2618 RTPrintf(" Only external copyright holders. Default: %RTbool\n", g_Defaults.fExternalCopyright);
2619 break;
2620 case SCMOPT_NO_UPDATE_LICENSE:
2621 RTPrintf(" License selection. Default: --license-ose-gpl\n");
2622 break;
2623
2624 case SCMOPT_LGPL_DISCLAIMER:
2625 RTPrintf(" Include LGPL version disclaimer. Default: --no-lgpl-disclaimer\n");
2626 break;
2627
2628 case SCMOPT_SET_SVN_EOL: RTPrintf(" Default: %RTbool\n", g_Defaults.fSetSvnEol); break;
2629 case SCMOPT_SET_SVN_EXECUTABLE: RTPrintf(" Default: %RTbool\n", g_Defaults.fSetSvnExecutable); break;
2630 case SCMOPT_SET_SVN_KEYWORDS: RTPrintf(" Default: %RTbool\n", g_Defaults.fSetSvnKeywords); break;
2631 case SCMOPT_TAB_SIZE: RTPrintf(" Default: %u\n", g_Defaults.cchTab); break;
2632 case SCMOPT_WIDTH: RTPrintf(" Default: %u\n", g_Defaults.cchWidth); break;
2633
2634 case SCMOPT_ONLY_SVN_DIRS: RTPrintf(" Default: %RTbool\n", g_Defaults.fOnlySvnDirs); break;
2635 case SCMOPT_ONLY_SVN_FILES: RTPrintf(" Default: %RTbool\n", g_Defaults.fOnlySvnFiles); break;
2636 case SCMOPT_FILTER_OUT_DIRS: RTPrintf(" Default: %s\n", g_Defaults.pszFilterOutDirs); break;
2637 case SCMOPT_FILTER_FILES: RTPrintf(" Default: %s\n", g_Defaults.pszFilterFiles); break;
2638 case SCMOPT_FILTER_OUT_FILES: RTPrintf(" Default: %s\n", g_Defaults.pszFilterOutFiles); break;
2639
2640 case SCMOPT_TREAT_AS:
2641 RTPrintf(" For treat the input file(s) differently, restting any --add-action.\n"
2642 " If the value is empty defaults will be used again. Possible values:\n");
2643 for (size_t iCfg = 0; iCfg < RT_ELEMENTS(g_aConfigs); iCfg++)
2644 RTPrintf(" %s (%s)\n", g_aConfigs[iCfg].pszName, g_aConfigs[iCfg].pszFilePattern);
2645 break;
2646
2647 case SCMOPT_ADD_ACTION:
2648 RTPrintf(" Adds a rewriter action. The first use after a --treat-as will copy and\n"
2649 " the action list selected by the --treat-as. The actuion list will be\n"
2650 " flushed by --treat-as.\n");
2651 break;
2652
2653 case SCMOPT_DEL_ACTION:
2654 RTPrintf(" Deletes one or more rewriter action (pattern). Best used after\n"
2655 " a --treat-as.\n");
2656 break;
2657
2658 default: AssertMsgFailed(("i=%d %d %s\n", i, paOpts[i].iShort, paOpts[i].pszLong));
2659 }
2660 i += cExtraAdvance;
2661 }
2662
2663 return RTEXITCODE_SUCCESS;
2664}
2665
2666int main(int argc, char **argv)
2667{
2668 int rc = RTR3InitExe(argc, &argv, 0);
2669 if (RT_FAILURE(rc))
2670 return 1;
2671
2672 /*
2673 * Init the current year.
2674 */
2675 RTTIMESPEC Now;
2676 RTTIME Time;
2677 RTTimeExplode(&Time, RTTimeNow(&Now));
2678 g_uYear = Time.i32Year;
2679
2680 /*
2681 * Init the settings.
2682 */
2683 PSCMSETTINGS pSettings;
2684 rc = scmSettingsCreate(&pSettings, &g_Defaults);
2685 if (RT_FAILURE(rc))
2686 {
2687 RTMsgError("scmSettingsCreate: %Rrc\n", rc);
2688 return 1;
2689 }
2690
2691 /*
2692 * Parse arguments and process input in order (because this is the only
2693 * thing that works at the moment).
2694 */
2695 static RTGETOPTDEF s_aOpts[14 + RT_ELEMENTS(g_aScmOpts)] =
2696 {
2697 { "--dry-run", 'd', RTGETOPT_REQ_NOTHING },
2698 { "--real-run", 'D', RTGETOPT_REQ_NOTHING },
2699 { "--check-run", SCMOPT_CHECK_RUN, RTGETOPT_REQ_NOTHING },
2700 { "--file-filter", 'f', RTGETOPT_REQ_STRING },
2701 { "--quiet", 'q', RTGETOPT_REQ_NOTHING },
2702 { "--verbose", 'v', RTGETOPT_REQ_NOTHING },
2703 { "--diff-ignore-eol", SCMOPT_DIFF_IGNORE_EOL, RTGETOPT_REQ_NOTHING },
2704 { "--diff-no-ignore-eol", SCMOPT_DIFF_NO_IGNORE_EOL, RTGETOPT_REQ_NOTHING },
2705 { "--diff-ignore-space", SCMOPT_DIFF_IGNORE_SPACE, RTGETOPT_REQ_NOTHING },
2706 { "--diff-no-ignore-space", SCMOPT_DIFF_NO_IGNORE_SPACE, RTGETOPT_REQ_NOTHING },
2707 { "--diff-ignore-leading-space", SCMOPT_DIFF_IGNORE_LEADING_SPACE, RTGETOPT_REQ_NOTHING },
2708 { "--diff-no-ignore-leading-space", SCMOPT_DIFF_NO_IGNORE_LEADING_SPACE, RTGETOPT_REQ_NOTHING },
2709 { "--diff-ignore-trailing-space", SCMOPT_DIFF_IGNORE_TRAILING_SPACE, RTGETOPT_REQ_NOTHING },
2710 { "--diff-no-ignore-trailing-space", SCMOPT_DIFF_NO_IGNORE_TRAILING_SPACE, RTGETOPT_REQ_NOTHING },
2711 { "--diff-special-chars", SCMOPT_DIFF_SPECIAL_CHARS, RTGETOPT_REQ_NOTHING },
2712 { "--diff-no-special-chars", SCMOPT_DIFF_NO_SPECIAL_CHARS, RTGETOPT_REQ_NOTHING },
2713 };
2714 memcpy(&s_aOpts[RT_ELEMENTS(s_aOpts) - RT_ELEMENTS(g_aScmOpts)], &g_aScmOpts[0], sizeof(g_aScmOpts));
2715
2716 bool fCheckRun = false;
2717 RTGETOPTUNION ValueUnion;
2718 RTGETOPTSTATE GetOptState;
2719 rc = RTGetOptInit(&GetOptState, argc, argv, &s_aOpts[0], RT_ELEMENTS(s_aOpts), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
2720 AssertReleaseRCReturn(rc, 1);
2721
2722 while ( (rc = RTGetOpt(&GetOptState, &ValueUnion)) != 0
2723 && rc != VINF_GETOPT_NOT_OPTION)
2724 {
2725 switch (rc)
2726 {
2727 case 'd':
2728 g_fDryRun = true;
2729 fCheckRun = false;
2730 break;
2731 case 'D':
2732 g_fDryRun = fCheckRun = false;
2733 break;
2734 case SCMOPT_CHECK_RUN:
2735 g_fDryRun = fCheckRun = true;
2736 break;
2737
2738 case 'f':
2739 g_pszFileFilter = ValueUnion.psz;
2740 break;
2741
2742 case 'h':
2743 return scmHelp(s_aOpts, RT_ELEMENTS(s_aOpts));
2744
2745 case SCMOPT_HELP_CONFIG:
2746 return scmHelpConfig();
2747
2748 case SCMOPT_HELP_ACTIONS:
2749 return scmHelpActions();
2750
2751 case 'q':
2752 g_iVerbosity = 0;
2753 break;
2754
2755 case 'v':
2756 g_iVerbosity++;
2757 break;
2758
2759 case 'V':
2760 {
2761 /* The following is assuming that svn does it's job here. */
2762 static const char s_szRev[] = "$Revision: 69753 $";
2763 const char *psz = RTStrStripL(strchr(s_szRev, ' '));
2764 RTPrintf("r%.*s\n", strchr(psz, ' ') - psz, psz);
2765 return 0;
2766 }
2767
2768 case SCMOPT_DIFF_IGNORE_EOL:
2769 g_fDiffIgnoreEol = true;
2770 break;
2771 case SCMOPT_DIFF_NO_IGNORE_EOL:
2772 g_fDiffIgnoreEol = false;
2773 break;
2774
2775 case SCMOPT_DIFF_IGNORE_SPACE:
2776 g_fDiffIgnoreTrailingWS = g_fDiffIgnoreLeadingWS = true;
2777 break;
2778 case SCMOPT_DIFF_NO_IGNORE_SPACE:
2779 g_fDiffIgnoreTrailingWS = g_fDiffIgnoreLeadingWS = false;
2780 break;
2781
2782 case SCMOPT_DIFF_IGNORE_LEADING_SPACE:
2783 g_fDiffIgnoreLeadingWS = true;
2784 break;
2785 case SCMOPT_DIFF_NO_IGNORE_LEADING_SPACE:
2786 g_fDiffIgnoreLeadingWS = false;
2787 break;
2788
2789 case SCMOPT_DIFF_IGNORE_TRAILING_SPACE:
2790 g_fDiffIgnoreTrailingWS = true;
2791 break;
2792 case SCMOPT_DIFF_NO_IGNORE_TRAILING_SPACE:
2793 g_fDiffIgnoreTrailingWS = false;
2794 break;
2795
2796 case SCMOPT_DIFF_SPECIAL_CHARS:
2797 g_fDiffSpecialChars = true;
2798 break;
2799 case SCMOPT_DIFF_NO_SPECIAL_CHARS:
2800 g_fDiffSpecialChars = false;
2801 break;
2802
2803 default:
2804 {
2805 int rc2 = scmSettingsBaseHandleOpt(&pSettings->Base, rc, &ValueUnion, "/", 1);
2806 if (RT_SUCCESS(rc2))
2807 break;
2808 if (rc2 != VERR_GETOPT_UNKNOWN_OPTION)
2809 return 2;
2810 return RTGetOptPrintError(rc, &ValueUnion);
2811 }
2812 }
2813 }
2814
2815 /*
2816 * Process non-options.
2817 */
2818 RTEXITCODE rcExit = RTEXITCODE_SUCCESS;
2819 if (rc == VINF_GETOPT_NOT_OPTION)
2820 {
2821 ScmSvnInit();
2822
2823 bool fWarned = g_fDryRun;
2824 while (rc == VINF_GETOPT_NOT_OPTION)
2825 {
2826 if (!fWarned)
2827 {
2828 RTPrintf("%s: Warning! This program will make changes to your source files and\n"
2829 "%s: there is a slight risk that bugs or a full disk may cause\n"
2830 "%s: LOSS OF DATA. So, please make sure you have checked in\n"
2831 "%s: all your changes already. If you didn't, then don't blame\n"
2832 "%s: anyone for not warning you!\n"
2833 "%s:\n"
2834 "%s: Press any key to continue...\n",
2835 g_szProgName, g_szProgName, g_szProgName, g_szProgName, g_szProgName,
2836 g_szProgName, g_szProgName);
2837 RTStrmGetCh(g_pStdIn);
2838 fWarned = true;
2839 }
2840
2841 rc = scmProcessSomething(ValueUnion.psz, pSettings);
2842 if (RT_FAILURE(rc))
2843 {
2844 rcExit = RTEXITCODE_FAILURE;
2845 break;
2846 }
2847
2848 /* next */
2849 rc = RTGetOpt(&GetOptState, &ValueUnion);
2850 if (RT_FAILURE(rc))
2851 rcExit = RTGetOptPrintError(rc, &ValueUnion);
2852 }
2853
2854 scmPrintStats();
2855 ScmSvnTerm();
2856 }
2857 else
2858 RTMsgWarning("No files or directories specified. Doing nothing");
2859
2860 scmSettingsDestroy(pSettings);
2861
2862 /* If we're in checking mode, fail if any files needed modification. */
2863 if ( rcExit == RTEXITCODE_SUCCESS
2864 && fCheckRun
2865 && g_cFilesModified > 0)
2866 {
2867 RTMsgError("Checking mode failed! %u file%s needs modifications", g_cFilesBinaries, g_cFilesBinaries > 1 ? "s" : "");
2868 rcExit = RTEXITCODE_FAILURE;
2869 }
2870
2871 return rcExit;
2872}
2873
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette