1 /* |
|
2 * cline.c |
|
3 * |
|
4 * Created on: 23.05.2011 |
|
5 * Author: Mike |
|
6 */ |
|
7 |
|
8 #include "cline.h" |
|
9 #include "scanner.h" |
|
10 #include "settings.h" |
|
11 #include "arguments.h" |
|
12 #include "stream.h" |
|
13 #include "regex_parser.h" |
|
14 |
|
15 void printHelpText() { |
|
16 const char* helpText = |
|
17 "\nUsage:" |
|
18 "\n cline [Options] [Directories...]" |
|
19 "\n cline [Options] [Directories...]" |
|
20 "\n\nCounts the line terminator characters (\\n) within all" |
|
21 " files in the specified\ndirectories." |
|
22 "\n\nOptions:" |
|
23 "\n -b <level> - binary file heuristics level (default medium)" |
|
24 "\n One of: ignore low medium high" |
|
25 "\n -E <pattern> - Excludes any line matching the <pattern>" |
|
26 "\n -e <start> <end> - Excludes lines between <start> and <end>" |
|
27 "\n You may use these options multiple times" |
|
28 "\n -h, --help - this help text" |
|
29 "\n -m - print information about matching files only" |
|
30 "\n -s <suffixes> - only count files with these suffixes (separated" |
|
31 "\n by commas)" |
|
32 "\n -S <suffixes> - count any file except those with these suffixes" |
|
33 "\n (separated by commas)" |
|
34 "\n -r, -R - includes subdirectories" |
|
35 "\n -v, --version - print out version information" |
|
36 "\n -V - turn verbose output off, print the result only" |
|
37 "\n\nShortcuts:" |
|
38 "\n --exclude-cstyle-comments" |
|
39 "\n = -E \"\\s*//\" -e \"\\s*/\\*\" \"\\*/\\s*\"" |
|
40 "\n\n" |
|
41 "The default call without any options is:" |
|
42 "\n cline ./\n\n" |
|
43 "So each file in the working directory is counted. If you want to count C" |
|
44 "\nsource code in your working directory and its subdirectories, type:" |
|
45 "\n cline -rs .c\n" |
|
46 "\nIf you want to exclude comment lines, you may use the -e/-E option." |
|
47 "\nAfter a line matches the regex pattern <start> any following line is" |
|
48 "\nnot counted unless a line matches the <end> pattern. A line is still " |
|
49 "\ncounted when it does not start or end with the respective patterns." |
|
50 "\nPlease note, that cline does not remove whitespace characters as this" |
|
51 "\nmight not be reasonable in some cases." |
|
52 "\n\nExample (C without comments):" |
|
53 "\n cline -s .c,.h --exclude-cstyle-comments"; |
|
54 |
|
55 printf(helpText); |
|
56 } |
|
57 |
|
58 int exit_with_version(settings_t* settings) { |
|
59 printf("cline - Revision: %s\n", VERSION); |
|
60 destroy_settings_t(settings); |
|
61 return 0; |
|
62 } |
|
63 |
|
64 int exit_with_help(settings_t* settings, int code) { |
|
65 printHelpText(); |
|
66 destroy_settings_t(settings); |
|
67 return code; |
|
68 } |
|
69 |
|
70 int main(int argc, char** argv) { |
|
71 |
|
72 /* Settings */ |
|
73 settings_t *settings = new_settings_t(); |
|
74 if (settings == NULL) { |
|
75 fprintf(stderr, "Memory allocation failed.\n"); |
|
76 return 1; |
|
77 } |
|
78 |
|
79 /* Get arguments */ |
|
80 string_list_t *directories = new_string_list_t(); |
|
81 if (directories == NULL) { |
|
82 fprintf(stderr, "Memory allocation failed.\n"); |
|
83 return 1; |
|
84 } |
|
85 char* includeSuffix = NULL; |
|
86 char* excludeSuffix = NULL; |
|
87 int checked = 0; |
|
88 |
|
89 for (int t = 1 ; t < argc ; t++) { |
|
90 |
|
91 int argflags = checkArgument(argv[t], "hsSrRmvVbeE"); |
|
92 int paropt = 0; |
|
93 |
|
94 /* s */ |
|
95 if ((argflags & 2) > 0) { |
|
96 if (!checkParamOpt(&paropt) || registerArgument(&checked, 2)) { |
|
97 return exit_with_help(settings, 1); |
|
98 } |
|
99 t++; |
|
100 if (t >= argc) { |
|
101 return exit_with_help(settings, 1); |
|
102 } |
|
103 includeSuffix = argv[t]; |
|
104 } |
|
105 /* S */ |
|
106 if ((argflags & 4) > 0) { |
|
107 if (!checkParamOpt(&paropt) || registerArgument(&checked, 4)) { |
|
108 return exit_with_help(settings, 1); |
|
109 } |
|
110 t++; |
|
111 if (t >= argc) { |
|
112 return exit_with_help(settings, 1); |
|
113 } |
|
114 excludeSuffix = argv[t]; |
|
115 } |
|
116 /* h */ |
|
117 if ((argflags & 1) > 0 || strcmp(argv[t], "--help") == 0) { |
|
118 return exit_with_help(settings, 0); |
|
119 } |
|
120 /* r, R */ |
|
121 if ((argflags & 24) > 0) { |
|
122 if (registerArgument(&checked, 24)) { |
|
123 return exit_with_help(settings, 1); |
|
124 } |
|
125 settings->recursive = true; |
|
126 } |
|
127 /* m */ |
|
128 if ((argflags & 32) > 0) { |
|
129 if (registerArgument(&checked, 32)) { |
|
130 return exit_with_help(settings, 1); |
|
131 } |
|
132 settings->matchesOnly = true; |
|
133 } |
|
134 /* v */ |
|
135 if ((argflags & 64) > 0 || strcmp(argv[t], "--version") == 0) { |
|
136 return exit_with_version(settings); |
|
137 } |
|
138 /* V */ |
|
139 if ((argflags & 128) > 0) { |
|
140 if (registerArgument(&checked, 128)) { |
|
141 return exit_with_help(settings, 1); |
|
142 } |
|
143 settings->verbose = false; |
|
144 } |
|
145 /* b */ |
|
146 if ((argflags & 256) > 0) { |
|
147 if (!checkParamOpt(&paropt) || registerArgument(&checked, 256)) { |
|
148 return exit_with_help(settings, 1); |
|
149 } |
|
150 t++; |
|
151 if (t >= argc) { |
|
152 return exit_with_help(settings, 1); |
|
153 } |
|
154 if (strcasecmp(argv[t], "ignore") == 0) { |
|
155 settings->bfileHeuristics->level = BFILE_IGNORE; |
|
156 } else if (strcasecmp(argv[t], "low") == 0) { |
|
157 settings->bfileHeuristics->level = BFILE_LOW_ACCURACY; |
|
158 } else if (strcasecmp(argv[t], "medium") == 0) { |
|
159 settings->bfileHeuristics->level = BFILE_MEDIUM_ACCURACY; |
|
160 } else if (strcasecmp(argv[t], "high") == 0) { |
|
161 settings->bfileHeuristics->level = BFILE_HIGH_ACCURACY; |
|
162 } else { |
|
163 return exit_with_help(settings, 1); |
|
164 } |
|
165 } |
|
166 /* e */ |
|
167 if ((argflags & 512) > 0) { |
|
168 if (!checkParamOpt(&paropt) || t + 2 >= argc) { |
|
169 return exit_with_help(settings, 1); |
|
170 } |
|
171 t++; add_string(settings->regex->pattern_list, argv[t]); |
|
172 t++; add_string(settings->regex->pattern_list, argv[t]); |
|
173 } |
|
174 /* E */ |
|
175 if ((argflags & 1024) > 0) { |
|
176 t++; |
|
177 if (!checkParamOpt(&paropt) || t >= argc) { |
|
178 return exit_with_help(settings, 1); |
|
179 } |
|
180 add_string(settings->regex->pattern_list, argv[t]); |
|
181 add_string(settings->regex->pattern_list, "$"); |
|
182 } |
|
183 if (argflags == 0) { |
|
184 /* SHORTCUTS */ |
|
185 /* exclude-cstyle-comments */ |
|
186 if (strcmp(argv[t], "--exclude-cstyle-comments") == 0) { |
|
187 add_string(settings->regex->pattern_list, "\\s*//"); |
|
188 add_string(settings->regex->pattern_list, "$"); |
|
189 add_string(settings->regex->pattern_list, "\\s*/\\*"); |
|
190 add_string(settings->regex->pattern_list, "\\*/\\s*"); |
|
191 } |
|
192 /* Path */ |
|
193 else { |
|
194 add_string(directories, argv[t]); |
|
195 } |
|
196 } |
|
197 } |
|
198 |
|
199 /* Configure output */ |
|
200 if (!settings->verbose) { |
|
201 close_stdout(); |
|
202 } |
|
203 |
|
204 /* Find tokens */ |
|
205 parseCSL(includeSuffix, settings->includeSuffixes); |
|
206 parseCSL(excludeSuffix, settings->excludeSuffixes); |
|
207 |
|
208 /* Scan directories */ |
|
209 if (regex_compile_all(settings->regex)) { |
|
210 int lines = 0; |
|
211 if (directories->count == 0) { |
|
212 add_string(directories, "."); |
|
213 } |
|
214 for (int t = 0 ; t < directories->count ; t++) { |
|
215 if (t > 0) { |
|
216 for (int u = 0 ; u < 79 ; u++) { |
|
217 printf("-"); |
|
218 } |
|
219 printf("\n"); |
|
220 } |
|
221 lines += scanDirectory((scanner_t){directories->items[t], 0}, settings); |
|
222 } |
|
223 destroy_string_list_t(directories); |
|
224 |
|
225 /* Print double line and line count */ |
|
226 for (int t = 0 ; t < 79 ; t++) { |
|
227 printf("="); |
|
228 } |
|
229 printf("\n%73d lines\n", lines); |
|
230 |
|
231 if (settings->confusing_lnlen && settings->regex->pattern_list->count > 0) { |
|
232 printf("\nSome files contain too long lines.\n" |
|
233 "The regex parser currently supports a maximum line length of %d." |
|
234 "\nThe result might be wrong.\n", REGEX_MAX_LINELENGTH); |
|
235 } |
|
236 |
|
237 if (!settings->verbose) { |
|
238 reopen_stdout(); |
|
239 printf("%d", lines); |
|
240 } |
|
241 destroy_settings_t(settings); |
|
242 } |
|
243 |
|
244 fflush(stdout); |
|
245 fflush(stderr); |
|
246 return 0; |
|
247 } |
|