sha1_object_info_extended(): expose a bit more info
The original interface for sha1_object_info() takes an object name and gives back a type and its size (the latter is given only when it was asked). The new interface wraps its implementation and exposes a bit more pieces of information that the interface used to discard, namely: - where the object is stored (loose? cached? packed?) - if packed, where in which packfile? Signed-off-by: Junio C Hamano <gitster@pobox.com> --- * In the earlier round, this used u.pack.delta to record the length of the delta chain, but the caller is not necessarily interested in the length of the delta chain per-se, but may only want to know if it is a delta against another object or is stored as a deflated data. Calling packed_object_info_detail() involves walking the reverse index chain to compute the store size of the object and is unnecessarily expensive. We could resurrect the code if a new caller wants to know, but I doubt it.
This commit is contained in:
parent
b9a62cbeb9
commit
9a49059022
28
cache.h
28
cache.h
@ -1022,6 +1022,34 @@ extern unsigned long unpack_object_header_buffer(const unsigned char *buf, unsig
|
|||||||
extern unsigned long get_size_from_delta(struct packed_git *, struct pack_window **, off_t);
|
extern unsigned long get_size_from_delta(struct packed_git *, struct pack_window **, off_t);
|
||||||
extern int packed_object_info_detail(struct packed_git *, off_t, unsigned long *, unsigned long *, unsigned int *, unsigned char *);
|
extern int packed_object_info_detail(struct packed_git *, off_t, unsigned long *, unsigned long *, unsigned int *, unsigned char *);
|
||||||
|
|
||||||
|
struct object_info {
|
||||||
|
/* Request */
|
||||||
|
unsigned long *sizep;
|
||||||
|
|
||||||
|
/* Response */
|
||||||
|
enum {
|
||||||
|
OI_CACHED,
|
||||||
|
OI_LOOSE,
|
||||||
|
OI_PACKED
|
||||||
|
} whence;
|
||||||
|
union {
|
||||||
|
/*
|
||||||
|
* struct {
|
||||||
|
* ... Nothing to expose in this case
|
||||||
|
* } cached;
|
||||||
|
* struct {
|
||||||
|
* ... Nothing to expose in this case
|
||||||
|
* } loose;
|
||||||
|
*/
|
||||||
|
struct {
|
||||||
|
struct packed_git *pack;
|
||||||
|
off_t offset;
|
||||||
|
unsigned int is_delta;
|
||||||
|
} packed;
|
||||||
|
} u;
|
||||||
|
};
|
||||||
|
extern int sha1_object_info_extended(const unsigned char *, struct object_info *);
|
||||||
|
|
||||||
/* Dumb servers support */
|
/* Dumb servers support */
|
||||||
extern int update_server_info(int);
|
extern int update_server_info(int);
|
||||||
|
|
||||||
|
42
sha1_file.c
42
sha1_file.c
@ -1481,7 +1481,7 @@ static off_t get_delta_base(struct packed_git *p,
|
|||||||
|
|
||||||
/* forward declaration for a mutually recursive function */
|
/* forward declaration for a mutually recursive function */
|
||||||
static int packed_object_info(struct packed_git *p, off_t offset,
|
static int packed_object_info(struct packed_git *p, off_t offset,
|
||||||
unsigned long *sizep);
|
unsigned long *sizep, int *rtype);
|
||||||
|
|
||||||
static int packed_delta_info(struct packed_git *p,
|
static int packed_delta_info(struct packed_git *p,
|
||||||
struct pack_window **w_curs,
|
struct pack_window **w_curs,
|
||||||
@ -1495,7 +1495,7 @@ static int packed_delta_info(struct packed_git *p,
|
|||||||
base_offset = get_delta_base(p, w_curs, &curpos, type, obj_offset);
|
base_offset = get_delta_base(p, w_curs, &curpos, type, obj_offset);
|
||||||
if (!base_offset)
|
if (!base_offset)
|
||||||
return OBJ_BAD;
|
return OBJ_BAD;
|
||||||
type = packed_object_info(p, base_offset, NULL);
|
type = packed_object_info(p, base_offset, NULL, NULL);
|
||||||
if (type <= OBJ_NONE) {
|
if (type <= OBJ_NONE) {
|
||||||
struct revindex_entry *revidx;
|
struct revindex_entry *revidx;
|
||||||
const unsigned char *base_sha1;
|
const unsigned char *base_sha1;
|
||||||
@ -1605,7 +1605,7 @@ int packed_object_info_detail(struct packed_git *p,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int packed_object_info(struct packed_git *p, off_t obj_offset,
|
static int packed_object_info(struct packed_git *p, off_t obj_offset,
|
||||||
unsigned long *sizep)
|
unsigned long *sizep, int *rtype)
|
||||||
{
|
{
|
||||||
struct pack_window *w_curs = NULL;
|
struct pack_window *w_curs = NULL;
|
||||||
unsigned long size;
|
unsigned long size;
|
||||||
@ -1613,6 +1613,8 @@ static int packed_object_info(struct packed_git *p, off_t obj_offset,
|
|||||||
enum object_type type;
|
enum object_type type;
|
||||||
|
|
||||||
type = unpack_object_header(p, &w_curs, &curpos, &size);
|
type = unpack_object_header(p, &w_curs, &curpos, &size);
|
||||||
|
if (rtype)
|
||||||
|
*rtype = type; /* representation type */
|
||||||
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case OBJ_OFS_DELTA:
|
case OBJ_OFS_DELTA:
|
||||||
@ -2093,24 +2095,28 @@ static int sha1_loose_object_info(const unsigned char *sha1, unsigned long *size
|
|||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
int sha1_object_info(const unsigned char *sha1, unsigned long *sizep)
|
/* returns enum object_type or negative */
|
||||||
|
int sha1_object_info_extended(const unsigned char *sha1, struct object_info *oi)
|
||||||
{
|
{
|
||||||
struct cached_object *co;
|
struct cached_object *co;
|
||||||
struct pack_entry e;
|
struct pack_entry e;
|
||||||
int status;
|
int status, rtype;
|
||||||
|
|
||||||
co = find_cached_object(sha1);
|
co = find_cached_object(sha1);
|
||||||
if (co) {
|
if (co) {
|
||||||
if (sizep)
|
if (oi->sizep)
|
||||||
*sizep = co->size;
|
*(oi->sizep) = co->size;
|
||||||
|
oi->whence = OI_CACHED;
|
||||||
return co->type;
|
return co->type;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!find_pack_entry(sha1, &e)) {
|
if (!find_pack_entry(sha1, &e)) {
|
||||||
/* Most likely it's a loose object. */
|
/* Most likely it's a loose object. */
|
||||||
status = sha1_loose_object_info(sha1, sizep);
|
status = sha1_loose_object_info(sha1, oi->sizep);
|
||||||
if (status >= 0)
|
if (status >= 0) {
|
||||||
|
oi->whence = OI_LOOSE;
|
||||||
return status;
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
/* Not a loose object; someone else may have just packed it. */
|
/* Not a loose object; someone else may have just packed it. */
|
||||||
reprepare_packed_git();
|
reprepare_packed_git();
|
||||||
@ -2118,15 +2124,29 @@ int sha1_object_info(const unsigned char *sha1, unsigned long *sizep)
|
|||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
status = packed_object_info(e.p, e.offset, sizep);
|
status = packed_object_info(e.p, e.offset, oi->sizep, &rtype);
|
||||||
if (status < 0) {
|
if (status < 0) {
|
||||||
mark_bad_packed_object(e.p, sha1);
|
mark_bad_packed_object(e.p, sha1);
|
||||||
status = sha1_object_info(sha1, sizep);
|
status = sha1_object_info_extended(sha1, oi);
|
||||||
|
} else {
|
||||||
|
oi->whence = OI_PACKED;
|
||||||
|
oi->u.packed.offset = e.offset;
|
||||||
|
oi->u.packed.pack = e.p;
|
||||||
|
oi->u.packed.is_delta = (rtype == OBJ_REF_DELTA ||
|
||||||
|
rtype == OBJ_OFS_DELTA);
|
||||||
}
|
}
|
||||||
|
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int sha1_object_info(const unsigned char *sha1, unsigned long *sizep)
|
||||||
|
{
|
||||||
|
struct object_info oi;
|
||||||
|
|
||||||
|
oi.sizep = sizep;
|
||||||
|
return sha1_object_info_extended(sha1, &oi);
|
||||||
|
}
|
||||||
|
|
||||||
static void *read_packed_sha1(const unsigned char *sha1,
|
static void *read_packed_sha1(const unsigned char *sha1,
|
||||||
enum object_type *type, unsigned long *size)
|
enum object_type *type, unsigned long *size)
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user