2009-10-19 20:38:32 +08:00
|
|
|
#include "cache.h"
|
|
|
|
#include "submodule.h"
|
|
|
|
#include "dir.h"
|
|
|
|
#include "diff.h"
|
|
|
|
#include "commit.h"
|
|
|
|
#include "revision.h"
|
2010-01-17 01:42:24 +08:00
|
|
|
#include "run-command.h"
|
2010-03-05 05:20:33 +08:00
|
|
|
#include "diffcore.h"
|
2010-07-07 21:39:13 +08:00
|
|
|
#include "refs.h"
|
2010-08-06 06:39:25 +08:00
|
|
|
#include "string-list.h"
|
2011-09-13 03:56:52 +08:00
|
|
|
#include "sha1-array.h"
|
2011-09-14 05:57:57 +08:00
|
|
|
#include "argv-array.h"
|
2013-07-31 03:50:34 +08:00
|
|
|
#include "blob.h"
|
2010-08-06 06:39:25 +08:00
|
|
|
|
Fix sparse warnings
Fix warnings from 'make check'.
- These files don't include 'builtin.h' causing sparse to complain that
cmd_* isn't declared:
builtin/clone.c:364, builtin/fetch-pack.c:797,
builtin/fmt-merge-msg.c:34, builtin/hash-object.c:78,
builtin/merge-index.c:69, builtin/merge-recursive.c:22
builtin/merge-tree.c:341, builtin/mktag.c:156, builtin/notes.c:426
builtin/notes.c:822, builtin/pack-redundant.c:596,
builtin/pack-refs.c:10, builtin/patch-id.c:60, builtin/patch-id.c:149,
builtin/remote.c:1512, builtin/remote-ext.c:240,
builtin/remote-fd.c:53, builtin/reset.c:236, builtin/send-pack.c:384,
builtin/unpack-file.c:25, builtin/var.c:75
- These files have symbols which should be marked static since they're
only file scope:
submodule.c:12, diff.c:631, replace_object.c:92, submodule.c:13,
submodule.c:14, trace.c:78, transport.c:195, transport-helper.c:79,
unpack-trees.c:19, url.c:3, url.c:18, url.c:104, url.c:117, url.c:123,
url.c:129, url.c:136, thread-utils.c:21, thread-utils.c:48
- These files redeclare symbols to be different types:
builtin/index-pack.c:210, parse-options.c:564, parse-options.c:571,
usage.c:49, usage.c:58, usage.c:63, usage.c:72
- These files use a literal integer 0 when they really should use a NULL
pointer:
daemon.c:663, fast-import.c:2942, imap-send.c:1072, notes-merge.c:362
While we're in the area, clean up some unused #includes in builtin files
(mostly exec_cmd.h).
Signed-off-by: Stephen Boyd <bebarino@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2011-03-22 15:51:05 +08:00
|
|
|
static struct string_list config_name_for_path;
|
|
|
|
static struct string_list config_fetch_recurse_submodules_for_name;
|
|
|
|
static struct string_list config_ignore_for_name;
|
2011-03-07 06:10:46 +08:00
|
|
|
static int config_fetch_recurse_submodules = RECURSE_SUBMODULES_ON_DEMAND;
|
2011-04-05 06:02:01 +08:00
|
|
|
static struct string_list changed_submodule_paths;
|
2011-09-13 03:56:52 +08:00
|
|
|
static int initialized_fetch_ref_tips;
|
|
|
|
static struct sha1_array ref_tips_before_fetch;
|
|
|
|
static struct sha1_array ref_tips_after_fetch;
|
|
|
|
|
2011-05-15 00:26:58 +08:00
|
|
|
/*
|
|
|
|
* The following flag is set if the .gitmodules file is unmerged. We then
|
|
|
|
* disable recursion for all submodules where .git/config doesn't have a
|
|
|
|
* matching config entry because we can't guess what might be configured in
|
|
|
|
* .gitmodules unless the user resolves the conflict. When a command line
|
|
|
|
* option is given (which always overrides configuration) this flag will be
|
|
|
|
* ignored.
|
|
|
|
*/
|
|
|
|
static int gitmodules_is_unmerged;
|
2009-10-19 20:38:32 +08:00
|
|
|
|
2013-07-31 03:50:34 +08:00
|
|
|
/*
|
|
|
|
* This flag is set if the .gitmodules file had unstaged modifications on
|
|
|
|
* startup. This must be checked before allowing modifications to the
|
|
|
|
* .gitmodules file with the intention to stage them later, because when
|
|
|
|
* continuing we would stage the modifications the user didn't stage herself
|
|
|
|
* too. That might change in a future version when we learn to stage the
|
|
|
|
* changes we do ourselves without staging any previous modifications.
|
|
|
|
*/
|
|
|
|
static int gitmodules_is_modified;
|
|
|
|
|
|
|
|
|
|
|
|
int is_staging_gitmodules_ok(void)
|
|
|
|
{
|
|
|
|
return !gitmodules_is_modified;
|
|
|
|
}
|
|
|
|
|
2013-08-07 03:15:11 +08:00
|
|
|
/*
|
|
|
|
* Try to update the "path" entry in the "submodule.<name>" section of the
|
|
|
|
* .gitmodules file. Return 0 only if a .gitmodules file was found, a section
|
|
|
|
* with the correct path=<oldpath> setting was found and we could update it.
|
|
|
|
*/
|
|
|
|
int update_path_in_gitmodules(const char *oldpath, const char *newpath)
|
|
|
|
{
|
|
|
|
struct strbuf entry = STRBUF_INIT;
|
|
|
|
struct string_list_item *path_option;
|
|
|
|
|
|
|
|
if (!file_exists(".gitmodules")) /* Do nothing without .gitmodules */
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (gitmodules_is_unmerged)
|
|
|
|
die(_("Cannot change unmerged .gitmodules, resolve merge conflicts first"));
|
|
|
|
|
|
|
|
path_option = unsorted_string_list_lookup(&config_name_for_path, oldpath);
|
|
|
|
if (!path_option) {
|
|
|
|
warning(_("Could not find section in .gitmodules where path=%s"), oldpath);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
strbuf_addstr(&entry, "submodule.");
|
|
|
|
strbuf_addstr(&entry, path_option->util);
|
|
|
|
strbuf_addstr(&entry, ".path");
|
|
|
|
if (git_config_set_in_file(".gitmodules", entry.buf, newpath) < 0) {
|
|
|
|
/* Maybe the user already did that, don't error out here */
|
|
|
|
warning(_("Could not update .gitmodules entry %s"), entry.buf);
|
|
|
|
strbuf_release(&entry);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
strbuf_release(&entry);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-08-07 03:15:25 +08:00
|
|
|
/*
|
|
|
|
* Try to remove the "submodule.<name>" section from .gitmodules where the given
|
|
|
|
* path is configured. Return 0 only if a .gitmodules file was found, a section
|
|
|
|
* with the correct path=<path> setting was found and we could remove it.
|
|
|
|
*/
|
|
|
|
int remove_path_from_gitmodules(const char *path)
|
|
|
|
{
|
|
|
|
struct strbuf sect = STRBUF_INIT;
|
|
|
|
struct string_list_item *path_option;
|
|
|
|
|
|
|
|
if (!file_exists(".gitmodules")) /* Do nothing without .gitmodules */
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (gitmodules_is_unmerged)
|
|
|
|
die(_("Cannot change unmerged .gitmodules, resolve merge conflicts first"));
|
|
|
|
|
|
|
|
path_option = unsorted_string_list_lookup(&config_name_for_path, path);
|
|
|
|
if (!path_option) {
|
|
|
|
warning(_("Could not find section in .gitmodules where path=%s"), path);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
strbuf_addstr(§, "submodule.");
|
|
|
|
strbuf_addstr(§, path_option->util);
|
|
|
|
if (git_config_rename_section_in_file(".gitmodules", sect.buf, NULL) < 0) {
|
|
|
|
/* Maybe the user already did that, don't error out here */
|
|
|
|
warning(_("Could not remove .gitmodules entry for %s"), path);
|
|
|
|
strbuf_release(§);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
strbuf_release(§);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-07-31 03:50:34 +08:00
|
|
|
void stage_updated_gitmodules(void)
|
|
|
|
{
|
2013-11-07 22:33:43 +08:00
|
|
|
if (add_file_to_cache(".gitmodules", 0))
|
2013-07-31 03:50:34 +08:00
|
|
|
die(_("staging updated .gitmodules failed"));
|
|
|
|
}
|
|
|
|
|
2010-01-12 14:31:58 +08:00
|
|
|
static int add_submodule_odb(const char *path)
|
2009-10-19 20:38:32 +08:00
|
|
|
{
|
|
|
|
struct strbuf objects_directory = STRBUF_INIT;
|
|
|
|
struct alternate_object_database *alt_odb;
|
2010-02-01 00:43:49 +08:00
|
|
|
int ret = 0;
|
2010-04-11 01:01:12 +08:00
|
|
|
const char *git_dir;
|
2009-10-19 20:38:32 +08:00
|
|
|
|
2010-04-11 01:01:12 +08:00
|
|
|
strbuf_addf(&objects_directory, "%s/.git", path);
|
2011-08-23 05:04:56 +08:00
|
|
|
git_dir = read_gitfile(objects_directory.buf);
|
2010-04-11 01:01:12 +08:00
|
|
|
if (git_dir) {
|
|
|
|
strbuf_reset(&objects_directory);
|
|
|
|
strbuf_addstr(&objects_directory, git_dir);
|
|
|
|
}
|
|
|
|
strbuf_addstr(&objects_directory, "/objects/");
|
2010-02-01 00:43:49 +08:00
|
|
|
if (!is_directory(objects_directory.buf)) {
|
|
|
|
ret = -1;
|
|
|
|
goto done;
|
|
|
|
}
|
2009-10-19 20:38:32 +08:00
|
|
|
/* avoid adding it twice */
|
|
|
|
for (alt_odb = alt_odb_list; alt_odb; alt_odb = alt_odb->next)
|
|
|
|
if (alt_odb->name - alt_odb->base == objects_directory.len &&
|
|
|
|
!strncmp(alt_odb->base, objects_directory.buf,
|
|
|
|
objects_directory.len))
|
2010-02-01 00:43:49 +08:00
|
|
|
goto done;
|
2009-10-19 20:38:32 +08:00
|
|
|
|
|
|
|
alt_odb = xmalloc(objects_directory.len + 42 + sizeof(*alt_odb));
|
|
|
|
alt_odb->next = alt_odb_list;
|
|
|
|
strcpy(alt_odb->base, objects_directory.buf);
|
|
|
|
alt_odb->name = alt_odb->base + objects_directory.len;
|
|
|
|
alt_odb->name[2] = '/';
|
|
|
|
alt_odb->name[40] = '\0';
|
|
|
|
alt_odb->name[41] = '\0';
|
|
|
|
alt_odb_list = alt_odb;
|
2012-05-15 00:24:45 +08:00
|
|
|
|
|
|
|
/* add possible alternates from the submodule */
|
|
|
|
read_info_alternates(objects_directory.buf, 0);
|
2009-10-19 20:38:32 +08:00
|
|
|
prepare_alt_odb();
|
2010-02-01 00:43:49 +08:00
|
|
|
done:
|
|
|
|
strbuf_release(&objects_directory);
|
|
|
|
return ret;
|
2009-10-19 20:38:32 +08:00
|
|
|
}
|
|
|
|
|
2010-08-06 06:39:25 +08:00
|
|
|
void set_diffopt_flags_from_submodule_config(struct diff_options *diffopt,
|
|
|
|
const char *path)
|
|
|
|
{
|
|
|
|
struct string_list_item *path_option, *ignore_option;
|
|
|
|
path_option = unsorted_string_list_lookup(&config_name_for_path, path);
|
|
|
|
if (path_option) {
|
|
|
|
ignore_option = unsorted_string_list_lookup(&config_ignore_for_name, path_option->util);
|
|
|
|
if (ignore_option)
|
|
|
|
handle_ignore_submodules_arg(diffopt, ignore_option->util);
|
2011-05-15 00:26:58 +08:00
|
|
|
else if (gitmodules_is_unmerged)
|
|
|
|
DIFF_OPT_SET(diffopt, IGNORE_SUBMODULES);
|
2010-08-06 06:39:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-11-12 20:54:52 +08:00
|
|
|
int submodule_config(const char *var, const char *value, void *cb)
|
2010-08-06 06:40:48 +08:00
|
|
|
{
|
2013-12-01 04:55:40 +08:00
|
|
|
if (starts_with(var, "submodule."))
|
2010-08-06 06:40:48 +08:00
|
|
|
return parse_submodule_config_option(var, value);
|
2010-11-11 07:55:02 +08:00
|
|
|
else if (!strcmp(var, "fetch.recursesubmodules")) {
|
2011-03-07 06:11:48 +08:00
|
|
|
config_fetch_recurse_submodules = parse_fetch_recurse_submodules_arg(var, value);
|
2010-11-11 07:55:02 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2010-08-06 06:40:48 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void gitmodules_config(void)
|
|
|
|
{
|
|
|
|
const char *work_tree = get_git_work_tree();
|
|
|
|
if (work_tree) {
|
|
|
|
struct strbuf gitmodules_path = STRBUF_INIT;
|
2011-05-15 00:26:58 +08:00
|
|
|
int pos;
|
2010-08-06 06:40:48 +08:00
|
|
|
strbuf_addstr(&gitmodules_path, work_tree);
|
|
|
|
strbuf_addstr(&gitmodules_path, "/.gitmodules");
|
2011-05-15 00:26:58 +08:00
|
|
|
if (read_cache() < 0)
|
|
|
|
die("index file corrupt");
|
|
|
|
pos = cache_name_pos(".gitmodules", 11);
|
|
|
|
if (pos < 0) { /* .gitmodules not found or isn't merged */
|
|
|
|
pos = -1 - pos;
|
|
|
|
if (active_nr > pos) { /* there is a .gitmodules */
|
|
|
|
const struct cache_entry *ce = active_cache[pos];
|
|
|
|
if (ce_namelen(ce) == 11 &&
|
|
|
|
!memcmp(ce->name, ".gitmodules", 11))
|
|
|
|
gitmodules_is_unmerged = 1;
|
|
|
|
}
|
2013-07-31 03:50:34 +08:00
|
|
|
} else if (pos < active_nr) {
|
|
|
|
struct stat st;
|
|
|
|
if (lstat(".gitmodules", &st) == 0 &&
|
|
|
|
ce_match_stat(active_cache[pos], &st, 0) & DATA_CHANGED)
|
|
|
|
gitmodules_is_modified = 1;
|
2011-05-15 00:26:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!gitmodules_is_unmerged)
|
|
|
|
git_config_from_file(submodule_config, gitmodules_path.buf, NULL);
|
2010-08-06 06:40:48 +08:00
|
|
|
strbuf_release(&gitmodules_path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-06 06:39:25 +08:00
|
|
|
int parse_submodule_config_option(const char *var, const char *value)
|
|
|
|
{
|
|
|
|
struct string_list_item *config;
|
2013-01-23 14:25:22 +08:00
|
|
|
const char *name, *key;
|
|
|
|
int namelen;
|
2010-08-06 06:39:25 +08:00
|
|
|
|
2013-01-23 14:25:22 +08:00
|
|
|
if (parse_config_key(var, "submodule", &name, &namelen, &key) < 0 || !name)
|
|
|
|
return 0;
|
2010-08-06 06:39:25 +08:00
|
|
|
|
2013-01-23 14:25:22 +08:00
|
|
|
if (!strcmp(key, "path")) {
|
2013-08-20 00:26:56 +08:00
|
|
|
if (!value)
|
|
|
|
return config_error_nonbool(var);
|
|
|
|
|
2010-08-06 06:39:25 +08:00
|
|
|
config = unsorted_string_list_lookup(&config_name_for_path, value);
|
|
|
|
if (config)
|
|
|
|
free(config->util);
|
|
|
|
else
|
|
|
|
config = string_list_append(&config_name_for_path, xstrdup(value));
|
2013-01-23 14:26:42 +08:00
|
|
|
config->util = xmemdupz(name, namelen);
|
2013-01-23 14:25:22 +08:00
|
|
|
} else if (!strcmp(key, "fetchrecursesubmodules")) {
|
2013-01-23 14:26:42 +08:00
|
|
|
char *name_cstr = xmemdupz(name, namelen);
|
|
|
|
config = unsorted_string_list_lookup(&config_fetch_recurse_submodules_for_name, name_cstr);
|
2010-11-11 07:55:41 +08:00
|
|
|
if (!config)
|
2013-01-23 14:26:42 +08:00
|
|
|
config = string_list_append(&config_fetch_recurse_submodules_for_name, name_cstr);
|
|
|
|
else
|
|
|
|
free(name_cstr);
|
2011-03-07 06:12:19 +08:00
|
|
|
config->util = (void *)(intptr_t)parse_fetch_recurse_submodules_arg(var, value);
|
2013-01-23 14:25:22 +08:00
|
|
|
} else if (!strcmp(key, "ignore")) {
|
2013-01-23 14:26:42 +08:00
|
|
|
char *name_cstr;
|
|
|
|
|
2013-08-20 00:26:56 +08:00
|
|
|
if (!value)
|
|
|
|
return config_error_nonbool(var);
|
|
|
|
|
2010-08-06 06:39:25 +08:00
|
|
|
if (strcmp(value, "untracked") && strcmp(value, "dirty") &&
|
|
|
|
strcmp(value, "all") && strcmp(value, "none")) {
|
|
|
|
warning("Invalid parameter \"%s\" for config option \"submodule.%s.ignore\"", value, var);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-01-23 14:26:42 +08:00
|
|
|
name_cstr = xmemdupz(name, namelen);
|
|
|
|
config = unsorted_string_list_lookup(&config_ignore_for_name, name_cstr);
|
|
|
|
if (config) {
|
2010-08-06 06:39:25 +08:00
|
|
|
free(config->util);
|
2013-01-23 14:26:42 +08:00
|
|
|
free(name_cstr);
|
|
|
|
} else
|
|
|
|
config = string_list_append(&config_ignore_for_name, name_cstr);
|
2010-08-06 06:39:25 +08:00
|
|
|
config->util = xstrdup(value);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-06-25 22:56:47 +08:00
|
|
|
void handle_ignore_submodules_arg(struct diff_options *diffopt,
|
|
|
|
const char *arg)
|
|
|
|
{
|
2010-08-05 16:49:55 +08:00
|
|
|
DIFF_OPT_CLR(diffopt, IGNORE_SUBMODULES);
|
|
|
|
DIFF_OPT_CLR(diffopt, IGNORE_UNTRACKED_IN_SUBMODULES);
|
|
|
|
DIFF_OPT_CLR(diffopt, IGNORE_DIRTY_SUBMODULES);
|
|
|
|
|
2010-06-25 22:56:47 +08:00
|
|
|
if (!strcmp(arg, "all"))
|
|
|
|
DIFF_OPT_SET(diffopt, IGNORE_SUBMODULES);
|
|
|
|
else if (!strcmp(arg, "untracked"))
|
|
|
|
DIFF_OPT_SET(diffopt, IGNORE_UNTRACKED_IN_SUBMODULES);
|
|
|
|
else if (!strcmp(arg, "dirty"))
|
|
|
|
DIFF_OPT_SET(diffopt, IGNORE_DIRTY_SUBMODULES);
|
2010-08-06 06:39:25 +08:00
|
|
|
else if (strcmp(arg, "none"))
|
2010-06-25 22:56:47 +08:00
|
|
|
die("bad --ignore-submodules argument: %s", arg);
|
|
|
|
}
|
|
|
|
|
2011-03-16 15:14:11 +08:00
|
|
|
static int prepare_submodule_summary(struct rev_info *rev, const char *path,
|
|
|
|
struct commit *left, struct commit *right,
|
|
|
|
int *fast_forward, int *fast_backward)
|
|
|
|
{
|
|
|
|
struct commit_list *merge_bases, *list;
|
|
|
|
|
|
|
|
init_revisions(rev, NULL);
|
|
|
|
setup_revisions(0, NULL, rev, NULL);
|
|
|
|
rev->left_right = 1;
|
|
|
|
rev->first_parent_only = 1;
|
|
|
|
left->object.flags |= SYMMETRIC_LEFT;
|
|
|
|
add_pending_object(rev, &left->object, path);
|
|
|
|
add_pending_object(rev, &right->object, path);
|
2014-10-31 03:20:44 +08:00
|
|
|
merge_bases = get_merge_bases(left, right);
|
2011-03-16 15:14:11 +08:00
|
|
|
if (merge_bases) {
|
|
|
|
if (merge_bases->item == left)
|
|
|
|
*fast_forward = 1;
|
|
|
|
else if (merge_bases->item == right)
|
|
|
|
*fast_backward = 1;
|
|
|
|
}
|
|
|
|
for (list = merge_bases; list; list = list->next) {
|
|
|
|
list->item->object.flags |= UNINTERESTING;
|
|
|
|
add_pending_object(rev, &list->item->object,
|
|
|
|
sha1_to_hex(list->item->object.sha1));
|
|
|
|
}
|
|
|
|
return prepare_revision_walk(rev);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void print_submodule_summary(struct rev_info *rev, FILE *f,
|
2013-04-06 00:12:08 +08:00
|
|
|
const char *line_prefix,
|
2011-03-16 15:14:11 +08:00
|
|
|
const char *del, const char *add, const char *reset)
|
|
|
|
{
|
|
|
|
static const char format[] = " %m %s";
|
|
|
|
struct strbuf sb = STRBUF_INIT;
|
|
|
|
struct commit *commit;
|
|
|
|
|
|
|
|
while ((commit = get_revision(rev))) {
|
|
|
|
struct pretty_print_context ctx = {0};
|
|
|
|
ctx.date_mode = rev->date_mode;
|
2013-06-26 18:19:50 +08:00
|
|
|
ctx.output_encoding = get_log_output_encoding();
|
2011-03-16 15:14:11 +08:00
|
|
|
strbuf_setlen(&sb, 0);
|
2013-04-06 00:12:08 +08:00
|
|
|
strbuf_addstr(&sb, line_prefix);
|
2011-03-16 15:14:11 +08:00
|
|
|
if (commit->object.flags & SYMMETRIC_LEFT) {
|
|
|
|
if (del)
|
|
|
|
strbuf_addstr(&sb, del);
|
|
|
|
}
|
|
|
|
else if (add)
|
|
|
|
strbuf_addstr(&sb, add);
|
|
|
|
format_commit_message(commit, format, &sb, &ctx);
|
|
|
|
if (reset)
|
|
|
|
strbuf_addstr(&sb, reset);
|
|
|
|
strbuf_addch(&sb, '\n');
|
|
|
|
fprintf(f, "%s", sb.buf);
|
|
|
|
}
|
|
|
|
strbuf_release(&sb);
|
|
|
|
}
|
|
|
|
|
2011-03-07 06:10:46 +08:00
|
|
|
int parse_fetch_recurse_submodules_arg(const char *opt, const char *arg)
|
|
|
|
{
|
|
|
|
switch (git_config_maybe_bool(opt, arg)) {
|
|
|
|
case 1:
|
|
|
|
return RECURSE_SUBMODULES_ON;
|
|
|
|
case 0:
|
|
|
|
return RECURSE_SUBMODULES_OFF;
|
|
|
|
default:
|
|
|
|
if (!strcmp(arg, "on-demand"))
|
|
|
|
return RECURSE_SUBMODULES_ON_DEMAND;
|
|
|
|
die("bad %s argument: %s", opt, arg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-19 20:38:32 +08:00
|
|
|
void show_submodule_summary(FILE *f, const char *path,
|
2013-04-06 00:12:08 +08:00
|
|
|
const char *line_prefix,
|
2009-10-19 20:38:32 +08:00
|
|
|
unsigned char one[20], unsigned char two[20],
|
2012-11-13 23:42:47 +08:00
|
|
|
unsigned dirty_submodule, const char *meta,
|
2009-10-19 20:38:32 +08:00
|
|
|
const char *del, const char *add, const char *reset)
|
|
|
|
{
|
|
|
|
struct rev_info rev;
|
submodule: clarify logic in show_submodule_summary
There are two uses of the "left" and "right" commit variables that
make it hard to be sure what values they have (both for the reader,
and for gcc, which wrongly complains that they might be used
uninitialized).
The function starts with a cascading if statement, checking that the
input sha1s exist, and finally working up to preparing a revision
walk. We only prepare the walk if the cascading conditional did not
find any problems, which we check by seeing whether it set the
"message" variable or not. It's simpler and more obvious to just add
a condition to the end of the cascade.
Later, we check the same "message" variable when deciding whether to
clear commit marks on the left/right commits; if it is set, we
presumably never started the walk. This is wrong, though; we might
have started the walk and munged commit flags, only to encounter an
error afterwards. We should always clear the flags on left/right if
they exist, whether the walk was successful or not.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2013-03-23 00:19:56 +08:00
|
|
|
struct commit *left = NULL, *right = NULL;
|
2009-10-19 20:38:32 +08:00
|
|
|
const char *message = NULL;
|
|
|
|
struct strbuf sb = STRBUF_INIT;
|
|
|
|
int fast_forward = 0, fast_backward = 0;
|
|
|
|
|
|
|
|
if (is_null_sha1(two))
|
|
|
|
message = "(submodule deleted)";
|
|
|
|
else if (add_submodule_odb(path))
|
|
|
|
message = "(not checked out)";
|
|
|
|
else if (is_null_sha1(one))
|
|
|
|
message = "(new submodule)";
|
|
|
|
else if (!(left = lookup_commit_reference(one)) ||
|
|
|
|
!(right = lookup_commit_reference(two)))
|
|
|
|
message = "(commits not present)";
|
submodule: clarify logic in show_submodule_summary
There are two uses of the "left" and "right" commit variables that
make it hard to be sure what values they have (both for the reader,
and for gcc, which wrongly complains that they might be used
uninitialized).
The function starts with a cascading if statement, checking that the
input sha1s exist, and finally working up to preparing a revision
walk. We only prepare the walk if the cascading conditional did not
find any problems, which we check by seeing whether it set the
"message" variable or not. It's simpler and more obvious to just add
a condition to the end of the cascade.
Later, we check the same "message" variable when deciding whether to
clear commit marks on the left/right commits; if it is set, we
presumably never started the walk. This is wrong, though; we might
have started the walk and munged commit flags, only to encounter an
error afterwards. We should always clear the flags on left/right if
they exist, whether the walk was successful or not.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2013-03-23 00:19:56 +08:00
|
|
|
else if (prepare_submodule_summary(&rev, path, left, right,
|
|
|
|
&fast_forward, &fast_backward))
|
2011-03-16 15:14:11 +08:00
|
|
|
message = "(revision walker failed)";
|
2009-10-19 20:38:32 +08:00
|
|
|
|
2010-03-05 05:20:33 +08:00
|
|
|
if (dirty_submodule & DIRTY_SUBMODULE_UNTRACKED)
|
2013-04-06 00:12:08 +08:00
|
|
|
fprintf(f, "%sSubmodule %s contains untracked content\n",
|
|
|
|
line_prefix, path);
|
2010-03-05 05:20:33 +08:00
|
|
|
if (dirty_submodule & DIRTY_SUBMODULE_MODIFIED)
|
2013-04-06 00:12:08 +08:00
|
|
|
fprintf(f, "%sSubmodule %s contains modified content\n",
|
|
|
|
line_prefix, path);
|
2010-03-05 05:20:33 +08:00
|
|
|
|
|
|
|
if (!hashcmp(one, two)) {
|
|
|
|
strbuf_release(&sb);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-04-06 00:12:08 +08:00
|
|
|
strbuf_addf(&sb, "%s%sSubmodule %s %s..", line_prefix, meta, path,
|
2009-10-19 20:38:32 +08:00
|
|
|
find_unique_abbrev(one, DEFAULT_ABBREV));
|
|
|
|
if (!fast_backward && !fast_forward)
|
|
|
|
strbuf_addch(&sb, '.');
|
|
|
|
strbuf_addf(&sb, "%s", find_unique_abbrev(two, DEFAULT_ABBREV));
|
|
|
|
if (message)
|
2012-11-13 23:42:47 +08:00
|
|
|
strbuf_addf(&sb, " %s%s\n", message, reset);
|
2009-10-19 20:38:32 +08:00
|
|
|
else
|
2012-11-13 23:42:47 +08:00
|
|
|
strbuf_addf(&sb, "%s:%s\n", fast_backward ? " (rewind)" : "", reset);
|
2009-10-19 20:38:32 +08:00
|
|
|
fwrite(sb.buf, sb.len, 1, f);
|
|
|
|
|
submodule: clarify logic in show_submodule_summary
There are two uses of the "left" and "right" commit variables that
make it hard to be sure what values they have (both for the reader,
and for gcc, which wrongly complains that they might be used
uninitialized).
The function starts with a cascading if statement, checking that the
input sha1s exist, and finally working up to preparing a revision
walk. We only prepare the walk if the cascading conditional did not
find any problems, which we check by seeing whether it set the
"message" variable or not. It's simpler and more obvious to just add
a condition to the end of the cascade.
Later, we check the same "message" variable when deciding whether to
clear commit marks on the left/right commits; if it is set, we
presumably never started the walk. This is wrong, though; we might
have started the walk and munged commit flags, only to encounter an
error afterwards. We should always clear the flags on left/right if
they exist, whether the walk was successful or not.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2013-03-23 00:19:56 +08:00
|
|
|
if (!message) /* only NULL if we succeeded in setting up the walk */
|
2013-04-06 00:12:08 +08:00
|
|
|
print_submodule_summary(&rev, f, line_prefix, del, add, reset);
|
submodule: clarify logic in show_submodule_summary
There are two uses of the "left" and "right" commit variables that
make it hard to be sure what values they have (both for the reader,
and for gcc, which wrongly complains that they might be used
uninitialized).
The function starts with a cascading if statement, checking that the
input sha1s exist, and finally working up to preparing a revision
walk. We only prepare the walk if the cascading conditional did not
find any problems, which we check by seeing whether it set the
"message" variable or not. It's simpler and more obvious to just add
a condition to the end of the cascade.
Later, we check the same "message" variable when deciding whether to
clear commit marks on the left/right commits; if it is set, we
presumably never started the walk. This is wrong, though; we might
have started the walk and munged commit flags, only to encounter an
error afterwards. We should always clear the flags on left/right if
they exist, whether the walk was successful or not.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2013-03-23 00:19:56 +08:00
|
|
|
if (left)
|
2009-10-19 20:38:32 +08:00
|
|
|
clear_commit_marks(left, ~0);
|
submodule: clarify logic in show_submodule_summary
There are two uses of the "left" and "right" commit variables that
make it hard to be sure what values they have (both for the reader,
and for gcc, which wrongly complains that they might be used
uninitialized).
The function starts with a cascading if statement, checking that the
input sha1s exist, and finally working up to preparing a revision
walk. We only prepare the walk if the cascading conditional did not
find any problems, which we check by seeing whether it set the
"message" variable or not. It's simpler and more obvious to just add
a condition to the end of the cascade.
Later, we check the same "message" variable when deciding whether to
clear commit marks on the left/right commits; if it is set, we
presumably never started the walk. This is wrong, though; we might
have started the walk and munged commit flags, only to encounter an
error afterwards. We should always clear the flags on left/right if
they exist, whether the walk was successful or not.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2013-03-23 00:19:56 +08:00
|
|
|
if (right)
|
2009-10-19 20:38:32 +08:00
|
|
|
clear_commit_marks(right, ~0);
|
2011-03-16 15:14:11 +08:00
|
|
|
|
2009-10-19 20:38:32 +08:00
|
|
|
strbuf_release(&sb);
|
|
|
|
}
|
2010-01-17 01:42:24 +08:00
|
|
|
|
2010-11-11 07:55:02 +08:00
|
|
|
void set_config_fetch_recurse_submodules(int value)
|
|
|
|
{
|
|
|
|
config_fetch_recurse_submodules = value;
|
|
|
|
}
|
|
|
|
|
2011-08-20 06:08:47 +08:00
|
|
|
static int has_remote(const char *refname, const unsigned char *sha1, int flags, void *cb_data)
|
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int submodule_needs_pushing(const char *path, const unsigned char sha1[20])
|
|
|
|
{
|
|
|
|
if (add_submodule_odb(path) || !lookup_commit_reference(sha1))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (for_each_remote_ref_submodule(path, has_remote, NULL) > 0) {
|
2014-08-20 03:09:35 +08:00
|
|
|
struct child_process cp = CHILD_PROCESS_INIT;
|
2011-08-20 06:08:47 +08:00
|
|
|
const char *argv[] = {"rev-list", NULL, "--not", "--remotes", "-n", "1" , NULL};
|
|
|
|
struct strbuf buf = STRBUF_INIT;
|
|
|
|
int needs_pushing = 0;
|
|
|
|
|
|
|
|
argv[1] = sha1_to_hex(sha1);
|
|
|
|
cp.argv = argv;
|
|
|
|
cp.env = local_repo_env;
|
|
|
|
cp.git_cmd = 1;
|
|
|
|
cp.no_stdin = 1;
|
|
|
|
cp.out = -1;
|
|
|
|
cp.dir = path;
|
|
|
|
if (start_command(&cp))
|
|
|
|
die("Could not run 'git rev-list %s --not --remotes -n 1' command in submodule %s",
|
|
|
|
sha1_to_hex(sha1), path);
|
|
|
|
if (strbuf_read(&buf, cp.out, 41))
|
|
|
|
needs_pushing = 1;
|
|
|
|
finish_command(&cp);
|
|
|
|
close(cp.out);
|
|
|
|
strbuf_release(&buf);
|
|
|
|
return needs_pushing;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void collect_submodules_from_diff(struct diff_queue_struct *q,
|
|
|
|
struct diff_options *options,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
int i;
|
2012-03-29 15:21:23 +08:00
|
|
|
struct string_list *needs_pushing = data;
|
2011-08-20 06:08:47 +08:00
|
|
|
|
|
|
|
for (i = 0; i < q->nr; i++) {
|
|
|
|
struct diff_filepair *p = q->queue[i];
|
|
|
|
if (!S_ISGITLINK(p->two->mode))
|
|
|
|
continue;
|
2012-03-29 15:21:23 +08:00
|
|
|
if (submodule_needs_pushing(p->two->path, p->two->sha1))
|
|
|
|
string_list_insert(needs_pushing, p->two->path);
|
2011-08-20 06:08:47 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-29 15:21:23 +08:00
|
|
|
static void find_unpushed_submodule_commits(struct commit *commit,
|
|
|
|
struct string_list *needs_pushing)
|
2011-08-20 06:08:47 +08:00
|
|
|
{
|
|
|
|
struct rev_info rev;
|
|
|
|
|
|
|
|
init_revisions(&rev, NULL);
|
|
|
|
rev.diffopt.output_format |= DIFF_FORMAT_CALLBACK;
|
|
|
|
rev.diffopt.format_callback = collect_submodules_from_diff;
|
|
|
|
rev.diffopt.format_callback_data = needs_pushing;
|
2011-12-17 18:27:19 +08:00
|
|
|
diff_tree_combined_merge(commit, 1, &rev);
|
2011-08-20 06:08:47 +08:00
|
|
|
}
|
|
|
|
|
2012-03-29 15:21:23 +08:00
|
|
|
int find_unpushed_submodules(unsigned char new_sha1[20],
|
|
|
|
const char *remotes_name, struct string_list *needs_pushing)
|
2011-08-20 06:08:47 +08:00
|
|
|
{
|
|
|
|
struct rev_info rev;
|
|
|
|
struct commit *commit;
|
|
|
|
const char *argv[] = {NULL, NULL, "--not", "NULL", NULL};
|
|
|
|
int argc = ARRAY_SIZE(argv) - 1;
|
|
|
|
char *sha1_copy;
|
2012-03-29 15:21:23 +08:00
|
|
|
|
2011-08-20 06:08:47 +08:00
|
|
|
struct strbuf remotes_arg = STRBUF_INIT;
|
|
|
|
|
|
|
|
strbuf_addf(&remotes_arg, "--remotes=%s", remotes_name);
|
|
|
|
init_revisions(&rev, NULL);
|
|
|
|
sha1_copy = xstrdup(sha1_to_hex(new_sha1));
|
|
|
|
argv[1] = sha1_copy;
|
|
|
|
argv[3] = remotes_arg.buf;
|
|
|
|
setup_revisions(argc, argv, &rev, NULL);
|
|
|
|
if (prepare_revision_walk(&rev))
|
|
|
|
die("revision walk setup failed");
|
|
|
|
|
2012-03-29 15:21:23 +08:00
|
|
|
while ((commit = get_revision(&rev)) != NULL)
|
|
|
|
find_unpushed_submodule_commits(commit, needs_pushing);
|
2011-08-20 06:08:47 +08:00
|
|
|
|
2012-03-29 15:21:21 +08:00
|
|
|
reset_revision_walk();
|
2011-08-20 06:08:47 +08:00
|
|
|
free(sha1_copy);
|
|
|
|
strbuf_release(&remotes_arg);
|
|
|
|
|
2012-03-29 15:21:23 +08:00
|
|
|
return needs_pushing->nr;
|
2011-08-20 06:08:47 +08:00
|
|
|
}
|
|
|
|
|
2012-03-29 15:21:24 +08:00
|
|
|
static int push_submodule(const char *path)
|
|
|
|
{
|
|
|
|
if (add_submodule_odb(path))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (for_each_remote_ref_submodule(path, has_remote, NULL) > 0) {
|
2014-08-20 03:09:35 +08:00
|
|
|
struct child_process cp = CHILD_PROCESS_INIT;
|
2012-03-29 15:21:24 +08:00
|
|
|
const char *argv[] = {"push", NULL};
|
|
|
|
|
|
|
|
cp.argv = argv;
|
|
|
|
cp.env = local_repo_env;
|
|
|
|
cp.git_cmd = 1;
|
|
|
|
cp.no_stdin = 1;
|
|
|
|
cp.dir = path;
|
|
|
|
if (run_command(&cp))
|
|
|
|
return 0;
|
|
|
|
close(cp.out);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int push_unpushed_submodules(unsigned char new_sha1[20], const char *remotes_name)
|
|
|
|
{
|
|
|
|
int i, ret = 1;
|
2014-07-18 17:19:00 +08:00
|
|
|
struct string_list needs_pushing = STRING_LIST_INIT_DUP;
|
2012-03-29 15:21:24 +08:00
|
|
|
|
|
|
|
if (!find_unpushed_submodules(new_sha1, remotes_name, &needs_pushing))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
for (i = 0; i < needs_pushing.nr; i++) {
|
|
|
|
const char *path = needs_pushing.items[i].string;
|
|
|
|
fprintf(stderr, "Pushing submodule '%s'\n", path);
|
|
|
|
if (!push_submodule(path)) {
|
|
|
|
fprintf(stderr, "Unable to push submodule '%s'\n", path);
|
|
|
|
ret = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
string_list_clear(&needs_pushing, 0);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-03-07 06:12:58 +08:00
|
|
|
static int is_submodule_commit_present(const char *path, unsigned char sha1[20])
|
|
|
|
{
|
|
|
|
int is_present = 0;
|
|
|
|
if (!add_submodule_odb(path) && lookup_commit_reference(sha1)) {
|
|
|
|
/* Even if the submodule is checked out and the commit is
|
|
|
|
* present, make sure it is reachable from a ref. */
|
2014-08-20 03:09:35 +08:00
|
|
|
struct child_process cp = CHILD_PROCESS_INIT;
|
2011-03-07 06:12:58 +08:00
|
|
|
const char *argv[] = {"rev-list", "-n", "1", NULL, "--not", "--all", NULL};
|
|
|
|
struct strbuf buf = STRBUF_INIT;
|
|
|
|
|
|
|
|
argv[3] = sha1_to_hex(sha1);
|
|
|
|
cp.argv = argv;
|
|
|
|
cp.env = local_repo_env;
|
|
|
|
cp.git_cmd = 1;
|
|
|
|
cp.no_stdin = 1;
|
|
|
|
cp.dir = path;
|
2015-03-23 11:53:56 +08:00
|
|
|
if (!capture_command(&cp, &buf, 1024) && !buf.len)
|
2011-03-07 06:12:58 +08:00
|
|
|
is_present = 1;
|
|
|
|
|
|
|
|
strbuf_release(&buf);
|
|
|
|
}
|
|
|
|
return is_present;
|
|
|
|
}
|
|
|
|
|
2011-03-07 06:10:46 +08:00
|
|
|
static void submodule_collect_changed_cb(struct diff_queue_struct *q,
|
|
|
|
struct diff_options *options,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < q->nr; i++) {
|
|
|
|
struct diff_filepair *p = q->queue[i];
|
|
|
|
if (!S_ISGITLINK(p->two->mode))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (S_ISGITLINK(p->one->mode)) {
|
|
|
|
/* NEEDSWORK: We should honor the name configured in
|
|
|
|
* the .gitmodules file of the commit we are examining
|
|
|
|
* here to be able to correctly follow submodules
|
|
|
|
* being moved around. */
|
|
|
|
struct string_list_item *path;
|
|
|
|
path = unsorted_string_list_lookup(&changed_submodule_paths, p->two->path);
|
2011-03-07 06:12:58 +08:00
|
|
|
if (!path && !is_submodule_commit_present(p->two->path, p->two->sha1))
|
2011-03-07 06:10:46 +08:00
|
|
|
string_list_append(&changed_submodule_paths, xstrdup(p->two->path));
|
|
|
|
} else {
|
|
|
|
/* Submodule is new or was moved here */
|
|
|
|
/* NEEDSWORK: When the .git directories of submodules
|
|
|
|
* live inside the superprojects .git directory some
|
|
|
|
* day we should fetch new submodules directly into
|
|
|
|
* that location too when config or options request
|
|
|
|
* that so they can be checked out from there. */
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-13 03:56:52 +08:00
|
|
|
static int add_sha1_to_array(const char *ref, const unsigned char *sha1,
|
|
|
|
int flags, void *data)
|
|
|
|
{
|
|
|
|
sha1_array_append(data, sha1);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-03-07 06:10:46 +08:00
|
|
|
void check_for_new_submodule_commits(unsigned char new_sha1[20])
|
2011-09-13 03:56:52 +08:00
|
|
|
{
|
|
|
|
if (!initialized_fetch_ref_tips) {
|
|
|
|
for_each_ref(add_sha1_to_array, &ref_tips_before_fetch);
|
|
|
|
initialized_fetch_ref_tips = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sha1_array_append(&ref_tips_after_fetch, new_sha1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void add_sha1_to_argv(const unsigned char sha1[20], void *data)
|
|
|
|
{
|
2011-09-14 05:57:57 +08:00
|
|
|
argv_array_push(data, sha1_to_hex(sha1));
|
2011-09-13 03:56:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void calculate_changed_submodule_paths(void)
|
2011-03-07 06:10:46 +08:00
|
|
|
{
|
|
|
|
struct rev_info rev;
|
|
|
|
struct commit *commit;
|
2011-09-14 05:57:57 +08:00
|
|
|
struct argv_array argv = ARGV_ARRAY_INIT;
|
2011-03-07 06:10:46 +08:00
|
|
|
|
2011-09-10 02:22:03 +08:00
|
|
|
/* No need to check if there are no submodules configured */
|
|
|
|
if (!config_name_for_path.nr)
|
|
|
|
return;
|
|
|
|
|
2011-03-07 06:10:46 +08:00
|
|
|
init_revisions(&rev, NULL);
|
2011-09-14 05:57:57 +08:00
|
|
|
argv_array_push(&argv, "--"); /* argv[0] program name */
|
2011-09-13 03:56:52 +08:00
|
|
|
sha1_array_for_each_unique(&ref_tips_after_fetch,
|
|
|
|
add_sha1_to_argv, &argv);
|
2011-09-14 05:57:57 +08:00
|
|
|
argv_array_push(&argv, "--not");
|
2011-09-13 03:56:52 +08:00
|
|
|
sha1_array_for_each_unique(&ref_tips_before_fetch,
|
|
|
|
add_sha1_to_argv, &argv);
|
|
|
|
setup_revisions(argv.argc, argv.argv, &rev, NULL);
|
2011-03-07 06:10:46 +08:00
|
|
|
if (prepare_revision_walk(&rev))
|
|
|
|
die("revision walk setup failed");
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Collect all submodules (whether checked out or not) for which new
|
|
|
|
* commits have been recorded upstream in "changed_submodule_paths".
|
|
|
|
*/
|
|
|
|
while ((commit = get_revision(&rev))) {
|
|
|
|
struct commit_list *parent = commit->parents;
|
|
|
|
while (parent) {
|
|
|
|
struct diff_options diff_opts;
|
|
|
|
diff_setup(&diff_opts);
|
2011-06-21 02:18:03 +08:00
|
|
|
DIFF_OPT_SET(&diff_opts, RECURSIVE);
|
2011-03-07 06:10:46 +08:00
|
|
|
diff_opts.output_format |= DIFF_FORMAT_CALLBACK;
|
|
|
|
diff_opts.format_callback = submodule_collect_changed_cb;
|
2012-08-03 20:16:24 +08:00
|
|
|
diff_setup_done(&diff_opts);
|
2011-03-07 06:10:46 +08:00
|
|
|
diff_tree_sha1(parent->item->object.sha1, commit->object.sha1, "", &diff_opts);
|
|
|
|
diffcore_std(&diff_opts);
|
|
|
|
diff_flush(&diff_opts);
|
|
|
|
parent = parent->next;
|
|
|
|
}
|
|
|
|
}
|
2011-09-13 03:56:52 +08:00
|
|
|
|
2011-09-14 05:57:57 +08:00
|
|
|
argv_array_clear(&argv);
|
2011-09-13 03:56:52 +08:00
|
|
|
sha1_array_clear(&ref_tips_before_fetch);
|
|
|
|
sha1_array_clear(&ref_tips_after_fetch);
|
|
|
|
initialized_fetch_ref_tips = 0;
|
2011-03-07 06:10:46 +08:00
|
|
|
}
|
|
|
|
|
2012-09-01 23:27:06 +08:00
|
|
|
int fetch_populated_submodules(const struct argv_array *options,
|
2011-03-07 06:11:21 +08:00
|
|
|
const char *prefix, int command_line_option,
|
2010-11-11 07:55:02 +08:00
|
|
|
int quiet)
|
2010-11-12 20:54:52 +08:00
|
|
|
{
|
2012-09-01 23:27:06 +08:00
|
|
|
int i, result = 0;
|
2014-08-20 03:09:35 +08:00
|
|
|
struct child_process cp = CHILD_PROCESS_INIT;
|
2012-09-01 23:27:06 +08:00
|
|
|
struct argv_array argv = ARGV_ARRAY_INIT;
|
2010-11-12 20:54:52 +08:00
|
|
|
struct string_list_item *name_for_path;
|
|
|
|
const char *work_tree = get_git_work_tree();
|
|
|
|
if (!work_tree)
|
2011-03-07 06:10:46 +08:00
|
|
|
goto out;
|
2010-11-12 20:54:52 +08:00
|
|
|
|
2013-06-10 00:33:45 +08:00
|
|
|
if (read_cache() < 0)
|
|
|
|
die("index file corrupt");
|
2010-11-12 20:54:52 +08:00
|
|
|
|
2012-09-01 23:27:06 +08:00
|
|
|
argv_array_push(&argv, "fetch");
|
|
|
|
for (i = 0; i < options->argc; i++)
|
|
|
|
argv_array_push(&argv, options->argv[i]);
|
|
|
|
argv_array_push(&argv, "--recurse-submodules-default");
|
|
|
|
/* default value, "--submodule-prefix" and its value are added later */
|
2010-11-12 20:54:52 +08:00
|
|
|
|
|
|
|
cp.env = local_repo_env;
|
|
|
|
cp.git_cmd = 1;
|
|
|
|
cp.no_stdin = 1;
|
|
|
|
|
2011-09-13 03:56:52 +08:00
|
|
|
calculate_changed_submodule_paths();
|
|
|
|
|
2010-11-12 20:54:52 +08:00
|
|
|
for (i = 0; i < active_nr; i++) {
|
|
|
|
struct strbuf submodule_path = STRBUF_INIT;
|
|
|
|
struct strbuf submodule_git_dir = STRBUF_INIT;
|
|
|
|
struct strbuf submodule_prefix = STRBUF_INIT;
|
Convert "struct cache_entry *" to "const ..." wherever possible
I attempted to make index_state->cache[] a "const struct cache_entry **"
to find out how existing entries in index are modified and where. The
question I have is what do we do if we really need to keep track of on-disk
changes in the index. The result is
- diff-lib.c: setting CE_UPTODATE
- name-hash.c: setting CE_HASHED
- preload-index.c, read-cache.c, unpack-trees.c and
builtin/update-index: obvious
- entry.c: write_entry() may refresh the checked out entry via
fill_stat_cache_info(). This causes "non-const struct cache_entry
*" in builtin/apply.c, builtin/checkout-index.c and
builtin/checkout.c
- builtin/ls-files.c: --with-tree changes stagemask and may set
CE_UPDATE
Of these, write_entry() and its call sites are probably most
interesting because it modifies on-disk info. But this is stat info
and can be retrieved via refresh, at least for porcelain
commands. Other just uses ce_flags for local purposes.
So, keeping track of "dirty" entries is just a matter of setting a
flag in index modification functions exposed by read-cache.c. Except
unpack-trees, the rest of the code base does not do anything funny
behind read-cache's back.
The actual patch is less valueable than the summary above. But if
anyone wants to re-identify the above sites. Applying this patch, then
this:
diff --git a/cache.h b/cache.h
index 430d021..1692891 100644
--- a/cache.h
+++ b/cache.h
@@ -267,7 +267,7 @@ static inline unsigned int canon_mode(unsigned int mode)
#define cache_entry_size(len) (offsetof(struct cache_entry,name) + (len) + 1)
struct index_state {
- struct cache_entry **cache;
+ const struct cache_entry **cache;
unsigned int version;
unsigned int cache_nr, cache_alloc, cache_changed;
struct string_list *resolve_undo;
will help quickly identify them without bogus warnings.
Signed-off-by: Nguyễn Thái Ngọc Duy <pclouds@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2013-07-09 23:29:00 +08:00
|
|
|
const struct cache_entry *ce = active_cache[i];
|
2011-03-07 06:10:46 +08:00
|
|
|
const char *git_dir, *name, *default_argv;
|
2010-11-12 20:54:52 +08:00
|
|
|
|
|
|
|
if (!S_ISGITLINK(ce->ce_mode))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
name = ce->name;
|
|
|
|
name_for_path = unsorted_string_list_lookup(&config_name_for_path, ce->name);
|
|
|
|
if (name_for_path)
|
|
|
|
name = name_for_path->util;
|
|
|
|
|
2011-03-07 06:10:46 +08:00
|
|
|
default_argv = "yes";
|
2011-03-07 06:11:21 +08:00
|
|
|
if (command_line_option == RECURSE_SUBMODULES_DEFAULT) {
|
2010-11-11 07:55:41 +08:00
|
|
|
struct string_list_item *fetch_recurse_submodules_option;
|
|
|
|
fetch_recurse_submodules_option = unsorted_string_list_lookup(&config_fetch_recurse_submodules_for_name, name);
|
|
|
|
if (fetch_recurse_submodules_option) {
|
2011-03-07 06:12:19 +08:00
|
|
|
if ((intptr_t)fetch_recurse_submodules_option->util == RECURSE_SUBMODULES_OFF)
|
2010-11-11 07:55:41 +08:00
|
|
|
continue;
|
2011-03-07 06:12:19 +08:00
|
|
|
if ((intptr_t)fetch_recurse_submodules_option->util == RECURSE_SUBMODULES_ON_DEMAND) {
|
|
|
|
if (!unsorted_string_list_lookup(&changed_submodule_paths, ce->name))
|
|
|
|
continue;
|
|
|
|
default_argv = "on-demand";
|
|
|
|
}
|
2010-11-11 07:55:41 +08:00
|
|
|
} else {
|
2011-05-15 00:26:58 +08:00
|
|
|
if ((config_fetch_recurse_submodules == RECURSE_SUBMODULES_OFF) ||
|
|
|
|
gitmodules_is_unmerged)
|
2010-11-11 07:55:41 +08:00
|
|
|
continue;
|
2011-03-07 06:10:46 +08:00
|
|
|
if (config_fetch_recurse_submodules == RECURSE_SUBMODULES_ON_DEMAND) {
|
|
|
|
if (!unsorted_string_list_lookup(&changed_submodule_paths, ce->name))
|
|
|
|
continue;
|
|
|
|
default_argv = "on-demand";
|
|
|
|
}
|
2010-11-11 07:55:41 +08:00
|
|
|
}
|
2011-03-07 06:11:21 +08:00
|
|
|
} else if (command_line_option == RECURSE_SUBMODULES_ON_DEMAND) {
|
|
|
|
if (!unsorted_string_list_lookup(&changed_submodule_paths, ce->name))
|
|
|
|
continue;
|
|
|
|
default_argv = "on-demand";
|
2010-11-11 07:55:02 +08:00
|
|
|
}
|
|
|
|
|
2010-11-12 20:54:52 +08:00
|
|
|
strbuf_addf(&submodule_path, "%s/%s", work_tree, ce->name);
|
|
|
|
strbuf_addf(&submodule_git_dir, "%s/.git", submodule_path.buf);
|
|
|
|
strbuf_addf(&submodule_prefix, "%s%s/", prefix, ce->name);
|
2011-08-23 05:04:56 +08:00
|
|
|
git_dir = read_gitfile(submodule_git_dir.buf);
|
2010-11-12 20:54:52 +08:00
|
|
|
if (!git_dir)
|
|
|
|
git_dir = submodule_git_dir.buf;
|
|
|
|
if (is_directory(git_dir)) {
|
|
|
|
if (!quiet)
|
|
|
|
printf("Fetching submodule %s%s\n", prefix, ce->name);
|
|
|
|
cp.dir = submodule_path.buf;
|
2012-09-01 23:27:06 +08:00
|
|
|
argv_array_push(&argv, default_argv);
|
|
|
|
argv_array_push(&argv, "--submodule-prefix");
|
|
|
|
argv_array_push(&argv, submodule_prefix.buf);
|
|
|
|
cp.argv = argv.argv;
|
2010-11-12 20:54:52 +08:00
|
|
|
if (run_command(&cp))
|
|
|
|
result = 1;
|
2012-09-01 23:27:06 +08:00
|
|
|
argv_array_pop(&argv);
|
|
|
|
argv_array_pop(&argv);
|
|
|
|
argv_array_pop(&argv);
|
2010-11-12 20:54:52 +08:00
|
|
|
}
|
|
|
|
strbuf_release(&submodule_path);
|
|
|
|
strbuf_release(&submodule_git_dir);
|
|
|
|
strbuf_release(&submodule_prefix);
|
|
|
|
}
|
2012-09-01 23:27:06 +08:00
|
|
|
argv_array_clear(&argv);
|
2011-03-07 06:10:46 +08:00
|
|
|
out:
|
|
|
|
string_list_clear(&changed_submodule_paths, 1);
|
2010-11-12 20:54:52 +08:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2010-03-14 06:00:27 +08:00
|
|
|
unsigned is_submodule_modified(const char *path, int ignore_untracked)
|
2010-01-17 01:42:24 +08:00
|
|
|
{
|
2010-03-05 05:20:33 +08:00
|
|
|
ssize_t len;
|
2014-08-20 03:09:35 +08:00
|
|
|
struct child_process cp = CHILD_PROCESS_INIT;
|
2010-01-17 01:42:24 +08:00
|
|
|
const char *argv[] = {
|
|
|
|
"status",
|
|
|
|
"--porcelain",
|
|
|
|
NULL,
|
2010-03-14 06:00:27 +08:00
|
|
|
NULL,
|
2010-01-17 01:42:24 +08:00
|
|
|
};
|
|
|
|
struct strbuf buf = STRBUF_INIT;
|
2010-03-05 05:20:33 +08:00
|
|
|
unsigned dirty_submodule = 0;
|
|
|
|
const char *line, *next_line;
|
2010-04-11 01:01:12 +08:00
|
|
|
const char *git_dir;
|
2010-01-17 01:42:24 +08:00
|
|
|
|
2010-04-11 01:01:12 +08:00
|
|
|
strbuf_addf(&buf, "%s/.git", path);
|
2011-08-23 05:04:56 +08:00
|
|
|
git_dir = read_gitfile(buf.buf);
|
2010-04-11 01:01:12 +08:00
|
|
|
if (!git_dir)
|
|
|
|
git_dir = buf.buf;
|
|
|
|
if (!is_directory(git_dir)) {
|
2010-01-17 01:42:24 +08:00
|
|
|
strbuf_release(&buf);
|
|
|
|
/* The submodule is not checked out, so it is not modified */
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
strbuf_reset(&buf);
|
|
|
|
|
2010-03-14 06:00:27 +08:00
|
|
|
if (ignore_untracked)
|
|
|
|
argv[2] = "-uno";
|
|
|
|
|
2010-01-17 01:42:24 +08:00
|
|
|
cp.argv = argv;
|
2010-04-11 01:01:12 +08:00
|
|
|
cp.env = local_repo_env;
|
2010-01-17 01:42:24 +08:00
|
|
|
cp.git_cmd = 1;
|
|
|
|
cp.no_stdin = 1;
|
|
|
|
cp.out = -1;
|
2010-04-11 01:01:12 +08:00
|
|
|
cp.dir = path;
|
2010-01-17 01:42:24 +08:00
|
|
|
if (start_command(&cp))
|
2011-12-08 05:50:14 +08:00
|
|
|
die("Could not run 'git status --porcelain' in submodule %s", path);
|
2010-01-17 01:42:24 +08:00
|
|
|
|
|
|
|
len = strbuf_read(&buf, cp.out, 1024);
|
2010-03-05 05:20:33 +08:00
|
|
|
line = buf.buf;
|
|
|
|
while (len > 2) {
|
|
|
|
if ((line[0] == '?') && (line[1] == '?')) {
|
|
|
|
dirty_submodule |= DIRTY_SUBMODULE_UNTRACKED;
|
|
|
|
if (dirty_submodule & DIRTY_SUBMODULE_MODIFIED)
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
dirty_submodule |= DIRTY_SUBMODULE_MODIFIED;
|
2010-03-14 06:00:27 +08:00
|
|
|
if (ignore_untracked ||
|
|
|
|
(dirty_submodule & DIRTY_SUBMODULE_UNTRACKED))
|
2010-03-05 05:20:33 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
next_line = strchr(line, '\n');
|
|
|
|
if (!next_line)
|
|
|
|
break;
|
|
|
|
next_line++;
|
|
|
|
len -= (next_line - line);
|
|
|
|
line = next_line;
|
|
|
|
}
|
2010-01-17 01:42:24 +08:00
|
|
|
close(cp.out);
|
|
|
|
|
|
|
|
if (finish_command(&cp))
|
2011-12-08 05:50:14 +08:00
|
|
|
die("'git status --porcelain' failed in submodule %s", path);
|
2010-01-17 01:42:24 +08:00
|
|
|
|
|
|
|
strbuf_release(&buf);
|
2010-03-05 05:20:33 +08:00
|
|
|
return dirty_submodule;
|
2010-01-17 01:42:24 +08:00
|
|
|
}
|
2010-07-07 21:39:13 +08:00
|
|
|
|
2012-09-27 02:21:13 +08:00
|
|
|
int submodule_uses_gitfile(const char *path)
|
|
|
|
{
|
2014-08-20 03:09:35 +08:00
|
|
|
struct child_process cp = CHILD_PROCESS_INIT;
|
2012-09-27 02:21:13 +08:00
|
|
|
const char *argv[] = {
|
|
|
|
"submodule",
|
|
|
|
"foreach",
|
|
|
|
"--quiet",
|
|
|
|
"--recursive",
|
|
|
|
"test -f .git",
|
|
|
|
NULL,
|
|
|
|
};
|
|
|
|
struct strbuf buf = STRBUF_INIT;
|
|
|
|
const char *git_dir;
|
|
|
|
|
|
|
|
strbuf_addf(&buf, "%s/.git", path);
|
|
|
|
git_dir = read_gitfile(buf.buf);
|
|
|
|
if (!git_dir) {
|
|
|
|
strbuf_release(&buf);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
strbuf_release(&buf);
|
|
|
|
|
|
|
|
/* Now test that all nested submodules use a gitfile too */
|
|
|
|
cp.argv = argv;
|
|
|
|
cp.env = local_repo_env;
|
|
|
|
cp.git_cmd = 1;
|
|
|
|
cp.no_stdin = 1;
|
|
|
|
cp.no_stderr = 1;
|
|
|
|
cp.no_stdout = 1;
|
|
|
|
cp.dir = path;
|
|
|
|
if (run_command(&cp))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ok_to_remove_submodule(const char *path)
|
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
ssize_t len;
|
2014-08-20 03:09:35 +08:00
|
|
|
struct child_process cp = CHILD_PROCESS_INIT;
|
2012-09-27 02:21:13 +08:00
|
|
|
const char *argv[] = {
|
|
|
|
"status",
|
|
|
|
"--porcelain",
|
|
|
|
"-u",
|
|
|
|
"--ignore-submodules=none",
|
|
|
|
NULL,
|
|
|
|
};
|
|
|
|
struct strbuf buf = STRBUF_INIT;
|
|
|
|
int ok_to_remove = 1;
|
|
|
|
|
|
|
|
if ((lstat(path, &st) < 0) || is_empty_dir(path))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (!submodule_uses_gitfile(path))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
cp.argv = argv;
|
|
|
|
cp.env = local_repo_env;
|
|
|
|
cp.git_cmd = 1;
|
|
|
|
cp.no_stdin = 1;
|
|
|
|
cp.out = -1;
|
|
|
|
cp.dir = path;
|
|
|
|
if (start_command(&cp))
|
|
|
|
die("Could not run 'git status --porcelain -uall --ignore-submodules=none' in submodule %s", path);
|
|
|
|
|
|
|
|
len = strbuf_read(&buf, cp.out, 1024);
|
|
|
|
if (len > 2)
|
|
|
|
ok_to_remove = 0;
|
|
|
|
close(cp.out);
|
|
|
|
|
|
|
|
if (finish_command(&cp))
|
|
|
|
die("'git status --porcelain -uall --ignore-submodules=none' failed in submodule %s", path);
|
|
|
|
|
|
|
|
strbuf_release(&buf);
|
|
|
|
return ok_to_remove;
|
|
|
|
}
|
|
|
|
|
2010-07-07 21:39:13 +08:00
|
|
|
static int find_first_merges(struct object_array *result, const char *path,
|
|
|
|
struct commit *a, struct commit *b)
|
|
|
|
{
|
|
|
|
int i, j;
|
2013-05-25 17:08:12 +08:00
|
|
|
struct object_array merges = OBJECT_ARRAY_INIT;
|
2010-07-07 21:39:13 +08:00
|
|
|
struct commit *commit;
|
|
|
|
int contains_another;
|
|
|
|
|
|
|
|
char merged_revision[42];
|
|
|
|
const char *rev_args[] = { "rev-list", "--merges", "--ancestry-path",
|
|
|
|
"--all", merged_revision, NULL };
|
|
|
|
struct rev_info revs;
|
|
|
|
struct setup_revision_opt rev_opts;
|
|
|
|
|
|
|
|
memset(result, 0, sizeof(struct object_array));
|
|
|
|
memset(&rev_opts, 0, sizeof(rev_opts));
|
|
|
|
|
|
|
|
/* get all revisions that merge commit a */
|
|
|
|
snprintf(merged_revision, sizeof(merged_revision), "^%s",
|
|
|
|
sha1_to_hex(a->object.sha1));
|
|
|
|
init_revisions(&revs, NULL);
|
|
|
|
rev_opts.submodule = path;
|
2014-06-27 20:11:39 +08:00
|
|
|
setup_revisions(ARRAY_SIZE(rev_args)-1, rev_args, &revs, &rev_opts);
|
2010-07-07 21:39:13 +08:00
|
|
|
|
|
|
|
/* save all revisions from the above list that contain b */
|
|
|
|
if (prepare_revision_walk(&revs))
|
|
|
|
die("revision walk setup failed");
|
|
|
|
while ((commit = get_revision(&revs)) != NULL) {
|
|
|
|
struct object *o = &(commit->object);
|
2012-08-28 05:46:01 +08:00
|
|
|
if (in_merge_bases(b, commit))
|
2010-07-07 21:39:13 +08:00
|
|
|
add_object_array(o, NULL, &merges);
|
|
|
|
}
|
2012-03-29 15:21:21 +08:00
|
|
|
reset_revision_walk();
|
2010-07-07 21:39:13 +08:00
|
|
|
|
|
|
|
/* Now we've got all merges that contain a and b. Prune all
|
|
|
|
* merges that contain another found merge and save them in
|
|
|
|
* result.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < merges.nr; i++) {
|
|
|
|
struct commit *m1 = (struct commit *) merges.objects[i].item;
|
|
|
|
|
|
|
|
contains_another = 0;
|
|
|
|
for (j = 0; j < merges.nr; j++) {
|
|
|
|
struct commit *m2 = (struct commit *) merges.objects[j].item;
|
2012-08-28 05:46:01 +08:00
|
|
|
if (i != j && in_merge_bases(m2, m1)) {
|
2010-07-07 21:39:13 +08:00
|
|
|
contains_another = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!contains_another)
|
2013-05-25 17:08:13 +08:00
|
|
|
add_object_array(merges.objects[i].item, NULL, result);
|
2010-07-07 21:39:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
free(merges.objects);
|
|
|
|
return result->nr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void print_commit(struct commit *commit)
|
|
|
|
{
|
|
|
|
struct strbuf sb = STRBUF_INIT;
|
|
|
|
struct pretty_print_context ctx = {0};
|
|
|
|
ctx.date_mode = DATE_NORMAL;
|
|
|
|
format_commit_message(commit, " %h: %m %s", &sb, &ctx);
|
|
|
|
fprintf(stderr, "%s\n", sb.buf);
|
|
|
|
strbuf_release(&sb);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define MERGE_WARNING(path, msg) \
|
|
|
|
warning("Failed to merge submodule %s (%s)", path, msg);
|
|
|
|
|
|
|
|
int merge_submodule(unsigned char result[20], const char *path,
|
|
|
|
const unsigned char base[20], const unsigned char a[20],
|
2011-10-13 20:59:05 +08:00
|
|
|
const unsigned char b[20], int search)
|
2010-07-07 21:39:13 +08:00
|
|
|
{
|
|
|
|
struct commit *commit_base, *commit_a, *commit_b;
|
|
|
|
int parent_count;
|
|
|
|
struct object_array merges;
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* store a in result in case we fail */
|
|
|
|
hashcpy(result, a);
|
|
|
|
|
|
|
|
/* we can not handle deletion conflicts */
|
|
|
|
if (is_null_sha1(base))
|
|
|
|
return 0;
|
|
|
|
if (is_null_sha1(a))
|
|
|
|
return 0;
|
|
|
|
if (is_null_sha1(b))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (add_submodule_odb(path)) {
|
|
|
|
MERGE_WARNING(path, "not checked out");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(commit_base = lookup_commit_reference(base)) ||
|
|
|
|
!(commit_a = lookup_commit_reference(a)) ||
|
|
|
|
!(commit_b = lookup_commit_reference(b))) {
|
|
|
|
MERGE_WARNING(path, "commits not present");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check whether both changes are forward */
|
2012-08-28 05:46:01 +08:00
|
|
|
if (!in_merge_bases(commit_base, commit_a) ||
|
|
|
|
!in_merge_bases(commit_base, commit_b)) {
|
2010-07-07 21:39:13 +08:00
|
|
|
MERGE_WARNING(path, "commits don't follow merge-base");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Case #1: a is contained in b or vice versa */
|
2012-08-28 05:46:01 +08:00
|
|
|
if (in_merge_bases(commit_a, commit_b)) {
|
2010-07-07 21:39:13 +08:00
|
|
|
hashcpy(result, b);
|
|
|
|
return 1;
|
|
|
|
}
|
2012-08-28 05:46:01 +08:00
|
|
|
if (in_merge_bases(commit_b, commit_a)) {
|
2010-07-07 21:39:13 +08:00
|
|
|
hashcpy(result, a);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Case #2: There are one or more merges that contain a and b in
|
|
|
|
* the submodule. If there is only one, then present it as a
|
|
|
|
* suggestion to the user, but leave it marked unmerged so the
|
|
|
|
* user needs to confirm the resolution.
|
|
|
|
*/
|
|
|
|
|
2011-10-13 20:59:05 +08:00
|
|
|
/* Skip the search if makes no sense to the calling context. */
|
|
|
|
if (!search)
|
|
|
|
return 0;
|
|
|
|
|
2010-07-07 21:39:13 +08:00
|
|
|
/* find commit which merges them */
|
|
|
|
parent_count = find_first_merges(&merges, path, commit_a, commit_b);
|
|
|
|
switch (parent_count) {
|
|
|
|
case 0:
|
|
|
|
MERGE_WARNING(path, "merge following commits not found");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
MERGE_WARNING(path, "not fast-forward");
|
|
|
|
fprintf(stderr, "Found a possible merge resolution "
|
|
|
|
"for the submodule:\n");
|
|
|
|
print_commit((struct commit *) merges.objects[0].item);
|
|
|
|
fprintf(stderr,
|
|
|
|
"If this is correct simply add it to the index "
|
|
|
|
"for example\n"
|
|
|
|
"by using:\n\n"
|
|
|
|
" git update-index --cacheinfo 160000 %s \"%s\"\n\n"
|
|
|
|
"which will accept this suggestion.\n",
|
|
|
|
sha1_to_hex(merges.objects[0].item->sha1), path);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
MERGE_WARNING(path, "multiple merges found");
|
|
|
|
for (i = 0; i < merges.nr; i++)
|
|
|
|
print_commit((struct commit *) merges.objects[i].item);
|
|
|
|
}
|
|
|
|
|
|
|
|
free(merges.objects);
|
|
|
|
return 0;
|
|
|
|
}
|
2013-07-31 03:50:03 +08:00
|
|
|
|
|
|
|
/* Update gitfile and core.worktree setting to connect work tree and git dir */
|
|
|
|
void connect_work_tree_and_git_dir(const char *work_tree, const char *git_dir)
|
|
|
|
{
|
|
|
|
struct strbuf file_name = STRBUF_INIT;
|
|
|
|
struct strbuf rel_path = STRBUF_INIT;
|
|
|
|
const char *real_work_tree = xstrdup(real_path(work_tree));
|
|
|
|
FILE *fp;
|
|
|
|
|
|
|
|
/* Update gitfile */
|
|
|
|
strbuf_addf(&file_name, "%s/.git", work_tree);
|
|
|
|
fp = fopen(file_name.buf, "w");
|
|
|
|
if (!fp)
|
|
|
|
die(_("Could not create git link %s"), file_name.buf);
|
|
|
|
fprintf(fp, "gitdir: %s\n", relative_path(git_dir, real_work_tree,
|
|
|
|
&rel_path));
|
|
|
|
fclose(fp);
|
|
|
|
|
|
|
|
/* Update core.worktree setting */
|
|
|
|
strbuf_reset(&file_name);
|
|
|
|
strbuf_addf(&file_name, "%s/config", git_dir);
|
|
|
|
if (git_config_set_in_file(file_name.buf, "core.worktree",
|
|
|
|
relative_path(real_work_tree, git_dir,
|
|
|
|
&rel_path)))
|
|
|
|
die(_("Could not set core.worktree in %s"),
|
|
|
|
file_name.buf);
|
|
|
|
|
|
|
|
strbuf_release(&file_name);
|
|
|
|
strbuf_release(&rel_path);
|
|
|
|
free((void *)real_work_tree);
|
|
|
|
}
|