3 const char *prefix_path(const char *prefix, int len, const char *path)
5 const char *orig = path;
31 /* Remove last component of the prefix */
34 die("'%s' is outside repository", orig);
36 } while (len && prefix[len-1] != '/');
40 int speclen = strlen(path);
41 char *n = xmalloc(speclen + len + 1);
43 memcpy(n, prefix, len);
44 memcpy(n + len, path, speclen+1);
51 * Unlike prefix_path, this should be used if the named file does
52 * not have to interact with index entry; i.e. name of a random file
55 const char *prefix_filename(const char *pfx, int pfx_len, const char *arg)
57 static char path[PATH_MAX];
58 if (!pfx || !*pfx || arg[0] == '/')
60 memcpy(path, pfx, pfx_len);
61 strcpy(path + pfx_len, arg);
66 * Verify a filename that we got as an argument for a pathspec
67 * entry. Note that a filename that begins with "-" never verifies
68 * as true, because even if such a filename were to exist, we want
69 * it to be preceded by the "--" marker (or we want the user to
70 * use a format like "./-filename")
72 void verify_filename(const char *prefix, const char *arg)
78 die("bad flag '%s' used after filename", arg);
79 name = prefix ? prefix_filename(prefix, strlen(prefix), arg) : arg;
80 if (!lstat(name, &st))
83 die("ambiguous argument '%s': unknown revision or path not in the working tree.\n"
84 "Use '--' to separate paths from revisions", arg);
85 die("'%s': %s", arg, strerror(errno));
89 * Opposite of the above: the command line did not have -- marker
90 * and we parsed the arg as a refname. It should not be interpretable
93 void verify_non_filename(const char *prefix, const char *arg)
98 if (!is_inside_work_tree() || is_inside_git_dir())
102 name = prefix ? prefix_filename(prefix, strlen(prefix), arg) : arg;
103 if (!lstat(name, &st))
104 die("ambiguous argument '%s': both revision and filename\n"
105 "Use '--' to separate filenames from revisions", arg);
107 die("'%s': %s", arg, strerror(errno));
110 const char **get_pathspec(const char *prefix, const char **pathspec)
112 const char *entry = *pathspec;
116 if (!prefix && !entry)
120 static const char *spec[2];
126 /* Otherwise we have to re-write the entries.. */
128 prefixlen = prefix ? strlen(prefix) : 0;
130 *p = prefix_path(prefix, prefixlen, entry);
131 } while ((entry = *++p) != NULL);
132 return (const char **) pathspec;
136 * Test if it looks like we're at a git directory.
139 * - either a objects/ directory _or_ the proper
140 * GIT_OBJECT_DIRECTORY environment variable
141 * - a refs/ directory
142 * - either a HEAD symlink or a HEAD file that is formatted as
143 * a proper "ref:", or a regular file HEAD that has a properly
144 * formatted sha1 object name.
146 static int is_git_directory(const char *suspect)
149 size_t len = strlen(suspect);
151 strcpy(path, suspect);
152 if (getenv(DB_ENVIRONMENT)) {
153 if (access(getenv(DB_ENVIRONMENT), X_OK))
157 strcpy(path + len, "/objects");
158 if (access(path, X_OK))
162 strcpy(path + len, "/refs");
163 if (access(path, X_OK))
166 strcpy(path + len, "/HEAD");
167 if (validate_headref(path))
173 static int inside_git_dir = -1;
175 int is_inside_git_dir(void)
177 if (inside_git_dir >= 0)
178 return inside_git_dir;
179 die("BUG: is_inside_git_dir called before setup_git_directory");
182 static int inside_work_tree = -1;
184 int is_inside_work_tree(void)
186 if (inside_git_dir >= 0)
187 return inside_work_tree;
188 die("BUG: is_inside_work_tree called before setup_git_directory");
191 static char *gitworktree_config;
193 static int git_setup_config(const char *var, const char *value)
195 if (!strcmp(var, "core.worktree")) {
196 if (gitworktree_config)
197 strlcpy(gitworktree_config, value, PATH_MAX);
200 return git_default_config(var, value);
203 const char *setup_git_directory_gently(int *nongit_ok)
205 static char cwd[PATH_MAX+1];
206 char worktree[PATH_MAX+1], gitdir[PATH_MAX+1];
207 const char *gitdirenv, *gitworktree;
208 int wt_rel_gitdir = 0;
210 gitdirenv = getenv(GIT_DIR_ENVIRONMENT);
214 if (!getcwd(cwd, sizeof(cwd)-1))
215 die("Unable to read current working directory");
217 offset = len = strlen(cwd);
219 if (is_git_directory(".git"))
226 while (cwd[--offset] != '/')
231 inside_work_tree = 1;
232 git_config(git_default_config);
240 inside_git_dir = !prefixcmp(cwd + offset + 1, ".git/");
241 return cwd + offset + 1;
245 die("Cannot come back to cwd");
246 if (!is_git_directory(".")) {
251 die("Not a git repository");
253 setenv(GIT_DIR_ENVIRONMENT, cwd, 1);
254 gitdirenv = getenv(GIT_DIR_ENVIRONMENT);
256 die("getenv after setenv failed");
259 if (PATH_MAX - 40 < strlen(gitdirenv)) {
264 die("$%s too big", GIT_DIR_ENVIRONMENT);
266 if (!is_git_directory(gitdirenv)) {
271 die("Not a git repository: '%s'", gitdirenv);
274 if (!getcwd(cwd, sizeof(cwd)-1))
275 die("Unable to read current working directory");
276 if (chdir(gitdirenv)) {
281 die("Cannot change directory to $%s '%s'",
282 GIT_DIR_ENVIRONMENT, gitdirenv);
284 if (!getcwd(gitdir, sizeof(gitdir)-1))
285 die("Unable to read current working directory");
287 die("Cannot come back to cwd");
290 * In case there is a work tree we may change the directory,
291 * therefore make GIT_DIR an absolute path.
293 if (gitdirenv[0] != '/') {
294 setenv(GIT_DIR_ENVIRONMENT, gitdir, 1);
295 gitdirenv = getenv(GIT_DIR_ENVIRONMENT);
297 die("getenv after setenv failed");
298 if (PATH_MAX - 40 < strlen(gitdirenv)) {
303 die("$%s too big after expansion to absolute path",
304 GIT_DIR_ENVIRONMENT);
310 inside_git_dir = !prefixcmp(cwd, gitdir);
312 gitworktree = getenv(GIT_WORK_TREE_ENVIRONMENT);
314 gitworktree_config = worktree;
317 git_config(git_setup_config);
319 gitworktree_config = NULL;
321 gitworktree = worktree;
322 if (gitworktree && gitworktree[0] != '/')
326 if (wt_rel_gitdir && chdir(gitdirenv))
327 die("Cannot change directory to $%s '%s'",
328 GIT_DIR_ENVIRONMENT, gitdirenv);
329 if (gitworktree && chdir(gitworktree)) {
331 if (wt_rel_gitdir && chdir(cwd))
332 die("Cannot come back to cwd");
337 die("Cannot change directory to working tree '%s'"
338 " from $%s", gitworktree, GIT_DIR_ENVIRONMENT);
340 die("Cannot change directory to working tree '%s'",
343 if (!getcwd(worktree, sizeof(worktree)-1))
344 die("Unable to read current working directory");
345 strcat(worktree, "/");
346 inside_work_tree = !prefixcmp(cwd, worktree);
348 if (gitworktree && inside_work_tree && !prefixcmp(worktree, gitdir) &&
349 strcmp(worktree, gitdir)) {
353 if (!inside_work_tree) {
355 die("Cannot come back to cwd");
359 if (!strcmp(cwd, worktree))
361 return cwd+strlen(worktree);
364 int git_config_perm(const char *var, const char *value)
368 if (!strcmp(value, "umask"))
370 if (!strcmp(value, "group"))
372 if (!strcmp(value, "all") ||
373 !strcmp(value, "world") ||
374 !strcmp(value, "everybody"))
375 return PERM_EVERYBODY;
380 return git_config_bool(var, value);
383 int check_repository_format_version(const char *var, const char *value)
385 if (strcmp(var, "core.repositoryformatversion") == 0)
386 repository_format_version = git_config_int(var, value);
387 else if (strcmp(var, "core.sharedrepository") == 0)
388 shared_repository = git_config_perm(var, value);
392 int check_repository_format(void)
394 git_config(check_repository_format_version);
395 if (GIT_REPO_VERSION < repository_format_version)
396 die ("Expected git repo version <= %d, found %d",
397 GIT_REPO_VERSION, repository_format_version);
401 const char *setup_git_directory(void)
403 const char *retval = setup_git_directory_gently(NULL);
404 check_repository_format();