From 3b059de54227e1fb0628f9cc47f2783e69c04401 Mon Sep 17 00:00:00 2001 From: Kevin Worm Date: Thu, 12 May 2022 10:33:33 +0200 Subject: [PATCH 1/4] Refactor early config parsing The Trace2 machinery wishes to parse the system and the global config early. To this end, it taps into the `common-main` framework to run first thing. There are more Git features that could benefit from such a handling, most notably the Windows-specific `core.longPaths` setting: If a user has worktrees whose path already requires long paths support, we cannot wait until we parse the the config settings in the usual way because the gitdir discovery needs to happen first and would fail because any `core.longPaths` setting in, say, `~/.gitconfig` would not have been parsed yet. To that end, let's refactor Trace2's early config parsing so that other users can tap into it, too. Signed-off-by: Kevin Worm Signed-off-by: Johannes Schindelin --- common-main.c | 10 ++++++++++ trace2.c | 2 +- trace2/tr2_sysenv.c | 20 +++++++++----------- trace2/tr2_sysenv.h | 3 ++- 4 files changed, 22 insertions(+), 13 deletions(-) diff --git a/common-main.c b/common-main.c index 29fb7452f8a0b7..269bb96bf7836c 100644 --- a/common-main.c +++ b/common-main.c @@ -1,6 +1,8 @@ #include "cache.h" +#include "config.h" #include "exec-cmd.h" #include "attr.h" +#include "trace2/tr2_sysenv.h" /* * Many parts of Git have subprograms communicate via pipe, expect the @@ -23,6 +25,11 @@ static void restore_sigpipe_to_default(void) signal(SIGPIPE, SIG_DFL); } +static int read_very_early_config_cb(const char *key, const char *value, void *d) +{ + return tr2_sysenv_cb(key, value, d); +} + int main(int argc, const char **argv) { int result; @@ -46,7 +53,10 @@ int main(int argc, const char **argv) attr_start(); + read_very_early_config(read_very_early_config_cb, NULL); + trace2_initialize(); + trace2_cmd_start(argv); trace2_collect_process_info(TRACE2_PROCESS_INFO_STARTUP); diff --git a/trace2.c b/trace2.c index e01cf77f1a894e..80b805074a6f1f 100644 --- a/trace2.c +++ b/trace2.c @@ -158,7 +158,7 @@ void trace2_initialize_fl(const char *file, int line) if (trace2_enabled) return; - tr2_sysenv_load(); + tr2_sysenv_check_size(); if (!tr2_tgt_want_builtins()) return; diff --git a/trace2/tr2_sysenv.c b/trace2/tr2_sysenv.c index a380dcf9105e8d..b7c371caf1fa7f 100644 --- a/trace2/tr2_sysenv.c +++ b/trace2/tr2_sysenv.c @@ -57,7 +57,14 @@ static struct tr2_sysenv_entry tr2_sysenv_settings[] = { }; /* clang-format on */ -static int tr2_sysenv_cb(const char *key, const char *value, void *d) +/* + * Load Trace2 settings from the system config (usually "/etc/gitconfig" + * unless we were built with a runtime-prefix). These are intended to + * define the default values for Trace2 as requested by the administrator. + * + * Then override with the Trace2 settings from the global config. + */ +int tr2_sysenv_cb(const char *key, const char *value, void *d) { int k; @@ -75,19 +82,10 @@ static int tr2_sysenv_cb(const char *key, const char *value, void *d) return 0; } -/* - * Load Trace2 settings from the system config (usually "/etc/gitconfig" - * unless we were built with a runtime-prefix). These are intended to - * define the default values for Trace2 as requested by the administrator. - * - * Then override with the Trace2 settings from the global config. - */ -void tr2_sysenv_load(void) +void tr2_sysenv_check_size(void) { if (ARRAY_SIZE(tr2_sysenv_settings) != TR2_SYSENV_MUST_BE_LAST) BUG("tr2_sysenv_settings size is wrong"); - - read_very_early_config(tr2_sysenv_cb, NULL); } /* diff --git a/trace2/tr2_sysenv.h b/trace2/tr2_sysenv.h index 3292ee15bc9676..685ba5d6ae0b92 100644 --- a/trace2/tr2_sysenv.h +++ b/trace2/tr2_sysenv.h @@ -30,7 +30,8 @@ enum tr2_sysenv_variable { TR2_SYSENV_MUST_BE_LAST }; -void tr2_sysenv_load(void); +int tr2_sysenv_cb(const char *key, const char *value, void *d); +void tr2_sysenv_check_size(void); const char *tr2_sysenv_get(enum tr2_sysenv_variable); const char *tr2_sysenv_display_name(enum tr2_sysenv_variable var); From 7a3a4c7a2a4fc0d7ac4bb1cd8ac2e6d9b6e52701 Mon Sep 17 00:00:00 2001 From: Kevin Worm Date: Thu, 12 May 2022 10:33:33 +0200 Subject: [PATCH 2/4] mingw: respect core.longPaths when initializing a new gitdir Use case: Allow git to checkout a repository or submodule in a directory with a long path when core.longpaths = true. Example: > ./git.exe config --global core.longpaths true > ./git.exe clone https://github.com/git/git.git --recurse-submodules \ /c/eval/git_test/loooooooooooooooooooooooooooooooooooooooooooooooooooooooo\ oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo\ oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo\ oooooooong Context: $ sh --version GNU bash, version 4.4.23(1)-release (x86_64-pc-msys) $ ./git.exe --version --build-options git version 2.36.1.windows.1 cpu: x86_64 built from commit: e2ff68a2d1426758c78d023f863bfa1e03cbc768 sizeof-long: 4 sizeof-size_t: 8 shell-path: /bin/sh feature: fsmonitor--daemon Error: fatal: '$GIT_DIR' too big. Problem analysis: setup_explicit_git_dir in setup.c uses PATH_MAX to check if the git dir is to long. On Windows PATH_MAX is set by limit.h to 260 and setup_explicit_git_dir ignores core.longpaths. Solution: The implementation is based on the solution proposed by Johannes Schindelin, see: https://github.com/git-for-windows/git/issues/3372#issuecomment-904598455 * Refactor the part of trace2_initialize() that reads the config. * Make tr2_sysenv_cb() a public function. * No longer calling it from trace2_initialize(), but from a static callback function in common-main.c. * Calling read_very_early_config() explicitly in main(), with that static callback function that calls into tr2_sysenv_cb(). * Extend the static callback function for Windows, and parse core.longPaths in that function. * Extend startup_info struct in cache.h with 'int max_long_path' so we can use it in setup_explicit_git_dir instead of PATH_MAX. This fixes https://github.com/git-for-windows/git/issues/3372 Signed-off-by: Kevin Worm Signed-off-by: Johannes Schindelin --- common-main.c | 5 ++++- compat/mingw.h | 6 ++++++ git-compat-util.h | 4 ++++ setup.c | 2 +- 4 files changed, 15 insertions(+), 2 deletions(-) diff --git a/common-main.c b/common-main.c index 269bb96bf7836c..3a64144960056b 100644 --- a/common-main.c +++ b/common-main.c @@ -27,7 +27,10 @@ static void restore_sigpipe_to_default(void) static int read_very_early_config_cb(const char *key, const char *value, void *d) { - return tr2_sysenv_cb(key, value, d); + if (starts_with(key, "core.")) + return platform_core_config(key, value, d); + else + return tr2_sysenv_cb(key, value, d); } int main(int argc, const char **argv) diff --git a/compat/mingw.h b/compat/mingw.h index c15f6c6303b76a..bead61287aa717 100644 --- a/compat/mingw.h +++ b/compat/mingw.h @@ -552,6 +552,12 @@ int is_valid_win32_path(const char *path, int allow_literal_nul); */ int handle_long_path(wchar_t *path, int len, int max_path, int expand); +static inline int mingw_get_max_path_size(void) +{ + return core_long_paths ? MAX_LONG_PATH : MAX_PATH; +} +#define get_max_path_size mingw_get_max_path_size + /** * Converts UTF-8 encoded string to UTF-16LE. * diff --git a/git-compat-util.h b/git-compat-util.h index 586d1a199a0f0d..38a01e050be208 100644 --- a/git-compat-util.h +++ b/git-compat-util.h @@ -372,6 +372,10 @@ char *gitdirname(char *); #define PATH_MAX 4096 #endif +#ifndef get_max_path_size +#define get_max_path_size() PATH_MAX +#endif + typedef uintmax_t timestamp_t; #define PRItime PRIuMAX #define parse_timestamp strtoumax diff --git a/setup.c b/setup.c index 96d0d6e51024b8..94c7c6b954cbe5 100644 --- a/setup.c +++ b/setup.c @@ -875,7 +875,7 @@ static const char *setup_explicit_git_dir(const char *gitdirenv, char *gitfile; int offset; - if (PATH_MAX - 40 < strlen(gitdirenv)) + if (get_max_path_size() - 40 < strlen(gitdirenv)) die(_("'$%s' too big"), GIT_DIR_ENVIRONMENT); gitfile = (char*)read_gitfile(gitdirenv); From 2ac3b25d03bac52a1f4baf694e1528e7ef97e503 Mon Sep 17 00:00:00 2001 From: Johannes Schindelin Date: Mon, 30 May 2022 12:24:13 +0200 Subject: [PATCH 3/4] t2031: verify that `git init` can create a long path This currently fails, with: ++ p=/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef ++ p=y/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef ++ test_config_global core.longpaths true ++ test_when_finished 'test_unconfig --global '\''core.longpaths'\''' ++ test 0 = 0 ++ test_cleanup='{ test_unconfig --global '\''core.longpaths'\'' } && (exit "$eval_ret"); eval_ret=$?; :' ++ git config --global core.longpaths true ++ git init y/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef fatal: cannot chdir to y/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef: No such file or directory error: last command exited with $?=128 not ok 7 - init with long path Signed-off-by: Johannes Schindelin --- t/t2031-checkout-long-paths.sh | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/t/t2031-checkout-long-paths.sh b/t/t2031-checkout-long-paths.sh index f30f8920ca689c..d38214beea241a 100755 --- a/t/t2031-checkout-long-paths.sh +++ b/t/t2031-checkout-long-paths.sh @@ -99,4 +99,11 @@ test_expect_success SHORTABSPATH 'clean up path close to MAX_PATH' ' test ! -d "$subdir1" ' +test_expect_success 'init with long path' ' + p=/123456789abcdef/123456789abcdef/123456789abcdef/123456789abc/ef && + p=y$p$p$p$p && + test_config_global core.longpaths true && + git init $p +' + test_done From 82225ba08e1ba9c492b293b239d0d131fe3acdf5 Mon Sep 17 00:00:00 2001 From: Kevin Worm Date: Thu, 2 Jun 2022 16:20:36 +0200 Subject: [PATCH 4/4] Fix long path issues for git init and git clone `t2031-checkout-long-paths.sh` runs successfully, but there are still issues remaining preventing submodules from working in a long path dir. Signed-off-by: Kevin Worm --- compat/mingw.c | 33 ++++++++++++++++++--------------- compat/win32/git.manifest | 9 +++++++-- git.rc | 2 +- 3 files changed, 26 insertions(+), 18 deletions(-) diff --git a/compat/mingw.c b/compat/mingw.c index 182498bbb48526..25293452d836c1 100644 --- a/compat/mingw.c +++ b/compat/mingw.c @@ -933,7 +933,7 @@ static int current_directory_len = 0; int mingw_chdir(const char *dirname) { - int result; + int result = 0; wchar_t wdirname[MAX_LONG_PATH]; if (xutftowcs_long_path(wdirname, dirname) < 0) return -1; @@ -954,8 +954,11 @@ int mingw_chdir(const char *dirname) CloseHandle(hnd); } - result = _wchdir(normalize_ntpath(wdirname)); + if(!SetCurrentDirectoryW(wdirname)) { + result = -1; + } current_directory_len = GetCurrentDirectoryW(0, NULL); + return result; } @@ -1254,15 +1257,11 @@ unsigned int sleep (unsigned int seconds) char *mingw_mktemp(char *template) { - wchar_t wtemplate[MAX_PATH]; + wchar_t wtemplate[MAX_LONG_PATH]; int offset = 0; - /* we need to return the path, thus no long paths here! */ - if (xutftowcsn(wtemplate, template, MAX_PATH, -1) < 0) { - if (errno == ERANGE) - errno = ENAMETOOLONG; + if (xutftowcs_long_path(wtemplate, template) < 0) return NULL; - } if (is_dir_sep(template[0]) && !is_dir_sep(template[1]) && iswalpha(wtemplate[0]) && wtemplate[1] == L':') { @@ -1278,10 +1277,14 @@ char *mingw_mktemp(char *template) int mkstemp(char *template) { - char *filename = mktemp(template); - if (filename == NULL) + wchar_t wtemplate[MAX_LONG_PATH]; + + if (xutftowcs_long_path(wtemplate, template) < 0) return -1; - return open(filename, O_RDWR | O_CREAT, 0600); + if (!_wmktemp(wtemplate)) + return -1; + + return _wopen(wtemplate, O_RDWR | O_CREAT, 0600); } int gettimeofday(struct timeval *tv, void *tz) @@ -1401,7 +1404,7 @@ char *mingw_strbuf_realpath(struct strbuf *resolved, const char *path) char *mingw_getcwd(char *pointer, int len) { - wchar_t cwd[MAX_PATH], wpointer[MAX_PATH]; + wchar_t cwd[MAX_LONG_PATH], wpointer[MAX_LONG_PATH]; DWORD ret = GetCurrentDirectoryW(ARRAY_SIZE(cwd), cwd); HANDLE hnd; @@ -3488,16 +3491,16 @@ static PSID get_current_user_sid(void) int is_path_owned_by_current_sid(const char *path) { - WCHAR wpath[MAX_PATH]; + WCHAR wpath[MAX_LONG_PATH]; PSID sid = NULL; PSECURITY_DESCRIPTOR descriptor = NULL; DWORD err; - static wchar_t home[MAX_PATH]; + static wchar_t home[MAX_LONG_PATH]; int result = 0; - if (xutftowcs_path(wpath, path) < 0) + if (xutftowcs_long_path(wpath, path) < 0) return 0; /* diff --git a/compat/win32/git.manifest b/compat/win32/git.manifest index 771e3cce431598..6777fd65618809 100644 --- a/compat/win32/git.manifest +++ b/compat/win32/git.manifest @@ -1,6 +1,11 @@ - + + + + true + + @@ -18,7 +23,7 @@ - + diff --git a/git.rc b/git.rc index cc3fdc6cc6cb83..cd37619e86421c 100644 --- a/git.rc +++ b/git.rc @@ -21,4 +21,4 @@ BEGIN END END -1 RT_MANIFEST "compat/win32/git.manifest" +1 24 "compat/win32/git.manifest"