Blame src/merge_driver.c

Packit ae9e2a
/*
Packit ae9e2a
 * Copyright (C) the libgit2 contributors. All rights reserved.
Packit ae9e2a
 *
Packit ae9e2a
 * This file is part of libgit2, distributed under the GNU GPL v2 with
Packit ae9e2a
 * a Linking Exception. For full terms see the included COPYING file.
Packit ae9e2a
 */
Packit ae9e2a
Packit ae9e2a
#include "common.h"
Packit ae9e2a
#include "vector.h"
Packit ae9e2a
#include "global.h"
Packit ae9e2a
#include "merge.h"
Packit ae9e2a
#include "merge_driver.h"
Packit ae9e2a
#include "git2/merge.h"
Packit ae9e2a
#include "git2/sys/merge.h"
Packit ae9e2a
Packit ae9e2a
static const char *merge_driver_name__text = "text";
Packit ae9e2a
static const char *merge_driver_name__union = "union";
Packit ae9e2a
static const char *merge_driver_name__binary = "binary";
Packit ae9e2a
Packit ae9e2a
struct merge_driver_registry {
Packit ae9e2a
	git_rwlock lock;
Packit ae9e2a
	git_vector drivers;
Packit ae9e2a
};
Packit ae9e2a
Packit ae9e2a
typedef struct {
Packit ae9e2a
	git_merge_driver *driver;
Packit ae9e2a
	int initialized;
Packit ae9e2a
	char name[GIT_FLEX_ARRAY];
Packit ae9e2a
} git_merge_driver_entry;
Packit ae9e2a
Packit ae9e2a
static struct merge_driver_registry merge_driver_registry;
Packit ae9e2a
Packit ae9e2a
static void git_merge_driver_global_shutdown(void);
Packit ae9e2a
Packit ae9e2a
const git_repository* git_merge_driver_source_repo(const git_merge_driver_source *src)
Packit ae9e2a
{
Packit ae9e2a
	assert(src);
Packit ae9e2a
	return src->repo;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
const git_index_entry* git_merge_driver_source_ancestor(const git_merge_driver_source *src)
Packit ae9e2a
{
Packit ae9e2a
	assert(src);
Packit ae9e2a
	return src->ancestor;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
const git_index_entry* git_merge_driver_source_ours(const git_merge_driver_source *src)
Packit ae9e2a
{
Packit ae9e2a
	assert(src);
Packit ae9e2a
	return src->ours;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
const git_index_entry* git_merge_driver_source_theirs(const git_merge_driver_source *src)
Packit ae9e2a
{
Packit ae9e2a
	assert(src);
Packit ae9e2a
	return src->theirs;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
const git_merge_file_options* git_merge_driver_source_file_options(const git_merge_driver_source *src)
Packit ae9e2a
{
Packit ae9e2a
	assert(src);
Packit ae9e2a
	return src->file_opts;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_merge_driver__builtin_apply(
Packit ae9e2a
	git_merge_driver *self,
Packit ae9e2a
	const char **path_out,
Packit ae9e2a
	uint32_t *mode_out,
Packit ae9e2a
	git_buf *merged_out,
Packit ae9e2a
	const char *filter_name,
Packit ae9e2a
	const git_merge_driver_source *src)
Packit ae9e2a
{
Packit ae9e2a
	git_merge_driver__builtin *driver = (git_merge_driver__builtin *)self;
Packit ae9e2a
	git_merge_file_options file_opts = GIT_MERGE_FILE_OPTIONS_INIT;
Packit ae9e2a
	git_merge_file_result result = {0};
Packit ae9e2a
	int error;
Packit ae9e2a
Packit ae9e2a
	GIT_UNUSED(filter_name);
Packit ae9e2a
Packit ae9e2a
	if (src->file_opts)
Packit ae9e2a
		memcpy(&file_opts, src->file_opts, sizeof(git_merge_file_options));
Packit ae9e2a
Packit ae9e2a
	if (driver->favor)
Packit ae9e2a
		file_opts.favor = driver->favor;
Packit ae9e2a
Packit ae9e2a
	if ((error = git_merge_file_from_index(&result, src->repo,
Packit ae9e2a
		src->ancestor, src->ours, src->theirs, &file_opts)) < 0)
Packit ae9e2a
		goto done;
Packit ae9e2a
Packit ae9e2a
	if (!result.automergeable &&
Packit ae9e2a
		!(file_opts.flags & GIT_MERGE_FILE_FAVOR__CONFLICTED)) {
Packit ae9e2a
		error = GIT_EMERGECONFLICT;
Packit ae9e2a
		goto done;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	*path_out = git_merge_file__best_path(
Packit ae9e2a
		src->ancestor ? src->ancestor->path : NULL,
Packit ae9e2a
		src->ours ? src->ours->path : NULL,
Packit ae9e2a
		src->theirs ? src->theirs->path : NULL);
Packit ae9e2a
Packit ae9e2a
	*mode_out = git_merge_file__best_mode(
Packit ae9e2a
		src->ancestor ? src->ancestor->mode : 0,
Packit ae9e2a
		src->ours ? src->ours->mode : 0,
Packit ae9e2a
		src->theirs ? src->theirs->mode : 0);
Packit ae9e2a
Packit ae9e2a
	merged_out->ptr = (char *)result.ptr;
Packit ae9e2a
	merged_out->size = result.len;
Packit ae9e2a
	merged_out->asize = result.len;
Packit ae9e2a
	result.ptr = NULL;
Packit ae9e2a
Packit ae9e2a
done:
Packit ae9e2a
	git_merge_file_result_free(&result);
Packit ae9e2a
	return error;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static int merge_driver_binary_apply(
Packit ae9e2a
	git_merge_driver *self,
Packit ae9e2a
	const char **path_out,
Packit ae9e2a
	uint32_t *mode_out,
Packit ae9e2a
	git_buf *merged_out,
Packit ae9e2a
	const char *filter_name,
Packit ae9e2a
	const git_merge_driver_source *src)
Packit ae9e2a
{
Packit ae9e2a
	GIT_UNUSED(self);
Packit ae9e2a
	GIT_UNUSED(path_out);
Packit ae9e2a
	GIT_UNUSED(mode_out);
Packit ae9e2a
	GIT_UNUSED(merged_out);
Packit ae9e2a
	GIT_UNUSED(filter_name);
Packit ae9e2a
	GIT_UNUSED(src);
Packit ae9e2a
Packit ae9e2a
	return GIT_EMERGECONFLICT;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static int merge_driver_entry_cmp(const void *a, const void *b)
Packit ae9e2a
{
Packit ae9e2a
	const git_merge_driver_entry *entry_a = a;
Packit ae9e2a
	const git_merge_driver_entry *entry_b = b;
Packit ae9e2a
Packit ae9e2a
	return strcmp(entry_a->name, entry_b->name);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static int merge_driver_entry_search(const void *a, const void *b)
Packit ae9e2a
{
Packit ae9e2a
	const char *name_a = a;
Packit ae9e2a
	const git_merge_driver_entry *entry_b = b;
Packit ae9e2a
Packit ae9e2a
	return strcmp(name_a, entry_b->name);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
git_merge_driver__builtin git_merge_driver__text = {
Packit ae9e2a
	{
Packit ae9e2a
		GIT_MERGE_DRIVER_VERSION,
Packit ae9e2a
		NULL,
Packit ae9e2a
		NULL,
Packit ae9e2a
		git_merge_driver__builtin_apply,
Packit ae9e2a
	},
Packit ae9e2a
	GIT_MERGE_FILE_FAVOR_NORMAL
Packit ae9e2a
};
Packit ae9e2a
Packit ae9e2a
git_merge_driver__builtin git_merge_driver__union = {
Packit ae9e2a
	{
Packit ae9e2a
		GIT_MERGE_DRIVER_VERSION,
Packit ae9e2a
		NULL,
Packit ae9e2a
		NULL,
Packit ae9e2a
		git_merge_driver__builtin_apply,
Packit ae9e2a
	},
Packit ae9e2a
	GIT_MERGE_FILE_FAVOR_UNION
Packit ae9e2a
};
Packit ae9e2a
Packit ae9e2a
git_merge_driver git_merge_driver__binary = {
Packit ae9e2a
	GIT_MERGE_DRIVER_VERSION,
Packit ae9e2a
	NULL,
Packit ae9e2a
	NULL,
Packit ae9e2a
	merge_driver_binary_apply
Packit ae9e2a
};
Packit ae9e2a
Packit ae9e2a
/* Note: callers must lock the registry before calling this function */
Packit ae9e2a
static int merge_driver_registry_insert(
Packit ae9e2a
	const char *name, git_merge_driver *driver)
Packit ae9e2a
{
Packit ae9e2a
	git_merge_driver_entry *entry;
Packit ae9e2a
Packit ae9e2a
	entry = git__calloc(1, sizeof(git_merge_driver_entry) + strlen(name) + 1);
Packit ae9e2a
	GITERR_CHECK_ALLOC(entry);
Packit ae9e2a
Packit ae9e2a
	strcpy(entry->name, name);
Packit ae9e2a
	entry->driver = driver;
Packit ae9e2a
Packit ae9e2a
	return git_vector_insert_sorted(
Packit ae9e2a
		&merge_driver_registry.drivers, entry, NULL);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_merge_driver_global_init(void)
Packit ae9e2a
{
Packit ae9e2a
	int error;
Packit ae9e2a
Packit ae9e2a
	if (git_rwlock_init(&merge_driver_registry.lock) < 0)
Packit ae9e2a
		return -1;
Packit ae9e2a
Packit ae9e2a
	if ((error = git_vector_init(&merge_driver_registry.drivers, 3,
Packit ae9e2a
		merge_driver_entry_cmp)) < 0)
Packit ae9e2a
		goto done;
Packit ae9e2a
Packit ae9e2a
	if ((error = merge_driver_registry_insert(
Packit ae9e2a
			merge_driver_name__text, &git_merge_driver__text.base)) < 0 ||
Packit ae9e2a
		(error = merge_driver_registry_insert(
Packit ae9e2a
			merge_driver_name__union, &git_merge_driver__union.base)) < 0 ||
Packit ae9e2a
		(error = merge_driver_registry_insert(
Packit ae9e2a
			merge_driver_name__binary, &git_merge_driver__binary)) < 0)
Packit ae9e2a
		goto done;
Packit ae9e2a
Packit ae9e2a
	git__on_shutdown(git_merge_driver_global_shutdown);
Packit ae9e2a
Packit ae9e2a
done:
Packit ae9e2a
	if (error < 0)
Packit ae9e2a
		git_vector_free_deep(&merge_driver_registry.drivers);
Packit ae9e2a
Packit ae9e2a
	return error;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static void git_merge_driver_global_shutdown(void)
Packit ae9e2a
{
Packit ae9e2a
	git_merge_driver_entry *entry;
Packit ae9e2a
	size_t i;
Packit ae9e2a
Packit ae9e2a
	if (git_rwlock_wrlock(&merge_driver_registry.lock) < 0)
Packit ae9e2a
		return;
Packit ae9e2a
Packit ae9e2a
	git_vector_foreach(&merge_driver_registry.drivers, i, entry) {
Packit ae9e2a
		if (entry->driver->shutdown)
Packit ae9e2a
			entry->driver->shutdown(entry->driver);
Packit ae9e2a
Packit ae9e2a
		git__free(entry);
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	git_vector_free(&merge_driver_registry.drivers);
Packit ae9e2a
Packit ae9e2a
	git_rwlock_wrunlock(&merge_driver_registry.lock);
Packit ae9e2a
	git_rwlock_free(&merge_driver_registry.lock);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
/* Note: callers must lock the registry before calling this function */
Packit ae9e2a
static int merge_driver_registry_find(size_t *pos, const char *name)
Packit ae9e2a
{
Packit ae9e2a
	return git_vector_search2(pos, &merge_driver_registry.drivers,
Packit ae9e2a
		merge_driver_entry_search, name);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
/* Note: callers must lock the registry before calling this function */
Packit ae9e2a
static git_merge_driver_entry *merge_driver_registry_lookup(
Packit ae9e2a
	size_t *pos, const char *name)
Packit ae9e2a
{
Packit ae9e2a
	git_merge_driver_entry *entry = NULL;
Packit ae9e2a
Packit ae9e2a
	if (!merge_driver_registry_find(pos, name))
Packit ae9e2a
		entry = git_vector_get(&merge_driver_registry.drivers, *pos);
Packit ae9e2a
Packit ae9e2a
	return entry;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_merge_driver_register(const char *name, git_merge_driver *driver)
Packit ae9e2a
{
Packit ae9e2a
	int error;
Packit ae9e2a
Packit ae9e2a
	assert(name && driver);
Packit ae9e2a
Packit ae9e2a
	if (git_rwlock_wrlock(&merge_driver_registry.lock) < 0) {
Packit ae9e2a
		giterr_set(GITERR_OS, "failed to lock merge driver registry");
Packit ae9e2a
		return -1;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	if (!merge_driver_registry_find(NULL, name)) {
Packit ae9e2a
		giterr_set(GITERR_MERGE, "attempt to reregister existing driver '%s'",
Packit ae9e2a
			name);
Packit ae9e2a
		error = GIT_EEXISTS;
Packit ae9e2a
		goto done;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	error = merge_driver_registry_insert(name, driver);
Packit ae9e2a
Packit ae9e2a
done:
Packit ae9e2a
	git_rwlock_wrunlock(&merge_driver_registry.lock);
Packit ae9e2a
	return error;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_merge_driver_unregister(const char *name)
Packit ae9e2a
{
Packit ae9e2a
	git_merge_driver_entry *entry;
Packit ae9e2a
	size_t pos;
Packit ae9e2a
	int error = 0;
Packit ae9e2a
Packit ae9e2a
	if (git_rwlock_wrlock(&merge_driver_registry.lock) < 0) {
Packit ae9e2a
		giterr_set(GITERR_OS, "failed to lock merge driver registry");
Packit ae9e2a
		return -1;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	if ((entry = merge_driver_registry_lookup(&pos, name)) == NULL) {
Packit ae9e2a
		giterr_set(GITERR_MERGE, "cannot find merge driver '%s' to unregister",
Packit ae9e2a
			name);
Packit ae9e2a
		error = GIT_ENOTFOUND;
Packit ae9e2a
		goto done;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	git_vector_remove(&merge_driver_registry.drivers, pos);
Packit ae9e2a
Packit ae9e2a
	if (entry->initialized && entry->driver->shutdown) {
Packit ae9e2a
		entry->driver->shutdown(entry->driver);
Packit ae9e2a
		entry->initialized = false;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	git__free(entry);
Packit ae9e2a
Packit ae9e2a
done:
Packit ae9e2a
	git_rwlock_wrunlock(&merge_driver_registry.lock);
Packit ae9e2a
	return error;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
git_merge_driver *git_merge_driver_lookup(const char *name)
Packit ae9e2a
{
Packit ae9e2a
	git_merge_driver_entry *entry;
Packit ae9e2a
	size_t pos;
Packit ae9e2a
	int error;
Packit ae9e2a
Packit ae9e2a
	/* If we've decided the merge driver to use internally - and not
Packit ae9e2a
	 * based on user configuration (in merge_driver_name_for_path)
Packit ae9e2a
	 * then we can use a hardcoded name to compare instead of bothering
Packit ae9e2a
	 * to take a lock and look it up in the vector.
Packit ae9e2a
	 */
Packit ae9e2a
	if (name == merge_driver_name__text)
Packit ae9e2a
		return &git_merge_driver__text.base;
Packit ae9e2a
	else if (name == merge_driver_name__binary)
Packit ae9e2a
		return &git_merge_driver__binary;
Packit ae9e2a
Packit ae9e2a
	if (git_rwlock_rdlock(&merge_driver_registry.lock) < 0) {
Packit ae9e2a
		giterr_set(GITERR_OS, "failed to lock merge driver registry");
Packit ae9e2a
		return NULL;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	entry = merge_driver_registry_lookup(&pos, name);
Packit ae9e2a
Packit ae9e2a
	git_rwlock_rdunlock(&merge_driver_registry.lock);
Packit ae9e2a
Packit ae9e2a
	if (entry == NULL) {
Packit ae9e2a
		giterr_set(GITERR_MERGE, "cannot use an unregistered filter");
Packit ae9e2a
		return NULL;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	if (!entry->initialized) {
Packit ae9e2a
		if (entry->driver->initialize &&
Packit ae9e2a
			(error = entry->driver->initialize(entry->driver)) < 0)
Packit ae9e2a
			return NULL;
Packit ae9e2a
Packit ae9e2a
		entry->initialized = 1;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	return entry->driver;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static int merge_driver_name_for_path(
Packit ae9e2a
	const char **out,
Packit ae9e2a
	git_repository *repo,
Packit ae9e2a
	const char *path,
Packit ae9e2a
	const char *default_driver)
Packit ae9e2a
{
Packit ae9e2a
	const char *value;
Packit ae9e2a
	int error;
Packit ae9e2a
Packit ae9e2a
	*out = NULL;
Packit ae9e2a
Packit ae9e2a
	if ((error = git_attr_get(&value, repo, 0, path, "merge")) < 0)
Packit ae9e2a
		return error;
Packit ae9e2a
Packit ae9e2a
	/* set: use the built-in 3-way merge driver ("text") */
Packit ae9e2a
	if (GIT_ATTR_TRUE(value))
Packit ae9e2a
		*out = merge_driver_name__text;
Packit ae9e2a
Packit ae9e2a
	/* unset: do not merge ("binary") */
Packit ae9e2a
	else if (GIT_ATTR_FALSE(value))
Packit ae9e2a
		*out = merge_driver_name__binary;
Packit ae9e2a
Packit ae9e2a
	else if (GIT_ATTR_UNSPECIFIED(value) && default_driver)
Packit ae9e2a
		*out = default_driver;
Packit ae9e2a
Packit ae9e2a
	else if (GIT_ATTR_UNSPECIFIED(value))
Packit ae9e2a
		*out = merge_driver_name__text;
Packit ae9e2a
Packit ae9e2a
	else
Packit ae9e2a
		*out = value;
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(git_merge_driver *) merge_driver_lookup_with_wildcard(
Packit ae9e2a
	const char *name)
Packit ae9e2a
{
Packit ae9e2a
	git_merge_driver *driver = git_merge_driver_lookup(name);
Packit ae9e2a
Packit ae9e2a
	if (driver == NULL)
Packit ae9e2a
		driver = git_merge_driver_lookup("*");
Packit ae9e2a
Packit ae9e2a
	return driver;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_merge_driver_for_source(
Packit ae9e2a
	const char **name_out,
Packit ae9e2a
	git_merge_driver **driver_out,
Packit ae9e2a
	const git_merge_driver_source *src)
Packit ae9e2a
{
Packit ae9e2a
	const char *path, *driver_name;
Packit ae9e2a
	int error = 0;
Packit ae9e2a
Packit ae9e2a
	path = git_merge_file__best_path(
Packit ae9e2a
		src->ancestor ? src->ancestor->path : NULL,
Packit ae9e2a
		src->ours ? src->ours->path : NULL,
Packit ae9e2a
		src->theirs ? src->theirs->path : NULL);
Packit ae9e2a
Packit ae9e2a
	if ((error = merge_driver_name_for_path(
Packit ae9e2a
			&driver_name, src->repo, path, src->default_driver)) < 0)
Packit ae9e2a
		return error;
Packit ae9e2a
Packit ae9e2a
	*name_out = driver_name;
Packit ae9e2a
	*driver_out = merge_driver_lookup_with_wildcard(driver_name);
Packit ae9e2a
	return error;
Packit ae9e2a
}
Packit ae9e2a