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 "common.h" |
9
|
|
|
|
|
|
|
|
10
|
|
|
|
|
|
|
#include "git2/attr.h" |
11
|
|
|
|
|
|
|
#include "git2/blob.h" |
12
|
|
|
|
|
|
|
#include "git2/index.h" |
13
|
|
|
|
|
|
|
#include "git2/sys/filter.h" |
14
|
|
|
|
|
|
|
|
15
|
|
|
|
|
|
|
#include "buf.h" |
16
|
|
|
|
|
|
|
#include "futils.h" |
17
|
|
|
|
|
|
|
#include "hash.h" |
18
|
|
|
|
|
|
|
#include "filter.h" |
19
|
|
|
|
|
|
|
#include "repository.h" |
20
|
|
|
|
|
|
|
|
21
|
|
|
|
|
|
|
typedef enum { |
22
|
|
|
|
|
|
|
GIT_CRLF_UNDEFINED, |
23
|
|
|
|
|
|
|
GIT_CRLF_BINARY, |
24
|
|
|
|
|
|
|
GIT_CRLF_TEXT, |
25
|
|
|
|
|
|
|
GIT_CRLF_TEXT_INPUT, |
26
|
|
|
|
|
|
|
GIT_CRLF_TEXT_CRLF, |
27
|
|
|
|
|
|
|
GIT_CRLF_AUTO, |
28
|
|
|
|
|
|
|
GIT_CRLF_AUTO_INPUT, |
29
|
|
|
|
|
|
|
GIT_CRLF_AUTO_CRLF |
30
|
|
|
|
|
|
|
} git_crlf_t; |
31
|
|
|
|
|
|
|
|
32
|
|
|
|
|
|
|
struct crlf_attrs { |
33
|
|
|
|
|
|
|
int attr_action; /* the .gitattributes setting */ |
34
|
|
|
|
|
|
|
int crlf_action; /* the core.autocrlf setting */ |
35
|
|
|
|
|
|
|
|
36
|
|
|
|
|
|
|
int auto_crlf; |
37
|
|
|
|
|
|
|
int safe_crlf; |
38
|
|
|
|
|
|
|
int core_eol; |
39
|
|
|
|
|
|
|
}; |
40
|
|
|
|
|
|
|
|
41
|
|
|
|
|
|
|
struct crlf_filter { |
42
|
|
|
|
|
|
|
git_filter f; |
43
|
|
|
|
|
|
|
}; |
44
|
|
|
|
|
|
|
|
45
|
520
|
|
|
|
|
|
static git_crlf_t check_crlf(const char *value) |
46
|
|
|
|
|
|
|
{ |
47
|
520
|
50
|
|
|
|
|
if (GIT_ATTR_IS_TRUE(value)) |
48
|
0
|
|
|
|
|
|
return GIT_CRLF_TEXT; |
49
|
520
|
50
|
|
|
|
|
else if (GIT_ATTR_IS_FALSE(value)) |
50
|
0
|
|
|
|
|
|
return GIT_CRLF_BINARY; |
51
|
520
|
50
|
|
|
|
|
else if (GIT_ATTR_IS_UNSPECIFIED(value)) |
52
|
|
|
|
|
|
|
; |
53
|
0
|
0
|
|
|
|
|
else if (strcmp(value, "input") == 0) |
54
|
0
|
|
|
|
|
|
return GIT_CRLF_TEXT_INPUT; |
55
|
0
|
0
|
|
|
|
|
else if (strcmp(value, "auto") == 0) |
56
|
0
|
|
|
|
|
|
return GIT_CRLF_AUTO; |
57
|
|
|
|
|
|
|
|
58
|
520
|
|
|
|
|
|
return GIT_CRLF_UNDEFINED; |
59
|
|
|
|
|
|
|
} |
60
|
|
|
|
|
|
|
|
61
|
260
|
|
|
|
|
|
static git_configmap_value check_eol(const char *value) |
62
|
|
|
|
|
|
|
{ |
63
|
260
|
50
|
|
|
|
|
if (GIT_ATTR_IS_UNSPECIFIED(value)) |
64
|
|
|
|
|
|
|
; |
65
|
0
|
0
|
|
|
|
|
else if (strcmp(value, "lf") == 0) |
66
|
0
|
|
|
|
|
|
return GIT_EOL_LF; |
67
|
0
|
0
|
|
|
|
|
else if (strcmp(value, "crlf") == 0) |
68
|
0
|
|
|
|
|
|
return GIT_EOL_CRLF; |
69
|
|
|
|
|
|
|
|
70
|
260
|
|
|
|
|
|
return GIT_EOL_UNSET; |
71
|
|
|
|
|
|
|
} |
72
|
|
|
|
|
|
|
|
73
|
57
|
|
|
|
|
|
static int has_cr_in_index(const git_filter_source *src) |
74
|
|
|
|
|
|
|
{ |
75
|
57
|
|
|
|
|
|
git_repository *repo = git_filter_source_repo(src); |
76
|
57
|
|
|
|
|
|
const char *path = git_filter_source_path(src); |
77
|
|
|
|
|
|
|
git_index *index; |
78
|
|
|
|
|
|
|
const git_index_entry *entry; |
79
|
|
|
|
|
|
|
git_blob *blob; |
80
|
|
|
|
|
|
|
const void *blobcontent; |
81
|
|
|
|
|
|
|
git_object_size_t blobsize; |
82
|
|
|
|
|
|
|
bool found_cr; |
83
|
|
|
|
|
|
|
|
84
|
57
|
50
|
|
|
|
|
if (!path) |
85
|
0
|
|
|
|
|
|
return false; |
86
|
|
|
|
|
|
|
|
87
|
57
|
50
|
|
|
|
|
if (git_repository_index__weakptr(&index, repo) < 0) { |
88
|
0
|
|
|
|
|
|
git_error_clear(); |
89
|
0
|
|
|
|
|
|
return false; |
90
|
|
|
|
|
|
|
} |
91
|
|
|
|
|
|
|
|
92
|
57
|
100
|
|
|
|
|
if (!(entry = git_index_get_bypath(index, path, 0)) && |
|
|
100
|
|
|
|
|
|
93
|
11
|
|
|
|
|
|
!(entry = git_index_get_bypath(index, path, 1))) |
94
|
10
|
|
|
|
|
|
return false; |
95
|
|
|
|
|
|
|
|
96
|
47
|
50
|
|
|
|
|
if (!S_ISREG(entry->mode)) /* don't crlf filter non-blobs */ |
97
|
0
|
|
|
|
|
|
return true; |
98
|
|
|
|
|
|
|
|
99
|
47
|
50
|
|
|
|
|
if (git_blob_lookup(&blob, repo, &entry->id) < 0) |
100
|
0
|
|
|
|
|
|
return false; |
101
|
|
|
|
|
|
|
|
102
|
47
|
|
|
|
|
|
blobcontent = git_blob_rawcontent(blob); |
103
|
47
|
|
|
|
|
|
blobsize = git_blob_rawsize(blob); |
104
|
47
|
50
|
|
|
|
|
if (!git__is_sizet(blobsize)) |
105
|
0
|
|
|
|
|
|
blobsize = (size_t)-1; |
106
|
|
|
|
|
|
|
|
107
|
47
|
50
|
|
|
|
|
found_cr = (blobcontent != NULL && |
108
|
94
|
50
|
|
|
|
|
blobsize > 0 && |
|
|
50
|
|
|
|
|
|
109
|
47
|
|
|
|
|
|
memchr(blobcontent, '\r', (size_t)blobsize) != NULL); |
110
|
|
|
|
|
|
|
|
111
|
47
|
|
|
|
|
|
git_blob_free(blob); |
112
|
57
|
|
|
|
|
|
return found_cr; |
113
|
|
|
|
|
|
|
} |
114
|
|
|
|
|
|
|
|
115
|
0
|
|
|
|
|
|
static int text_eol_is_crlf(struct crlf_attrs *ca) |
116
|
|
|
|
|
|
|
{ |
117
|
0
|
0
|
|
|
|
|
if (ca->auto_crlf == GIT_AUTO_CRLF_TRUE) |
118
|
0
|
|
|
|
|
|
return 1; |
119
|
0
|
0
|
|
|
|
|
else if (ca->auto_crlf == GIT_AUTO_CRLF_INPUT) |
120
|
0
|
|
|
|
|
|
return 0; |
121
|
|
|
|
|
|
|
|
122
|
0
|
0
|
|
|
|
|
if (ca->core_eol == GIT_EOL_CRLF) |
123
|
0
|
|
|
|
|
|
return 1; |
124
|
|
|
|
|
|
|
if (ca->core_eol == GIT_EOL_UNSET && GIT_EOL_NATIVE == GIT_EOL_CRLF) |
125
|
|
|
|
|
|
|
return 1; |
126
|
|
|
|
|
|
|
|
127
|
0
|
|
|
|
|
|
return 0; |
128
|
|
|
|
|
|
|
} |
129
|
|
|
|
|
|
|
|
130
|
16
|
|
|
|
|
|
static git_configmap_value output_eol(struct crlf_attrs *ca) |
131
|
|
|
|
|
|
|
{ |
132
|
16
|
|
|
|
|
|
switch (ca->crlf_action) { |
133
|
|
|
|
|
|
|
case GIT_CRLF_BINARY: |
134
|
0
|
|
|
|
|
|
return GIT_EOL_UNSET; |
135
|
|
|
|
|
|
|
case GIT_CRLF_TEXT_CRLF: |
136
|
0
|
|
|
|
|
|
return GIT_EOL_CRLF; |
137
|
|
|
|
|
|
|
case GIT_CRLF_TEXT_INPUT: |
138
|
0
|
|
|
|
|
|
return GIT_EOL_LF; |
139
|
|
|
|
|
|
|
case GIT_CRLF_UNDEFINED: |
140
|
|
|
|
|
|
|
case GIT_CRLF_AUTO_CRLF: |
141
|
6
|
|
|
|
|
|
return GIT_EOL_CRLF; |
142
|
|
|
|
|
|
|
case GIT_CRLF_AUTO_INPUT: |
143
|
10
|
|
|
|
|
|
return GIT_EOL_LF; |
144
|
|
|
|
|
|
|
case GIT_CRLF_TEXT: |
145
|
|
|
|
|
|
|
case GIT_CRLF_AUTO: |
146
|
0
|
0
|
|
|
|
|
return text_eol_is_crlf(ca) ? GIT_EOL_CRLF : GIT_EOL_LF; |
147
|
|
|
|
|
|
|
} |
148
|
|
|
|
|
|
|
|
149
|
|
|
|
|
|
|
/* TODO: warn when available */ |
150
|
0
|
|
|
|
|
|
return ca->core_eol; |
151
|
|
|
|
|
|
|
} |
152
|
|
|
|
|
|
|
|
153
|
57
|
|
|
|
|
|
GIT_INLINE(int) check_safecrlf( |
154
|
|
|
|
|
|
|
struct crlf_attrs *ca, |
155
|
|
|
|
|
|
|
const git_filter_source *src, |
156
|
|
|
|
|
|
|
git_str_text_stats *stats) |
157
|
|
|
|
|
|
|
{ |
158
|
57
|
|
|
|
|
|
const char *filename = git_filter_source_path(src); |
159
|
|
|
|
|
|
|
|
160
|
57
|
50
|
|
|
|
|
if (!ca->safe_crlf) |
161
|
57
|
|
|
|
|
|
return 0; |
162
|
|
|
|
|
|
|
|
163
|
0
|
0
|
|
|
|
|
if (output_eol(ca) == GIT_EOL_LF) { |
164
|
|
|
|
|
|
|
/* |
165
|
|
|
|
|
|
|
* CRLFs would not be restored by checkout: |
166
|
|
|
|
|
|
|
* check if we'd remove CRLFs |
167
|
|
|
|
|
|
|
*/ |
168
|
0
|
0
|
|
|
|
|
if (stats->crlf) { |
169
|
0
|
0
|
|
|
|
|
if (ca->safe_crlf == GIT_SAFE_CRLF_WARN) { |
170
|
|
|
|
|
|
|
/* TODO: issue a warning when available */ |
171
|
|
|
|
|
|
|
} else { |
172
|
0
|
0
|
|
|
|
|
if (filename && *filename) |
|
|
0
|
|
|
|
|
|
173
|
0
|
|
|
|
|
|
git_error_set( |
174
|
|
|
|
|
|
|
GIT_ERROR_FILTER, "CRLF would be replaced by LF in '%s'", |
175
|
|
|
|
|
|
|
filename); |
176
|
|
|
|
|
|
|
else |
177
|
0
|
|
|
|
|
|
git_error_set( |
178
|
|
|
|
|
|
|
GIT_ERROR_FILTER, "CRLF would be replaced by LF"); |
179
|
|
|
|
|
|
|
|
180
|
0
|
|
|
|
|
|
return -1; |
181
|
|
|
|
|
|
|
} |
182
|
|
|
|
|
|
|
} |
183
|
0
|
0
|
|
|
|
|
} else if (output_eol(ca) == GIT_EOL_CRLF) { |
184
|
|
|
|
|
|
|
/* |
185
|
|
|
|
|
|
|
* CRLFs would be added by checkout: |
186
|
|
|
|
|
|
|
* check if we have "naked" LFs |
187
|
|
|
|
|
|
|
*/ |
188
|
0
|
0
|
|
|
|
|
if (stats->crlf != stats->lf) { |
189
|
0
|
0
|
|
|
|
|
if (ca->safe_crlf == GIT_SAFE_CRLF_WARN) { |
190
|
|
|
|
|
|
|
/* TODO: issue a warning when available */ |
191
|
|
|
|
|
|
|
} else { |
192
|
0
|
0
|
|
|
|
|
if (filename && *filename) |
|
|
0
|
|
|
|
|
|
193
|
0
|
|
|
|
|
|
git_error_set( |
194
|
|
|
|
|
|
|
GIT_ERROR_FILTER, "LF would be replaced by CRLF in '%s'", |
195
|
|
|
|
|
|
|
filename); |
196
|
|
|
|
|
|
|
else |
197
|
0
|
|
|
|
|
|
git_error_set( |
198
|
|
|
|
|
|
|
GIT_ERROR_FILTER, "LF would be replaced by CRLF"); |
199
|
|
|
|
|
|
|
|
200
|
0
|
|
|
|
|
|
return -1; |
201
|
|
|
|
|
|
|
} |
202
|
|
|
|
|
|
|
} |
203
|
|
|
|
|
|
|
} |
204
|
|
|
|
|
|
|
|
205
|
0
|
|
|
|
|
|
return 0; |
206
|
|
|
|
|
|
|
} |
207
|
|
|
|
|
|
|
|
208
|
59
|
|
|
|
|
|
static int crlf_apply_to_odb( |
209
|
|
|
|
|
|
|
struct crlf_attrs *ca, |
210
|
|
|
|
|
|
|
git_str *to, |
211
|
|
|
|
|
|
|
const git_str *from, |
212
|
|
|
|
|
|
|
const git_filter_source *src) |
213
|
|
|
|
|
|
|
{ |
214
|
|
|
|
|
|
|
git_str_text_stats stats; |
215
|
|
|
|
|
|
|
bool is_binary; |
216
|
|
|
|
|
|
|
int error; |
217
|
|
|
|
|
|
|
|
218
|
|
|
|
|
|
|
/* Binary attribute? Empty file? Nothing to do */ |
219
|
59
|
50
|
|
|
|
|
if (ca->crlf_action == GIT_CRLF_BINARY || from->size == 0) |
|
|
100
|
|
|
|
|
|
220
|
2
|
|
|
|
|
|
return GIT_PASSTHROUGH; |
221
|
|
|
|
|
|
|
|
222
|
57
|
|
|
|
|
|
is_binary = git_str_gather_text_stats(&stats, from, false); |
223
|
|
|
|
|
|
|
|
224
|
|
|
|
|
|
|
/* Heuristics to see if we can skip the conversion. |
225
|
|
|
|
|
|
|
* Straight from Core Git. |
226
|
|
|
|
|
|
|
*/ |
227
|
57
|
50
|
|
|
|
|
if (ca->crlf_action == GIT_CRLF_AUTO || |
|
|
100
|
|
|
|
|
|
228
|
12
|
50
|
|
|
|
|
ca->crlf_action == GIT_CRLF_AUTO_INPUT || |
229
|
12
|
|
|
|
|
|
ca->crlf_action == GIT_CRLF_AUTO_CRLF) { |
230
|
|
|
|
|
|
|
|
231
|
57
|
50
|
|
|
|
|
if (is_binary) |
232
|
0
|
|
|
|
|
|
return GIT_PASSTHROUGH; |
233
|
|
|
|
|
|
|
|
234
|
|
|
|
|
|
|
/* |
235
|
|
|
|
|
|
|
* If the file in the index has any CR in it, do not convert. |
236
|
|
|
|
|
|
|
* This is the new safer autocrlf handling. |
237
|
|
|
|
|
|
|
*/ |
238
|
57
|
50
|
|
|
|
|
if (has_cr_in_index(src)) |
239
|
0
|
|
|
|
|
|
return GIT_PASSTHROUGH; |
240
|
|
|
|
|
|
|
} |
241
|
|
|
|
|
|
|
|
242
|
57
|
50
|
|
|
|
|
if ((error = check_safecrlf(ca, src, &stats)) < 0) |
243
|
0
|
|
|
|
|
|
return error; |
244
|
|
|
|
|
|
|
|
245
|
|
|
|
|
|
|
/* If there are no CR characters to filter out, then just pass */ |
246
|
57
|
100
|
|
|
|
|
if (!stats.crlf) |
247
|
54
|
|
|
|
|
|
return GIT_PASSTHROUGH; |
248
|
|
|
|
|
|
|
|
249
|
|
|
|
|
|
|
/* Actually drop the carriage returns */ |
250
|
59
|
|
|
|
|
|
return git_str_crlf_to_lf(to, from); |
251
|
|
|
|
|
|
|
} |
252
|
|
|
|
|
|
|
|
253
|
16
|
|
|
|
|
|
static int crlf_apply_to_workdir( |
254
|
|
|
|
|
|
|
struct crlf_attrs *ca, |
255
|
|
|
|
|
|
|
git_str *to, |
256
|
|
|
|
|
|
|
const git_str *from) |
257
|
|
|
|
|
|
|
{ |
258
|
|
|
|
|
|
|
git_str_text_stats stats; |
259
|
|
|
|
|
|
|
bool is_binary; |
260
|
|
|
|
|
|
|
|
261
|
|
|
|
|
|
|
/* Empty file? Nothing to do. */ |
262
|
16
|
50
|
|
|
|
|
if (git_str_len(from) == 0 || output_eol(ca) != GIT_EOL_CRLF) |
|
|
100
|
|
|
|
|
|
263
|
10
|
|
|
|
|
|
return GIT_PASSTHROUGH; |
264
|
|
|
|
|
|
|
|
265
|
6
|
|
|
|
|
|
is_binary = git_str_gather_text_stats(&stats, from, false); |
266
|
|
|
|
|
|
|
|
267
|
|
|
|
|
|
|
/* If there are no LFs, or all LFs are part of a CRLF, nothing to do */ |
268
|
6
|
100
|
|
|
|
|
if (stats.lf == 0 || stats.lf == stats.crlf) |
|
|
50
|
|
|
|
|
|
269
|
3
|
|
|
|
|
|
return GIT_PASSTHROUGH; |
270
|
|
|
|
|
|
|
|
271
|
3
|
50
|
|
|
|
|
if (ca->crlf_action == GIT_CRLF_AUTO || |
|
|
50
|
|
|
|
|
|
272
|
3
|
50
|
|
|
|
|
ca->crlf_action == GIT_CRLF_AUTO_INPUT || |
273
|
3
|
|
|
|
|
|
ca->crlf_action == GIT_CRLF_AUTO_CRLF) { |
274
|
|
|
|
|
|
|
|
275
|
|
|
|
|
|
|
/* If we have any existing CR or CRLF line endings, do nothing */ |
276
|
3
|
50
|
|
|
|
|
if (stats.cr > 0) |
277
|
0
|
|
|
|
|
|
return GIT_PASSTHROUGH; |
278
|
|
|
|
|
|
|
|
279
|
|
|
|
|
|
|
/* Don't filter binary files */ |
280
|
3
|
50
|
|
|
|
|
if (is_binary) |
281
|
0
|
|
|
|
|
|
return GIT_PASSTHROUGH; |
282
|
|
|
|
|
|
|
} |
283
|
|
|
|
|
|
|
|
284
|
16
|
|
|
|
|
|
return git_str_lf_to_crlf(to, from); |
285
|
|
|
|
|
|
|
} |
286
|
|
|
|
|
|
|
|
287
|
260
|
|
|
|
|
|
static int convert_attrs( |
288
|
|
|
|
|
|
|
struct crlf_attrs *ca, |
289
|
|
|
|
|
|
|
const char **attr_values, |
290
|
|
|
|
|
|
|
const git_filter_source *src) |
291
|
|
|
|
|
|
|
{ |
292
|
|
|
|
|
|
|
int error; |
293
|
|
|
|
|
|
|
|
294
|
260
|
|
|
|
|
|
memset(ca, 0, sizeof(struct crlf_attrs)); |
295
|
|
|
|
|
|
|
|
296
|
260
|
50
|
|
|
|
|
if ((error = git_repository__configmap_lookup(&ca->auto_crlf, |
297
|
260
|
50
|
|
|
|
|
git_filter_source_repo(src), GIT_CONFIGMAP_AUTO_CRLF)) < 0 || |
298
|
260
|
|
|
|
|
|
(error = git_repository__configmap_lookup(&ca->safe_crlf, |
299
|
260
|
50
|
|
|
|
|
git_filter_source_repo(src), GIT_CONFIGMAP_SAFE_CRLF)) < 0 || |
300
|
260
|
|
|
|
|
|
(error = git_repository__configmap_lookup(&ca->core_eol, |
301
|
|
|
|
|
|
|
git_filter_source_repo(src), GIT_CONFIGMAP_EOL)) < 0) |
302
|
0
|
|
|
|
|
|
return error; |
303
|
|
|
|
|
|
|
|
304
|
|
|
|
|
|
|
/* downgrade FAIL to WARN if ALLOW_UNSAFE option is used */ |
305
|
260
|
100
|
|
|
|
|
if ((git_filter_source_flags(src) & GIT_FILTER_ALLOW_UNSAFE) && |
|
|
50
|
|
|
|
|
|
306
|
160
|
|
|
|
|
|
ca->safe_crlf == GIT_SAFE_CRLF_FAIL) |
307
|
0
|
|
|
|
|
|
ca->safe_crlf = GIT_SAFE_CRLF_WARN; |
308
|
|
|
|
|
|
|
|
309
|
260
|
50
|
|
|
|
|
if (attr_values) { |
310
|
|
|
|
|
|
|
/* load the text attribute */ |
311
|
260
|
|
|
|
|
|
ca->crlf_action = check_crlf(attr_values[2]); /* text */ |
312
|
|
|
|
|
|
|
|
313
|
260
|
50
|
|
|
|
|
if (ca->crlf_action == GIT_CRLF_UNDEFINED) |
314
|
260
|
|
|
|
|
|
ca->crlf_action = check_crlf(attr_values[0]); /* crlf */ |
315
|
|
|
|
|
|
|
|
316
|
260
|
50
|
|
|
|
|
if (ca->crlf_action != GIT_CRLF_BINARY) { |
317
|
|
|
|
|
|
|
/* load the eol attribute */ |
318
|
260
|
|
|
|
|
|
int eol_attr = check_eol(attr_values[1]); |
319
|
|
|
|
|
|
|
|
320
|
260
|
50
|
|
|
|
|
if (ca->crlf_action == GIT_CRLF_AUTO && eol_attr == GIT_EOL_LF) |
|
|
0
|
|
|
|
|
|
321
|
0
|
|
|
|
|
|
ca->crlf_action = GIT_CRLF_AUTO_INPUT; |
322
|
260
|
50
|
|
|
|
|
else if (ca->crlf_action == GIT_CRLF_AUTO && eol_attr == GIT_EOL_CRLF) |
|
|
0
|
|
|
|
|
|
323
|
0
|
|
|
|
|
|
ca->crlf_action = GIT_CRLF_AUTO_CRLF; |
324
|
260
|
50
|
|
|
|
|
else if (eol_attr == GIT_EOL_LF) |
325
|
0
|
|
|
|
|
|
ca->crlf_action = GIT_CRLF_TEXT_INPUT; |
326
|
260
|
50
|
|
|
|
|
else if (eol_attr == GIT_EOL_CRLF) |
327
|
0
|
|
|
|
|
|
ca->crlf_action = GIT_CRLF_TEXT_CRLF; |
328
|
|
|
|
|
|
|
} |
329
|
|
|
|
|
|
|
|
330
|
260
|
|
|
|
|
|
ca->attr_action = ca->crlf_action; |
331
|
|
|
|
|
|
|
} else { |
332
|
0
|
|
|
|
|
|
ca->crlf_action = GIT_CRLF_UNDEFINED; |
333
|
|
|
|
|
|
|
} |
334
|
|
|
|
|
|
|
|
335
|
260
|
50
|
|
|
|
|
if (ca->crlf_action == GIT_CRLF_TEXT) |
336
|
0
|
0
|
|
|
|
|
ca->crlf_action = text_eol_is_crlf(ca) ? GIT_CRLF_TEXT_CRLF : GIT_CRLF_TEXT_INPUT; |
337
|
260
|
50
|
|
|
|
|
if (ca->crlf_action == GIT_CRLF_UNDEFINED && ca->auto_crlf == GIT_AUTO_CRLF_FALSE) |
|
|
100
|
|
|
|
|
|
338
|
189
|
|
|
|
|
|
ca->crlf_action = GIT_CRLF_BINARY; |
339
|
260
|
100
|
|
|
|
|
if (ca->crlf_action == GIT_CRLF_UNDEFINED && ca->auto_crlf == GIT_AUTO_CRLF_TRUE) |
|
|
100
|
|
|
|
|
|
340
|
16
|
|
|
|
|
|
ca->crlf_action = GIT_CRLF_AUTO_CRLF; |
341
|
260
|
100
|
|
|
|
|
if (ca->crlf_action == GIT_CRLF_UNDEFINED && ca->auto_crlf == GIT_AUTO_CRLF_INPUT) |
|
|
50
|
|
|
|
|
|
342
|
55
|
|
|
|
|
|
ca->crlf_action = GIT_CRLF_AUTO_INPUT; |
343
|
|
|
|
|
|
|
|
344
|
260
|
|
|
|
|
|
return 0; |
345
|
|
|
|
|
|
|
} |
346
|
|
|
|
|
|
|
|
347
|
260
|
|
|
|
|
|
static int crlf_check( |
348
|
|
|
|
|
|
|
git_filter *self, |
349
|
|
|
|
|
|
|
void **payload, /* points to NULL ptr on entry, may be set */ |
350
|
|
|
|
|
|
|
const git_filter_source *src, |
351
|
|
|
|
|
|
|
const char **attr_values) |
352
|
|
|
|
|
|
|
{ |
353
|
|
|
|
|
|
|
struct crlf_attrs ca; |
354
|
|
|
|
|
|
|
|
355
|
260
|
|
|
|
|
|
GIT_UNUSED(self); |
356
|
|
|
|
|
|
|
|
357
|
260
|
|
|
|
|
|
convert_attrs(&ca, attr_values, src); |
358
|
|
|
|
|
|
|
|
359
|
260
|
100
|
|
|
|
|
if (ca.crlf_action == GIT_CRLF_BINARY) |
360
|
189
|
|
|
|
|
|
return GIT_PASSTHROUGH; |
361
|
|
|
|
|
|
|
|
362
|
71
|
|
|
|
|
|
*payload = git__malloc(sizeof(ca)); |
363
|
71
|
50
|
|
|
|
|
GIT_ERROR_CHECK_ALLOC(*payload); |
364
|
71
|
|
|
|
|
|
memcpy(*payload, &ca, sizeof(ca)); |
365
|
|
|
|
|
|
|
|
366
|
260
|
|
|
|
|
|
return 0; |
367
|
|
|
|
|
|
|
} |
368
|
|
|
|
|
|
|
|
369
|
75
|
|
|
|
|
|
static int crlf_apply( |
370
|
|
|
|
|
|
|
git_filter *self, |
371
|
|
|
|
|
|
|
void **payload, /* may be read and/or set */ |
372
|
|
|
|
|
|
|
git_str *to, |
373
|
|
|
|
|
|
|
const git_str *from, |
374
|
|
|
|
|
|
|
const git_filter_source *src) |
375
|
|
|
|
|
|
|
{ |
376
|
75
|
|
|
|
|
|
int error = 0; |
377
|
|
|
|
|
|
|
|
378
|
|
|
|
|
|
|
/* initialize payload in case `check` was bypassed */ |
379
|
75
|
50
|
|
|
|
|
if (!*payload) { |
380
|
0
|
0
|
|
|
|
|
if ((error = crlf_check(self, payload, src, NULL)) < 0) |
381
|
0
|
|
|
|
|
|
return error; |
382
|
|
|
|
|
|
|
} |
383
|
|
|
|
|
|
|
|
384
|
75
|
100
|
|
|
|
|
if (git_filter_source_mode(src) == GIT_FILTER_SMUDGE) |
385
|
16
|
|
|
|
|
|
error = crlf_apply_to_workdir(*payload, to, from); |
386
|
|
|
|
|
|
|
else |
387
|
59
|
|
|
|
|
|
error = crlf_apply_to_odb(*payload, to, from, src); |
388
|
|
|
|
|
|
|
|
389
|
75
|
|
|
|
|
|
return error; |
390
|
|
|
|
|
|
|
} |
391
|
|
|
|
|
|
|
|
392
|
75
|
|
|
|
|
|
static int crlf_stream( |
393
|
|
|
|
|
|
|
git_writestream **out, |
394
|
|
|
|
|
|
|
git_filter *self, |
395
|
|
|
|
|
|
|
void **payload, |
396
|
|
|
|
|
|
|
const git_filter_source *src, |
397
|
|
|
|
|
|
|
git_writestream *next) |
398
|
|
|
|
|
|
|
{ |
399
|
75
|
|
|
|
|
|
return git_filter_buffered_stream_new(out, |
400
|
|
|
|
|
|
|
self, crlf_apply, NULL, payload, src, next); |
401
|
|
|
|
|
|
|
} |
402
|
|
|
|
|
|
|
|
403
|
71
|
|
|
|
|
|
static void crlf_cleanup( |
404
|
|
|
|
|
|
|
git_filter *self, |
405
|
|
|
|
|
|
|
void *payload) |
406
|
|
|
|
|
|
|
{ |
407
|
71
|
|
|
|
|
|
GIT_UNUSED(self); |
408
|
71
|
|
|
|
|
|
git__free(payload); |
409
|
71
|
|
|
|
|
|
} |
410
|
|
|
|
|
|
|
|
411
|
87
|
|
|
|
|
|
git_filter *git_crlf_filter_new(void) |
412
|
|
|
|
|
|
|
{ |
413
|
87
|
|
|
|
|
|
struct crlf_filter *f = git__calloc(1, sizeof(struct crlf_filter)); |
414
|
87
|
50
|
|
|
|
|
if (f == NULL) |
415
|
0
|
|
|
|
|
|
return NULL; |
416
|
|
|
|
|
|
|
|
417
|
87
|
|
|
|
|
|
f->f.version = GIT_FILTER_VERSION; |
418
|
87
|
|
|
|
|
|
f->f.attributes = "crlf eol text"; |
419
|
87
|
|
|
|
|
|
f->f.initialize = NULL; |
420
|
87
|
|
|
|
|
|
f->f.shutdown = git_filter_free; |
421
|
87
|
|
|
|
|
|
f->f.check = crlf_check; |
422
|
87
|
|
|
|
|
|
f->f.stream = crlf_stream; |
423
|
87
|
|
|
|
|
|
f->f.cleanup = crlf_cleanup; |
424
|
|
|
|
|
|
|
|
425
|
87
|
|
|
|
|
|
return (git_filter *)f; |
426
|
|
|
|
|
|
|
} |