Git: git9

git9 @ 7c5c8a7e0a33a2c83860dc9dcb2dcf3d6c23b2e4

#include <u.h>
#include <libc.h>
#include <ctype.h>
#include "git.h"

enum {
	Sinit,
	Siter,
};

static int
crackidx(char *path, int *np)
{
	int fd;
	char buf[4];

	if((fd = open(path, OREAD)) == -1)
		return -1;
	if(seek(fd, 8 + 255*4, 0) == -1)
		return -1;
	if(readn(fd, buf, sizeof(buf)) != sizeof(buf))
		return -1;
	*np = GETBE32(buf);
	return fd;
}

int
isloosedir(char *s)
{
	return strlen(s) == 2 && isxdigit(s[0]) && isxdigit(s[1]);
}

int
endswith(char *n, char *s)
{
	int nn, ns;

	nn = strlen(n);
	ns = strlen(s);
	return nn > ns && strcmp(n + nn - ns, s) == 0;
}

int
olsreadpacked(Objlist *ols, Hash *h)
{
	char *p;
	int i, j;

	i = ols->packidx;
	j = ols->entidx;

	if(ols->state == Siter)
		goto step;
	for(i = 0; i < ols->npack; i++){
		if(!endswith(ols->pack[i].name, ".idx"))
			continue;
		if((p = smprint(".git/objects/pack/%s", ols->pack[i].name)) == nil)
			sysfatal("smprint: %r");
		ols->fd = crackidx(p, &ols->nent);
		free(p);
		if(ols->fd == -1)
			continue;
		j = 0;
		while(j < ols->nent){
			if(readn(ols->fd, h->h, sizeof(h->h)) != sizeof(h->h))
				continue;
			ols->state = Siter;
			ols->packidx = i;
			ols->entidx = j;
			return 0;
step:
			j++;
		}
		close(ols->fd);
	}
	ols->state = Sinit;
	return -1;
}


int
olsreadloose(Objlist *ols, Hash *h)
{
	char buf[64], *p;
	int i, j, n;

	i = ols->topidx;
	j = ols->looseidx;
	if(ols->state == Siter)
		goto step;
	for(i = 0; i < ols->ntop; i++){
		if(!isloosedir(ols->top[i].name))
			continue;
		if((p = smprint(".git/objects/%s", ols->top[i].name)) == nil)
			sysfatal("smprint: %r");
		ols->fd = open(p, OREAD);
		free(p);
		if(ols->fd == -1)
			continue;
		while((ols->nloose = dirread(ols->fd, &ols->loose)) > 0){
			j = 0;
			while(j < ols->nloose){
				n = snprint(buf, sizeof(buf), "%s%s", ols->top[i].name, ols->loose[j].name);
				if(n >= sizeof(buf))
					goto step;
				if(hparse(h, buf) == -1)
					goto step;
				ols->state = Siter;
				ols->topidx = i;
				ols->looseidx = j;
				return 0;
step:
				j++;
			}
			free(ols->loose);
			ols->loose = nil;
		}
		close(ols->fd);
		ols->fd = -1;
	}
	ols->state = Sinit;
	return -1;
}

Objlist*
mkols(void)
{
	Objlist *ols;

	ols = emalloc(sizeof(Objlist));
	if((ols->ntop = slurpdir(".git/objects", &ols->top)) == -1)
		sysfatal("read top level: %r");
	if((ols->npack = slurpdir(".git/objects/pack", &ols->pack)) == -1)
		ols->pack = nil;
	ols->fd = -1;
	return ols;
}

void
olsfree(Objlist *ols)
{
	if(ols == nil)
		return;
	if(ols->fd != -1)
		close(ols->fd);
	free(ols->top);
	free(ols->loose);
	free(ols->pack);
	free(ols);
}

int
olsnext(Objlist *ols, Hash *h)
{
	if(ols->stage == 0){
		if(olsreadloose(ols, h) != -1){
			ols->idx++;
			return 0;
		}
		ols->stage++;
	}
	if(ols->stage == 1){
		if(olsreadpacked(ols, h) != -1){
			ols->idx++;
			return 0;
		}
		ols->stage++;
	}
	return -1;
}