Merge branch 'eb/core-eol'

* eb/core-eol:
  Add "core.eol" config variable
  Rename the "crlf" attribute "text"
  Add per-repository eol normalization
  Add tests for per-repository eol normalization

Conflicts:
	Documentation/config.txt
	Makefile
maint
Junio C Hamano 2010-06-21 06:02:49 -07:00
commit d5cff17eda
12 changed files with 570 additions and 110 deletions

View File

@ -196,20 +196,17 @@ core.quotepath::
quoted without `-z` regardless of the setting of this quoted without `-z` regardless of the setting of this
variable. variable.


core.autocrlf:: core.eol::
If true, makes git convert `CRLF` at the end of lines in text files to Sets the line ending type to use in the working directory for
`LF` when reading from the work tree, and convert in reverse when files that have the `text` property set. Alternatives are
writing to the work tree. The variable can be set to 'lf', 'crlf' and 'native', which uses the platform's native
'input', in which case the conversion happens only while line ending. The default value is `native`. See
reading from the work tree but files are written out to the work linkgit:gitattributes[5] for more information on end-of-line
tree with `LF` at the end of lines. A file is considered conversion.
"text" (i.e. be subjected to the autocrlf mechanism) based on
the file's `crlf` attribute, or if `crlf` is unspecified,
based on the file's contents. See linkgit:gitattributes[5].


core.safecrlf:: core.safecrlf::
If true, makes git check if converting `CRLF` as controlled by If true, makes git check if converting `CRLF` is reversible when
`core.autocrlf` is reversible. Git will verify if a command end-of-line conversion is active. Git will verify if a command
modifies a file in the work tree either directly or indirectly. modifies a file in the work tree either directly or indirectly.
For example, committing a file followed by checking out the For example, committing a file followed by checking out the
same file should yield the original file in the work tree. If same file should yield the original file in the work tree. If
@ -219,7 +216,7 @@ core.safecrlf::
irreversible conversion but continue the operation. irreversible conversion but continue the operation.
+ +
CRLF conversion bears a slight chance of corrupting data. CRLF conversion bears a slight chance of corrupting data.
autocrlf=true will convert CRLF to LF during commit and LF to When it is enabled, git will convert CRLF to LF during commit and LF to
CRLF during checkout. A file that contains a mixture of LF and CRLF during checkout. A file that contains a mixture of LF and
CRLF before the commit cannot be recreated by git. For text CRLF before the commit cannot be recreated by git. For text
files this is the right thing to do: it corrects line endings files this is the right thing to do: it corrects line endings
@ -243,15 +240,25 @@ converting CRLFs corrupts data.
+ +
Note, this safety check does not mean that a checkout will generate a Note, this safety check does not mean that a checkout will generate a
file identical to the original file for a different setting of file identical to the original file for a different setting of
`core.autocrlf`, but only for the current one. For example, a text `core.eol` and `core.autocrlf`, but only for the current one. For
file with `LF` would be accepted with `core.autocrlf=input` and could example, a text file with `LF` would be accepted with `core.eol=lf`
later be checked out with `core.autocrlf=true`, in which case the and could later be checked out with `core.eol=crlf`, in which case the
resulting file would contain `CRLF`, although the original file resulting file would contain `CRLF`, although the original file
contained `LF`. However, in both work trees the line endings would be contained `LF`. However, in both work trees the line endings would be
consistent, that is either all `LF` or all `CRLF`, but never mixed. A consistent, that is either all `LF` or all `CRLF`, but never mixed. A
file with mixed line endings would be reported by the `core.safecrlf` file with mixed line endings would be reported by the `core.safecrlf`
mechanism. mechanism.


core.autocrlf::
Setting this variable to "true" is almost the same as setting
the `text` attribute to "auto" on all files except that text
files are not guaranteed to be normalized: files that contain
`CRLF` in the repository will not be touched. Use this
setting if you want to have `CRLF` line endings in your
working directory even though the repository does not have
normalized line endings. This variable can be set to 'input',
in which case no output conversion is performed.

core.symlinks:: core.symlinks::
If false, symbolic links are checked out as small plain files that If false, symbolic links are checked out as small plain files that
contain the link text. linkgit:git-update-index[1] and contain the link text. linkgit:git-update-index[1] and
@ -979,13 +986,15 @@ gitcvs.logfile::
various stuff. See linkgit:git-cvsserver[1]. various stuff. See linkgit:git-cvsserver[1].


gitcvs.usecrlfattr:: gitcvs.usecrlfattr::
If true, the server will look up the `crlf` attribute for If true, the server will look up the end-of-line conversion
files to determine the '-k' modes to use. If `crlf` is set, attributes for files to determine the '-k' modes to use. If
the '-k' mode will be left blank, so cvs clients will the attributes force git to treat a file as text,
treat it as text. If `crlf` is explicitly unset, the file the '-k' mode will be left blank so cvs clients will
treat it as text. If they suppress text conversion, the file
will be set with '-kb' mode, which suppresses any newline munging will be set with '-kb' mode, which suppresses any newline munging
the client might otherwise do. If `crlf` is not specified, the client might otherwise do. If the attributes do not allow
then 'gitcvs.allbinary' is used. See linkgit:gitattributes[5]. the file type to be determined, then 'gitcvs.allbinary' is
used. See linkgit:gitattributes[5].


gitcvs.allbinary:: gitcvs.allbinary::
This is used if 'gitcvs.usecrlfattr' does not resolve This is used if 'gitcvs.usecrlfattr' does not resolve

View File

@ -369,16 +369,13 @@ By default the server leaves the '-k' mode blank for all files,
which causes the cvs client to treat them as a text files, subject which causes the cvs client to treat them as a text files, subject
to crlf conversion on some platforms. to crlf conversion on some platforms.


You can make the server use `crlf` attributes to set the '-k' modes You can make the server use the end-of-line conversion attributes to
for files by setting the `gitcvs.usecrlfattr` config variable. set the '-k' modes for files by setting the `gitcvs.usecrlfattr`
In this case, if `crlf` is explicitly unset ('-crlf'), then the config variable. See linkgit:gitattributes[5] for more information
server will set '-kb' mode for binary files. If `crlf` is set, about end-of-line conversion.
then the '-k' mode will explicitly be left blank. See
also linkgit:gitattributes[5] for more information about the `crlf`
attribute.


Alternatively, if `gitcvs.usecrlfattr` config is not enabled Alternatively, if `gitcvs.usecrlfattr` config is not enabled
or if the `crlf` attribute is unspecified for a filename, then or the attributes do not allow automatic detection for a filename, then
the server uses the `gitcvs.allbinary` config for the default setting. the server uses the `gitcvs.allbinary` config for the default setting.
If `gitcvs.allbinary` is set, then file not otherwise If `gitcvs.allbinary` is set, then file not otherwise
specified will default to '-kb' mode. Otherwise the '-k' mode specified will default to '-kb' mode. Otherwise the '-k' mode

View File

@ -92,53 +92,154 @@ such as 'git checkout' and 'git merge' run. They also affect how
git stores the contents you prepare in the working tree in the git stores the contents you prepare in the working tree in the
repository upon 'git add' and 'git commit'. repository upon 'git add' and 'git commit'.


`crlf` `text`
^^^^^^ ^^^^^^


This attribute controls the line-ending convention. This attribute enables and controls end-of-line normalization. When a
text file is normalized, its line endings are converted to LF in the
repository. To control what line ending style is used in the working
directory, use the `eol` attribute for a single file and the
`core.eol` configuration variable for all text files.


Set:: Set::


Setting the `crlf` attribute on a path is meant to mark Setting the `text` attribute on a path enables end-of-line
the path as a "text" file. 'core.autocrlf' conversion normalization and marks the path as a text file. End-of-line
takes place without guessing the content type by conversion takes place without guessing the content type.
inspection.


Unset:: Unset::


Unsetting the `crlf` attribute on a path tells git not to Unsetting the `text` attribute on a path tells git not to
attempt any end-of-line conversion upon checkin or checkout. attempt any end-of-line conversion upon checkin or checkout.


Set to string value "auto"::

When `text` is set to "auto", the path is marked for automatic
end-of-line normalization. If git decides that the content is
text, its line endings are normalized to LF on checkin.

Unspecified:: Unspecified::


Unspecified `crlf` attribute tells git to apply the If the `text` attribute is unspecified, git uses the
`core.autocrlf` conversion when the file content looks `core.autocrlf` configuration variable to determine if the
like text. file should be converted.


Set to string value "input":: Any other value causes git to act as if `text` has been left
unspecified.


This is similar to setting the attribute to `true`, but `eol`
also forces git to act as if `core.autocrlf` is set to ^^^^^
`input` for the path.


Any other value set to `crlf` attribute is ignored and git acts This attribute sets a specific line-ending style to be used in the
as if the attribute is left unspecified. working directory. It enables end-of-line normalization without any
content checks, effectively setting the `text` attribute.


Set to string value "crlf"::


The `core.autocrlf` conversion This setting forces git to normalize line endings for this
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ file on checkin and convert them to CRLF when the file is
checked out.


If the configuration variable `core.autocrlf` is false, no Set to string value "lf"::
conversion is done.


When `core.autocrlf` is true, it means that the platform wants This setting forces git to normalize line endings to LF on
CRLF line endings for files in the working tree, and you want to checkin and prevents conversion to CRLF when the file is
convert them back to the normal LF line endings when checking checked out.
in to the repository.


When `core.autocrlf` is set to "input", line endings are Backwards compatibility with `crlf` attribute
converted to LF upon checkin, but there is no conversion done ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
upon checkout.
For backwards compatibility, the `crlf` attribute is interpreted as
follows:

------------------------
crlf text
-crlf -text
crlf=input eol=lf
------------------------

End-of-line conversion
^^^^^^^^^^^^^^^^^^^^^^

While git normally leaves file contents alone, it can be configured to
normalize line endings to LF in the repository and, optionally, to
convert them to CRLF when files are checked out.

Here is an example that will make git normalize .txt, .vcproj and .sh
files, ensure that .vcproj files have CRLF and .sh files have LF in
the working directory, and prevent .jpg files from being normalized
regardless of their content.

------------------------
*.txt text
*.vcproj eol=crlf
*.sh eol=lf
*.jpg -text
------------------------

Other source code management systems normalize all text files in their
repositories, and there are two ways to enable similar automatic
normalization in git.

If you simply want to have CRLF line endings in your working directory
regardless of the repository you are working with, you can set the
config variable "core.autocrlf" without changing any attributes.

------------------------
[core]
autocrlf = true
------------------------

This does not force normalization of all text files, but does ensure
that text files that you introduce to the repository have their line
endings normalized to LF when they are added, and that files that are
already normalized in the repository stay normalized.

If you want to interoperate with a source code management system that
enforces end-of-line normalization, or you simply want all text files
in your repository to be normalized, you should instead set the `text`
attribute to "auto" for _all_ files.

------------------------
* text=auto
------------------------

This ensures that all files that git considers to be text will have
normalized (LF) line endings in the repository. The `core.eol`
configuration variable controls which line endings git will use for
normalized files in your working directory; the default is to use the
native line ending for your platform, or CRLF if `core.autocrlf` is
set.

NOTE: When `text=auto` normalization is enabled in an existing
repository, any text files containing CRLFs should be normalized. If
they are not they will be normalized the next time someone tries to
change them, causing unfortunate misattribution. From a clean working
directory:

-------------------------------------------------
$ echo "* text=auto" >>.gitattributes
$ rm .git/index # Remove the index to force git to
$ git reset # re-scan the working directory
$ git status # Show files that will be normalized
$ git add -u
$ git add .gitattributes
$ git commit -m "Introduce end-of-line normalization"
-------------------------------------------------

If any files that should not be normalized show up in 'git status',
unset their `text` attribute before running 'git add -u'.

------------------------
manual.pdf -text
------------------------

Conversely, text files that git does not detect can have normalization
enabled manually.

------------------------
weirdchars.txt text
------------------------


If `core.safecrlf` is set to "true" or "warn", git verifies if If `core.safecrlf` is set to "true" or "warn", git verifies if
the conversion is reversible for the current setting of the conversion is reversible for the current setting of
@ -223,11 +324,11 @@ Interaction between checkin/checkout attributes
In the check-in codepath, the worktree file is first converted In the check-in codepath, the worktree file is first converted
with `filter` driver (if specified and corresponding driver with `filter` driver (if specified and corresponding driver
defined), then the result is processed with `ident` (if defined), then the result is processed with `ident` (if
specified), and then finally with `crlf` (again, if specified specified), and then finally with `text` (again, if specified
and applicable). and applicable).


In the check-out codepath, the blob content is first converted In the check-out codepath, the blob content is first converted
with `crlf`, and then `ident` and fed to `filter`. with `text`, and then `ident` and fed to `filter`.




Generating diff text Generating diff text
@ -651,7 +752,7 @@ You do not want any end-of-line conversions applied to, nor textual diffs
produced for, any binary file you track. You would need to specify e.g. produced for, any binary file you track. You would need to specify e.g.


------------ ------------
*.jpg -crlf -diff *.jpg -text -diff
------------ ------------


but that may become cumbersome, when you have many attributes. Using but that may become cumbersome, when you have many attributes. Using
@ -664,7 +765,7 @@ the same time. The system knows a built-in attribute macro, `binary`:


which is equivalent to the above. Note that the attribute macros can only which is equivalent to the above. Note that the attribute macros can only
be "Set" (see the above example that sets "binary" macro as if it were an be "Set" (see the above example that sets "binary" macro as if it were an
ordinary attribute --- setting it in turn unsets "crlf" and "diff"). ordinary attribute --- setting it in turn unsets "text" and "diff").




DEFINING ATTRIBUTE MACROS DEFINING ATTRIBUTE MACROS
@ -675,7 +776,7 @@ at the toplevel (i.e. not in any subdirectory). The built-in attribute
macro "binary" is equivalent to: macro "binary" is equivalent to:


------------ ------------
[attr]binary -diff -crlf [attr]binary -diff -text
------------ ------------





View File

@ -233,6 +233,8 @@ all::
# #
# Define CHECK_HEADER_DEPENDENCIES to check for problems in the hard-coded # Define CHECK_HEADER_DEPENDENCIES to check for problems in the hard-coded
# dependency rules. # dependency rules.
#
# Define NATIVE_CRLF if your platform uses CRLF for line endings.


GIT-VERSION-FILE: FORCE GIT-VERSION-FILE: FORCE
@$(SHELL_PATH) ./GIT-VERSION-GEN @$(SHELL_PATH) ./GIT-VERSION-GEN
@ -1056,6 +1058,7 @@ ifeq ($(uname_S),Windows)
NO_CURL = YesPlease NO_CURL = YesPlease
NO_PYTHON = YesPlease NO_PYTHON = YesPlease
BLK_SHA1 = YesPlease BLK_SHA1 = YesPlease
NATIVE_CRLF = YesPlease


CC = compat/vcbuild/scripts/clink.pl CC = compat/vcbuild/scripts/clink.pl
AR = compat/vcbuild/scripts/lib.pl AR = compat/vcbuild/scripts/lib.pl

2
attr.c
View File

@ -287,7 +287,7 @@ static void free_attr_elem(struct attr_stack *e)
} }


static const char *builtin_attr[] = { static const char *builtin_attr[] = {
"[attr]binary -diff -crlf", "[attr]binary -diff -text",
NULL, NULL,
}; };



22
cache.h
View File

@ -547,7 +547,6 @@ extern int core_compression_seen;
extern size_t packed_git_window_size; extern size_t packed_git_window_size;
extern size_t packed_git_limit; extern size_t packed_git_limit;
extern size_t delta_base_cache_limit; extern size_t delta_base_cache_limit;
extern int auto_crlf;
extern int read_replace_refs; extern int read_replace_refs;
extern int fsync_object_files; extern int fsync_object_files;
extern int core_preload_index; extern int core_preload_index;
@ -561,6 +560,27 @@ enum safe_crlf {


extern enum safe_crlf safe_crlf; extern enum safe_crlf safe_crlf;


enum auto_crlf {
AUTO_CRLF_FALSE = 0,
AUTO_CRLF_TRUE = 1,
AUTO_CRLF_INPUT = -1,
};

extern enum auto_crlf auto_crlf;

enum eol {
EOL_UNSET,
EOL_CRLF,
EOL_LF,
#ifdef NATIVE_CRLF
EOL_NATIVE = EOL_CRLF
#else
EOL_NATIVE = EOL_LF
#endif
};

extern enum eol eol;

enum branch_track { enum branch_track {
BRANCH_TRACK_UNSPECIFIED = -1, BRANCH_TRACK_UNSPECIFIED = -1,
BRANCH_TRACK_NEVER = 0, BRANCH_TRACK_NEVER = 0,

View File

@ -517,7 +517,9 @@ static int git_default_core_config(const char *var, const char *value)


if (!strcmp(var, "core.autocrlf")) { if (!strcmp(var, "core.autocrlf")) {
if (value && !strcasecmp(value, "input")) { if (value && !strcasecmp(value, "input")) {
auto_crlf = -1; if (eol == EOL_CRLF)
return error("core.autocrlf=input conflicts with core.eol=crlf");
auto_crlf = AUTO_CRLF_INPUT;
return 0; return 0;
} }
auto_crlf = git_config_bool(var, value); auto_crlf = git_config_bool(var, value);
@ -533,6 +535,20 @@ static int git_default_core_config(const char *var, const char *value)
return 0; return 0;
} }


if (!strcmp(var, "core.eol")) {
if (value && !strcasecmp(value, "lf"))
eol = EOL_LF;
else if (value && !strcasecmp(value, "crlf"))
eol = EOL_CRLF;
else if (value && !strcasecmp(value, "native"))
eol = EOL_NATIVE;
else
eol = EOL_UNSET;
if (eol == EOL_CRLF && auto_crlf == AUTO_CRLF_INPUT)
return error("core.autocrlf=input conflicts with core.eol=crlf");
return 0;
}

if (!strcmp(var, "core.notesref")) { if (!strcmp(var, "core.notesref")) {
notes_ref_name = xstrdup(value); notes_ref_name = xstrdup(value);
return 0; return 0;

150
convert.c
View File

@ -8,13 +8,17 @@
* This should use the pathname to decide on whether it wants to do some * This should use the pathname to decide on whether it wants to do some
* more interesting conversions (automatic gzip/unzip, general format * more interesting conversions (automatic gzip/unzip, general format
* conversions etc etc), but by default it just does automatic CRLF<->LF * conversions etc etc), but by default it just does automatic CRLF<->LF
* translation when the "auto_crlf" option is set. * translation when the "text" attribute or "auto_crlf" option is set.
*/ */


#define CRLF_GUESS (-1) enum action {
#define CRLF_BINARY 0 CRLF_GUESS = -1,
#define CRLF_TEXT 1 CRLF_BINARY = 0,
#define CRLF_INPUT 2 CRLF_TEXT,
CRLF_INPUT,
CRLF_CRLF,
CRLF_AUTO,
};


struct text_stat { struct text_stat {
/* NUL, CR, LF and CRLF counts */ /* NUL, CR, LF and CRLF counts */
@ -89,31 +93,55 @@ static int is_binary(unsigned long size, struct text_stat *stats)
return 0; return 0;
} }


static void check_safe_crlf(const char *path, int action, static enum eol determine_output_conversion(enum action action) {
switch (action) {
case CRLF_BINARY:
return EOL_UNSET;
case CRLF_CRLF:
return EOL_CRLF;
case CRLF_INPUT:
return EOL_LF;
case CRLF_GUESS:
if (!auto_crlf)
return EOL_UNSET;
/* fall through */
case CRLF_TEXT:
case CRLF_AUTO:
if (auto_crlf == AUTO_CRLF_TRUE)
return EOL_CRLF;
else if (auto_crlf == AUTO_CRLF_INPUT)
return EOL_LF;
else if (eol == EOL_UNSET)
return EOL_NATIVE;
}
return eol;
}

static void check_safe_crlf(const char *path, enum action action,
struct text_stat *stats, enum safe_crlf checksafe) struct text_stat *stats, enum safe_crlf checksafe)
{ {
if (!checksafe) if (!checksafe)
return; return;


if (action == CRLF_INPUT || auto_crlf <= 0) { if (determine_output_conversion(action) == EOL_LF) {
/* /*
* CRLFs would not be restored by checkout: * CRLFs would not be restored by checkout:
* check if we'd remove CRLFs * check if we'd remove CRLFs
*/ */
if (stats->crlf) { if (stats->crlf) {
if (checksafe == SAFE_CRLF_WARN) if (checksafe == SAFE_CRLF_WARN)
warning("CRLF will be replaced by LF in %s.", path); warning("CRLF will be replaced by LF in %s.\nThe file will have its original line endings in your working directory.", path);
else /* i.e. SAFE_CRLF_FAIL */ else /* i.e. SAFE_CRLF_FAIL */
die("CRLF would be replaced by LF in %s.", path); die("CRLF would be replaced by LF in %s.", path);
} }
} else if (auto_crlf > 0) { } else if (determine_output_conversion(action) == EOL_CRLF) {
/* /*
* CRLFs would be added by checkout: * CRLFs would be added by checkout:
* check if we have "naked" LFs * check if we have "naked" LFs
*/ */
if (stats->lf != stats->crlf) { if (stats->lf != stats->crlf) {
if (checksafe == SAFE_CRLF_WARN) if (checksafe == SAFE_CRLF_WARN)
warning("LF will be replaced by CRLF in %s", path); warning("LF will be replaced by CRLF in %s.\nThe file will have its original line endings in your working directory.", path);
else /* i.e. SAFE_CRLF_FAIL */ else /* i.e. SAFE_CRLF_FAIL */
die("LF would be replaced by CRLF in %s", path); die("LF would be replaced by CRLF in %s", path);
} }
@ -158,17 +186,18 @@ static int has_cr_in_index(const char *path)
} }


static int crlf_to_git(const char *path, const char *src, size_t len, static int crlf_to_git(const char *path, const char *src, size_t len,
struct strbuf *buf, int action, enum safe_crlf checksafe) struct strbuf *buf, enum action action, enum safe_crlf checksafe)
{ {
struct text_stat stats; struct text_stat stats;
char *dst; char *dst;


if ((action == CRLF_BINARY) || !auto_crlf || !len) if (action == CRLF_BINARY ||
(action == CRLF_GUESS && auto_crlf == AUTO_CRLF_FALSE) || !len)
return 0; return 0;


gather_stats(src, len, &stats); gather_stats(src, len, &stats);


if (action == CRLF_GUESS) { if (action == CRLF_AUTO || action == CRLF_GUESS) {
/* /*
* We're currently not going to even try to convert stuff * We're currently not going to even try to convert stuff
* that has bare CR characters. Does anybody do that crazy * that has bare CR characters. Does anybody do that crazy
@ -183,12 +212,14 @@ static int crlf_to_git(const char *path, const char *src, size_t len,
if (is_binary(len, &stats)) if (is_binary(len, &stats))
return 0; return 0;


/* if (action == CRLF_GUESS) {
* If the file in the index has any CR in it, do not convert. /*
* This is the new safer autocrlf handling. * If the file in the index has any CR in it, do not convert.
*/ * This is the new safer autocrlf handling.
if (has_cr_in_index(path)) */
return 0; if (has_cr_in_index(path))
return 0;
}
} }


check_safe_crlf(path, action, &stats, checksafe); check_safe_crlf(path, action, &stats, checksafe);
@ -201,7 +232,7 @@ static int crlf_to_git(const char *path, const char *src, size_t len,
if (strbuf_avail(buf) + buf->len < len) if (strbuf_avail(buf) + buf->len < len)
strbuf_grow(buf, len - buf->len); strbuf_grow(buf, len - buf->len);
dst = buf->buf; dst = buf->buf;
if (action == CRLF_GUESS) { if (action == CRLF_AUTO || action == CRLF_GUESS) {
/* /*
* If we guessed, we already know we rejected a file with * If we guessed, we already know we rejected a file with
* lone CR, and we can strip a CR without looking at what * lone CR, and we can strip a CR without looking at what
@ -224,16 +255,12 @@ static int crlf_to_git(const char *path, const char *src, size_t len,
} }


static int crlf_to_worktree(const char *path, const char *src, size_t len, static int crlf_to_worktree(const char *path, const char *src, size_t len,
struct strbuf *buf, int action) struct strbuf *buf, enum action action)
{ {
char *to_free = NULL; char *to_free = NULL;
struct text_stat stats; struct text_stat stats;


if ((action == CRLF_BINARY) || (action == CRLF_INPUT) || if (!len || determine_output_conversion(action) != EOL_CRLF)
auto_crlf <= 0)
return 0;

if (!len)
return 0; return 0;


gather_stats(src, len, &stats); gather_stats(src, len, &stats);
@ -246,11 +273,13 @@ static int crlf_to_worktree(const char *path, const char *src, size_t len,
if (stats.lf == stats.crlf) if (stats.lf == stats.crlf)
return 0; return 0;


if (action == CRLF_GUESS) { if (action == CRLF_AUTO || action == CRLF_GUESS) {
/* If we have any CR or CRLF line endings, we do not touch it */ if (action == CRLF_GUESS) {
/* This is the new safer autocrlf-handling */ /* If we have any CR or CRLF line endings, we do not touch it */
if (stats.cr > 0 || stats.crlf > 0) /* This is the new safer autocrlf-handling */
return 0; if (stats.cr > 0 || stats.crlf > 0)
return 0;
}


/* If we have any bare CR characters, we're not going to touch it */ /* If we have any bare CR characters, we're not going to touch it */
if (stats.cr != stats.crlf) if (stats.cr != stats.crlf)
@ -425,12 +454,16 @@ static int read_convert_config(const char *var, const char *value, void *cb)


static void setup_convert_check(struct git_attr_check *check) static void setup_convert_check(struct git_attr_check *check)
{ {
static struct git_attr *attr_text;
static struct git_attr *attr_crlf; static struct git_attr *attr_crlf;
static struct git_attr *attr_eol;
static struct git_attr *attr_ident; static struct git_attr *attr_ident;
static struct git_attr *attr_filter; static struct git_attr *attr_filter;


if (!attr_crlf) { if (!attr_text) {
attr_text = git_attr("text");
attr_crlf = git_attr("crlf"); attr_crlf = git_attr("crlf");
attr_eol = git_attr("eol");
attr_ident = git_attr("ident"); attr_ident = git_attr("ident");
attr_filter = git_attr("filter"); attr_filter = git_attr("filter");
user_convert_tail = &user_convert; user_convert_tail = &user_convert;
@ -439,6 +472,8 @@ static void setup_convert_check(struct git_attr_check *check)
check[0].attr = attr_crlf; check[0].attr = attr_crlf;
check[1].attr = attr_ident; check[1].attr = attr_ident;
check[2].attr = attr_filter; check[2].attr = attr_filter;
check[3].attr = attr_eol;
check[4].attr = attr_text;
} }


static int count_ident(const char *cp, unsigned long size) static int count_ident(const char *cp, unsigned long size)
@ -619,9 +654,24 @@ static int git_path_check_crlf(const char *path, struct git_attr_check *check)
; ;
else if (!strcmp(value, "input")) else if (!strcmp(value, "input"))
return CRLF_INPUT; return CRLF_INPUT;
else if (!strcmp(value, "auto"))
return CRLF_AUTO;
return CRLF_GUESS; return CRLF_GUESS;
} }


static int git_path_check_eol(const char *path, struct git_attr_check *check)
{
const char *value = check->value;

if (ATTR_UNSET(value))
;
else if (!strcmp(value, "lf"))
return EOL_LF;
else if (!strcmp(value, "crlf"))
return EOL_CRLF;
return EOL_UNSET;
}

static struct convert_driver *git_path_check_convert(const char *path, static struct convert_driver *git_path_check_convert(const char *path,
struct git_attr_check *check) struct git_attr_check *check)
{ {
@ -643,20 +693,34 @@ static int git_path_check_ident(const char *path, struct git_attr_check *check)
return !!ATTR_TRUE(value); return !!ATTR_TRUE(value);
} }


enum action determine_action(enum action text_attr, enum eol eol_attr) {
if (text_attr == CRLF_BINARY)
return CRLF_BINARY;
if (eol_attr == EOL_LF)
return CRLF_INPUT;
if (eol_attr == EOL_CRLF)
return CRLF_CRLF;
return text_attr;
}

int convert_to_git(const char *path, const char *src, size_t len, int convert_to_git(const char *path, const char *src, size_t len,
struct strbuf *dst, enum safe_crlf checksafe) struct strbuf *dst, enum safe_crlf checksafe)
{ {
struct git_attr_check check[3]; struct git_attr_check check[5];
int crlf = CRLF_GUESS; enum action action = CRLF_GUESS;
enum eol eol_attr = EOL_UNSET;
int ident = 0, ret = 0; int ident = 0, ret = 0;
const char *filter = NULL; const char *filter = NULL;


setup_convert_check(check); setup_convert_check(check);
if (!git_checkattr(path, ARRAY_SIZE(check), check)) { if (!git_checkattr(path, ARRAY_SIZE(check), check)) {
struct convert_driver *drv; struct convert_driver *drv;
crlf = git_path_check_crlf(path, check + 0); action = git_path_check_crlf(path, check + 4);
if (action == CRLF_GUESS)
action = git_path_check_crlf(path, check + 0);
ident = git_path_check_ident(path, check + 1); ident = git_path_check_ident(path, check + 1);
drv = git_path_check_convert(path, check + 2); drv = git_path_check_convert(path, check + 2);
eol_attr = git_path_check_eol(path, check + 3);
if (drv && drv->clean) if (drv && drv->clean)
filter = drv->clean; filter = drv->clean;
} }
@ -666,7 +730,8 @@ int convert_to_git(const char *path, const char *src, size_t len,
src = dst->buf; src = dst->buf;
len = dst->len; len = dst->len;
} }
ret |= crlf_to_git(path, src, len, dst, crlf, checksafe); action = determine_action(action, eol_attr);
ret |= crlf_to_git(path, src, len, dst, action, checksafe);
if (ret) { if (ret) {
src = dst->buf; src = dst->buf;
len = dst->len; len = dst->len;
@ -676,17 +741,21 @@ int convert_to_git(const char *path, const char *src, size_t len,


int convert_to_working_tree(const char *path, const char *src, size_t len, struct strbuf *dst) int convert_to_working_tree(const char *path, const char *src, size_t len, struct strbuf *dst)
{ {
struct git_attr_check check[3]; struct git_attr_check check[5];
int crlf = CRLF_GUESS; enum action action = CRLF_GUESS;
enum eol eol_attr = EOL_UNSET;
int ident = 0, ret = 0; int ident = 0, ret = 0;
const char *filter = NULL; const char *filter = NULL;


setup_convert_check(check); setup_convert_check(check);
if (!git_checkattr(path, ARRAY_SIZE(check), check)) { if (!git_checkattr(path, ARRAY_SIZE(check), check)) {
struct convert_driver *drv; struct convert_driver *drv;
crlf = git_path_check_crlf(path, check + 0); action = git_path_check_crlf(path, check + 4);
if (action == CRLF_GUESS)
action = git_path_check_crlf(path, check + 0);
ident = git_path_check_ident(path, check + 1); ident = git_path_check_ident(path, check + 1);
drv = git_path_check_convert(path, check + 2); drv = git_path_check_convert(path, check + 2);
eol_attr = git_path_check_eol(path, check + 3);
if (drv && drv->smudge) if (drv && drv->smudge)
filter = drv->smudge; filter = drv->smudge;
} }
@ -696,7 +765,8 @@ int convert_to_working_tree(const char *path, const char *src, size_t len, struc
src = dst->buf; src = dst->buf;
len = dst->len; len = dst->len;
} }
ret |= crlf_to_worktree(path, src, len, dst, crlf); action = determine_action(action, eol_attr);
ret |= crlf_to_worktree(path, src, len, dst, action);
if (ret) { if (ret) {
src = dst->buf; src = dst->buf;
len = dst->len; len = dst->len;

View File

@ -38,8 +38,9 @@ const char *pager_program;
int pager_use_color = 1; int pager_use_color = 1;
const char *editor_program; const char *editor_program;
const char *excludes_file; const char *excludes_file;
int auto_crlf = 0; /* 1: both ways, -1: only when adding git objects */ enum auto_crlf auto_crlf = AUTO_CRLF_FALSE;
int read_replace_refs = 1; int read_replace_refs = 1;
enum eol eol = EOL_UNSET;
enum safe_crlf safe_crlf = SAFE_CRLF_WARN; enum safe_crlf safe_crlf = SAFE_CRLF_WARN;
unsigned whitespace_rule_cfg = WS_DEFAULT_RULE; unsigned whitespace_rule_cfg = WS_DEFAULT_RULE;
enum branch_track git_branch_track = BRANCH_TRACK_REMOTE; enum branch_track git_branch_track = BRANCH_TRACK_REMOTE;

View File

@ -2415,15 +2415,20 @@ sub kopts_from_path
if ( defined ( $cfg->{gitcvs}{usecrlfattr} ) and if ( defined ( $cfg->{gitcvs}{usecrlfattr} ) and
$cfg->{gitcvs}{usecrlfattr} =~ /\s*(1|true|yes)\s*$/i ) $cfg->{gitcvs}{usecrlfattr} =~ /\s*(1|true|yes)\s*$/i )
{ {
my ($val) = check_attr( "crlf", $path ); my ($val) = check_attr( "text", $path );
if ( $val eq "set" ) if ( $val eq "unspecified" )
{ {
return ""; $val = check_attr( "crlf", $path );
} }
elsif ( $val eq "unset" ) if ( $val eq "unset" )
{ {
return "-kb" return "-kb"
} }
elsif ( check_attr( "eol", $path ) ne "unspecified" ||
$val eq "set" || $val eq "input" )
{
return "";
}
else else
{ {
$log->info("Unrecognized check_attr crlf $path : $val"); $log->info("Unrecognized check_attr crlf $path : $val");

155
t/t0025-crlf-auto.sh Executable file
View File

@ -0,0 +1,155 @@
#!/bin/sh

test_description='CRLF conversion'

. ./test-lib.sh

has_cr() {
tr '\015' Q <"$1" | grep Q >/dev/null
}

test_expect_success setup '

git config core.autocrlf false &&

for w in Hello world how are you; do echo $w; done >one &&
for w in I am very very fine thank you; do echo ${w}Q; done | q_to_cr >two &&
for w in Oh here is a QNUL byte how alarming; do echo ${w}; done | q_to_nul >three &&
git add . &&

git commit -m initial &&

one=`git rev-parse HEAD:one` &&
two=`git rev-parse HEAD:two` &&
three=`git rev-parse HEAD:three` &&

echo happy.
'

test_expect_success 'default settings cause no changes' '

rm -f .gitattributes tmp one two three &&
git read-tree --reset -u HEAD &&

! has_cr one &&
has_cr two &&
onediff=`git diff one` &&
twodiff=`git diff two` &&
threediff=`git diff three` &&
test -z "$onediff" -a -z "$twodiff" -a -z "$threediff"
'

test_expect_success 'crlf=true causes a CRLF file to be normalized' '

# Backwards compatibility check
rm -f .gitattributes tmp one two three &&
echo "two crlf" > .gitattributes &&
git read-tree --reset -u HEAD &&

# Note, "normalized" means that git will normalize it if added
has_cr two &&
twodiff=`git diff two` &&
test -n "$twodiff"
'

test_expect_success 'text=true causes a CRLF file to be normalized' '

rm -f .gitattributes tmp one two three &&
echo "two text" > .gitattributes &&
git read-tree --reset -u HEAD &&

# Note, "normalized" means that git will normalize it if added
has_cr two &&
twodiff=`git diff two` &&
test -n "$twodiff"
'

test_expect_success 'eol=crlf gives a normalized file CRLFs with autocrlf=false' '

rm -f .gitattributes tmp one two three &&
git config core.autocrlf false &&
echo "one eol=crlf" > .gitattributes &&
git read-tree --reset -u HEAD &&

has_cr one &&
onediff=`git diff one` &&
test -z "$onediff"
'

test_expect_success 'eol=crlf gives a normalized file CRLFs with autocrlf=input' '

rm -f .gitattributes tmp one two three &&
git config core.autocrlf input &&
echo "one eol=crlf" > .gitattributes &&
git read-tree --reset -u HEAD &&

has_cr one &&
onediff=`git diff one` &&
test -z "$onediff"
'

test_expect_success 'eol=lf gives a normalized file LFs with autocrlf=true' '

rm -f .gitattributes tmp one two three &&
git config core.autocrlf true &&
echo "one eol=lf" > .gitattributes &&
git read-tree --reset -u HEAD &&

! has_cr one &&
onediff=`git diff one` &&
test -z "$onediff"
'

test_expect_success 'autocrlf=true does not normalize CRLF files' '

rm -f .gitattributes tmp one two three &&
git config core.autocrlf true &&
git read-tree --reset -u HEAD &&

has_cr one &&
has_cr two &&
onediff=`git diff one` &&
twodiff=`git diff two` &&
threediff=`git diff three` &&
test -z "$onediff" -a -z "$twodiff" -a -z "$threediff"
'

test_expect_success 'text=auto, autocrlf=true _does_ normalize CRLF files' '

rm -f .gitattributes tmp one two three &&
git config core.autocrlf true &&
echo "* text=auto" > .gitattributes &&
git read-tree --reset -u HEAD &&

has_cr one &&
has_cr two &&
onediff=`git diff one` &&
twodiff=`git diff two` &&
threediff=`git diff three` &&
test -z "$onediff" -a -n "$twodiff" -a -z "$threediff"
'

test_expect_success 'text=auto, autocrlf=true does not normalize binary files' '

rm -f .gitattributes tmp one two three &&
git config core.autocrlf true &&
echo "* text=auto" > .gitattributes &&
git read-tree --reset -u HEAD &&

! has_cr three &&
threediff=`git diff three` &&
test -z "$threediff"
'

test_expect_success 'eol=crlf _does_ normalize binary files' '

rm -f .gitattributes tmp one two three &&
echo "three eol=crlf" > .gitattributes &&
git read-tree --reset -u HEAD &&

has_cr three &&
threediff=`git diff three` &&
test -z "$threediff"
'

test_done

83
t/t0026-eol-config.sh Executable file
View File

@ -0,0 +1,83 @@
#!/bin/sh

test_description='CRLF conversion'

. ./test-lib.sh

has_cr() {
tr '\015' Q <"$1" | grep Q >/dev/null
}

test_expect_success setup '

git config core.autocrlf false &&

echo "one text" > .gitattributes

for w in Hello world how are you; do echo $w; done >one &&
for w in I am very very fine thank you; do echo $w; done >two &&
git add . &&

git commit -m initial &&

one=`git rev-parse HEAD:one` &&
two=`git rev-parse HEAD:two` &&

echo happy.
'

test_expect_success 'eol=lf puts LFs in normalized file' '

rm -f .gitattributes tmp one two &&
git config core.eol lf &&
git read-tree --reset -u HEAD &&

! has_cr one &&
! has_cr two &&
onediff=`git diff one` &&
twodiff=`git diff two` &&
test -z "$onediff" -a -z "$twodiff"
'

test_expect_success 'eol=crlf puts CRLFs in normalized file' '

rm -f .gitattributes tmp one two &&
git config core.eol crlf &&
git read-tree --reset -u HEAD &&

has_cr one &&
! has_cr two &&
onediff=`git diff one` &&
twodiff=`git diff two` &&
test -z "$onediff" -a -z "$twodiff"
'

test_expect_success 'autocrlf=true overrides eol=lf' '

rm -f .gitattributes tmp one two &&
git config core.eol lf &&
git config core.autocrlf true &&
git read-tree --reset -u HEAD &&

has_cr one &&
has_cr two &&
onediff=`git diff one` &&
twodiff=`git diff two` &&
test -z "$onediff" -a -z "$twodiff"
'

test_expect_success 'autocrlf=true overrides unset eol' '

rm -f .gitattributes tmp one two &&
git config --unset-all core.eol &&
git config core.autocrlf true &&
git read-tree --reset -u HEAD &&

has_cr one &&
has_cr two &&
onediff=`git diff one` &&
twodiff=`git diff two` &&
test -z "$onediff" -a -z "$twodiff"
'

test_done