Since in the future we want to support an additional hash algorithm, add a structure that represents a hash algorithm and all the data that must go along with it. Add a constant to allow easy enumeration of hash algorithms. Implement function typedefs to create an abstract API that can be used by any hash algorithm, and wrappers for the existing SHA1 functions that conform to this API.
Don't include an entry in the hash algorithm structure for the null object ID. As this value is all zeros, any suitably sized all-zero object ID can be used, and there's no need to store a given one on a per-hash basis. Signed-off-by: brian m. carlson <sand...@crustytoothpaste.net> --- cache.h | 36 ++++++++++++++++++++++++++++++++++++ sha1_file.c | 29 +++++++++++++++++++++++++++++ 2 files changed, 65 insertions(+) diff --git a/cache.h b/cache.h index 1c69d2a05a..375a7fb15e 100644 --- a/cache.h +++ b/cache.h @@ -76,6 +76,42 @@ struct object_id { unsigned char hash[GIT_MAX_RAWSZ]; }; +#define GIT_HASH_SHA1 0 + +typedef void (*git_hash_init_fn)(void *ctx); +typedef void (*git_hash_update_fn)(void *ctx, const void *in, size_t len); +typedef void (*git_hash_final_fn)(unsigned char *hash, void *ctx); + +struct git_hash_algo { + /* The name of the algorithm, as appears in the config file. */ + const char *name; + + /* The size of a hash context (e.g. git_SHA_CTX). */ + size_t ctxsz; + + /* The length of the hash in bytes. */ + size_t rawsz; + + /* The length of the hash in hex characters. */ + size_t hexsz; + + /* The hash initialization function. */ + git_hash_init_fn init_fn; + + /* The hash update function. */ + git_hash_update_fn update_fn; + + /* The hash finalization function. */ + git_hash_final_fn final_fn; + + /* The OID of the empty tree. */ + const struct object_id *empty_tree; + + /* The OID of the empty blob. */ + const struct object_id *empty_blob; +}; +extern const struct git_hash_algo hash_algos[1]; + #if defined(DT_UNKNOWN) && !defined(NO_D_TYPE_IN_DIRENT) #define DTYPE(de) ((de)->d_type) #else diff --git a/sha1_file.c b/sha1_file.c index b60ae15f70..bd9ff02802 100644 --- a/sha1_file.c +++ b/sha1_file.c @@ -41,6 +41,35 @@ const struct object_id empty_blob_oid = { EMPTY_BLOB_SHA1_BIN_LITERAL }; +static inline void git_hash_sha1_init(void *ctx) +{ + git_SHA1_Init((git_SHA_CTX *)ctx); +} + +static inline void git_hash_sha1_update(void *ctx, const void *data, size_t len) +{ + git_SHA1_Update((git_SHA_CTX *)ctx, data, len); +} + +static inline void git_hash_sha1_final(unsigned char *hash, void *ctx) +{ + git_SHA1_Final(hash, (git_SHA_CTX *)ctx); +} + +const struct git_hash_algo hash_algos[1] = { + [GIT_HASH_SHA1] = { + .name = "sha-1", + .ctxsz = sizeof(git_SHA_CTX), + .rawsz = GIT_SHA1_RAWSZ, + .hexsz = GIT_SHA1_HEXSZ, + .init_fn = git_hash_sha1_init, + .update_fn = git_hash_sha1_update, + .final_fn = git_hash_sha1_final, + .empty_tree = &empty_tree_oid, + .empty_blob = &empty_blob_oid, + }, +}; + /* * This is meant to hold a *small* number of objects that you would * want read_sha1_file() to be able to return, but yet you do not want