summaryrefslogtreecommitdiff
path: root/src/annotated_commit.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/annotated_commit.c')
-rw-r--r--src/annotated_commit.c89
1 files changed, 71 insertions, 18 deletions
diff --git a/src/annotated_commit.c b/src/annotated_commit.c
index 3f2d2ed17..e53b95dee 100644
--- a/src/annotated_commit.c
+++ b/src/annotated_commit.c
@@ -7,12 +7,16 @@
#include "common.h"
#include "annotated_commit.h"
+#include "refs.h"
+#include "cache.h"
#include "git2/commit.h"
#include "git2/refs.h"
#include "git2/repository.h"
#include "git2/annotated_commit.h"
#include "git2/revparse.h"
+#include "git2/tree.h"
+#include "git2/index.h"
static int annotated_commit_init(
git_annotated_commit **out,
@@ -22,14 +26,17 @@ static int annotated_commit_init(
const char *remote_url)
{
git_annotated_commit *annotated_commit;
+ git_commit *commit = NULL;
int error = 0;
assert(out && id);
*out = NULL;
- annotated_commit = git__calloc(1, sizeof(git_annotated_commit));
- GITERR_CHECK_ALLOC(annotated_commit);
+ if ((error = git_commit_lookup(&commit, repo, id)) < 0 ||
+ (error = git_annotated_commit_from_commit(&annotated_commit,
+ commit)) < 0)
+ goto done;
if (ref_name) {
annotated_commit->ref_name = git__strdup(ref_name);
@@ -41,15 +48,10 @@ static int annotated_commit_init(
GITERR_CHECK_ALLOC(annotated_commit->remote_url);
}
- git_oid_fmt(annotated_commit->id_str, id);
- annotated_commit->id_str[GIT_OID_HEXSZ] = '\0';
-
- if ((error = git_commit_lookup(&annotated_commit->commit, repo, id)) < 0) {
- git_annotated_commit_free(annotated_commit);
- return error;
- }
-
*out = annotated_commit;
+
+done:
+ git_commit_free(commit);
return error;
}
@@ -75,6 +77,51 @@ int git_annotated_commit_from_ref(
return error;
}
+int git_annotated_commit_from_head(
+ git_annotated_commit **out,
+ git_repository *repo)
+{
+ git_reference *head;
+ int error;
+
+ assert(out && repo);
+
+ *out = NULL;
+
+ if ((error = git_reference_lookup(&head, repo, GIT_HEAD_FILE)) < 0)
+ return -1;
+
+ error = git_annotated_commit_from_ref(out, repo, head);
+
+ git_reference_free(head);
+ return error;
+}
+
+int git_annotated_commit_from_commit(
+ git_annotated_commit **out,
+ git_commit *commit)
+{
+ git_annotated_commit *annotated_commit;
+
+ assert(out && commit);
+
+ *out = NULL;
+
+ annotated_commit = git__calloc(1, sizeof(git_annotated_commit));
+ GITERR_CHECK_ALLOC(annotated_commit);
+
+ annotated_commit->type = GIT_ANNOTATED_COMMIT_REAL;
+
+ git_cached_obj_incref(commit);
+ annotated_commit->commit = commit;
+
+ git_oid_fmt(annotated_commit->id_str, git_commit_id(commit));
+ annotated_commit->id_str[GIT_OID_HEXSZ] = '\0';
+
+ *out = annotated_commit;
+ return 0;
+}
+
int git_annotated_commit_lookup(
git_annotated_commit **out,
git_repository *repo,
@@ -136,14 +183,20 @@ void git_annotated_commit_free(git_annotated_commit *annotated_commit)
if (annotated_commit == NULL)
return;
- if (annotated_commit->commit != NULL)
- git_commit_free(annotated_commit->commit);
-
- if (annotated_commit->ref_name != NULL)
- git__free(annotated_commit->ref_name);
-
- if (annotated_commit->remote_url != NULL)
- git__free(annotated_commit->remote_url);
+ switch (annotated_commit->type) {
+ case GIT_ANNOTATED_COMMIT_REAL:
+ git_commit_free(annotated_commit->commit);
+ git_tree_free(annotated_commit->tree);
+ git__free(annotated_commit->ref_name);
+ git__free(annotated_commit->remote_url);
+ break;
+ case GIT_ANNOTATED_COMMIT_VIRTUAL:
+ git_index_free(annotated_commit->index);
+ git_array_clear(annotated_commit->parents);
+ break;
+ default:
+ abort();
+ }
git__free(annotated_commit);
}