diff options
Diffstat (limited to 'src/refs.c')
-rw-r--r-- | src/refs.c | 547 |
1 files changed, 547 insertions, 0 deletions
diff --git a/src/refs.c b/src/refs.c new file mode 100644 index 000000000..91f13aed0 --- /dev/null +++ b/src/refs.c @@ -0,0 +1,547 @@ +/* + * This file is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License, version 2, + * as published by the Free Software Foundation. + * + * In addition to the permissions in the GNU General Public License, + * the authors give you unlimited permission to link the compiled + * version of this file into combinations with other programs, + * and to distribute those combinations without any restriction + * coming from the use of this file. (The General Public License + * restrictions do apply in other respects; for example, they cover + * modification of the file, and distribution when not linked into + * a combined executable.) + * + * This file is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; see the file COPYING. If not, write to + * the Free Software Foundation, 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301, USA. + */ + +#include "refs.h" +#include "hash.h" +#include "repository.h" +#include "fileops.h" + +#define HASH_SEED 2147483647 +#define MAX_NESTING_LEVEL 5 + +static const int default_table_size = 32; + +static struct { + size_t size; /* size in bytes of the object structure */ +} git_references_table[] = { + {sizeof(git_reference_object_id)}, /* 0 = GIT_REF_OBJECT_ID */ + {sizeof(git_reference_symbolic)}, /* 1 = GIT_REF_SYMBOLIC */ +}; + +static uint32_t reftable_hash(const void *key) +{ + return git__hash(key, strlen((const char *)key), HASH_SEED); +} + +static int reftable_haskey(void *reference, const void *key) +{ + git_reference *ref; + char *name; + + ref = (git_reference *)reference; + name = (char *)key; + + return strcmp(name, ref->name) == 0; +} + +git_reference_database *git_reference_database__alloc() +{ + git_reference_database *ref_database = git__malloc(sizeof(git_reference_database)); + if (!ref_database) + return NULL; + + memset(ref_database, 0x0, sizeof(git_reference_database)); + + ref_database->references = git_hashtable_alloc( + default_table_size, + reftable_hash, + reftable_haskey); + + if (ref_database->references == NULL) { + free(ref_database); + return NULL; + } + + return ref_database; +} + +static void reference__free(git_reference *reference) +{ + assert(reference); + + switch (reference->type) { + case GIT_REF_SYMBOLIC: + // The target of the symbolic ref has to be freed by itself. + + /* Fallthrough */ + + case GIT_REF_ANY: + case GIT_REF_OBJECT_ID: + if (reference->name) + free(reference->name); + + /* Fallthrough */ + + default: + free(reference); + break; + } +} + +void git_reference_database__free(git_reference_database *ref_database) +{ + git_hashtable_iterator it; + git_reference *reference; + + assert(ref_database); + + git_hashtable_iterator_init(ref_database->references, &it); + + while ((reference = (git_reference *)git_hashtable_iterator_next(&it)) != NULL) { + git_hashtable_remove(ref_database->references, reference->name); + reference__free(reference); + } + + git_hashtable_free(ref_database->references); + free(ref_database); +} + + +static int check_refname_validity(const char *name) { + int error = GIT_SUCCESS; + + // TODO : To be implemented + + return error; +} + +static int reference_newobject(git_reference **reference_out, git_rtype type, const char *name) +{ + git_reference *reference = NULL; + + assert(reference_out && name); + + *reference_out = NULL; + + switch (type) { + case GIT_REF_OBJECT_ID: + case GIT_REF_SYMBOLIC: + break; + + default: + return GIT_EINVALIDTYPE; + } + + reference = git__malloc(git_references_table[type].size); + + if (reference == NULL) + return GIT_ENOMEM; + + memset(reference, 0x0, git_references_table[type].size); + + reference->name = git__malloc(strlen(name) + 1); + strcpy(reference->name, name); + + reference->type = type; + + + *reference_out = reference; + + return GIT_SUCCESS; +} + +static int symbolic_reference_target_name__parse(char *target_name_out, const char *name, gitfo_buf *buffer) { + int error = GIT_SUCCESS; + char *refname_start, *refname_end; + const char *buffer_end; + int refname_len; + + refname_start = (char *)buffer->data; + buffer_end = (const char *)(buffer->data) + buffer->len; + + if (git__prefixcmp(refname_start, GIT_SYMREF)) + return GIT_EREFCORRUPTED; + + refname_start += strlen(GIT_SYMREF); + + /* Skip the potential white spaces */ + while (isspace(refname_start[0]) && refname_start < buffer_end) + refname_start++; + + refname_end = refname_start; + + /* Seek the end of the target reference name */ + while(!isspace(refname_end[0]) && refname_end < buffer_end) + refname_end++; + + refname_len = refname_end - refname_start; + + memcpy(target_name_out, refname_start, refname_len); + target_name_out[refname_len] = 0; + + return error; +} + +static int object_id_reference__parse(git_reference **reference_out, const char *name, gitfo_buf *buffer) { + int error = GIT_SUCCESS; + git_oid target_oid; + git_reference *reference; + char *buffer_start; + const char *buffer_end; + + buffer_start = (char *)buffer->data; + buffer_end = (const char *)(buffer_start) + buffer->len; + + /* Is this a valid object id ? */ + error = git__parse_oid(&target_oid, &buffer_start, buffer_end, ""); + if (error < GIT_SUCCESS) + return error; + + error = reference_newobject(&reference, GIT_REF_OBJECT_ID, name); + if (error < GIT_SUCCESS) + return error; + + git_oid_cpy(&((git_reference_object_id *)reference)->id, &target_oid); + + *reference_out = reference; + + return error; +} + +static int read_loose_reference(gitfo_buf *file_content, const char *name, const char *path_repository) +{ + int error = GIT_SUCCESS; + char ref_path[GIT_PATH_MAX]; + + /* Determine the full path of the ref */ + strcpy(ref_path, path_repository); + strcat(ref_path, name); + + /* Does it even exist ? */ + if (gitfo_exists(ref_path) < GIT_SUCCESS) + return GIT_ENOTFOUND; + + /* A ref can not be a directory */ + if (!gitfo_isdir(ref_path)) + return GIT_EINVALIDREFNAME; + + error = gitfo_read_file(file_content, ref_path); + + return error; +} + +static int try_to_find_an_existing_loose_reference(git_reference **reference_out, git_reference_database *ref_database, const char *name, const char *path_repository, int *nesting_level) +{ + int error = GIT_SUCCESS; + gitfo_buf file_content = GITFO_BUF_INIT; + git_reference *reference, *target_reference; + git_reference_symbolic *peeled_reference; + char target_name[MAX_GITDIR_TREE_STRUCTURE_PATH_LENGTH]; + + + error = read_loose_reference(&file_content, name, path_repository); + if (error < GIT_SUCCESS) + goto cleanup; + + /* Does this look like a symbolic ref ? */ + if (!git__prefixcmp((const char *)(file_content.data), GIT_SYMREF)) { + error = symbolic_reference_target_name__parse(target_name, name, &file_content); + if (error < GIT_SUCCESS) + goto cleanup; + + error = git_reference_lookup(&target_reference, ref_database, target_name, path_repository, nesting_level); + if (error < GIT_SUCCESS) + goto cleanup; + + error = reference_newobject((git_reference **)&peeled_reference, GIT_REF_SYMBOLIC, name); + if (error < GIT_SUCCESS) + goto cleanup; + + peeled_reference->target = target_reference; + + reference = (git_reference *)peeled_reference; + + goto found; + } + + if (object_id_reference__parse(&reference, name, &file_content) < GIT_SUCCESS) { + error = GIT_EREFCORRUPTED; + goto cleanup; + } + +found: + *reference_out = reference; + +cleanup: + if (file_content.data) + gitfo_free_buf(&file_content); + + return error; +} + +static int read_packed_refs_content(gitfo_buf *file_content, const char *path_repository) +{ + int error = GIT_SUCCESS; + char ref_path[GIT_PATH_MAX]; + + /* Determine the full path of the file */ + strcpy(ref_path, path_repository); + strcat(ref_path, GIT_PACKEDREFS_FILE); + + /* Does it even exist ? */ + if (gitfo_exists(ref_path) < GIT_SUCCESS) + return GIT_ENOTFOUND; + + error = gitfo_read_file(file_content, ref_path); + + return error; +} + +static int packed_tag_peeled_reference__parse(git_oid *peeled_oid_out, git_reference *tag_reference, char** buffer_out, const char *buffer_end) +{ + int error = GIT_SUCCESS; + + /* Ensure it's not the first entry of the file */ + if (tag_reference == NULL) + return GIT_EPACKEDREFSCORRUPTED; + + /* Ensure reference is a tag */ + if (git__prefixcmp(tag_reference->name, GIT_REFS_TAGS_DIR)) + return GIT_EPACKEDREFSCORRUPTED; + + /* Is this a valid object id ? */ + if (git__parse_oid(peeled_oid_out, buffer_out, buffer_end, "^") < GIT_SUCCESS) { + error = GIT_EPACKEDREFSCORRUPTED; + } + + return error; +} + +static int packed_reference__parse(git_oid *oid_out, char *reference_name_out, char** buffer_out, const char *buffer_end) +{ + int error = GIT_SUCCESS; + char *refname_end; + int refname_len; + + /* This should be the beginning of a line containing an object id, a space and its name */ + if ((*buffer_out + GIT_OID_HEXSZ)[0] != ' ') + return GIT_EPACKEDREFSCORRUPTED; + + /* Slight hack to reuse git__parse_oid() which assumes that the id is LF terminated */ + (*buffer_out + GIT_OID_HEXSZ)[0] = '\n'; + + /* Is this a valid object id ? */ + if (git__parse_oid(oid_out, buffer_out, buffer_end, "") < GIT_SUCCESS) + return GIT_EPACKEDREFSCORRUPTED; + + /* We should be at the begining of the name of the reference */ + if (isspace(*buffer_out[0])) + return GIT_EPACKEDREFSCORRUPTED; + + refname_end = *buffer_out; + + /* Seek the end of the target reference name */ + while(!isspace(refname_end[0]) && refname_end < buffer_end) + refname_end++; + + refname_len = refname_end - *buffer_out; + + memcpy(reference_name_out, *buffer_out, refname_len); + reference_name_out[refname_len] = 0; + + *buffer_out = refname_end + 1; + + return error; +} + +static int packed_reference_file__parse(git_reference **reference_out, git_reference_database *ref_database, const char *name, const char *path_repository, int *nesting_level) +{ + int error = GIT_SUCCESS; + gitfo_buf file_content = GITFO_BUF_INIT; + char *buffer_start; + const char *buffer_end; + char reference_name[MAX_GITDIR_TREE_STRUCTURE_PATH_LENGTH]; + git_oid oid, peeled_oid; + git_reference *reference = NULL, *found_reference = NULL; + + error = read_packed_refs_content(&file_content, path_repository); + if (error < GIT_SUCCESS) + goto cleanup; + + buffer_start = (char *)file_content.data; + buffer_end = (const char *)(buffer_start) + file_content.len; + + /* Does the header look like valid ? */ + if (git__prefixcmp((const char *)(buffer_start), GIT_PACKEDREFS_HEADER)) { + error = GIT_EPACKEDREFSCORRUPTED; + goto cleanup; + } + + /* Let's skip the header */ + buffer_start += strlen(GIT_PACKEDREFS_HEADER); + + while (buffer_start < buffer_end) { + /* Is it a peeled reference pointed at by a tag ? */ + if (buffer_start[0] == '^') { + error = packed_tag_peeled_reference__parse(&peeled_oid, reference, &buffer_start, buffer_end); + if (error < GIT_SUCCESS) + goto cleanup; + + /* As we do not _currently_ need the peeled object pointed at by the tag, we just don't use the parsed object id. Maybe later ? */ + + /* Reinit the reference to catch potential successive lines starting by '^' */ + reference = NULL; + + continue; + } + + error = packed_reference__parse(&oid, reference_name, &buffer_start, buffer_end); + if (error < GIT_SUCCESS) + goto cleanup; + + /* Does a more up-to-date loose reference exist ? */ + reference = git_hashtable_lookup(ref_database->references, reference_name); + if (reference == NULL) { + error = reference_newobject(&reference, GIT_REF_OBJECT_ID, reference_name); + if (error < GIT_SUCCESS) + goto cleanup; + + reference->is_packed = 1; + + git_oid_cpy(&((git_reference_object_id *)reference)->id, &oid); + + git_hashtable_insert(ref_database->references, reference->name, reference); + + /* Is it the reference we're looking for ? */ + if (!strcmp(reference_name, name)) + found_reference = reference; // TODO : Should we guard against two found references in the same packed-refs file ? + } + } + + ref_database->have_packed_refs_been_parsed = 1; + + if (found_reference == NULL) { + error = GIT_ENOTFOUND; + goto cleanup; + } + + *reference_out = found_reference; + +cleanup: + if (file_content.data) + gitfo_free_buf(&file_content); + + return error; +} + +int git_reference_lookup(git_reference **reference_out, git_reference_database *ref_database, const char *name, const char *path_repository, int *nesting_level) +{ + int error = GIT_SUCCESS; + git_reference *reference, *packed_reference = NULL; + + if (*nesting_level == MAX_NESTING_LEVEL) { + return GIT_ETOONESTEDSYMREF; + } + + error = check_refname_validity(name); + if (error < GIT_SUCCESS) + return error; + + /* Has the ref already been previously parsed ? */ + reference = git_hashtable_lookup(ref_database->references, name); + + /* Has a loose reference been found ? */ + if (reference != NULL && !reference->is_packed) { + *reference_out = reference; + return GIT_SUCCESS; + } + + /* Has every available ref already been parsed ? */ + if (ref_database->is_fully_loaded) { + + if (reference == NULL) { + return GIT_ENOTFOUND; + } else { + /* Is is safe to consider the packed reference as the most up-to-date reference */ + assert(reference->is_packed); + *reference_out = reference; + return GIT_SUCCESS; + } + } + + /* We temporarily store the packed reference until we're sure no more up-to-date loose reference exists. */ + if (reference != NULL) { + assert(reference->is_packed); + packed_reference = reference; + } + + if (*nesting_level == 0) { + /* Is the database being populated */ + if (ref_database->is_busy) + return GIT_EBUSY; + + ref_database->is_busy = 1; + } + + (*nesting_level)++; + + /* Does the reference exist as a loose file based reference ? */ + error = try_to_find_an_existing_loose_reference(&reference, ref_database, name, path_repository, nesting_level); + + /* Have we found a more up-to-date loose reference than the packed reference we stored ? */ + if (error == GIT_SUCCESS && packed_reference != NULL) { + git_hashtable_remove(ref_database->references, packed_reference->name); + reference__free(packed_reference); + } + + if (error == GIT_SUCCESS) { + git_hashtable_insert(ref_database->references, reference->name, reference); + goto found; + } + + if (error != GIT_ENOTFOUND) + goto cleanup; + + /* Nothing has been found in the loose refs */ + assert(error == GIT_ENOTFOUND); + + /* If we've stored a pack reference, now is the time to return it */ + if (packed_reference != NULL) { + reference = packed_reference; + error = GIT_SUCCESS; + goto found; + } + + /* Have the dormant references already been parsed ? */ + if (ref_database->have_packed_refs_been_parsed) + return GIT_ENOTFOUND; + + /* has the reference previously been packed ? */ + error = packed_reference_file__parse(&reference, ref_database, name, path_repository, nesting_level); + if (error < GIT_SUCCESS) { + goto cleanup; + } + +found: + *reference_out = reference; + +cleanup: + (*nesting_level)--; + + if (*nesting_level == 0) + ref_database->is_busy = 0; + + return error; +} |