diff options
Diffstat (limited to 'builtin/init-db.c')
| -rw-r--r-- | builtin/init-db.c | 79 | 
1 files changed, 46 insertions, 33 deletions
| diff --git a/builtin/init-db.c b/builtin/init-db.c index 56f85e239a..4335738135 100644 --- a/builtin/init-db.c +++ b/builtin/init-db.c @@ -119,15 +119,18 @@ static void copy_templates(const char *template_dir)  	DIR *dir;  	const char *git_dir = get_git_dir();  	int len = strlen(git_dir); +	char *to_free = NULL;  	if (!template_dir)  		template_dir = getenv(TEMPLATE_DIR_ENVIRONMENT);  	if (!template_dir)  		template_dir = init_db_template_dir;  	if (!template_dir) -		template_dir = system_path(DEFAULT_GIT_TEMPLATE_DIR); -	if (!template_dir[0]) +		template_dir = to_free = system_path(DEFAULT_GIT_TEMPLATE_DIR); +	if (!template_dir[0]) { +		free(to_free);  		return; +	}  	template_len = strlen(template_dir);  	if (PATH_MAX <= (template_len+strlen("/config")))  		die(_("insanely long template path %s"), template_dir); @@ -139,7 +142,7 @@ static void copy_templates(const char *template_dir)  	dir = opendir(template_path);  	if (!dir) {  		warning(_("templates not found %s"), template_dir); -		return; +		goto free_return;  	}  	/* Make sure that template is from the correct vintage */ @@ -155,8 +158,7 @@ static void copy_templates(const char *template_dir)  			"a wrong format version %d from '%s'"),  			repository_format_version,  			template_dir); -		closedir(dir); -		return; +		goto close_free_return;  	}  	memcpy(path, git_dir, len); @@ -166,7 +168,10 @@ static void copy_templates(const char *template_dir)  	copy_templates_1(path, len,  			 template_path, template_len,  			 dir); +close_free_return:  	closedir(dir); +free_return: +	free(to_free);  }  static int git_init_db_config(const char *k, const char *v, void *cb) @@ -177,6 +182,20 @@ static int git_init_db_config(const char *k, const char *v, void *cb)  	return 0;  } +/* + * If the git_dir is not directly inside the working tree, then git will not + * find it by default, and we need to set the worktree explicitly. + */ +static int needs_work_tree_config(const char *git_dir, const char *work_tree) +{ +	if (!strcmp(work_tree, "/") && !strcmp(git_dir, "/.git")) +		return 0; +	if (skip_prefix(git_dir, work_tree, &git_dir) && +	    !strcmp(git_dir, "/.git")) +		return 0; +	return 1; +} +  static int create_default_files(const char *template_path)  {  	const char *git_dir = get_git_dir(); @@ -254,7 +273,10 @@ static int create_default_files(const char *template_path)  		struct stat st2;  		filemode = (!chmod(path, st1.st_mode ^ S_IXUSR) &&  				!lstat(path, &st2) && -				st1.st_mode != st2.st_mode); +				st1.st_mode != st2.st_mode && +				!chmod(path, st1.st_mode)); +		if (filemode && !reinit && (st1.st_mode & S_IXUSR)) +			filemode = 0;  	}  	git_config_set("core.filemode", filemode ? "true" : "false"); @@ -266,10 +288,8 @@ static int create_default_files(const char *template_path)  		/* allow template config file to override the default */  		if (log_all_ref_updates == -1)  		    git_config_set("core.logallrefupdates", "true"); -		if (!starts_with(git_dir, work_tree) || -		    strcmp(git_dir + strlen(work_tree), "/.git")) { +		if (needs_work_tree_config(git_dir, work_tree))  			git_config_set("core.worktree", work_tree); -		}  	}  	if (!reinit) { @@ -330,19 +350,18 @@ int set_git_dir_init(const char *git_dir, const char *real_git_dir,  		 * moving the target repo later on in separate_git_dir()  		 */  		git_link = xstrdup(real_path(git_dir)); +		set_git_dir(real_path(real_git_dir));  	}  	else { -		real_git_dir = real_path(git_dir); +		set_git_dir(real_path(git_dir));  		git_link = NULL;  	} -	set_git_dir(real_path(real_git_dir));  	return 0;  }  static void separate_git_dir(const char *git_dir)  {  	struct stat st; -	FILE *fp;  	if (!stat(git_link, &st)) {  		const char *src; @@ -358,11 +377,7 @@ static void separate_git_dir(const char *git_dir)  			die_errno(_("unable to move %s to %s"), src, git_dir);  	} -	fp = fopen(git_link, "w"); -	if (!fp) -		die(_("Could not create git link %s"), git_link); -	fprintf(fp, "gitdir: %s\n", git_dir); -	fclose(fp); +	write_file(git_link, 1, "gitdir: %s\n", git_dir);  }  int init_db(const char *template_dir, unsigned int flags) @@ -426,8 +441,9 @@ int init_db(const char *template_dir, unsigned int flags)  static int guess_repository_type(const char *git_dir)  { -	char cwd[PATH_MAX];  	const char *slash; +	char *cwd; +	int cwd_is_git_dir;  	/*  	 * "GIT_DIR=. git init" is always bare. @@ -435,9 +451,10 @@ static int guess_repository_type(const char *git_dir)  	 */  	if (!strcmp(".", git_dir))  		return 1; -	if (!getcwd(cwd, sizeof(cwd))) -		die_errno(_("cannot tell cwd")); -	if (!strcmp(git_dir, cwd)) +	cwd = xgetcwd(); +	cwd_is_git_dir = !strcmp(git_dir, cwd); +	free(cwd); +	if (cwd_is_git_dir)  		return 1;  	/*  	 * "GIT_DIR=.git or GIT_DIR=something/.git is usually not. @@ -462,7 +479,7 @@ static int shared_callback(const struct option *opt, const char *arg, int unset)  }  static const char *const init_db_usage[] = { -	N_("git init [-q | --quiet] [--bare] [--template=<template-directory>] [--shared[=<permissions>]] [directory]"), +	N_("git init [-q | --quiet] [--bare] [--template=<template-directory>] [--shared[=<permissions>]] [<directory>]"),  	NULL  }; @@ -535,10 +552,9 @@ int cmd_init_db(int argc, const char **argv, const char *prefix)  		usage(init_db_usage[0]);  	}  	if (is_bare_repository_cfg == 1) { -		static char git_dir[PATH_MAX+1]; - -		setenv(GIT_DIR_ENVIRONMENT, -			getcwd(git_dir, sizeof(git_dir)), argc > 0); +		char *cwd = xgetcwd(); +		setenv(GIT_DIR_ENVIRONMENT, cwd, argc > 0); +		free(cwd);  	}  	if (init_shared_repository != -1) @@ -572,13 +588,10 @@ int cmd_init_db(int argc, const char **argv, const char *prefix)  			git_work_tree_cfg = xstrdup(real_path(rel));  			free(rel);  		} -		if (!git_work_tree_cfg) { -			git_work_tree_cfg = xcalloc(PATH_MAX, 1); -			if (!getcwd(git_work_tree_cfg, PATH_MAX)) -				die_errno (_("Cannot access current working directory")); -		} +		if (!git_work_tree_cfg) +			git_work_tree_cfg = xgetcwd();  		if (work_tree) -			set_git_work_tree(real_path(work_tree)); +			set_git_work_tree(work_tree);  		else  			set_git_work_tree(git_work_tree_cfg);  		if (access(get_git_work_tree(), X_OK)) @@ -587,7 +600,7 @@ int cmd_init_db(int argc, const char **argv, const char *prefix)  	}  	else {  		if (work_tree) -			set_git_work_tree(real_path(work_tree)); +			set_git_work_tree(work_tree);  	}  	set_git_dir_init(git_dir, real_git_dir, 1); | 
