Skip to content

Commit 3bb8d69

Browse files
committed
Merge branch 'cn/maint-lf-to-crlf-filter' into maint
* cn/maint-lf-to-crlf-filter: lf_to_crlf_filter(): tell the caller we added "\n" when draining convert: track state in LF-to-CRLF filter
2 parents 1a7bd4f + 87afe9a commit 3bb8d69

File tree

1 file changed

+40
-14
lines changed

1 file changed

+40
-14
lines changed

convert.c

Lines changed: 40 additions & 14 deletions
Original file line numberDiff line numberDiff line change
@@ -876,24 +876,42 @@ int is_null_stream_filter(struct stream_filter *filter)
876876
/*
877877
* LF-to-CRLF filter
878878
*/
879+
880+
struct lf_to_crlf_filter {
881+
struct stream_filter filter;
882+
unsigned want_lf:1;
883+
};
884+
879885
static int lf_to_crlf_filter_fn(struct stream_filter *filter,
880886
const char *input, size_t *isize_p,
881887
char *output, size_t *osize_p)
882888
{
883-
size_t count;
889+
size_t count, o = 0;
890+
struct lf_to_crlf_filter *lf_to_crlf = (struct lf_to_crlf_filter *)filter;
891+
892+
/* Output a pending LF if we need to */
893+
if (lf_to_crlf->want_lf) {
894+
output[o++] = '\n';
895+
lf_to_crlf->want_lf = 0;
896+
}
897+
898+
/* We are told to drain */
899+
if (!input) {
900+
*osize_p -= o;
901+
return 0;
902+
}
884903

885-
if (!input)
886-
return 0; /* we do not keep any states */
887904
count = *isize_p;
888905
if (count) {
889-
size_t i, o;
890-
for (i = o = 0; o < *osize_p && i < count; i++) {
906+
size_t i;
907+
for (i = 0; o < *osize_p && i < count; i++) {
891908
char ch = input[i];
892909
if (ch == '\n') {
893-
if (o + 1 < *osize_p)
894-
output[o++] = '\r';
895-
else
896-
break;
910+
output[o++] = '\r';
911+
if (o >= *osize_p) {
912+
lf_to_crlf->want_lf = 1;
913+
continue; /* We need to increase i */
914+
}
897915
}
898916
output[o++] = ch;
899917
}
@@ -904,15 +922,23 @@ static int lf_to_crlf_filter_fn(struct stream_filter *filter,
904922
return 0;
905923
}
906924

925+
static void lf_to_crlf_free_fn(struct stream_filter *filter)
926+
{
927+
free(filter);
928+
}
929+
907930
static struct stream_filter_vtbl lf_to_crlf_vtbl = {
908931
lf_to_crlf_filter_fn,
909-
null_free_fn,
932+
lf_to_crlf_free_fn,
910933
};
911934

912-
static struct stream_filter lf_to_crlf_filter_singleton = {
913-
&lf_to_crlf_vtbl,
914-
};
935+
static struct stream_filter *lf_to_crlf_filter(void)
936+
{
937+
struct lf_to_crlf_filter *lf_to_crlf = xcalloc(1, sizeof(*lf_to_crlf));
915938

939+
lf_to_crlf->filter.vtbl = &lf_to_crlf_vtbl;
940+
return (struct stream_filter *)lf_to_crlf;
941+
}
916942

917943
/*
918944
* Cascade filter
@@ -1194,7 +1220,7 @@ struct stream_filter *get_stream_filter(const char *path, const unsigned char *s
11941220

11951221
else if (output_eol(crlf_action) == EOL_CRLF &&
11961222
!(crlf_action == CRLF_AUTO || crlf_action == CRLF_GUESS))
1197-
filter = cascade_filter(filter, &lf_to_crlf_filter_singleton);
1223+
filter = cascade_filter(filter, lf_to_crlf_filter());
11981224

11991225
return filter;
12001226
}

0 commit comments

Comments
 (0)