Commit 967e073d by Edward Thomson

merge driver: correct global initialization

parent 7a3ab14f
...@@ -9,6 +9,7 @@ ...@@ -9,6 +9,7 @@
#include "hash.h" #include "hash.h"
#include "sysdir.h" #include "sysdir.h"
#include "filter.h" #include "filter.h"
#include "merge_driver.h"
#include "openssl_stream.h" #include "openssl_stream.h"
#include "thread-utils.h" #include "thread-utils.h"
#include "git2/global.h" #include "git2/global.h"
...@@ -59,6 +60,7 @@ static int init_common(void) ...@@ -59,6 +60,7 @@ static int init_common(void)
if ((ret = git_hash_global_init()) == 0 && if ((ret = git_hash_global_init()) == 0 &&
(ret = git_sysdir_global_init()) == 0 && (ret = git_sysdir_global_init()) == 0 &&
(ret = git_filter_global_init()) == 0 && (ret = git_filter_global_init()) == 0 &&
(ret = git_merge_driver_global_init()) == 0 &&
(ret = git_transport_ssh_global_init()) == 0) (ret = git_transport_ssh_global_init()) == 0)
ret = git_openssl_stream_global_init(); ret = git_openssl_stream_global_init();
......
...@@ -29,6 +29,7 @@ ...@@ -29,6 +29,7 @@
#include "annotated_commit.h" #include "annotated_commit.h"
#include "commit.h" #include "commit.h"
#include "oidarray.h" #include "oidarray.h"
#include "merge_driver.h"
#include "git2/types.h" #include "git2/types.h"
#include "git2/repository.h" #include "git2/repository.h"
......
...@@ -36,37 +36,6 @@ enum { ...@@ -36,37 +36,6 @@ enum {
}; };
/* Merge drivers */
struct git_merge_driver_source {
git_repository *repo;
const char *default_driver;
const git_merge_file_options *file_opts;
const git_index_entry *ancestor;
const git_index_entry *ours;
const git_index_entry *theirs;
};
extern int git_merge_driver_for_path(
char **name_out,
git_merge_driver **driver_out,
git_repository *repo,
const char *path);
/* Basic (normal) merge driver, takes favor type as the payload argument */
extern git_merge_driver git_merge_driver__normal;
/* Merge driver for text files, performs a standard three-way merge */
extern git_merge_driver git_merge_driver__text;
/* Merge driver for union-style merging */
extern git_merge_driver git_merge_driver__union;
/* Merge driver for unmergeable (binary) files: always produces conflicts */
extern git_merge_driver git_merge_driver__binary;
/** Types of changes when files are merged from branch to branch. */ /** Types of changes when files are merged from branch to branch. */
typedef enum { typedef enum {
/* No conflict - a change only occurs in one branch. */ /* No conflict - a change only occurs in one branch. */
......
...@@ -9,6 +9,7 @@ ...@@ -9,6 +9,7 @@
#include "vector.h" #include "vector.h"
#include "global.h" #include "global.h"
#include "merge.h" #include "merge.h"
#include "merge_driver.h"
#include "git2/merge.h" #include "git2/merge.h"
#include "git2/sys/merge.h" #include "git2/sys/merge.h"
...@@ -17,6 +18,7 @@ static const char *merge_driver_name__union = "union"; ...@@ -17,6 +18,7 @@ static const char *merge_driver_name__union = "union";
static const char *merge_driver_name__binary = "binary"; static const char *merge_driver_name__binary = "binary";
struct merge_driver_registry { struct merge_driver_registry {
git_rwlock lock;
git_vector drivers; git_vector drivers;
}; };
...@@ -26,11 +28,14 @@ typedef struct { ...@@ -26,11 +28,14 @@ typedef struct {
char name[GIT_FLEX_ARRAY]; char name[GIT_FLEX_ARRAY];
} git_merge_driver_entry; } git_merge_driver_entry;
static struct merge_driver_registry *merge_driver_registry = NULL; static struct merge_driver_registry merge_driver_registry;
static git_merge_file_favor_t merge_favor_normal = GIT_MERGE_FILE_FAVOR_NORMAL; static git_merge_file_favor_t merge_favor_normal = GIT_MERGE_FILE_FAVOR_NORMAL;
static git_merge_file_favor_t merge_favor_union = GIT_MERGE_FILE_FAVOR_UNION; static git_merge_file_favor_t merge_favor_union = GIT_MERGE_FILE_FAVOR_UNION;
static void git_merge_driver_global_shutdown(void);
static int merge_driver_apply( static int merge_driver_apply(
git_merge_driver *self, git_merge_driver *self,
void **payload, void **payload,
...@@ -144,26 +149,6 @@ static int merge_driver_entry_search(const void *a, const void *b) ...@@ -144,26 +149,6 @@ static int merge_driver_entry_search(const void *a, const void *b)
return strcmp(name_a, entry_b->name); return strcmp(name_a, entry_b->name);
} }
static void merge_driver_registry_shutdown(void)
{
struct merge_driver_registry *reg;
git_merge_driver_entry *entry;
size_t i;
if ((reg = git__swap(merge_driver_registry, NULL)) == NULL)
return;
git_vector_foreach(&reg->drivers, i, entry) {
if (entry && entry->driver->shutdown)
entry->driver->shutdown(entry->driver);
git__free(entry);
}
git_vector_free(&reg->drivers);
git__free(reg);
}
git_merge_driver git_merge_driver__normal = { git_merge_driver git_merge_driver__normal = {
GIT_MERGE_DRIVER_VERSION, GIT_MERGE_DRIVER_VERSION,
NULL, NULL,
...@@ -196,73 +181,133 @@ git_merge_driver git_merge_driver__binary = { ...@@ -196,73 +181,133 @@ git_merge_driver git_merge_driver__binary = {
merge_driver_binary_apply merge_driver_binary_apply
}; };
static int merge_driver_registry_initialize(void) /* Note: callers must lock the registry before calling this function */
static int merge_driver_registry_insert(
const char *name, git_merge_driver *driver)
{ {
struct merge_driver_registry *reg; git_merge_driver_entry *entry;
int error = 0;
if (merge_driver_registry) entry = git__calloc(1, sizeof(git_merge_driver_entry) + strlen(name) + 1);
return 0; GITERR_CHECK_ALLOC(entry);
strcpy(entry->name, name);
entry->driver = driver;
reg = git__calloc(1, sizeof(struct merge_driver_registry)); return git_vector_insert_sorted(
GITERR_CHECK_ALLOC(reg); &merge_driver_registry.drivers, entry, NULL);
}
if ((error = git_vector_init(&reg->drivers, 3, merge_driver_entry_cmp)) < 0) int git_merge_driver_global_init(void)
goto done; {
int error;
reg = git__compare_and_swap(&merge_driver_registry, NULL, reg); if (git_rwlock_init(&merge_driver_registry.lock) < 0)
return -1;
if (reg != NULL) if ((error = git_vector_init(&merge_driver_registry.drivers, 3,
merge_driver_entry_cmp)) < 0)
goto done; goto done;
git__on_shutdown(merge_driver_registry_shutdown); if ((error = merge_driver_registry_insert(
if ((error = git_merge_driver_register(
merge_driver_name__text, &git_merge_driver__text)) < 0 || merge_driver_name__text, &git_merge_driver__text)) < 0 ||
(error = git_merge_driver_register( (error = merge_driver_registry_insert(
merge_driver_name__union, &git_merge_driver__union)) < 0 || merge_driver_name__union, &git_merge_driver__union)) < 0 ||
(error = git_merge_driver_register( (error = merge_driver_registry_insert(
merge_driver_name__binary, &git_merge_driver__binary)) < 0) merge_driver_name__binary, &git_merge_driver__binary)) < 0)
goto done;
git__on_shutdown(git_merge_driver_global_shutdown);
done: done:
if (error < 0) if (error < 0)
merge_driver_registry_shutdown(); git_vector_free_deep(&merge_driver_registry.drivers);
return error; return error;
} }
int git_merge_driver_register(const char *name, git_merge_driver *driver) static void git_merge_driver_global_shutdown(void)
{ {
git_merge_driver_entry *entry; git_merge_driver_entry *entry;
size_t i;
if (git_rwlock_wrlock(&merge_driver_registry.lock) < 0)
return;
git_vector_foreach(&merge_driver_registry.drivers, i, entry) {
if (entry->driver->shutdown)
entry->driver->shutdown(entry->driver);
git__free(entry);
}
git_vector_free(&merge_driver_registry.drivers);
git_rwlock_wrunlock(&merge_driver_registry.lock);
git_rwlock_free(&merge_driver_registry.lock);
}
/* Note: callers must lock the registry before calling this function */
static int merge_driver_registry_find(size_t *pos, const char *name)
{
return git_vector_search2(pos, &merge_driver_registry.drivers,
merge_driver_entry_search, name);
}
/* Note: callers must lock the registry before calling this function */
static git_merge_driver_entry *merge_driver_registry_lookup(
size_t *pos, const char *name)
{
git_merge_driver_entry *entry = NULL;
if (!merge_driver_registry_find(pos, name))
entry = git_vector_get(&merge_driver_registry.drivers, *pos);
return entry;
}
int git_merge_driver_register(const char *name, git_merge_driver *driver)
{
int error;
assert(name && driver); assert(name && driver);
if (merge_driver_registry_initialize() < 0) if (git_rwlock_wrlock(&merge_driver_registry.lock) < 0) {
giterr_set(GITERR_OS, "failed to lock merge driver registry");
return -1; return -1;
}
entry = git__calloc(1, sizeof(git_merge_driver_entry) + strlen(name) + 1); if (!merge_driver_registry_find(NULL, name)) {
GITERR_CHECK_ALLOC(entry); giterr_set(GITERR_MERGE, "attempt to reregister existing driver '%s'",
name);
error = GIT_EEXISTS;
goto done;
}
strcpy(entry->name, name); error = merge_driver_registry_insert(name, driver);
entry->driver = driver;
return git_vector_insert_sorted( done:
&merge_driver_registry->drivers, entry, NULL); git_rwlock_wrunlock(&merge_driver_registry.lock);
return error;
} }
int git_merge_driver_unregister(const char *name) int git_merge_driver_unregister(const char *name)
{ {
git_merge_driver_entry *entry; git_merge_driver_entry *entry;
size_t pos; size_t pos;
int error; int error = 0;
if ((error = git_vector_search2(&pos, &merge_driver_registry->drivers, if (git_rwlock_wrlock(&merge_driver_registry.lock) < 0) {
merge_driver_entry_search, name)) < 0) giterr_set(GITERR_OS, "failed to lock merge driver registry");
return error; return -1;
}
if ((entry = merge_driver_registry_lookup(&pos, name)) == NULL) {
giterr_set(GITERR_MERGE, "cannot find merge driver '%s' to unregister",
name);
error = GIT_ENOTFOUND;
goto done;
}
entry = git_vector_get(&merge_driver_registry->drivers, pos); git_vector_remove(&merge_driver_registry.drivers, pos);
git_vector_remove(&merge_driver_registry->drivers, pos);
if (entry->initialized && entry->driver->shutdown) { if (entry->initialized && entry->driver->shutdown) {
entry->driver->shutdown(entry->driver); entry->driver->shutdown(entry->driver);
...@@ -271,7 +316,9 @@ int git_merge_driver_unregister(const char *name) ...@@ -271,7 +316,9 @@ int git_merge_driver_unregister(const char *name)
git__free(entry); git__free(entry);
return 0; done:
git_rwlock_wrunlock(&merge_driver_registry.lock);
return error;
} }
git_merge_driver *git_merge_driver_lookup(const char *name) git_merge_driver *git_merge_driver_lookup(const char *name)
...@@ -282,24 +329,27 @@ git_merge_driver *git_merge_driver_lookup(const char *name) ...@@ -282,24 +329,27 @@ git_merge_driver *git_merge_driver_lookup(const char *name)
/* If we've decided the merge driver to use internally - and not /* If we've decided the merge driver to use internally - and not
* based on user configuration (in merge_driver_name_for_path) * based on user configuration (in merge_driver_name_for_path)
* then we can use a hardcoded name instead of looking it up in * then we can use a hardcoded name to compare instead of bothering
* the vector. * to take a lock and look it up in the vector.
*/ */
if (name == merge_driver_name__text) if (name == merge_driver_name__text)
return &git_merge_driver__text; return &git_merge_driver__text;
else if (name == merge_driver_name__binary) else if (name == merge_driver_name__binary)
return &git_merge_driver__binary; return &git_merge_driver__binary;
if (merge_driver_registry_initialize() < 0) if (git_rwlock_rdlock(&merge_driver_registry.lock) < 0) {
giterr_set(GITERR_OS, "failed to lock merge driver registry");
return NULL; return NULL;
}
error = git_vector_search2(&pos, &merge_driver_registry->drivers, entry = merge_driver_registry_lookup(&pos, name);
merge_driver_entry_search, name);
if (error == GIT_ENOTFOUND) git_rwlock_rdunlock(&merge_driver_registry.lock);
return NULL;
entry = git_vector_get(&merge_driver_registry->drivers, pos); if (entry == NULL) {
giterr_set(GITERR_MERGE, "cannot use an unregistered filter");
return NULL;
}
if (!entry->initialized) { if (!entry->initialized) {
if (entry->driver->initialize && if (entry->driver->initialize &&
......
/*
* Copyright (C) the libgit2 contributors. All rights reserved.
*
* This file is part of libgit2, distributed under the GNU GPL v2 with
* a Linking Exception. For full terms see the included COPYING file.
*/
#ifndef INCLUDE_merge_driver_h__
#define INCLUDE_merge_driver_h__
#include "git2/merge.h"
#include "git2/index.h"
#include "git2/sys/merge.h"
struct git_merge_driver_source {
git_repository *repo;
const char *default_driver;
const git_merge_file_options *file_opts;
const git_index_entry *ancestor;
const git_index_entry *ours;
const git_index_entry *theirs;
};
extern int git_merge_driver_global_init(void);
extern int git_merge_driver_for_path(
char **name_out,
git_merge_driver **driver_out,
git_repository *repo,
const char *path);
/* Basic (normal) merge driver, takes favor type as the payload argument */
extern git_merge_driver git_merge_driver__normal;
/* Merge driver for text files, performs a standard three-way merge */
extern git_merge_driver git_merge_driver__text;
/* Merge driver for union-style merging */
extern git_merge_driver git_merge_driver__union;
/* Merge driver for unmergeable (binary) files: always produces conflicts */
extern git_merge_driver git_merge_driver__binary;
#endif
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment