File Coverage

deps/libgit2/src/diff_driver.c
Criterion Covered Total %
statement 38 237 16.0
branch 15 178 8.4
condition n/a
subroutine n/a
pod n/a
total 53 415 12.7


line stmt bran cond sub pod time code
1             /*
2             * Copyright (C) the libgit2 contributors. All rights reserved.
3             *
4             * This file is part of libgit2, distributed under the GNU GPL v2 with
5             * a Linking Exception. For full terms see the included COPYING file.
6             */
7              
8             #include "diff_driver.h"
9              
10             #include "git2/attr.h"
11              
12             #include "common.h"
13             #include "diff.h"
14             #include "strmap.h"
15             #include "map.h"
16             #include "buf_text.h"
17             #include "config.h"
18             #include "regexp.h"
19             #include "repository.h"
20              
21             typedef enum {
22             DIFF_DRIVER_AUTO = 0,
23             DIFF_DRIVER_BINARY = 1,
24             DIFF_DRIVER_TEXT = 2,
25             DIFF_DRIVER_PATTERNLIST = 3,
26             } git_diff_driver_t;
27              
28             typedef struct {
29             git_regexp re;
30             int flags;
31             } git_diff_driver_pattern;
32              
33             enum {
34             REG_NEGATE = (1 << 15) /* get out of the way of existing flags */
35             };
36              
37             /* data for finding function context for a given file type */
38             struct git_diff_driver {
39             git_diff_driver_t type;
40             uint32_t binary_flags;
41             uint32_t other_flags;
42             git_array_t(git_diff_driver_pattern) fn_patterns;
43             git_regexp word_pattern;
44             char name[GIT_FLEX_ARRAY];
45             };
46              
47             #include "userdiff.h"
48              
49             struct git_diff_driver_registry {
50             git_strmap *drivers;
51             };
52              
53             #define FORCE_DIFFABLE (GIT_DIFF_FORCE_TEXT | GIT_DIFF_FORCE_BINARY)
54              
55             static git_diff_driver global_drivers[3] = {
56             { DIFF_DRIVER_AUTO, 0, 0, },
57             { DIFF_DRIVER_BINARY, GIT_DIFF_FORCE_BINARY, 0 },
58             { DIFF_DRIVER_TEXT, GIT_DIFF_FORCE_TEXT, 0 },
59             };
60              
61 0           git_diff_driver_registry *git_diff_driver_registry_new(void)
62             {
63 0           git_diff_driver_registry *reg =
64 0           git__calloc(1, sizeof(git_diff_driver_registry));
65 0 0         if (!reg)
66 0           return NULL;
67              
68 0 0         if (git_strmap_new(®->drivers) < 0) {
69 0           git_diff_driver_registry_free(reg);
70 0           return NULL;
71             }
72              
73 0           return reg;
74             }
75              
76 44           void git_diff_driver_registry_free(git_diff_driver_registry *reg)
77             {
78             git_diff_driver *drv;
79              
80 44 50         if (!reg)
81 44           return;
82              
83 0 0         git_strmap_foreach_value(reg->drivers, drv, git_diff_driver_free(drv));
84 0           git_strmap_free(reg->drivers);
85 0           git__free(reg);
86             }
87              
88 0           static int diff_driver_add_patterns(
89             git_diff_driver *drv, const char *regex_str, int regex_flags)
90             {
91 0           int error = 0;
92             const char *scan, *end;
93 0           git_diff_driver_pattern *pat = NULL;
94 0           git_buf buf = GIT_BUF_INIT;
95              
96 0 0         for (scan = regex_str; scan; scan = end) {
97             /* get pattern to fill in */
98 0 0         if ((pat = git_array_alloc(drv->fn_patterns)) == NULL) {
    0          
    0          
99 0           return -1;
100             }
101              
102 0           pat->flags = regex_flags;
103 0 0         if (*scan == '!') {
104 0           pat->flags |= REG_NEGATE;
105 0           ++scan;
106             }
107              
108 0 0         if ((end = strchr(scan, '\n')) != NULL) {
109 0           error = git_buf_set(&buf, scan, end - scan);
110 0           end++;
111             } else {
112 0           error = git_buf_sets(&buf, scan);
113             }
114 0 0         if (error < 0)
115 0           break;
116              
117 0           if ((error = git_regexp_compile(&pat->re, buf.ptr, regex_flags)) != 0) {
118             /*
119             * TODO: issue a warning
120             */
121             }
122             }
123              
124 0 0         if (error && pat != NULL)
    0          
125 0 0         (void)git_array_pop(drv->fn_patterns); /* release last item */
126 0           git_buf_dispose(&buf);
127              
128             /* We want to ignore bad patterns, so return success regardless */
129 0           return 0;
130             }
131              
132 0           static int diff_driver_xfuncname(const git_config_entry *entry, void *payload)
133             {
134 0           return diff_driver_add_patterns(payload, entry->value, 0);
135             }
136              
137 0           static int diff_driver_funcname(const git_config_entry *entry, void *payload)
138             {
139 0           return diff_driver_add_patterns(payload, entry->value, 0);
140             }
141              
142 0           static git_diff_driver_registry *git_repository_driver_registry(
143             git_repository *repo)
144             {
145 0 0         if (!repo->diff_drivers) {
146 0           git_diff_driver_registry *reg = git_diff_driver_registry_new();
147 0           reg = git__compare_and_swap(&repo->diff_drivers, NULL, reg);
148              
149 0 0         if (reg != NULL) /* if we race, free losing allocation */
150 0           git_diff_driver_registry_free(reg);
151             }
152              
153 0 0         if (!repo->diff_drivers)
154 0           git_error_set(GIT_ERROR_REPOSITORY, "unable to create diff driver registry");
155              
156 0           return repo->diff_drivers;
157             }
158              
159 0           static int diff_driver_alloc(
160             git_diff_driver **out, size_t *namelen_out, const char *name)
161             {
162             git_diff_driver *driver;
163 0           size_t driverlen = sizeof(git_diff_driver),
164 0           namelen = strlen(name),
165             alloclen;
166              
167 0 0         GIT_ERROR_CHECK_ALLOC_ADD(&alloclen, driverlen, namelen);
    0          
168 0 0         GIT_ERROR_CHECK_ALLOC_ADD(&alloclen, alloclen, 1);
    0          
169              
170 0           driver = git__calloc(1, alloclen);
171 0 0         GIT_ERROR_CHECK_ALLOC(driver);
172              
173 0           memcpy(driver->name, name, namelen);
174              
175 0           *out = driver;
176              
177 0 0         if (namelen_out)
178 0           *namelen_out = namelen;
179              
180 0           return 0;
181             }
182              
183 0           static int git_diff_driver_builtin(
184             git_diff_driver **out,
185             git_diff_driver_registry *reg,
186             const char *driver_name)
187             {
188 0           git_diff_driver_definition *ddef = NULL;
189 0           git_diff_driver *drv = NULL;
190 0           int error = 0;
191             size_t idx;
192              
193 0 0         for (idx = 0; idx < ARRAY_SIZE(builtin_defs); ++idx) {
194 0 0         if (!strcasecmp(driver_name, builtin_defs[idx].name)) {
195 0           ddef = &builtin_defs[idx];
196 0           break;
197             }
198             }
199 0 0         if (!ddef)
200 0           goto done;
201              
202 0 0         if ((error = diff_driver_alloc(&drv, NULL, ddef->name)) < 0)
203 0           goto done;
204              
205 0           drv->type = DIFF_DRIVER_PATTERNLIST;
206              
207 0 0         if (ddef->fns &&
    0          
208 0           (error = diff_driver_add_patterns(
209             drv, ddef->fns, ddef->flags)) < 0)
210 0           goto done;
211              
212 0 0         if (ddef->words &&
    0          
213 0           (error = git_regexp_compile(&drv->word_pattern, ddef->words, ddef->flags)) < 0)
214 0           goto done;
215              
216 0 0         if ((error = git_strmap_set(reg->drivers, drv->name, drv)) < 0)
217 0           goto done;
218              
219             done:
220 0 0         if (error && drv)
    0          
221 0           git_diff_driver_free(drv);
222             else
223 0           *out = drv;
224              
225 0           return error;
226             }
227              
228 0           static int git_diff_driver_load(
229             git_diff_driver **out, git_repository *repo, const char *driver_name)
230             {
231 0           int error = 0;
232             git_diff_driver_registry *reg;
233             git_diff_driver *drv;
234             size_t namelen;
235 0           git_config *cfg = NULL;
236 0           git_buf name = GIT_BUF_INIT;
237 0           git_config_entry *ce = NULL;
238 0           bool found_driver = false;
239              
240 0 0         if ((reg = git_repository_driver_registry(repo)) == NULL)
241 0           return -1;
242              
243 0 0         if ((drv = git_strmap_get(reg->drivers, driver_name)) != NULL) {
244 0           *out = drv;
245 0           return 0;
246             }
247              
248 0 0         if ((error = diff_driver_alloc(&drv, &namelen, driver_name)) < 0)
249 0           goto done;
250              
251 0           drv->type = DIFF_DRIVER_AUTO;
252              
253             /* if you can't read config for repo, just use default driver */
254 0 0         if (git_repository_config_snapshot(&cfg, repo) < 0) {
255 0           git_error_clear();
256 0           goto done;
257             }
258              
259 0 0         if ((error = git_buf_printf(&name, "diff.%s.binary", driver_name)) < 0)
260 0           goto done;
261              
262 0           switch (git_config__get_bool_force(cfg, name.ptr, -1)) {
263             case true:
264             /* if diff..binary is true, just return the binary driver */
265 0           *out = &global_drivers[DIFF_DRIVER_BINARY];
266 0           goto done;
267             case false:
268             /* if diff..binary is false, force binary checks off */
269             /* but still may have custom function context patterns, etc. */
270 0           drv->binary_flags = GIT_DIFF_FORCE_TEXT;
271 0           found_driver = true;
272 0           break;
273             default:
274             /* diff..binary unspecified or "auto", so just continue */
275 0           break;
276             }
277              
278             /* TODO: warn if diff..command or diff..textconv are set */
279              
280 0           git_buf_truncate(&name, namelen + strlen("diff.."));
281 0 0         if ((error = git_buf_PUTS(&name, "xfuncname")) < 0)
282 0           goto done;
283              
284 0 0         if ((error = git_config_get_multivar_foreach(
285 0           cfg, name.ptr, NULL, diff_driver_xfuncname, drv)) < 0) {
286 0 0         if (error != GIT_ENOTFOUND)
287 0           goto done;
288 0           git_error_clear(); /* no diff..xfuncname, so just continue */
289             }
290              
291 0           git_buf_truncate(&name, namelen + strlen("diff.."));
292 0 0         if ((error = git_buf_PUTS(&name, "funcname")) < 0)
293 0           goto done;
294              
295 0 0         if ((error = git_config_get_multivar_foreach(
296 0           cfg, name.ptr, NULL, diff_driver_funcname, drv)) < 0) {
297 0 0         if (error != GIT_ENOTFOUND)
298 0           goto done;
299 0           git_error_clear(); /* no diff..funcname, so just continue */
300             }
301              
302             /* if we found any patterns, set driver type to use correct callback */
303 0 0         if (git_array_size(drv->fn_patterns) > 0) {
304 0           drv->type = DIFF_DRIVER_PATTERNLIST;
305 0           found_driver = true;
306             }
307              
308 0           git_buf_truncate(&name, namelen + strlen("diff.."));
309 0 0         if ((error = git_buf_PUTS(&name, "wordregex")) < 0)
310 0           goto done;
311              
312 0 0         if ((error = git_config__lookup_entry(&ce, cfg, name.ptr, false)) < 0)
313 0           goto done;
314 0 0         if (!ce || !ce->value)
    0          
315             /* no diff..wordregex, so just continue */;
316 0 0         else if (!(error = git_regexp_compile(&drv->word_pattern, ce->value, 0)))
317 0           found_driver = true;
318             else {
319             /* TODO: warn about bad regex instead of failure */
320 0           goto done;
321             }
322              
323             /* TODO: look up diff..algorithm to turn on minimal / patience
324             * diff in drv->other_flags
325             */
326              
327             /* if no driver config found at all, fall back on AUTO driver */
328 0 0         if (!found_driver)
329 0           goto done;
330              
331             /* store driver in registry */
332 0 0         if ((error = git_strmap_set(reg->drivers, drv->name, drv)) < 0)
333 0           goto done;
334              
335 0           *out = drv;
336              
337             done:
338 0           git_config_entry_free(ce);
339 0           git_buf_dispose(&name);
340 0           git_config_free(cfg);
341              
342 0 0         if (!*out) {
343 0           int error2 = git_diff_driver_builtin(out, reg, driver_name);
344 0 0         if (!error)
345 0           error = error2;
346             }
347              
348 0 0         if (drv && drv != *out)
    0          
349 0           git_diff_driver_free(drv);
350              
351 0           return error;
352             }
353              
354 100           int git_diff_driver_lookup(
355             git_diff_driver **out, git_repository *repo,
356             git_attr_session *attrsession, const char *path)
357             {
358 100           int error = 0;
359 100           const char *values[1], *attrs[] = { "diff" };
360              
361 100 50         assert(out);
362 100           *out = NULL;
363              
364 100 50         if (!repo || !path || !strlen(path))
    50          
    50          
365             /* just use the auto value */;
366 100 50         else if ((error = git_attr_get_many_with_session(values, repo,
367             attrsession, 0, path, 1, attrs)) < 0)
368             /* return error below */;
369              
370 100 50         else if (GIT_ATTR_IS_UNSPECIFIED(values[0]))
371             /* just use the auto value */;
372 0 0         else if (GIT_ATTR_IS_FALSE(values[0]))
373 0           *out = &global_drivers[DIFF_DRIVER_BINARY];
374 0 0         else if (GIT_ATTR_IS_TRUE(values[0]))
375 0           *out = &global_drivers[DIFF_DRIVER_TEXT];
376              
377             /* otherwise look for driver information in config and build driver */
378 0 0         else if ((error = git_diff_driver_load(out, repo, values[0])) < 0) {
379 0 0         if (error == GIT_ENOTFOUND) {
380 0           error = 0;
381 0           git_error_clear();
382             }
383             }
384              
385 100 50         if (!*out)
386 100           *out = &global_drivers[DIFF_DRIVER_AUTO];
387              
388 100           return error;
389             }
390              
391 0           void git_diff_driver_free(git_diff_driver *driver)
392             {
393             size_t i;
394              
395 0 0         if (!driver)
396 0           return;
397              
398 0 0         for (i = 0; i < git_array_size(driver->fn_patterns); ++i)
399 0 0         git_regexp_dispose(& git_array_get(driver->fn_patterns, i)->re);
400 0           git_array_clear(driver->fn_patterns);
401              
402 0           git_regexp_dispose(&driver->word_pattern);
403              
404 0           git__free(driver);
405             }
406              
407 100           void git_diff_driver_update_options(
408             uint32_t *option_flags, git_diff_driver *driver)
409             {
410 100 100         if ((*option_flags & FORCE_DIFFABLE) == 0)
411 96           *option_flags |= driver->binary_flags;
412              
413 100           *option_flags |= driver->other_flags;
414 100           }
415              
416 60           int git_diff_driver_content_is_binary(
417             git_diff_driver *driver, const char *content, size_t content_len)
418             {
419 60           git_buf search = GIT_BUF_INIT;
420              
421             GIT_UNUSED(driver);
422              
423 60           git_buf_attach_notowned(&search, content,
424             min(content_len, GIT_FILTER_BYTES_TO_CHECK_NUL));
425              
426             /* TODO: provide encoding / binary detection callbacks that can
427             * be UTF-8 aware, etc. For now, instead of trying to be smart,
428             * let's just use the simple NUL-byte detection that core git uses.
429             */
430              
431             /* previously was: if (git_buf_text_is_binary(&search)) */
432 60 50         if (git_buf_text_contains_nul(&search))
433 0           return 1;
434              
435 60           return 0;
436             }
437              
438 0           static int diff_context_line__simple(
439             git_diff_driver *driver, git_buf *line)
440             {
441 0           char firstch = line->ptr[0];
442             GIT_UNUSED(driver);
443 0 0         return (git__isalpha(firstch) || firstch == '_' || firstch == '$');
    0          
    0          
444             }
445              
446 0           static int diff_context_line__pattern_match(
447             git_diff_driver *driver, git_buf *line)
448             {
449 0           size_t i, maxi = git_array_size(driver->fn_patterns);
450             git_regmatch pmatch[2];
451              
452 0 0         for (i = 0; i < maxi; ++i) {
453 0 0         git_diff_driver_pattern *pat = git_array_get(driver->fn_patterns, i);
454              
455 0 0         if (!git_regexp_search(&pat->re, line->ptr, 2, pmatch)) {
456 0 0         if (pat->flags & REG_NEGATE)
457 0           return false;
458              
459             /* use pmatch data to trim line data */
460 0           i = (pmatch[1].start >= 0) ? 1 : 0;
461 0           git_buf_consume(line, git_buf_cstr(line) + pmatch[i].start);
462 0           git_buf_truncate(line, pmatch[i].end - pmatch[i].start);
463 0           git_buf_rtrim(line);
464              
465 0           return true;
466             }
467             }
468              
469 0           return false;
470             }
471              
472 0           static long diff_context_find(
473             const char *line,
474             long line_len,
475             char *out,
476             long out_size,
477             void *payload)
478             {
479 0           git_diff_find_context_payload *ctxt = payload;
480              
481 0 0         if (git_buf_set(&ctxt->line, line, (size_t)line_len) < 0)
482 0           return -1;
483 0           git_buf_rtrim(&ctxt->line);
484              
485 0 0         if (!ctxt->line.size)
486 0           return -1;
487              
488 0 0         if (!ctxt->match_line || !ctxt->match_line(ctxt->driver, &ctxt->line))
    0          
489 0           return -1;
490              
491 0 0         if (out_size > (long)ctxt->line.size)
492 0           out_size = (long)ctxt->line.size;
493 0           memcpy(out, ctxt->line.ptr, (size_t)out_size);
494              
495 0           return out_size;
496             }
497              
498 38           void git_diff_find_context_init(
499             git_diff_find_context_fn *findfn_out,
500             git_diff_find_context_payload *payload_out,
501             git_diff_driver *driver)
502             {
503 38 50         *findfn_out = driver ? diff_context_find : NULL;
504              
505 38           memset(payload_out, 0, sizeof(*payload_out));
506 38 50         if (driver) {
507 38           payload_out->driver = driver;
508 76           payload_out->match_line = (driver->type == DIFF_DRIVER_PATTERNLIST) ?
509 38 50         diff_context_line__pattern_match : diff_context_line__simple;
510 38           git_buf_init(&payload_out->line, 0);
511             }
512 38           }
513              
514 38           void git_diff_find_context_clear(git_diff_find_context_payload *payload)
515             {
516 38 50         if (payload) {
517 38           git_buf_dispose(&payload->line);
518 38           payload->driver = NULL;
519             }
520 38           }