Merge branch 'mm/line-log-limited-ops' into seen

"git log -L<range>:<path>" learned to limit various "diff" operations
like --stat, --check, -G, to the specified range:path.

* mm/line-log-limited-ops:
  diffcore-pickaxe: scope -G to the -L tracked range
  diff: support --check with -L line ranges
  line-log: support diff stat formats with -L
  diff: extract a line-range diff helper for reuse
  diff: emit -L hunk headers via xdiff's formatter
  diff: simplify the line-range filter by classifying removals immediately
  diff: rename and group the line-range filter for clarity
seen
Junio C Hamano 2026-07-01 11:10:53 -07:00
commit 49c15eb4e4
11 changed files with 827 additions and 228 deletions

View File

@ -9,13 +9,20 @@
_<start>_ and _<end>_ (or _<funcname>_) must exist in the starting revision.
You can specify this option more than once. Implies `--patch`.
Patch output can be suppressed using `--no-patch`.
Non-patch diff formats `--raw`, `--name-only`, `--name-status`,
and `--summary` are supported. Diff stat formats
(`--stat`, `--numstat`, `--shortstat`, `--dirstat`) are not
currently implemented.
The following non-patch diff formats are supported: `--raw`,
`--name-only`, `--name-status`, `--summary`, `--check`,
`--stat`, `--numstat`, and `--shortstat`.
The stat formats count only lines within the tracked range.
`--dirstat` is not supported
with `-L`: it summarizes change as each directory's share of
the total churn, not as counts for the tracked lines. Use
`--numstat` for exact per-file counts within the range.
+
Patch formatting options such as `--word-diff`, `--color-moved`,
`--no-prefix`, and whitespace options (`-w`, `-b`) are supported,
as are pickaxe options (`-S`, `-G`) and `--diff-filter`.
as are pickaxe options (`-S`, `-G`) and `--diff-filter`. `-G` is
scoped to the tracked range; `-S` is still evaluated over the whole
file, so an `-S` query may select a commit for a change outside the
range.
+
include::line-range-format.adoc[]

493
diff.c
View File

@ -611,28 +611,74 @@ struct emit_callback {
};

/*
* State for the line-range callback wrappers that sit between
* xdi_diff_outf() and fn_out_consume(). xdiff produces a normal,
* unfiltered diff; the wrappers intercept each hunk header and line,
* track post-image position, and forward only lines that fall within
* the requested ranges. Contiguous in-range lines are collected into
* range hunks and flushed with a synthetic @@ header so that
* fn_out_consume() sees well-formed unified-diff fragments.
* Line-range filter: scopes "git log -L" output to the tracked ranges.
*
* Removal lines ('-') cannot be classified by post-image position, so
* they are buffered in pending_rm until the next '+' or ' ' line
* reveals whether they precede an in-range line (flush into range hunk) or
* an out-of-range line (discard).
* It sits between xdi_diff_outf() and an output callback (fn_out_consume,
* diffstat_consume, checkdiff_consume). xdiff produces a normal diff; the
* filter forwards only the lines inside the requested ranges, collecting
* contiguous in-range lines into a "range hunk" emitted with a synthetic
* @@ header so the callback sees well-formed unified-diff fragments.
*
* A diff describes the change from a pre-image to a post-image. Each
* line is context (' ', in both), a removal ('-', pre-image only), or
* an addition ('+', post-image only). -L tracks ranges in the
* post-image, so a line is in range by its post-image position.
*
* Two 1-based cursors track the next line in each image, named as in
* struct emit_callback and seeded from the xdiff hunk header:
*
* lno_in_postimage advances on '+' and ' ' (lines in the post-image)
* lno_in_preimage advances on '-' and ' ' (lines in the pre-image)
*
* Ranges are 0-based half-open [start, end), so a line is tested at the
* 0-based index idx_in_postimage = lno_in_postimage - 1.
*
* A '-' is not present in the post-image, so it has no post-image line
* number of its own. Since it does not advance lno_in_postimage, it is
* classified at the idx_in_postimage that the following '+'/' ' will
* occupy. xdiff emits a change's removals before its additions, so that
* index is already known when the '-' arrives.
*
* The synthetic "@@ -<old> +<new> @@" header has two sides, old (the
* pre-image) and new (the post-image), matching the xdiff_emit_hunk_fn
* callback; the hunk.old_begin / hunk.new_begin fields below hold those
* begins, and flush_range_hunk() derives the counts from the buffered
* lines.
*
* Example, tracking post-image line 2 (range [1, 2)) of:
*
* pre-image post-image
* 1 a 1 a
* 2 b 2 X (b -> X)
* 3 c 3 c
*
* classify each line by idx_in_postimage. The pre and post columns
* are each cursor's value while that line is classified, i.e. before
* the line advances them (pre = lno_in_preimage,
* post = lno_in_postimage, idx = idx_in_postimage):
* ' a' pre 1 post 1 idx 0 -> before start, skip
* '-b' pre 2 post 2 idx 1 -> keep (removal)
* '+X' pre 3 post 2 idx 1 -> keep (addition)
* ' c' pre 3 post 3 idx 2 -> past end, flush
*
* -b and +X share idx = 1 because -b did not advance lno_in_postimage;
* both land in the range hunk, flushed when ' c' crosses the range end.
*/
struct line_range_callback {
struct line_range_filter {
xdiff_emit_line_fn orig_line_fn;
/*
* Optional; consumers that report file line numbers (e.g.
* checkdiff) need the synthetic hunk header to set their
* post-image position before in-range lines are replayed.
*/
xdiff_emit_hunk_fn orig_hunk_fn;
void *orig_cb_data;
const struct range_set *ranges; /* 0-based [start, end) */
unsigned int cur_range; /* index into the range_set */

/* Post/pre-image line counters (1-based, set from hunk headers) */
long lno_post;
long lno_pre;
long lno_in_postimage;
long lno_in_preimage;

/*
* Function name from most recent xdiff hunk header;
@ -641,17 +687,17 @@ struct line_range_callback {
char func[80];
long funclen;

/* Range hunk being accumulated for the current range */
struct strbuf rhunk;
long rhunk_old_begin, rhunk_old_count;
long rhunk_new_begin, rhunk_new_count;
int rhunk_active;
int rhunk_has_changes; /* any '+' or '-' lines? */

/* Removal lines not yet known to be in-range */
struct strbuf pending_rm;
int pending_rm_count;
long pending_rm_pre_begin; /* pre-image line of first pending */
/*
* The range hunk being accumulated. At most one is live at a time:
* it is flushed and reset as the cursor leaves each range (and once
* more at end of diff), then reused for the next range.
*/
struct {
struct strbuf lines; /* buffered in-range diff lines */
long old_begin;
long new_begin;
int active;
} hunk;

int ret; /* latched error from orig_line_fn */
};
@ -2541,26 +2587,60 @@ static int quick_consume(void *priv, char *line UNUSED, unsigned long len UNUSED
return 1;
}

static void discard_pending_rm(struct line_range_callback *s)
static void line_range_filter_init(struct line_range_filter *filter,
const struct range_set *ranges,
xdiff_emit_line_fn line_fn,
void *cb_data)
{
strbuf_reset(&s->pending_rm);
s->pending_rm_count = 0;
memset(filter, 0, sizeof(*filter));
filter->orig_line_fn = line_fn;
filter->orig_cb_data = cb_data;
filter->ranges = ranges;
strbuf_init(&filter->hunk.lines, 0);
}

static void flush_rhunk(struct line_range_callback *s)
/*
* Begin a range hunk at the first in-range line. Its position fixes the
* hunk's begins, taken from the two image cursors before they advance:
* new_begin from the post-image, old_begin from the pre-image. The line
* counts are not tracked here; flush_range_hunk() derives them from the
* buffered lines.
*/
static void begin_range_hunk(struct line_range_filter *filter)
{
filter->hunk.active = 1;
filter->hunk.new_begin = filter->lno_in_postimage;
filter->hunk.old_begin = filter->lno_in_preimage;
strbuf_reset(&filter->hunk.lines);
}

static void flush_range_hunk(struct line_range_filter *filter)
{
struct strbuf hdr = STRBUF_INIT;
const char *p, *end;
long old_count = 0, new_count = 0;
int has_changes = 0;

if (!s->rhunk_active || s->ret)
if (!filter->hunk.active || filter->ret)
return;

/* Drain any pending removal lines into the range hunk */
if (s->pending_rm_count) {
strbuf_addbuf(&s->rhunk, &s->pending_rm);
s->rhunk_old_count += s->pending_rm_count;
s->rhunk_has_changes = 1;
discard_pending_rm(s);
/*
* Derive the hunk's geometry from the buffered lines: a ' '
* counts on both sides, a '-' on the old side, a '+' on the new.
* A '-' or '+' marks a real change; the "\ No newline at end of
* file" marker (line[0] == '\\') counts on neither side.
*/
p = filter->hunk.lines.buf;
end = p + filter->hunk.lines.len;
while (p < end) {
const char *eol = memchr(p, '\n', end - p);
if (*p == ' ' || *p == '-')
old_count++;
if (*p == ' ' || *p == '+')
new_count++;
if (*p == '-' || *p == '+')
has_changes = 1;
p = eol ? eol + 1 : end;
}

/*
@ -2569,22 +2649,28 @@ static void flush_rhunk(struct line_range_callback *s)
* ctxlen causes xdiff to emit context covering a range that
* has no changes in this commit.
*/
if (!s->rhunk_has_changes) {
s->rhunk_active = 0;
strbuf_reset(&s->rhunk);
if (!has_changes) {
filter->hunk.active = 0;
strbuf_reset(&filter->hunk.lines);
return;
}

strbuf_addf(&hdr, "@@ -%ld,%ld +%ld,%ld @@",
s->rhunk_old_begin, s->rhunk_old_count,
s->rhunk_new_begin, s->rhunk_new_count);
if (s->funclen > 0) {
strbuf_addch(&hdr, ' ');
strbuf_add(&hdr, s->func, s->funclen);
}
strbuf_addch(&hdr, '\n');
xdiff_emit_hunk_header(&hdr, filter->hunk.old_begin, old_count,
filter->hunk.new_begin, new_count,
filter->func, filter->funclen);

s->ret = s->orig_line_fn(s->orig_cb_data, hdr.buf, hdr.len);
/*
* Inform a line-numbering consumer of the post-image position
* before replaying lines, mirroring the hunk callback xdiff
* would have issued for a non-scoped diff.
*/
if (filter->orig_hunk_fn)
filter->orig_hunk_fn(filter->orig_cb_data,
filter->hunk.old_begin, old_count,
filter->hunk.new_begin, new_count,
filter->func, filter->funclen);

filter->ret = filter->orig_line_fn(filter->orig_cb_data, hdr.buf, hdr.len);
strbuf_release(&hdr);

/*
@ -2592,135 +2678,159 @@ static void flush_rhunk(struct line_range_callback *s)
* The cast discards const because xdiff_emit_line_fn takes
* char *, though fn_out_consume does not modify the buffer.
*/
p = s->rhunk.buf;
end = p + s->rhunk.len;
while (!s->ret && p < end) {
p = filter->hunk.lines.buf;
end = p + filter->hunk.lines.len;
while (!filter->ret && p < end) {
const char *eol = memchr(p, '\n', end - p);
unsigned long line_len = eol ? (unsigned long)(eol - p + 1)
: (unsigned long)(end - p);
s->ret = s->orig_line_fn(s->orig_cb_data, (char *)p, line_len);
filter->ret = filter->orig_line_fn(filter->orig_cb_data, (char *)p, line_len);
p += line_len;
}

s->rhunk_active = 0;
strbuf_reset(&s->rhunk);
filter->hunk.active = 0;
strbuf_reset(&filter->hunk.lines);
}

static void line_range_hunk_fn(void *data,
long old_begin, long old_nr UNUSED,
long new_begin, long new_nr UNUSED,
long old_begin, long old_nr,
long new_begin, long new_nr,
const char *func, long funclen)
{
struct line_range_callback *s = data;
struct line_range_filter *filter = data;

/*
* When count > 0, begin is 1-based. When count == 0, begin is
* adjusted down by 1 by xdl_emit_hunk_hdr(), but no lines of
* that type will arrive, so the value is unused.
*
* Any pending removal lines from the previous xdiff hunk are
* intentionally left in pending_rm: the line callback will
* flush or discard them when the next content line reveals
* whether the removals precede in-range content.
* Seed the per-image line cursors from the hunk header's begins. For
* a side with no lines (count 0), xdiff's callback has already moved
* its begin to the line before the change, so add one back to recover
* the true 1-based start. xdiff_emit_hunk_header() reapplies that -1
* when the clipped hunk is emitted.
*/
s->lno_post = new_begin;
s->lno_pre = old_begin;
filter->lno_in_postimage = new_nr ? new_begin : new_begin + 1;
filter->lno_in_preimage = old_nr ? old_begin : old_begin + 1;

if (funclen > 0) {
if (funclen > (long)sizeof(s->func))
funclen = sizeof(s->func);
memcpy(s->func, func, funclen);
if (funclen > (long)sizeof(filter->func))
funclen = sizeof(filter->func);
memcpy(filter->func, func, funclen);
}
s->funclen = funclen;
filter->funclen = funclen;
}

static int line_range_line_fn(void *priv, char *line, unsigned long len)
{
struct line_range_callback *s = priv;
const struct range *cur;
long lno_0, cur_pre;
struct line_range_filter *filter = priv;
long idx_in_postimage;
int in_range;

if (s->ret)
return s->ret;

if (line[0] == '-') {
if (!s->pending_rm_count)
s->pending_rm_pre_begin = s->lno_pre;
s->lno_pre++;
strbuf_add(&s->pending_rm, line, len);
s->pending_rm_count++;
return s->ret;
}
if (filter->ret)
return filter->ret;

if (line[0] == '\\') {
if (s->pending_rm_count)
strbuf_add(&s->pending_rm, line, len);
else if (s->rhunk_active)
strbuf_add(&s->rhunk, line, len);
/* otherwise outside tracked range; drop silently */
return s->ret;
if (filter->hunk.active)
strbuf_add(&filter->hunk.lines, line, len);
return filter->ret;
}

if (line[0] != '+' && line[0] != ' ')
if (line[0] != '+' && line[0] != ' ' && line[0] != '-')
BUG("unexpected diff line type '%c'", line[0]);

lno_0 = s->lno_post - 1;
cur_pre = s->lno_pre; /* save before advancing for context lines */
s->lno_post++;
if (line[0] == ' ')
s->lno_pre++;
/*
* idx_in_postimage is this line's 0-based post-image index (see the model on
* struct line_range_filter). The cursors are advanced only after
* the line is classified, so a '-' is tested at the same idx_in_postimage as
* the '+'/' ' that follows it.
*/
idx_in_postimage = filter->lno_in_postimage - 1;

/* Advance past ranges we've passed */
while (s->cur_range < s->ranges->nr &&
lno_0 >= s->ranges->ranges[s->cur_range].end) {
if (s->rhunk_active)
flush_rhunk(s);
discard_pending_rm(s);
s->cur_range++;
/* Retire ranges we have passed, flushing the one we leave. */
while (filter->cur_range < filter->ranges->nr &&
idx_in_postimage >= filter->ranges->ranges[filter->cur_range].end) {
if (filter->hunk.active)
flush_range_hunk(filter);
filter->cur_range++;
}

/* Past all ranges */
if (s->cur_range >= s->ranges->nr) {
discard_pending_rm(s);
return s->ret;
in_range = filter->cur_range < filter->ranges->nr &&
idx_in_postimage >= filter->ranges->ranges[filter->cur_range].start &&
idx_in_postimage < filter->ranges->ranges[filter->cur_range].end;

if (in_range) {
if (!filter->hunk.active)
begin_range_hunk(filter);

strbuf_add(&filter->hunk.lines, line, len);
}

cur = &s->ranges->ranges[s->cur_range];
/*
* Advance each image's cursor: a line present in that image (see
* the model) consumes one of its line numbers.
*/
if (line[0] != '-')
filter->lno_in_postimage++;
if (line[0] != '+')
filter->lno_in_preimage++;

/* Before current range */
if (lno_0 < cur->start) {
discard_pending_rm(s);
return s->ret;
return filter->ret;
}

/*
* Run an xdiff pass through an initialized line-range filter, flush the
* final range hunk, and release the filter. Inflates ctxlen to the largest
* range span first, so that every change within a single range lands in one
* xdiff hunk and the inter-change context is emitted; the filter then clips
* back to range boundaries. The optimal ctxlen depends on where changes fall
* within the range, which is only known after xdiff runs, so the max span is
* the upper bound that guarantees correctness in a single pass. Every
* consumer (patch, diffstat, check) relies on one xdiff hunk per range, so
* this lives here rather than at each call site. Also clears
* XDL_EMIT_NO_HUNK_HDR: the filter seeds its per-image position from the hunk
* headers, so a consumer that otherwise suppresses them (diffstat) still gets
* them here. Returns non-zero if xdiff or any forwarded callback failed.
*/
static int line_range_filter_diff(struct line_range_filter *filter,
mmfile_t *mf1, mmfile_t *mf2,
xpparam_t *xpp, xdemitconf_t *xecfg)
{
const struct range_set *ranges = filter->ranges;
long max_span = 0;
unsigned int i;
int ret;

for (i = 0; i < ranges->nr; i++) {
long span = ranges->ranges[i].end - ranges->ranges[i].start;
if (span > max_span)
max_span = span;
}
if (max_span > xecfg->ctxlen)
xecfg->ctxlen = max_span;

/* In range so start a new range hunk if needed */
if (!s->rhunk_active) {
s->rhunk_active = 1;
s->rhunk_has_changes = 0;
s->rhunk_new_begin = lno_0 + 1;
s->rhunk_old_begin = s->pending_rm_count
? s->pending_rm_pre_begin : cur_pre;
s->rhunk_old_count = 0;
s->rhunk_new_count = 0;
strbuf_reset(&s->rhunk);
/* the filter seeds its per-image position from hunk headers */
xecfg->flags &= ~XDL_EMIT_NO_HUNK_HDR;

ret = xdi_diff_outf(mf1, mf2, line_range_hunk_fn,
line_range_line_fn, filter, xpp, xecfg);
if (!ret) {
flush_range_hunk(filter);
ret = filter->ret;
}
strbuf_release(&filter->hunk.lines);
return ret;
}

/* Flush pending removals into range hunk */
if (s->pending_rm_count) {
strbuf_addbuf(&s->rhunk, &s->pending_rm);
s->rhunk_old_count += s->pending_rm_count;
s->rhunk_has_changes = 1;
discard_pending_rm(s);
}
/*
* Expose the in-file line-range filter to callers outside diff.c (e.g.
* pickaxe -G); see xdiff-interface.h for the contract.
*/
int diff_emit_line_ranges(mmfile_t *one, mmfile_t *two,
const struct range_set *ranges,
xdiff_emit_line_fn line_fn, void *cb_data,
xpparam_t *xpp, xdemitconf_t *xecfg)
{
struct line_range_filter filter;

strbuf_add(&s->rhunk, line, len);
s->rhunk_new_count++;
if (line[0] == '+')
s->rhunk_has_changes = 1;
else
s->rhunk_old_count++;

return s->ret;
line_range_filter_init(&filter, ranges, line_fn, cb_data);
return line_range_filter_diff(&filter, one, two, xpp, xecfg);
}

static void pprint_rename(struct strbuf *name, const char *a, const char *b)
@ -4099,51 +4209,15 @@ static void builtin_diff(const char *name_a,
xdi_diff_outf(&mf1, &mf2, NULL, quick_consume,
&ecbdata, &xpp, &xecfg);
} else if (line_ranges) {
struct line_range_callback lr_state;
unsigned int i;
long max_span = 0;
struct line_range_filter lr_filter;

memset(&lr_state, 0, sizeof(lr_state));
lr_state.orig_line_fn = fn_out_consume;
lr_state.orig_cb_data = &ecbdata;
lr_state.ranges = line_ranges;
strbuf_init(&lr_state.rhunk, 0);
strbuf_init(&lr_state.pending_rm, 0);
line_range_filter_init(&lr_filter, line_ranges,
fn_out_consume, &ecbdata);

/*
* Inflate ctxlen so that all changes within
* any single range are merged into one xdiff
* hunk and the inter-change context is emitted.
* The callback clips back to range boundaries.
*
* The optimal ctxlen depends on where changes
* fall within the range, which is only known
* after xdiff runs; the max range span is the
* upper bound that guarantees correctness in a
* single pass.
*/
for (i = 0; i < line_ranges->nr; i++) {
long span = line_ranges->ranges[i].end -
line_ranges->ranges[i].start;
if (span > max_span)
max_span = span;
}
if (max_span > xecfg.ctxlen)
xecfg.ctxlen = max_span;

if (xdi_diff_outf(&mf1, &mf2,
line_range_hunk_fn,
line_range_line_fn,
&lr_state, &xpp, &xecfg))
if (line_range_filter_diff(&lr_filter, &mf1, &mf2,
&xpp, &xecfg))
die("unable to generate diff for %s",
one->path);

flush_rhunk(&lr_state);
if (lr_state.ret)
die("unable to generate diff for %s",
one->path);
strbuf_release(&lr_state.rhunk);
strbuf_release(&lr_state.pending_rm);
} else if (xdi_diff_outf(&mf1, &mf2, NULL, fn_out_consume,
&ecbdata, &xpp, &xecfg))
die("unable to generate diff for %s", one->path);
@ -4261,7 +4335,18 @@ static void builtin_diffstat(const char *name_a, const char *name_b,
xecfg.ctxlen = o->context;
xecfg.interhunkctxlen = o->interhunkcontext;
xecfg.flags = XDL_EMIT_NO_HUNK_HDR;
if (xdi_diff_outf(&mf1, &mf2, NULL,

if (p->line_ranges) {
struct line_range_filter lr_filter;

line_range_filter_init(&lr_filter, p->line_ranges,
diffstat_consume, diffstat);

if (line_range_filter_diff(&lr_filter, &mf1, &mf2,
&xpp, &xecfg))
die("unable to generate diffstat for %s",
one->path);
} else if (xdi_diff_outf(&mf1, &mf2, NULL,
diffstat_consume, diffstat, &xpp, &xecfg))
die("unable to generate diffstat for %s", one->path);

@ -4291,11 +4376,29 @@ static void builtin_diffstat(const char *name_a, const char *name_b,
diff_free_filespec_data(two);
}

/*
* Is the 0-based line index within any of the tracked ranges?
* (range_set ranges are 0-based, half-open [start, end).) This is a
* one-shot query for a single line and scans; the streaming filter
* (line_range_line_fn) uses a forward cursor instead.
*/
static int idx_in_ranges(const struct range_set *ranges, long idx)
{
unsigned int i;

for (i = 0; i < ranges->nr; i++)
if (idx >= ranges->ranges[i].start &&
idx < ranges->ranges[i].end)
return 1;
return 0;
}

static void builtin_checkdiff(const char *name_a, const char *name_b,
const char *attr_path,
struct diff_filespec *one,
struct diff_filespec *two,
struct diff_options *o)
struct diff_options *o,
const struct range_set *line_ranges)
{
mmfile_t mf1, mf2;
struct checkdiff_t data;
@ -4335,7 +4438,19 @@ static void builtin_checkdiff(const char *name_a, const char *name_b,
memset(&xecfg, 0, sizeof(xecfg));
xecfg.ctxlen = 1; /* at least one context line */
xpp.flags = 0;
if (xdi_diff_outf(&mf1, &mf2, checkdiff_consume_hunk,

if (line_ranges) {
struct line_range_filter lr_filter;

line_range_filter_init(&lr_filter, line_ranges,
checkdiff_consume, &data);
lr_filter.orig_hunk_fn = checkdiff_consume_hunk;

if (line_range_filter_diff(&lr_filter, &mf1, &mf2,
&xpp, &xecfg))
die("unable to generate checkdiff for %s",
one->path);
} else if (xdi_diff_outf(&mf1, &mf2, checkdiff_consume_hunk,
checkdiff_consume, &data,
&xpp, &xecfg))
die("unable to generate checkdiff for %s", one->path);
@ -4348,6 +4463,17 @@ static void builtin_checkdiff(const char *name_a, const char *name_b,
check_blank_at_eof(&mf1, &mf2, &ecbdata);
blank_at_eof = ecbdata.blank_at_eof_in_postimage;

/*
* check_blank_at_eof() scans the whole file; with -L,
* keep the report only when its line is in a tracked
* range. The error's location is the first trailing
* blank line (blank_at_eof, 1-based; ranges 0-based), so
* we scope by that line.
*/
if (blank_at_eof && line_ranges &&
!idx_in_ranges(line_ranges, blank_at_eof - 1))
blank_at_eof = 0;

if (blank_at_eof) {
static char *err;
if (!err)
@ -5143,7 +5269,8 @@ static void run_checkdiff(struct diff_filepair *p, struct diff_options *o)
diff_fill_oid_info(p->one, o->repo->index);
diff_fill_oid_info(p->two, o->repo->index);

builtin_checkdiff(name, other, attr_path, p->one, p->two, o);
builtin_checkdiff(name, other, attr_path, p->one, p->two, o,
p->line_ranges);
}

void repo_diff_setup(struct repository *r, struct diff_options *options)

View File

@ -16,7 +16,8 @@

typedef int (*pickaxe_fn)(mmfile_t *one, mmfile_t *two,
struct diff_options *o,
regex_t *regexp, kwset_t kws);
regex_t *regexp, kwset_t kws,
const struct range_set *ranges);

struct diffgrep_cb {
regex_t *regexp;
@ -42,7 +43,8 @@ static int diffgrep_consume(void *priv, char *line, unsigned long len)

static int diff_grep(mmfile_t *one, mmfile_t *two,
struct diff_options *o,
regex_t *regexp, kwset_t kws UNUSED)
regex_t *regexp, kwset_t kws UNUSED,
const struct range_set *ranges)
{
struct diffgrep_cb ecbdata;
xpparam_t xpp;
@ -50,8 +52,11 @@ static int diff_grep(mmfile_t *one, mmfile_t *two,
int ret;

/*
* We have both sides; need to run textual diff and see if
* the pattern appears on added/deleted lines.
* We have both sides; need to run textual diff and see if the
* pattern appears on added/deleted lines. Under -L (ranges set),
* forward only the tracked range's lines so the match is scoped.
* -G needs only a hit/no-hit answer, so the line-number bookkeeping
* the filter does for -L patch and check output is irrelevant here.
*/
memset(&xpp, 0, sizeof(xpp));
memset(&xecfg, 0, sizeof(xecfg));
@ -65,8 +70,12 @@ static int diff_grep(mmfile_t *one, mmfile_t *two,
* An xdiff error might be our "data->hit" from above. See the
* comment for xdiff_emit_line_fn in xdiff-interface.h
*/
ret = xdi_diff_outf(one, two, NULL, diffgrep_consume,
&ecbdata, &xpp, &xecfg);
if (ranges)
ret = diff_emit_line_ranges(one, two, ranges, diffgrep_consume,
&ecbdata, &xpp, &xecfg);
else
ret = xdi_diff_outf(one, two, NULL, diffgrep_consume,
&ecbdata, &xpp, &xecfg);
if (ecbdata.hit)
return 1;
if (ret)
@ -119,8 +128,13 @@ static unsigned int contains(mmfile_t *mf, regex_t *regexp, kwset_t kws,

static int has_changes(mmfile_t *one, mmfile_t *two,
struct diff_options *o UNUSED,
regex_t *regexp, kwset_t kws)
regex_t *regexp, kwset_t kws,
const struct range_set *ranges UNUSED)
{
/*
* -S counts needle occurrences in each whole blob. Scoping this to
* a -L range is left to a follow-up; for now -S ignores the range.
*/
unsigned int c1 = one ? contains(one, regexp, kws, 0) : 0;
unsigned int c2 = two ? contains(two, regexp, kws, c1 + 1) : 0;
return c1 != c2;
@ -132,6 +146,7 @@ static int pickaxe_match(struct diff_filepair *p, struct diff_options *o,
struct userdiff_driver *textconv_one = NULL;
struct userdiff_driver *textconv_two = NULL;
mmfile_t mf1, mf2;
const struct range_set *ranges;
int ret;

/* ignore unmerged */
@ -169,7 +184,13 @@ static int pickaxe_match(struct diff_filepair *p, struct diff_options *o,
mf1.size = fill_textconv(o->repo, textconv_one, p->one, &mf1.ptr);
mf2.size = fill_textconv(o->repo, textconv_two, p->two, &mf2.ptr);

ret = fn(&mf1, &mf2, o, regexp, kws);
/*
* -L scopes the search to the tracked range, but the range is in
* original-file line coordinates that do not map onto textconv
* output, so search the whole file when textconv is in play.
*/
ranges = (textconv_one || textconv_two) ? NULL : p->line_ranges;
ret = fn(&mf1, &mf2, o, regexp, kws, ranges);

if (textconv_one)
free(mf1.ptr);

View File

@ -3207,8 +3207,10 @@ int setup_revisions(int argc, const char **argv, struct rev_info *revs, struct s
(revs->diffopt.output_format &
~(DIFF_FORMAT_PATCH | DIFF_FORMAT_NO_OUTPUT |
DIFF_FORMAT_RAW | DIFF_FORMAT_NAME |
DIFF_FORMAT_NAME_STATUS | DIFF_FORMAT_SUMMARY))))
die(_("-L does not yet support the requested diff format"));
DIFF_FORMAT_NAME_STATUS | DIFF_FORMAT_SUMMARY |
DIFF_FORMAT_NUMSTAT | DIFF_FORMAT_DIFFSTAT |
DIFF_FORMAT_SHORTSTAT | DIFF_FORMAT_CHECKDIFF))))
die(_("-L does not support the requested diff format"));

if (revs->expand_tabs_in_log < 0)
revs->expand_tabs_in_log = revs->expand_tabs_in_log_default;

View File

@ -176,24 +176,15 @@ test_expect_success '--name-status shows status and path' '
test_grep ! "^@@" actual
'

test_expect_success '--stat is not yet supported with -L' '
test_must_fail git log -L1,24:b.c --stat 2>err &&
test_grep "does not yet support" err
'

test_expect_success '--numstat is not yet supported with -L' '
test_must_fail git log -L1,24:b.c --numstat 2>err &&
test_grep "does not yet support" err
'

test_expect_success '--shortstat is not yet supported with -L' '
test_must_fail git log -L1,24:b.c --shortstat 2>err &&
test_grep "does not yet support" err
'

test_expect_success '--dirstat is not yet supported with -L' '
test_expect_success '--dirstat is not supported with -L' '
# --dirstat is not supported with -L: its default mode measures
# whole-file change, not the tracked lines, and the
# --dirstat=lines variant is deferred too, so both forms are
# rejected like any other unsupported format.
test_must_fail git log -L1,24:b.c --dirstat 2>err &&
test_grep "does not yet support" err
test_grep "does not support" err &&
test_must_fail git log -L1,24:b.c --dirstat=lines 2>err &&
test_grep "does not support" err
'

test_expect_success 'setup for checking fancy rename following' '
@ -731,13 +722,138 @@ test_expect_success '-L with -S filters to string-count changes' '
test_expect_success '-L with -G filters to diff-text matches' '
git checkout parent-oids &&
git log -L:func2:file.c -G "F2 [+] 2" --format= >actual &&
# -G greps the whole-file diff text, not just the tracked range;
# combined with -L, this selects commits that both touch func2
# and have "F2 + 2" in their diff.
# -G greps the diff text, and under -L only the lines in the
# tracked range (unlike -S above, which searches the whole file);
# this selects commits whose change to func2 contains "F2 + 2".
test $(grep -c "^diff --git" actual) = 1 &&
test_grep "F2 + 2" actual
'

test_expect_success 'setup for trailing deletion test' '
git checkout --orphan trailing-del &&
git reset --hard &&
cat >file.c <<-\EOF &&
void tracked()
{
return 1;
}
// trailing comment
EOF
git add file.c &&
test_tick &&
git commit -m "add file with trailing comment" &&
# Modify tracked() AND delete the trailing comment in
# one commit, so the commit touches the tracked range
# and is not filtered out by the revision walker.
cat >file.c <<-\EOF &&
void tracked()
{
return 2;
}
EOF
git commit -a -m "modify tracked and delete trailing comment"
'

test_expect_success '-L does not include deletions past end of tracked range' '
git log -L:tracked:file.c --format= -1 -p >actual &&
# The trailing comment deletion is outside the tracked
# range and should not appear in the patch output.
test_grep "return 2" actual &&
test_grep ! "trailing comment" actual
'

test_expect_success '-L includes leading deletions resolved by in-range line' '
git checkout --orphan leading-del &&
git reset --hard &&
cat >file.c <<-\EOF &&
// leading comment
void tracked()
{
return 1;
}
EOF
git add file.c &&
test_tick &&
git commit -m "add file with leading comment" &&
cat >file.c <<-\EOF &&
void tracked()
{
return 2;
}
EOF
git commit -a -m "modify tracked and delete leading comment" &&
git log -L:tracked:file.c --format= -1 -p >actual &&
# The leading comment deletion is resolved by the next
# non-removal line (void tracked), which is in range: a
# removal is classified by the position of the following
# line, so it joins the range that line falls in.
test_grep "return 2" actual &&
test_grep "leading comment" actual
'

test_expect_success 'setup for line-range filter edge cases' '
git checkout --orphan filter-edge &&
git reset --hard &&
cat >file.c <<-\EOF &&
void before()
{
return 0;
}

void tracked()
{
int a = 1;
int b = 2;
int c = 3;
return a + b + c;
}

void after()
{
return 9;
}
EOF
git add file.c &&
test_tick &&
git commit -m "initial"
'

test_expect_success '-L change at exact first line of range' '
git checkout filter-edge &&
# Change the function signature (first line of range)
sed "s/void tracked/int tracked/" file.c >tmp &&
mv tmp file.c &&
git commit -a -m "change first line" &&
git log -L:tracked:file.c -p --format=%s -1 >actual &&
test_grep "change first line" actual &&
test_grep "+int tracked" actual &&
test_grep "\\-void tracked" actual
'

test_expect_success '-L change at exact last line of range' '
git checkout filter-edge &&
git reset --hard HEAD~1 &&
# Change the closing brace line (last line of range)
sed "s/^}$/} \/\/ end tracked/" file.c >tmp &&
mv tmp file.c &&
git commit -a -m "change last line" &&
git log -L:tracked:file.c -p --format=%s -1 >actual &&
test_grep "change last line" actual &&
test_grep "end tracked" actual
'

test_expect_success '-L pure deletion in range (no additions)' '
git checkout filter-edge &&
git reset --hard HEAD~1 &&
# Delete a line inside tracked() without adding anything
sed "/int c/d" file.c >tmp &&
mv tmp file.c &&
git commit -a -m "pure deletion" &&
git log -L:tracked:file.c -p --format=%s -1 >actual &&
test_grep "pure deletion" actual &&
test_grep "\\-.*int c" actual
'

test_expect_success '-L with --diff-filter=M excludes root commit' '
git checkout parent-oids &&
git log -L:func2:file.c --diff-filter=M --format=%s --no-patch >actual &&
@ -762,9 +878,9 @@ test_expect_success '-L with -S suppresses non-matching commits' '
test_cmp expect actual
'

test_expect_success '--full-diff is not yet supported with -L' '
test_expect_success '--full-diff is not supported with -L' '
test_must_fail git log -L1,24:b.c --full-diff 2>err &&
test_grep "does not yet support" err
test_grep "does not support" err
'

test_expect_success '-L --oneline has no extra blank line before diff' '
@ -775,10 +891,289 @@ test_expect_success '-L --oneline has no extra blank line before diff' '
test_grep "^diff --git" line2
'

test_expect_success 'setup for stat range-scoping tests' '
git checkout --orphan stat-scoping &&
git reset --hard &&
cat >file.c <<-\EOF &&
int func1()
{
return F1;
}

int func2()
{
return F2;
}
EOF
git add file.c &&
test_tick &&
git commit -m "Add func1() and func2()" &&

# Modify both functions in a single commit so that
# whole-file stats differ from the counts for the tracked range.
sed -e "s/F1/F1 + 1/" -e "s/F2/F2 + 2/" file.c >tmp &&
mv tmp file.c &&
git commit -a -m "Modify both functions"
'

test_expect_success '--numstat counts only lines in tracked range' '
# "Modify both functions" changes one line in func1 and one in
# func2. Whole-file numstat would show 2 added, 2 deleted.
# numstat for func2 within the tracked range should show only 1 and 1.
git log -L:func2:file.c --numstat --format=%s -1 >actual &&
test_grep "Modify both functions" actual &&
test_grep "^1 1 file.c$" actual &&
test_grep ! "^diff --git" actual
'

test_expect_success '--numstat counts only additions for root commit' '
# Root commit creates both func1 (4 lines) and func2 (4 lines).
# Whole-file numstat would show 9 lines added. numstat for func2
# within the tracked range should show only 4.
git log -L:func2:file.c --numstat --format=%s >actual &&
test_grep "Add func1() and func2()" actual &&
test_grep "^4 0 file.c$" actual &&
test_grep ! "^diff --git" actual
'

test_expect_success '--stat counts only lines in tracked range' '
git log -L:func2:file.c --stat --format=%s -1 >actual &&
test_grep "Modify both functions" actual &&
test_grep "file.c |" actual &&
test_grep "1 insertion" actual &&
test_grep "1 deletion" actual &&
test_grep ! "^diff --git" actual
'

test_expect_success '--shortstat counts only lines in tracked range' '
# --shortstat prints only the summary line: no per-file "file.c |"
# line. Counts cover only the tracked range, as for --numstat above.
git log -L:func2:file.c --shortstat --format=%s -1 >actual &&
test_grep "Modify both functions" actual &&
test_grep "1 insertion" actual &&
test_grep "1 deletion" actual &&
test_grep ! "file.c |" actual &&
test_grep ! "^diff --git" actual
'

test_expect_success '--numstat across renames and multiple commits' '
# parallel-change carries the tracked function f across an a.c -> b.c
# rename and a merge of two parallel histories. With -M, --numstat
# follows the rename and reports added/removed counts for f within
# the tracked range (not whole-file) per commit; the file column flips from
# b.c to a.c at the rename as the walk goes back in time. Commits
# that do not change the range of f emit no row (the merge and the
# pure file-move produce nothing), so there are fewer rows than
# commits.
git checkout parallel-change &&
git log -M -L ":f:b.c" --format= --numstat >actual &&
cat >expect <<-\EOF &&
1 1 b.c
1 1 a.c
1 1 a.c
1 1 a.c
1 0 a.c
13 0 a.c
EOF
test_cmp expect actual
'

test_expect_success '-L multiple ranges with --numstat excludes untracked change' '
git checkout --orphan multi-range &&
git reset --hard &&
cat >m.c <<-\EOF &&
int func1()
{
return F1;
}

int func2()
{
return F2;
}

int func3()
{
return F3;
}
EOF
git add m.c &&
test_tick &&
git commit -m "add m.c" &&
# Change all three functions but track only func1 and func2.
# Whole-file numstat would be 3 3; a 2 2 result proves the
# untracked func3 change is excluded and the two ranges just sum.
sed -e "s/F1/F1 + 1/" -e "s/F2/F2 + 2/" -e "s/F3/F3 + 3/" m.c >tmp &&
mv tmp m.c &&
git commit -a -m "Modify all three functions" &&
git log -L:func1:m.c -L:func2:m.c --numstat --format=%s -1 >actual &&
test_grep "Modify all three functions" actual &&
test_grep "^2 2 m.c$" actual &&
test_grep ! "^3 3 m.c$" actual
'

test_expect_success '--summary shows new file on root commit' '
git checkout parent-oids &&
git log -L:func2:file.c --summary --format= >actual &&
test_grep "create mode 100644 file.c" actual
'

test_expect_success 'setup for --check test' '
git checkout --orphan check-test &&
git reset --hard &&
cat >check.c <<-\EOF &&
void tracked()
{
return;
}

void other()
{
return;
}
EOF
git add check.c &&
test_tick &&
git commit -m "add check.c" &&
# Introduce trailing whitespace errors in both functions
sed "s/return;/return; /" check.c >check.c.tmp &&
mv check.c.tmp check.c &&
git commit -a -m "introduce trailing whitespace"
'

test_expect_success '--check scoped to tracked range with correct file line' '
# tracked() trailing whitespace is at check.c:3; report it with the
# real file line number, not a count from the start of the range
# hunk. other() at check.c:8 is outside the range and is excluded.
test_must_fail git log -L:tracked:check.c --check --format= >actual &&
test_grep "check.c:3: trailing whitespace" actual &&
test_grep ! "check.c:8:" actual
'

test_expect_success '--check reports each of several tracked ranges' '
# Track both functions as separate ranges. Each range is flushed
# as its own hunk, so the second error must report its real file
# line (check.c:8), not continue the numbering from the first
# range (check.c:3).
test_must_fail git log -L:tracked:check.c -L:other:check.c \
--check --format= >actual &&
test_grep "check.c:3: trailing whitespace" actual &&
test_grep "check.c:8: trailing whitespace" actual
'

test_expect_success '--check line numbers stay correct across a gap in one range' '
git checkout --orphan check-gap &&
git reset --hard &&
cat >gap.c <<-\EOF &&
void tracked()
{
int a = 1;
int b = 2;
int c = 3;
int d = 4;
int e = 5;
int g = 7;
return;
}
EOF
git add gap.c &&
test_tick &&
git commit -m "add gap.c" &&
# Two trailing-whitespace errors within one tracked range,
# separated by clean lines. ctxlen is inflated to the range span,
# so they land in a single xdiff hunk with the gap as context;
# both must report their real file line number, with the context
# lines between them counted.
sed -e "s/int a = 1;/int a = 1; /" -e "s/int g = 7;/int g = 7; /" gap.c >tmp &&
mv tmp gap.c &&
git commit -a -m "ws errors with a gap" &&
test_must_fail git log -L:tracked:gap.c --check --format= >actual &&
test_grep "gap.c:3: trailing whitespace" actual &&
test_grep "gap.c:8: trailing whitespace" actual
'

test_expect_success '--check does not report blank-at-eof outside the range' '
git checkout --orphan check-eof &&
git reset --hard &&
printf "void tracked()\n{\n return;\n}\n\nint tail = 1;\n" >eof.c &&
git add eof.c &&
test_tick &&
git commit -m "add eof.c" &&
# One commit introduces a trailing-whitespace error inside tracked()
# (line 3) and a blank line at end of file (line 7, outside the
# range). The blank-at-eof check scans the whole file, so it must be
# scoped: report the in-range error, not the out-of-range EOF blank.
printf "void tracked()\n{\n return; \n}\n\nint tail = 1;\n\n" >eof.c &&
git commit -a -m "ws in range, blank at eof out of range" &&
test_must_fail git log -L:tracked:eof.c --check --format= >actual &&
test_grep "eof.c:3: trailing whitespace" actual &&
test_grep ! "blank line at EOF" actual
'

test_expect_success '-L -G is scoped to the tracked range' '
git checkout --orphan grep-scope &&
git reset --hard &&
cat >gp.c <<-\EOF &&
int func1()
{
return ALPHA;
}

int func2()
{
return BETA;
}
EOF
git add gp.c &&
test_tick &&
git commit -m "add gp.c" &&
sed -e "s/ALPHA/ALPHA2/" -e "s/BETA/BETA2/" gp.c >tmp &&
mv tmp gp.c &&
git commit -a -m "touch both functions" &&
# The commit changes ALPHA (func1) and BETA (func2). Tracking func2,
# -G BETA matches its in-range change; -G ALPHA must not, since ALPHA
# changes only outside the tracked range.
git log -L:func2:gp.c -G BETA --format=%s >actual &&
test_grep "touch both functions" actual &&
git log -L:func2:gp.c -G ALPHA --format=%s >actual &&
test_grep ! "touch both functions" actual
'

test_expect_success '-L -G searches the whole file under textconv' '
git checkout --orphan grep-textconv &&
git reset --hard &&
cat >tc.c <<-\EOF &&
int func1()
{
return F1;
}

int func2()
{
return F2;
}
EOF
git add tc.c &&
test_tick &&
git commit -m "add tc.c" &&
# One commit changes func1 and func2; MAGIC lands only in the
# func2 change, outside func1.
sed -e "s/F1/F1 + 1/" -e "s/return F2/return MAGIC/" tc.c >tmp &&
mv tmp tc.c &&
git commit -a -m "change both funcs" &&
echo "tc.c diff=tc" >.gitattributes &&

# Without a textconv driver, -G is scoped to func1, so MAGIC (only
# in the func2 change) does not select the commit.
git log -L:func1:tc.c -G MAGIC --format=%s --no-patch >actual &&
test_must_be_empty actual &&

# A textconv driver makes the range (original-file line numbers)
# meaningless against the driver output, so -G falls back to the
# whole file and MAGIC now selects the commit.
git config diff.tc.textconv cat &&
git log -L:func1:tc.c -G MAGIC --format=%s --no-patch >actual &&
test_grep "change both funcs" actual
'

test_done

View File

@ -8,7 +8,7 @@ diff --git a/b.c b/b.c
index bf79c2f..27c829c 100644
--- a/b.c
+++ b/b.c
@@ -25,0 +18,9 @@
@@ -24,0 +18,9 @@
+long f(long x)
+{
+ int s = 0;

View File

@ -8,7 +8,7 @@ diff --git a/a.c b/a.c
index 0b9cae5..5de3ea4 100644
--- a/a.c
+++ b/a.c
@@ -23,0 +24,1 @@ int main ()
@@ -22,0 +24 @@ int main ()
+/* incomplete lines are bad! */

commit 100b61a6f2f720f812620a9d10afb3a960ccb73c
@ -21,7 +21,7 @@ diff --git a/a.c b/a.c
index 5e709a1..0b9cae5 100644
--- a/a.c
+++ b/a.c
@@ -22,1 +22,1 @@ int main ()
@@ -22 +22 @@ int main ()
-}
+}
\ No newline at end of file
@ -37,5 +37,5 @@ new file mode 100644
index 0000000..444e415
--- /dev/null
+++ b/a.c
@@ -0,0 +20,1 @@
@@ -0,0 +20 @@
+}

View File

@ -8,7 +8,7 @@ diff --git a/b.c b/b.c
index 69cb69c..a0d566e 100644
--- a/b.c
+++ b/b.c
@@ -25,0 +18,9 @@
@@ -24,0 +18,9 @@
+long f(long x)
+{
+ int s = 0;

View File

@ -8,7 +8,7 @@ diff --git a/a.c b/a.c
index e4fa1d8..62c1fc2 100644
--- a/a.c
+++ b/a.c
@@ -23,0 +24,1 @@ int main ()
@@ -22,0 +24 @@ int main ()
+/* incomplete lines are bad! */

commit 29f32ac3141c48b22803e5c4127b719917b67d0f8ca8c5248bebfa2a19f7da10
@ -21,7 +21,7 @@ diff --git a/a.c b/a.c
index d325124..e4fa1d8 100644
--- a/a.c
+++ b/a.c
@@ -22,1 +22,1 @@ int main ()
@@ -22 +22 @@ int main ()
-}
+}
\ No newline at end of file
@ -37,5 +37,5 @@ new file mode 100644
index 0000000..9f550c3
--- /dev/null
+++ b/a.c
@@ -0,0 +20,1 @@
@@ -0,0 +20 @@
+}

View File

@ -91,6 +91,25 @@ static int xdiff_outf(void *priv_, mmbuffer_t *mb, int nbuf)
return 0;
}

static int strbuf_out_line(void *priv, mmbuffer_t *mb, int nbuf)
{
struct strbuf *out = priv;
int i;
for (i = 0; i < nbuf; i++)
strbuf_add(out, mb[i].ptr, mb[i].size);
return 0;
}

void xdiff_emit_hunk_header(struct strbuf *out,
long old_begin, long old_count,
long new_begin, long new_count,
const char *func, long funclen)
{
xdemitcb_t ecb = { .priv = out, .out_line = strbuf_out_line };
xdl_emit_hunk_hdr(old_begin, old_count, new_begin, new_count,
func, funclen, &ecb);
}

/*
* Trim down common substring at the end of the buffers,
* but end on a complete line.

View File

@ -46,6 +46,19 @@ int xdi_diff_outf(mmfile_t *mf1, mmfile_t *mf2,
xdiff_emit_line_fn line_fn,
void *consume_callback_data,
xpparam_t const *xpp, xdemitconf_t const *xecfg);

struct range_set;
/*
* Like xdi_diff_outf(), but forwards only the lines within the given
* (post-image) line ranges to line_fn, as "git log -L" scopes its output.
* Returns line_fn's latched return value (so a consumer can signal a hit
* with a non-zero return), or non-zero on xdiff failure. Defined in
* diff.c (it reuses the line-range filter there).
*/
int diff_emit_line_ranges(mmfile_t *mf1, mmfile_t *mf2,
const struct range_set *ranges,
xdiff_emit_line_fn line_fn, void *cb_data,
xpparam_t *xpp, xdemitconf_t *xecfg);
int read_mmfile(mmfile_t *ptr, const char *filename);
void read_mmblob(mmfile_t *ptr, struct object_database *odb,
const struct object_id *oid);
@ -76,4 +89,19 @@ int xdiff_compare_lines(const char *l1, long s1,
*/
unsigned long xdiff_hash_string(const char *s, size_t len, long flags);

struct strbuf;

/*
* Append a unified-diff hunk header to `out`, e.g.
* "@@ -<old> +<new> @@ func\n". The header comes from wrapping xdiff's
* own hunk-header emitter, so it matches what a normal diff would
* produce for these begins and counts. For a side with no lines
* (count 0) the begin is the line before the change, and a count of 1
* is omitted.
*/
void xdiff_emit_hunk_header(struct strbuf *out,
long old_begin, long old_count,
long new_begin, long new_count,
const char *func, long funclen);

#endif