diff: don't attempt to strip prefix from absolute Windows paths
commitffd04e92e2c512cbbcb99526b064af302b443cb2
authorJohannes Sixt <j6t@kdbg.org>
Fri, 19 Oct 2018 16:58:07 +0000 (19 18:58 +0200)
committerJunio C Hamano <gitster@pobox.com>
Mon, 22 Oct 2018 01:17:50 +0000 (22 10:17 +0900)
tree9b07ef173b3efae42d683ec9a1f276ae88fe3908
parent27d05d1a1a62273aa3749f4d0ab8a126ef11ff66
diff: don't attempt to strip prefix from absolute Windows paths

git diff can be invoked with absolute paths. Typically, this triggers
the --no-index case. Then the absolute paths remain in the file names
that are printed in the output.

There is one peculiarity, though: When the command is invoked from a
a sub-directory in a repository, then it is attempted to strip the
sub-directory from the beginning of relative paths. Yet, to detect a
relative path the code just checks for an initial forward slash.
This mistakes a Windows style path like "D:/base" as a relative path
and the output looks like this, for example:

  D:\dir\test\one>git -P diff --numstat D:\dir\base D:\dir\diff
  1       1       ir/{base => diff}/1.txt

where the correct output should be

  D:\dir\test\one>git -P diff --numstat D:\dir\base D:\dir\diff
  1       1       D:/dir/{base => diff}/1.txt

If the sub-directory where 'git diff' is invoked is sufficiently deep
that the prefix becomes longer than the path to be printed, then the
subsequent code accesses the path out of bounds.

Use is_absolute_path() to detect Windows style absolute paths.

One might wonder whether the check for a directory separator that
is visible in the patch context should be changed from == '/' to
is_dir_sep() or not. It turns out not to be necessary. That code
only ever investigates paths that have undergone pathspec
normalization, after which there are only forward slashes even on
Windows.

Signed-off-by: Johannes Sixt <j6t@kdbg.org>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
diff.c
t/t4053-diff-no-index.sh