setup.c 9.1 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 42 43 44 45 46 47 48 49
	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);
	
		memcpy(n, prefix, len);
		memcpy(n + len, path, speclen+1);
		path = n;
	}
	return path;
}

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 99
	if (is_inside_git_dir())
		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 214 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 254 255 256 257 258
	if (!gitdirenv) {
		int len, offset;

		if (!getcwd(cwd, sizeof(cwd)-1) || cwd[0] != '/')
			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);
	}

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

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

286 287 288 289 290 291 292 293 294 295 296
	/*
	 * 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);
		if (PATH_MAX - 40 < strlen(gitdirenv)) {
			if (nongit_ok) {
				*nongit_ok = 1;
				return NULL;
297
			}
298 299 300 301 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
			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);
337
	}
338 339 340 341
	if (!getcwd(worktree, sizeof(worktree)-1) || worktree[0] != '/')
		die("Unable to read current working directory");
	strcat(worktree, "/");
	inside_work_tree = !prefixcmp(cwd, worktree);
342

343 344 345 346 347 348 349 350
	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");
351
		return NULL;
352
	}
353

354 355 356
	if (!strcmp(cwd, worktree))
		return NULL;
	return cwd+strlen(worktree);
357
}
358

359 360 361 362 363 364 365 366 367 368 369 370 371 372 373
int git_config_perm(const char *var, const char *value)
{
	if (value) {
		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;
	}
	return git_config_bool(var, value);
}

J
Junio C Hamano 已提交
374 375 376 377
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);
378
	else if (strcmp(var, "core.sharedrepository") == 0)
379
		shared_repository = git_config_perm(var, value);
J
Junio C Hamano 已提交
380 381 382 383 384 385 386 387 388 389 390 391
       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;
}

392 393
const char *setup_git_directory(void)
{
394
	const char *retval = setup_git_directory_gently(NULL);
395
	check_repository_format();
396 397
	return retval;
}