setup.c 9.2 KB
Newer Older
1 2
#include "cache.h"

J
Junio C Hamano 已提交
3
const char *prefix_path(const char *prefix, int len, const char *path)
4
{
J
Junio C Hamano 已提交
5
	const char *orig = path;
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41
	for (;;) {
		char c;
		if (*path != '.')
			break;
		c = path[1];
		/* "." */
		if (!c) {
			path++;
			break;
		}
		/* "./" */
		if (c == '/') {
			path += 2;
			continue;
		}
		if (c != '.')
			break;
		c = path[2];
		if (!c)
			path += 2;
		else if (c == '/')
			path += 3;
		else
			break;
		/* ".." and "../" */
		/* Remove last component of the prefix */
		do {
			if (!len)
				die("'%s' is outside repository", orig);
			len--;
		} while (len && prefix[len-1] != '/');
		continue;
	}
	if (len) {
		int speclen = strlen(path);
		char *n = xmalloc(speclen + len + 1);
J
Junio C Hamano 已提交
42

43 44 45 46 47 48 49
		memcpy(n, prefix, len);
		memcpy(n + len, path, speclen+1);
		path = n;
	}
	return path;
}

J
Junio C Hamano 已提交
50
/*
51 52 53 54 55 56 57 58 59 60 61 62 63 64
 * Unlike prefix_path, this should be used if the named file does
 * not have to interact with index entry; i.e. name of a random file
 * on the filesystem.
 */
const char *prefix_filename(const char *pfx, int pfx_len, const char *arg)
{
	static char path[PATH_MAX];
	if (!pfx || !*pfx || arg[0] == '/')
		return arg;
	memcpy(path, pfx, pfx_len);
	strcpy(path + pfx_len, arg);
	return path;
}

65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82
/*
 * Verify a filename that we got as an argument for a pathspec
 * entry. Note that a filename that begins with "-" never verifies
 * as true, because even if such a filename were to exist, we want
 * it to be preceded by the "--" marker (or we want the user to
 * use a format like "./-filename")
 */
void verify_filename(const char *prefix, const char *arg)
{
	const char *name;
	struct stat st;

	if (*arg == '-')
		die("bad flag '%s' used after filename", arg);
	name = prefix ? prefix_filename(prefix, strlen(prefix), arg) : arg;
	if (!lstat(name, &st))
		return;
	if (errno == ENOENT)
83 84
		die("ambiguous argument '%s': unknown revision or path not in the working tree.\n"
		    "Use '--' to separate paths from revisions", arg);
85 86 87
	die("'%s': %s", arg, strerror(errno));
}

88 89 90 91 92 93 94 95 96 97
/*
 * Opposite of the above: the command line did not have -- marker
 * and we parsed the arg as a refname.  It should not be interpretable
 * as a filename.
 */
void verify_non_filename(const char *prefix, const char *arg)
{
	const char *name;
	struct stat st;

98
	if (!is_inside_work_tree() || is_inside_git_dir())
99
		return;
100 101 102 103 104 105 106 107 108 109
	if (*arg == '-')
		return; /* flag */
	name = prefix ? prefix_filename(prefix, strlen(prefix), arg) : arg;
	if (!lstat(name, &st))
		die("ambiguous argument '%s': both revision and filename\n"
		    "Use '--' to separate filenames from revisions", arg);
	if (errno != ENOENT)
		die("'%s': %s", arg, strerror(errno));
}

J
Junio C Hamano 已提交
110
const char **get_pathspec(const char *prefix, const char **pathspec)
111
{
J
Junio C Hamano 已提交
112 113
	const char *entry = *pathspec;
	const char **p;
114 115
	int prefixlen;

116 117
	if (!prefix && !entry)
		return NULL;
118 119 120 121 122 123 124 125 126 127

	if (!entry) {
		static const char *spec[2];
		spec[0] = prefix;
		spec[1] = NULL;
		return spec;
	}

	/* Otherwise we have to re-write the entries.. */
	p = pathspec;
128
	prefixlen = prefix ? strlen(prefix) : 0;
129
	do {
130
		*p = prefix_path(prefix, prefixlen, entry);
131 132 133 134
	} while ((entry = *++p) != NULL);
	return (const char **) pathspec;
}

135
/*
136
 * Test if it looks like we're at a git directory.
137
 * We want to see:
138
 *
139
 *  - either a objects/ directory _or_ the proper
140
 *    GIT_OBJECT_DIRECTORY environment variable
141
 *  - a refs/ directory
J
Junio C Hamano 已提交
142
 *  - either a HEAD symlink or a HEAD file that is formatted as
J
Junio C Hamano 已提交
143 144
 *    a proper "ref:", or a regular file HEAD that has a properly
 *    formatted sha1 object name.
145
 */
146
static int is_git_directory(const char *suspect)
147
{
148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163
	char path[PATH_MAX];
	size_t len = strlen(suspect);

	strcpy(path, suspect);
	if (getenv(DB_ENVIRONMENT)) {
		if (access(getenv(DB_ENVIRONMENT), X_OK))
			return 0;
	}
	else {
		strcpy(path + len, "/objects");
		if (access(path, X_OK))
			return 0;
	}

	strcpy(path + len, "/refs");
	if (access(path, X_OK))
J
Junio C Hamano 已提交
164
		return 0;
165 166

	strcpy(path + len, "/HEAD");
J
Junio C Hamano 已提交
167
	if (validate_headref(path))
168 169
		return 0;

J
Junio C Hamano 已提交
170
	return 1;
171 172
}

173 174 175 176
static int inside_git_dir = -1;

int is_inside_git_dir(void)
{
177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198
	if (inside_git_dir >= 0)
		return inside_git_dir;
	die("BUG: is_inside_git_dir called before setup_git_directory");
}

static int inside_work_tree = -1;

int is_inside_work_tree(void)
{
	if (inside_git_dir >= 0)
		return inside_work_tree;
	die("BUG: is_inside_work_tree called before setup_git_directory");
}

static char *gitworktree_config;

static int git_setup_config(const char *var, const char *value)
{
	if (!strcmp(var, "core.worktree")) {
		if (gitworktree_config)
			strlcpy(gitworktree_config, value, PATH_MAX);
		return 0;
199
	}
200
	return git_default_config(var, value);
201 202
}

203
const char *setup_git_directory_gently(int *nongit_ok)
204 205
{
	static char cwd[PATH_MAX+1];
206 207 208
	char worktree[PATH_MAX+1], gitdir[PATH_MAX+1];
	const char *gitdirenv, *gitworktree;
	int wt_rel_gitdir = 0;
209

210
	gitdirenv = getenv(GIT_DIR_ENVIRONMENT);
211 212 213
	if (!gitdirenv) {
		int len, offset;

214
		if (!getcwd(cwd, sizeof(cwd)-1))
215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253
			die("Unable to read current working directory");

		offset = len = strlen(cwd);
		for (;;) {
			if (is_git_directory(".git"))
				break;
			if (offset == 0) {
				offset = -1;
				break;
			}
			chdir("..");
			while (cwd[--offset] != '/')
				; /* do nothing */
		}

		if (offset >= 0) {
			inside_work_tree = 1;
			git_config(git_default_config);
			if (offset == len) {
				inside_git_dir = 0;
				return NULL;
			}

			cwd[len++] = '/';
			cwd[len] = '\0';
			inside_git_dir = !prefixcmp(cwd + offset + 1, ".git/");
			return cwd + offset + 1;
		}

		if (chdir(cwd))
			die("Cannot come back to cwd");
		if (!is_git_directory(".")) {
			if (nongit_ok) {
				*nongit_ok = 1;
				return NULL;
			}
			die("Not a git repository");
		}
		setenv(GIT_DIR_ENVIRONMENT, cwd, 1);
254 255 256
		gitdirenv = getenv(GIT_DIR_ENVIRONMENT);
		if (!gitdirenv)
			die("getenv after setenv failed");
257 258 259 260 261
	}

	if (PATH_MAX - 40 < strlen(gitdirenv)) {
		if (nongit_ok) {
			*nongit_ok = 1;
262
			return NULL;
263 264 265 266
		}
		die("$%s too big", GIT_DIR_ENVIRONMENT);
	}
	if (!is_git_directory(gitdirenv)) {
267
		if (nongit_ok) {
268 269 270
			*nongit_ok = 1;
			return NULL;
		}
271
		die("Not a git repository: '%s'", gitdirenv);
272
	}
273

274
	if (!getcwd(cwd, sizeof(cwd)-1))
275
		die("Unable to read current working directory");
276 277 278 279 280 281 282 283
	if (chdir(gitdirenv)) {
		if (nongit_ok) {
			*nongit_ok = 1;
			return NULL;
		}
		die("Cannot change directory to $%s '%s'",
			GIT_DIR_ENVIRONMENT, gitdirenv);
	}
284
	if (!getcwd(gitdir, sizeof(gitdir)-1))
285 286 287
		die("Unable to read current working directory");
	if (chdir(cwd))
		die("Cannot come back to cwd");
288

289 290 291 292 293 294 295
	/*
	 * In case there is a work tree we may change the directory,
	 * therefore make GIT_DIR an absolute path.
	 */
	if (gitdirenv[0] != '/') {
		setenv(GIT_DIR_ENVIRONMENT, gitdir, 1);
		gitdirenv = getenv(GIT_DIR_ENVIRONMENT);
296 297
		if (!gitdirenv)
			die("getenv after setenv failed");
298 299 300 301
		if (PATH_MAX - 40 < strlen(gitdirenv)) {
			if (nongit_ok) {
				*nongit_ok = 1;
				return NULL;
302
			}
303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341
			die("$%s too big after expansion to absolute path",
				GIT_DIR_ENVIRONMENT);
		}
	}

	strcat(cwd, "/");
	strcat(gitdir, "/");
	inside_git_dir = !prefixcmp(cwd, gitdir);

	gitworktree = getenv(GIT_WORK_TREE_ENVIRONMENT);
	if (!gitworktree) {
		gitworktree_config = worktree;
		worktree[0] = '\0';
	}
	git_config(git_setup_config);
	if (!gitworktree) {
		gitworktree_config = NULL;
		if (worktree[0])
			gitworktree = worktree;
		if (gitworktree && gitworktree[0] != '/')
			wt_rel_gitdir = 1;
	}

	if (wt_rel_gitdir && chdir(gitdirenv))
		die("Cannot change directory to $%s '%s'",
			GIT_DIR_ENVIRONMENT, gitdirenv);
	if (gitworktree && chdir(gitworktree)) {
		if (nongit_ok) {
			if (wt_rel_gitdir && chdir(cwd))
				die("Cannot come back to cwd");
			*nongit_ok = 1;
			return NULL;
		}
		if (wt_rel_gitdir)
			die("Cannot change directory to working tree '%s'"
				" from $%s", gitworktree, GIT_DIR_ENVIRONMENT);
		else
			die("Cannot change directory to working tree '%s'",
				gitworktree);
342
	}
343
	if (!getcwd(worktree, sizeof(worktree)-1))
344 345 346
		die("Unable to read current working directory");
	strcat(worktree, "/");
	inside_work_tree = !prefixcmp(cwd, worktree);
347

348 349 350 351 352 353 354 355
	if (gitworktree && inside_work_tree && !prefixcmp(worktree, gitdir) &&
	    strcmp(worktree, gitdir)) {
		inside_git_dir = 0;
	}

	if (!inside_work_tree) {
		if (chdir(cwd))
			die("Cannot come back to cwd");
356
		return NULL;
357
	}
358

359 360 361
	if (!strcmp(cwd, worktree))
		return NULL;
	return cwd+strlen(worktree);
362
}
363

364 365 366
int git_config_perm(const char *var, const char *value)
{
	if (value) {
367
		int i;
368 369 370 371 372 373 374 375
		if (!strcmp(value, "umask"))
			return PERM_UMASK;
		if (!strcmp(value, "group"))
			return PERM_GROUP;
		if (!strcmp(value, "all") ||
		    !strcmp(value, "world") ||
		    !strcmp(value, "everybody"))
			return PERM_EVERYBODY;
376 377 378
		i = atoi(value);
		if (i > 1)
			return i;
379 380 381 382
	}
	return git_config_bool(var, value);
}

J
Junio C Hamano 已提交
383 384 385 386
int check_repository_format_version(const char *var, const char *value)
{
       if (strcmp(var, "core.repositoryformatversion") == 0)
               repository_format_version = git_config_int(var, value);
387
	else if (strcmp(var, "core.sharedrepository") == 0)
388
		shared_repository = git_config_perm(var, value);
J
Junio C Hamano 已提交
389 390 391 392 393 394 395 396 397 398 399 400
       return 0;
}

int check_repository_format(void)
{
	git_config(check_repository_format_version);
	if (GIT_REPO_VERSION < repository_format_version)
		die ("Expected git repo version <= %d, found %d",
		     GIT_REPO_VERSION, repository_format_version);
	return 0;
}

401 402
const char *setup_git_directory(void)
{
403
	const char *retval = setup_git_directory_gently(NULL);
404
	check_repository_format();
405 406
	return retval;
}