shithub: git9

Download patch

ref: ec28e68d5f5d72748d4b2d0be2861956b856ef4f
author: Ori Bernstein <[email protected]>
date: Fri Jun 28 22:18:37 EDT 2019

Import git9 from mercurial repository

--- /dev/null
+++ b/LICENSE
@@ -1,0 +1,19 @@
+Copyright (c) 2019 Ori Bernstein
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
--- /dev/null
+++ b/README
@@ -1,0 +1,151 @@
+Git for Plan 9: git/fs
+======================
+
+Plan 9 is a non-posix system that currently has no git port.  Git
+itself feels distinctly un-plan9ish.
+
+Git/fs implements a git client for plan 9.  The intent is to support
+working with git repositories, without cloning the git interface
+directly.
+
+Git/fs is alpha software. It more or less works for me, but has many
+bugs, and many more missing tools, but it's at the point where the
+hard parts are done, and "the rest is just scripting", using standard
+command line tools (cp, echo, diff, patch, and diff3).
+
+Structure
+---------
+
+The git/fs program provides a file system mounted on /mnt/git.  It
+provides a read-only view into the repository contents to allow
+scripts to inspect the data.  Surrounding scripts and binaries will
+manipulate the repository contents directly.  These changes will be
+immediately mirrored in the file system.
+
+Scripts will generally mount git/fs as needed to do
+their work, but if you want to browse the repository
+manually, run it yourself. You'll get `/mnt/git` mounted,
+with the following contents:
+
+	/mnt/git/object:	The objects in the repo.
+	/mnt/git/branch:	The branches in the repo.
+	/mnt/git/ctl:		A file showing the status of the repo.
+				Currently, it only shows the current branch.
+	/mnt/git/HEAD		An alias for the currently checked out
+				commit directory.
+
+Visible Differences
+-------------------
+
+The most obvious difference is that Git's index is a bit boneheaded, so I'm
+ignoring it.  The index doesn't affect the wire protocol, so this
+isn't an interoperability issue, unless you share the same physical
+repository on both Plan 9 and Unix.  If you do, expect them to disagree
+about the files that have been modified in the working copy.
+
+In fact, the entire concept of the staging area has been dropped, as
+it's both confusing and clunky.  There are now only three states that
+files can be in: 'untracked', 'dirty', and 'committed'.  Tracking is
+done with empty files under .git/index9/{removed,tracked}/path/to/file.
+
+It's implemented in Plan 9 flavor C, and provides tools for writing
+repository contents, and a file system for read-only access, which
+will mirror the current state of the repository.
+
+Installation
+------------
+
+Install with `mk install`.
+
+Examples
+--------
+
+Some usage examples:
+
+	git/clone git://git.eigenstate.org/ori/mc.git
+	git/log
+	cd subdir/name
+	git/add foo.c
+	diff bar.c /mnt/git/HEAD/
+	git/commit
+	git/push
+
+Commits are presented as directories with the following
+contents:
+
+	author:	A file containing the author name
+	hash:	A file containing the commit hash
+	parent:	A file containing the commit parents, one per line.
+	msg:	A file containing the log message for that commit
+	tree:	A directory containing a view of the repository.
+
+So, for example:
+
+	% ls /mnt/git/branch/heads/master
+	/mnt/git/branch/heads/master/author
+	/mnt/git/branch/heads/master/hash
+	/mnt/git/branch/heads/master/msg
+	/mnt/git/branch/heads/master/parent
+	/mnt/git/branch/heads/master/tree
+	% cat /mnt/git/branch/heads/master/hash
+	7d539a7c08aba3f31b3913e0efef11c43ea9
+
+	# This is the same commit, with the same contents.
+	% ls /mnt/git/object/7d539a7c08aba3f31b3913e0efef11c43ea9f9ef
+	/mnt/git/object/7d539a7c08aba3f31b3913e0efef11c43ea9f9ef/author
+	/mnt/git/object/7d539a7c08aba3f31b3913e0efef11c43ea9f9ef/hash
+	/mnt/git/object/7d539a7c08aba3f31b3913e0efef11c43ea9f9ef/msg
+	/mnt/git/object/7d539a7c08aba3f31b3913e0efef11c43ea9f9ef/parent
+	/mnt/git/object/7d539a7c08aba3f31b3913e0efef11c43ea9f9ef/tree
+
+	# what git/diff will hopefully do more concisely soon, filtering
+	# out the non-git files.
+	ape/diff -ur /mnt/git/branch/heads/master/tree .
+	Only in .: .git
+	Only in .: debug
+	diff -ur /mnt/git/branch/heads/master/tree/fold.myr ./fold.myr
+	--- /mnt/git/branch/heads/master/tree/fold.myr	Wed Dec 31 16:00:00 1969
+	+++ ./fold.myr	Mon Apr  1 21:39:06 2019
+	@@ -6,6 +6,8 @@
+	 	const foldexpr : (e : expr# -> std.option(constval))
+	 ;;
+	 
+	+/* Look, diffing files just works, and I don't need any fancy glue! */
+	+
+	 const foldexpr = {e
+	 	match e
+	 	| &(`Eident &[.sc=`Sclassenum, .name=name, .ty=`Tyenum &(`Body enum)]):
+	Only in .: refs
+
+	
+The following utilities and binaries are provided:
+
+	fs:	The git filesystem.
+	fetch:	The protocol bits for getting data from a git server.
+	send:	The protocol bits for sending data to a git server.
+	save:	The gnarly bits for storing the files for a commit.
+	conf:	A program to extract information from a config file.
+	clone:	Clones a repository.
+	commit:	Commits a snapshot of the working directory.
+	log:	Prints the contents of a commmit log.
+	add:	Tells the repository to add a file to the next commit.
+	walk:	`du`, but for git status.
+
+
+Supported protocols: git:// and git+ssh://. If someone
+implements others, I'll gladly accept patches.
+
+TODOs
+-----
+
+Documentation has not yet been written.  You'll need to read the
+source. Notably missing functionality includes:
+
+	git/mkpatch:	Generate a 'git am' compatible patch.
+	git/apply:	Apply a diff.
+	git/diff:	Wrapper wrapper around git/walk that
+			diffs the changed files.
+	git/merge:	Yup, what it says on the label. Should
+			also be a script around git/fs.
+	git/log:	Need to figure out how to make it filter
+			by files.
--- /dev/null
+++ b/add
@@ -1,0 +1,46 @@
+#!/bin/rc -e
+
+rfork ne
+
+fn usage {
+	echo 'usage: '$argv0' [-r] files..' >[1=2]
+	echo '	-r:		remove instead of adding' >[1=2]
+}
+
+add='tracked'
+del='removed'
+while(~ $1 -*){
+	switch($1){
+	case -r
+		add='removed'
+		del='tracked'
+	case --
+		break
+	case *; usage
+	}
+	shift
+}
+
+dir=`{pwd}
+base=`{git/conf -r}
+if(! ~ $status ''){
+	echo 'not in git repository' `{pwd} >[1=2]
+	exit notrepo
+}
+
+cd $base
+rel=`{sed 's@^'$base'/*@@' <{echo $dir}}
+if(~ $#rel 0)
+	rel=''
+for(f in $*){
+	if(! test -f $base/$rel/$f){
+		echo 'could not add '$base/$rel/$f': does not exist' >[1=2]
+		exit 'nofile'
+	}
+	addpath=.git/index9/$add/$rel/$f
+	delpath=.git/index9/$del/$rel/$f
+	mkdir -p `{basename -d $addpath}
+	mkdir -p `{basename -d $delpath}
+	touch $addpath
+	rm -f $delpath
+}
--- /dev/null
+++ b/branch
@@ -1,0 +1,81 @@
+#!/bin/rc -e
+
+rfork en
+
+fn usage{
+	echo usage: $argv0 [-b base] [-o origin] new >[2=1]
+	echo '	'-b base:	use "base" for branch (default: current branch) >[2=1]
+	echo '	'-o origin:	use "origin" for remote branch >[2=1]
+	echo '	'new:		name of new branch
+	exit usage
+}
+
+if(! cd `{git/conf -r}){
+	exit 'not in git repository'
+	exit notgit
+}
+git/fs
+
+nl='
+'
+stay=''
+create=''
+cur=`{awk '$1=="branch"{print $2}' < /mnt/git/ctl}
+while(~ $1 -* && ! ~ $1 --){
+	switch($1){
+	case -c; create=true
+	case -s; stay=true
+	case -o; origin=$1
+	case *; usage
+	}
+	shift
+}
+if(~ $1 --) shift
+
+if(~ $#* 0){
+	echo $cur
+	exit
+}
+if(! ~ $#* 1)
+	usage
+new=$1
+
+if(~ $create ''){
+	if(! test -e .git/refs/heads/$new){
+		echo branch $new: does not exist >[1=2]
+		exit exists
+	}
+}
+if not{
+	if(test -e .git/refs/heads/$new){
+		echo could not create $new: already exists >[1=2]
+		exit exists
+	}
+	branched=''
+	candidates=(.git/refs/$cur .git/refs/heads/$cur .git/refs/remotes/$cur .git/refs/remotes/*/$cur)
+	for(br in $candidates){
+		if(test -f $br){
+			echo 'creating new branch '$new
+			cp $br .git/refs/heads/$new
+			branched="ok"
+		}
+	}
+	if(~ $branched ''){
+		echo 'could not find branch '$cur >[1=2]
+		exit notfound
+	}
+}
+
+if(~ $stay ''){
+	rm -f `$nl{git/walk -cfT}
+	echo 'ref: refs/heads/'$new > .git/HEAD
+	tree=/mnt/git/HEAD/tree
+	@{builtin cd $tree && tar cif /fd/1 .} | @{tar xf /fd/0}
+	for(f in `$nl{walk -f $tree | sed 's@^'$tree'/*@@'}){
+		if(! ~ $#f 0){
+			idx=.git/index9/tracked/$f
+			mkdir -p `{basename -d $idx}
+			walk -eq $f > $idx
+		}
+	}
+}
--- /dev/null
+++ b/clone
@@ -1,0 +1,98 @@
+#!/bin/rc
+
+rfork en
+nl='
+'
+
+if(~ $#* 1){
+	remote=$1
+	local=`{basename $1 .git}
+}
+if not if(~ $#* 2){
+	remote=$1
+	local=$2
+}
+if not{
+	echo usage: git/clone remote [local] >[1=2]
+	exit usage
+}
+
+if(test -e $local){
+	echo $local already exists
+	exit exists
+}
+	
+fn clone{
+	mkdir -p $local/.git
+	mkdir -p $local/.git/objects/pack/
+	mkdir -p $local/.git/refs/heads/
+	
+	cd $local
+	
+	dircp /sys/lib/git/template .git
+	echo '[remote "origin"]' 				>> .git/config
+	echo '	url='$remote					>> .git/config
+	echo '	fetch=+refs/heads/*:refs/remotes/origin/*'	>> .git/config
+	{git/fetch $remote >[2=3] | awk '
+		/^remote/{
+			if($2=="HEAD"){
+				headhash=$3
+				headref=""
+			}else{
+				gsub("^refs/heads", "refs/remotes/origin", $2)
+				if($2 == "refs/remotes/origin/master" || $3 == headhash)
+					headref=$2
+				outfile = ".git/" $2
+				system("mkdir -p `{basename -d "outfile"}")
+				print $3 > outfile
+				close(outfile)
+			}
+		}
+		END{
+			if(headref != ""){
+				remote = headref
+				gsub("^refs/remotes/origin", "refs/heads", headref)
+				system("mkdir -p `{basename -d .git/" headref"}");
+				system("cp .git/" remote " .git/" headref)
+				print "ref: " headref > ".git/HEAD"
+			}else{
+				print "warning: detached head "headhash > "/fd/2"
+				print headhash > ".git/HEAD"
+			}
+		}
+	'} |[3] tr '\x0d' '\x0a'
+	if(! ~ $status '|')
+		exit 'clone:' $status
+
+	tree=/mnt/git/branch/heads/master/tree
+	echo checking out repository...
+	if(test -f .git/refs/remotes/origin/master){
+		cp .git/refs/remotes/origin/master .git/refs/heads/master
+		git/fs
+		@ {builtin cd $tree && tar cif /fd/1 .} | @ {tar xf /fd/0}
+		if(! ~ $status '')
+			exit 'checkout:' $status
+		for(f in `$nl{walk -f $tree | sed 's@^'$tree'/*@@'}){
+			if(! ~ $#f 0){
+				idx=.git/index9/tracked/$f
+				mkdir -p `{basename -d $idx}
+				walk -eq $f > $idx
+			}
+		}
+	}
+	if not{
+		echo no master branch >[1=2]
+		echo check out your code with git/branch >[1=2]
+	}
+}
+
+@{clone}
+st=$status
+if(~ $st ''){
+	echo done.
+}
+if not{
+	echo clone failed: $st >[2=1]
+	echo cleaning up $local >[2=1]
+	rm -rf $local
+}
--- /dev/null
+++ b/commit
@@ -1,0 +1,119 @@
+#!/bin/rc
+
+rfork ne
+
+nl='
+'
+
+fn whoami{
+	name=`{git/conf user.name}
+	email=`{git/conf user.email}
+	msgfile=.git/git-msg.$pid
+	if(test -f /adm/keys.who){
+		if(~ $name '')
+			name=`{cat /adm/keys.who | awk -F'|' '$1=="'$user'" {print $3}'}
+		if(~ $email '')
+			email=`{cat /adm/keys.who | awk -F'|' '$1=="'$user'" {print $5}'}
+	}
+	if(~ $name '')
+		name=glenda
+	if(~ $email '')
+		[email protected]
+}
+
+fn findbranch{
+	branch=`{awk '$1=="branch"{print $2}' < /mnt/git/ctl}
+	if(test -e /mnt/git/branch/$branch/tree){
+		refpath=.git/refs/$branch
+		initial=false
+	}
+	if not if(test -e /mnt/git/object/$branch/tree){
+		refpath=.git/HEAD
+		initial=false
+	}
+	if not if(! test -e /mnt/git/HEAD/tree){
+		refpath=.git/refs/$branch
+		initial=true
+	}
+	if not{
+		echo invalid branch $branch >[1=2]
+		exit badbranch
+	}
+
+}
+
+fn editmsg{
+	echo '' > $msgfile.tmp
+	echo '# Commit message goes here.' >> $msgfile.tmp
+	echo '# Author: '$name' <'$email'>' >> $msgfile.tmp
+	echo '#' $nl'# ' ^ `$nl{git/walk -fAMR} >> $msgfile.tmp
+	sam $msgfile.tmp
+	if(! test -s $msgfile.tmp || ~ `{wc -l <{grep -v '^[ 	]*[\n#]' $msgfile.tmp}} 0){
+		echo 'cancelling commit: empty message' >[1=2]
+		exit 'nocommit'
+	}
+	grep -v '^[ 	]*#' < $msgfile.tmp > $msgfile
+}
+
+fn parents{
+	if(test -f .git/index9/merge-parents)
+		parents=`{cat .git/index9/merge-parents}
+	if not if(~ $initial true)
+		parents=()
+	if not
+		parents=$branch
+}
+
+fn commit{
+	msg=`"{cat $msgfile}
+	if(! ~ $#parents 0)
+		pflags='-p'^$parents
+	hash=`{git/save -n $"name -e $"email  -m $"msg $pflags}
+	st=$status
+	if(~ $hash '')
+		exit nocommit
+}
+
+fn update{
+	mkdir -p `{basename -d $refpath}
+	echo $hash > $refpath
+	for(f in `$nl{git/walk -cfAM}){
+		mkdir -p `{basename -d $f}
+		walk -eq $f > .git/index9/tracked/$f
+	}
+	for(f in `$nl{git/walk -cfR}){
+		rm -f .git/index9/tracked/$f
+		rm -f .git/index9/removed/$f
+	}
+
+}
+
+fn cleanup{
+	rm -f $msgfile
+	rm -f .git/index9/merge-parents
+}
+
+msgfile=/tmp/git-msg.$pid
+if(! cd `{git/conf -r})
+	exit 'not in git repository'
+git/fs
+mkdir -p .git/refs
+if(git/walk -q){
+	echo no changes to commit >[1=2]
+	exit clean
+}
+@{
+	flag e +
+	whoami
+	findbranch
+	parents
+	editmsg
+	commit
+	update
+	cleanup
+}
+st=$status
+if(! ~ $status ''){
+	echo 'could not commit:' $st >[1=2]
+	exit $st
+}
--- /dev/null
+++ b/conf.c
@@ -1,0 +1,102 @@
+#include <u.h>
+#include <libc.h>
+#include <ctype.h>
+
+#include "git.h"
+
+static int
+showconf(char *cfg, char *sect, char *key)
+{
+	char *ln, *p;
+	Biobuf *f;
+	int foundsect, nsect, nkey;
+
+	if((f = Bopen(cfg, OREAD)) == nil)
+		return 0;
+
+	nsect = sect ? strlen(sect) : 0;
+	nkey = strlen(key);
+	foundsect = (sect == nil);
+	while((ln = Brdstr(f, '\n', 1)) != nil){
+		p = strip(ln);
+		if(*p == '[' && sect){
+			foundsect = strncmp(sect, ln, nsect) == 0;
+		}else if(foundsect && strncmp(p, key, nkey) == 0){
+			p = strip(p + nkey);
+			if(*p != '=')
+				continue;
+			p = strip(p + 1);
+			print("%s\n", p);
+			free(ln);
+			return 1;
+		}
+		free(ln);
+	}
+	return 0;
+}
+
+static void
+showroot(void)
+{
+	char path[256], buf[256], *p;
+
+	if((getwd(path, sizeof(path))) == nil)
+		sysfatal("could not get wd: %r");
+	while((p = strrchr(path, '/')) != nil){
+		snprint(buf, sizeof(buf), "%s/.git", path);
+		if(access(buf, AEXIST) == 0){
+			print("%s\n", path);
+			return;
+		}
+		*p = '\0';
+	}
+	sysfatal("not a git repository");
+}
+
+
+void
+usage(void)
+{
+	fprint(2, "usage: %s [-f file] [-r] keys..\n", argv0);
+	fprint(2, "\t-f:	use file 'file' (default: .git/config)\n");
+	fprint(2, "\t r:	print repository root\n");
+	exits("usage");
+}
+
+void
+main(int argc, char **argv)
+{
+	char *file[32], *p, *s;
+	int i, j, nfile, findroot;
+
+	nfile = 0;
+	findroot = 0;
+	ARGBEGIN{
+	case 'f':	file[nfile++]=EARGF(usage());	break;
+	case 'r':	findroot++;			break;
+	default:	usage();			break;
+	}ARGEND;
+
+	if(findroot)
+		showroot();
+	if(nfile == 0){
+		file[nfile++] = ".git/config";
+		if((p = getenv("home")) != nil)
+			file[nfile++] = smprint("%s/lib/git/config", p);
+	}
+
+	for(i = 0; i < argc; i++){
+		if((p = strchr(argv[i], '.')) == nil){
+			s = nil;
+			p = argv[i];
+		}else{
+			*p = 0;
+			p++;
+			s = smprint("[%s]", argv[i]);
+		}
+		for(j = 0; j < nfile; j++)
+			if(showconf(file[j], s, p))
+				break;
+	}
+	exits(nil);
+}
--- /dev/null
+++ b/diff
@@ -1,0 +1,40 @@
+#!/bin/rc
+
+rfork e
+
+fn usage{
+	echo git/diff '[-b branch] [file ...]' >[2=1]
+	exit usage
+}
+
+if(! cd `{git/conf -r}){
+	echo 'not a git repository' >[1=2]
+	exit notgit
+}
+git/fs
+
+while(~ $1 -*){
+	switch($1){
+	case -b; branch=$2; shift
+	case *; usage
+	}
+	shift
+}
+
+if(~ $#branch 0)
+	branch=`{git/branch}
+dirty=`{git/walk | awk '/^[MAR]/ {print $2}'}
+if(! ~ $#* 0){
+	echo $dirty | sed 's/ /\n/g' | sort >/tmp/git.$pid.diff.dirty
+	echo $* | sed 's/ /\n/g' | sort >/tmp/git.$pid.diff.args
+	dirty=`{join /tmp/git.$pid.diff.dirty /tmp/git.$pid.diff.args}
+}
+for(f in $dirty){
+	orig=/mnt/git/branch/$branch/tree/$f
+	if(! test -f $orig)
+		orig=/dev/null
+	if(! test -f $f)
+		f=/dev/null
+	ape/diff -up $orig $f
+}
+rm -f /tmp/git.$pid.diff.dirty /tmp/git.$pid.diff.args
--- /dev/null
+++ b/fetch.c
@@ -1,0 +1,285 @@
+#include <u.h>
+#include <libc.h>
+
+#include "git.h"
+
+Object *indexed;
+char *fetchbranch;
+char *upstream = "origin";
+char *packtmp = ".git/objects/pack/fetch.tmp";
+
+int
+resolveremote(Hash *h, char *ref)
+{
+	char buf[128], *s;
+	int r, f;
+
+	ref = strip(ref);
+	if((r = hparse(h, ref)) != -1)
+		return r;
+	/* Slightly special handling: translate remote refs to local ones. */
+	if(strcmp(ref, "HEAD") == 0){
+		snprint(buf, sizeof(buf), ".git/HEAD");
+	}else if(strstr(ref, "refs/heads") == ref){
+		ref += strlen("refs/heads");
+		snprint(buf, sizeof(buf), ".git/refs/remotes/%s/%s", upstream, ref);
+	}else if(strstr(ref, "refs/tags") == ref){
+		ref += strlen("refs/tags");
+		snprint(buf, sizeof(buf), ".git/refs/tags/%s/%s", upstream, ref);
+	}else{
+		return -1;
+	}
+
+	s = strip(buf);
+	if((f = open(s, OREAD)) == -1)
+		return -1;
+	if(readn(f, buf, sizeof(buf)) >= 40)
+		r = hparse(h, buf);
+	close(f);
+
+	if(r == -1 && strstr(buf, "ref:") == buf)
+		return resolveremote(h, buf + strlen("ref:"));
+	
+	return r;
+}
+
+int
+rename(char *pack, char *idx, Hash h)
+{
+	char name[128];
+	Dir st;
+
+	nulldir(&st);
+	st.name = name;
+	snprint(name, sizeof(name), "%H.pack", h);
+	if(access(name, AEXIST) == 0)
+		fprint(2, "warning, pack %s already fetched\n", name);
+	else if(dirwstat(pack, &st) == -1)
+		return -1;
+	snprint(name, sizeof(name), "%H.idx", h);
+	if(access(name, AEXIST) == 0)
+		fprint(2, "warning, pack %s already indexed\n", name);
+	else if(dirwstat(idx, &st) == -1)
+		return -1;
+	return 0;
+}
+
+int
+checkhash(int fd, vlong sz, Hash *hcomp)
+{
+	DigestState *st;
+	Hash hexpect;
+	char buf[65536];
+	vlong n, r;
+	int nr;
+	
+	if(sz < 28){
+		werrstr("undersize packfile");
+		return -1;
+	}
+
+	st = nil;
+	n = 0;
+	while(n != sz - 20){
+		nr = sizeof(buf);
+		if(sz - n - 20 < sizeof(buf))
+			nr = sz - n - 20;
+		r = readn(fd, buf, nr);
+		if(r != nr)
+			return -1;
+		st = sha1((uchar*)buf, nr, nil, st);
+		n += r;
+	}
+	sha1(nil, 0, hcomp->h, st);
+	if(readn(fd, hexpect.h, sizeof(hexpect.h)) != sizeof(hexpect.h))
+		sysfatal("truncated packfile");
+	if(!hasheq(hcomp, &hexpect)){
+		werrstr("bad hash: %H != %H", *hcomp, hexpect);
+		return -1;
+	}
+	return 0;
+}
+
+int
+mkoutpath(char *path)
+{
+	char s[128];
+	char *p;
+	int fd;
+
+	snprint(s, sizeof(s), "%s", path);
+	for(p=strchr(s+1, '/'); p; p=strchr(p+1, '/')){
+		*p = 0;
+		if(access(s, AEXIST) != 0){
+			fd = create(s, OREAD, DMDIR | 0755);
+			if(fd == -1)
+				return -1;
+			close(fd);
+		}		
+		*p = '/';
+	}
+	return 0;
+}
+
+int
+branchmatch(char *br, char *pat)
+{
+	char name[128];
+
+	if(strstr(pat, "refs/heads") == pat)
+		snprint(name, sizeof(name), "%s", pat);
+	else if(strstr(pat, "heads"))
+		snprint(name, sizeof(name), "refs/%s", pat);
+	else
+		snprint(name, sizeof(name), "refs/heads/%s", pat);
+	return strcmp(br, name) == 0;
+}
+
+int
+fetchpack(int fd, int pfd, char *packtmp)
+{
+	char buf[65536];
+	char idxtmp[256];
+	char *sp[3];
+	Hash h, *have, *want;
+	int nref, refsz;
+	int i, n, req;
+	vlong packsz;
+
+	nref = 0;
+	refsz = 16;
+	have = emalloc(refsz * sizeof(have[0]));
+	want = emalloc(refsz * sizeof(want[0]));
+	while(1){
+		n = readpkt(fd, buf, sizeof(buf));
+		if(n == -1)
+			return -1;
+		if(n == 0)
+			break;
+		if(strncmp(buf, "ERR ", 4) == 0)
+			sysfatal("%s", buf + 4);
+		getfields(buf, sp, nelem(sp), 1, " \t\n\r");
+		if(strstr(sp[1], "^{}"))
+			continue;
+		if(fetchbranch && !branchmatch(sp[1], fetchbranch))
+			continue;
+		if(refsz == nref + 1){
+			refsz *= 2;
+			have = erealloc(have, refsz * sizeof(have[0]));
+			want = erealloc(want, refsz * sizeof(want[0]));
+		}
+		if(hparse(&want[nref], sp[0]) == -1)
+			sysfatal("invalid hash %s", sp[0]);
+		if (resolveremote(&have[nref], sp[1]) == -1)
+			memset(&have[nref], 0, sizeof(have[nref]));
+		print("remote %s %H local %H\n", sp[1], want[nref], have[nref]);
+		nref++;
+	}
+
+	req = 0;
+	for(i = 0; i < nref; i++){
+		if(memcmp(have[i].h, want[i].h, sizeof(have[i].h)) == 0)
+			continue;
+		n = snprint(buf, sizeof(buf), "want %H", want[i]);
+		print("want %H\n", want[i]);
+		if(writepkt(fd, buf, n) == -1)
+			sysfatal("could not send want for %H", want[i]);
+		req = 1; 
+	}
+	flushpkt(fd);
+	for(i = 0; i < nref; i++){
+		if(memcmp(have[i].h, Zhash.h, sizeof(Zhash.h)) == 0)
+			continue;
+		n = snprint(buf, sizeof(buf), "have %H\n", have[i]);
+		if(writepkt(fd, buf, n + 1) == -1)
+			sysfatal("could not send have for %H", have[i]);
+	}
+	if(!req){
+		fprint(2, "up to date\n");
+		flushpkt(fd);
+	}
+	n = snprint(buf, sizeof(buf), "done\n");
+	if(writepkt(fd, buf, n) == -1)
+		sysfatal("lost connection write");
+	if(!req)
+		return 0;
+
+	if((n = readpkt(fd, buf, sizeof(buf))) == -1)
+		sysfatal("lost connection read");
+	buf[n] = 0;
+
+	fprint(2, "fetching...\n");
+	packsz = 0;
+	while(1){
+		n = readn(fd, buf, sizeof buf);
+		if(n == 0)
+			break;
+		if(n == -1 || write(pfd, buf, n) != n)
+			sysfatal("could not fetch packfile: %r");
+		packsz += n;
+	}
+	if(seek(pfd, 0, 0) == -1)
+		sysfatal("packfile seek: %r");
+	if(checkhash(pfd, packsz, &h) == -1)
+		sysfatal("corrupt packfile: %r");
+	close(pfd);
+	n = strlen(packtmp) - strlen(".tmp");
+	memcpy(idxtmp, packtmp, n);
+	memcpy(idxtmp + n, ".idx", strlen(".idx") + 1);
+	if(indexpack(packtmp, idxtmp, h) == -1)
+		sysfatal("could not index fetched pack: %r");
+	if(rename(packtmp, idxtmp, h) == -1)
+		sysfatal("could not rename indexed pack: %r");
+	return 0;
+}
+
+void
+usage(void)
+{
+	fprint(2, "usage: %s [-b br] remote\n", argv0);
+	fprint(2, "\t-b br:	only fetch matching branch 'br'\n");
+	fprint(2, "remote:	fetch from this repository\n");
+	exits("usage");
+}
+
+void
+main(int argc, char **argv)
+{
+	char proto[Nproto], host[Nhost], port[Nport];
+	char repo[Nrepo], path[Npath];
+	int fd, pfd;
+
+	ARGBEGIN{
+	case 'b':	fetchbranch=EARGF(usage());	break;
+	case 'u':	upstream=EARGF(usage());	break;
+	case 'd':	chattygit++;			break;
+	default:	usage();			break;
+	}ARGEND;
+
+	gitinit();
+	if(argc != 1)
+		usage();
+	fd = -1;
+
+	if(mkoutpath(packtmp) == -1)
+		sysfatal("could not create %s: %r", packtmp);
+	if((pfd = create(packtmp, ORDWR, 0644)) == -1)
+		sysfatal("could not create %s: %r", packtmp);
+
+	if(parseuri(argv[0], proto, host, port, path, repo) == -1)
+		sysfatal("bad uri %s", argv[0]);
+	if(strcmp(proto, "ssh") == 0 || strcmp(proto, "git+ssh") == 0)
+		fd = dialssh(host, port, path, "upload");
+	else if(strcmp(proto, "git") == 0)
+		fd = dialgit(host, port, path, "upload");
+	else if(strcmp(proto, "http") == 0 || strcmp(proto, "git+http") == 0)
+		sysfatal("http clone not implemented");
+	else
+		sysfatal("unknown protocol %s", proto);
+	
+	if(fd == -1)
+		sysfatal("could not dial %s:%s: %r", proto, host);
+	if(fetchpack(fd, pfd, packtmp) == -1)
+		sysfatal("fetch failed: %r");
+	exits(nil);
+}
--- /dev/null
+++ b/fs.c
@@ -1,0 +1,756 @@
+#include <u.h>
+#include <libc.h>
+#include <ctype.h>
+#include <fcall.h>
+#include <thread.h>
+#include <9p.h>
+
+#include "git.h"
+
+typedef struct Ols Ols;
+
+char *Eperm = "permission denied";
+char *Eexist = "does not exist";
+char *E2long = "path too long";
+char *Enodir = "not a directory";
+char *Erepo = "unable to read repo";
+char *Egreg = "wat";
+
+enum {
+	Qroot,
+	Qhead,
+	Qbranch,
+	Qcommit,
+	Qcommitmsg,
+	Qcommitparent,
+	Qcommittree,
+	Qcommitdata,
+	Qcommithash,
+	Qcommitauthor,
+	Qobject,
+	Qctl,
+	Qmax,
+	Internal=1<<7,
+};
+
+typedef struct Gitaux Gitaux;
+struct Gitaux {
+	int	 npath;
+	Qid	 path[Npath];
+	Object  *opath[Npath];
+	char 	*refpath;
+	int	 qdir;
+	vlong	 mtime;
+	Object	*obj;
+
+	/* For listing object dir */
+	Ols	*ols;
+	Object	*olslast;
+};
+
+char *qroot[] = {
+	"HEAD",
+	"branch",
+	"object",
+	"ctl",
+};
+
+char *username;
+char *mtpt = "/mnt/git";
+char **branches = nil;
+
+static vlong
+findbranch(Gitaux *aux, char *path)
+{
+	int i;
+
+	for(i = 0; branches[i]; i++)
+		if(strcmp(path, branches[i]) == 0)
+			goto found;
+	branches = realloc(branches, sizeof(char *)*(i + 2));
+	branches[i] = estrdup(path);
+	branches[i + 1] = nil;
+
+found:
+	if(aux)
+		aux->refpath = estrdup(branches[i]);
+	return QPATH(i, Qbranch|Internal);
+}
+
+static void
+obj2dir(Dir *d, Object *o, long qdir, vlong mtime)
+{
+	char name[64];
+
+	snprint(name, sizeof(name), "%H", o->hash);
+	d->name = estrdup9p(name);
+	d->qid.type = QTDIR;
+	d->qid.path = QPATH(o->id, qdir);
+	d->atime = mtime;
+	d->mtime = mtime;
+	d->uid = estrdup9p(username);
+	d->gid = estrdup9p(username);
+	d->muid = estrdup9p(username);
+	d->mode = 0755 | DMDIR;
+	if(o->type == GBlob || o->type == GTag){
+		d->qid.type = 0;
+		d->mode = 0644;
+		d->length = o->size;
+	}
+
+}
+
+static int
+rootgen(int i, Dir *d, void *p)
+{
+	Gitaux *aux;
+
+	aux = p;
+	if (i >= nelem(qroot))
+		return -1;
+	d->mode = 0555 | DMDIR;
+	d->name = estrdup9p(qroot[i]);
+	d->qid.vers = 0;
+	d->qid.type = strcmp(qroot[i], "ctl") == 0 ? 0 : QTDIR;
+	d->qid.path = QPATH(i, Qroot);
+	d->uid = estrdup9p(username);
+	d->gid = estrdup9p(username);
+	d->muid = estrdup9p(username);
+	d->mtime = aux->mtime;
+	return 0;
+}
+
+static int
+branchgen(int i, Dir *d, void *p)
+{
+	Gitaux *aux;
+	Dir *refs;
+	int n;
+
+	aux = p;
+	refs = nil;
+	d->qid.vers = 0;
+	d->qid.type = QTDIR;
+	d->qid.path = findbranch(nil, aux->refpath);
+	d->mode = 0555 | DMDIR;
+	d->uid = estrdup9p(username);
+	d->gid = estrdup9p(username);
+	d->muid = estrdup9p(username);
+	d->mtime = aux->mtime;
+	d->atime = aux->mtime;
+	if((n = slurpdir(aux->refpath, &refs)) < 0)
+		return -1;
+	if(i < n){
+		d->name = estrdup9p(refs[i].name);
+		free(refs);
+		return 0;
+	}else{
+		free(refs);
+		return -1;
+	}
+}
+
+/* FIXME: walk to the appropriate submodule.. */
+static Object*
+modrefobj(Dirent *e)
+{
+	Object *m;
+
+	m = emalloc(sizeof(Object));
+	m->hash = e->h;
+	m->type = GTree;
+	m->tree = emalloc(sizeof(Tree));
+	m->tree->ent = nil;
+	m->tree->nent = 0;
+	m->flag |= Cloaded|Cparsed;
+	m->off = -1;
+	ref(m);
+	cache(m);
+	return m;
+}
+
+static int
+gtreegen(int i, Dir *d, void *p)
+{
+	Gitaux *aux;
+	Object *o, *e;
+
+	aux = p;
+	e = aux->obj;
+	if(i >= e->tree->nent)
+		return -1;
+	if((o = readobject(e->tree->ent[i].h)) == nil)
+		if(e->tree->ent[i].modref)
+			o = modrefobj(&e->tree->ent[i]);
+		else
+			die("could not read object %H: %r", e->tree->ent[i].h, e->hash);
+	d->qid.vers = 0;
+	d->qid.type = o->type == GTree ? QTDIR : 0;
+	d->qid.path = QPATH(o->id, aux->qdir);
+	d->mode = e->tree->ent[i].mode;
+	d->atime = aux->mtime;
+	d->mtime = aux->mtime;
+	d->uid = estrdup9p(username);
+	d->gid = estrdup9p(username);
+	d->muid = estrdup9p(username);
+	d->name = estrdup9p(e->tree->ent[i].name);
+	d->length = o->size;
+	return 0;
+}
+
+static int
+gcommitgen(int i, Dir *d, void *p)
+{
+	Object *o;
+
+	o = ((Gitaux*)p)->obj;
+	d->uid = estrdup9p(username);
+	d->gid = estrdup9p(username);
+	d->muid = estrdup9p(username);
+	d->mode = 0444;
+	d->atime = o->commit->ctime;
+	d->mtime = o->commit->ctime;
+	d->qid.type = 0;
+	d->qid.vers = 0;
+
+	switch(i){
+	case 0:
+		d->mode = 0555 | DMDIR;
+		d->name = estrdup9p("tree");
+		d->qid.type = QTDIR;
+		d->qid.path = QPATH(o->id, Qcommittree);
+		break;
+	case 1:
+		d->name = estrdup9p("parent");
+		d->qid.path = QPATH(o->id, Qcommitparent);
+		break;
+	case 2:
+		d->name = estrdup9p("msg");
+		d->qid.path = QPATH(o->id, Qcommitmsg);
+		break;
+	case 3:
+		d->name = estrdup9p("hash");
+		d->qid.path = QPATH(o->id, Qcommithash);
+		break;
+	case 4:
+		d->name = estrdup9p("author");
+		d->qid.path = QPATH(o->id, Qcommitauthor);
+		break;
+	default:
+		return -1;
+	}
+	return 0;
+}
+
+
+static int
+objgen(int i, Dir *d, void *p)
+{
+	Gitaux *aux;
+	Object *o;
+	Ols *ols;
+	Hash h;
+
+	aux = p;
+	if(!aux->ols)
+		aux->ols = mkols();
+	ols = aux->ols;
+	o = nil;
+	/* We tried to sent it, but it didn't fit */
+	if(aux->olslast && ols->idx == i + 1){
+		obj2dir(d, aux->olslast, Qobject, aux->mtime);
+		return 0;
+	}
+	while(ols->idx <= i){
+		if(olsnext(ols, &h) == -1)
+			return -1;
+		if((o = readobject(h)) == nil)
+			return -1;
+	}
+	if(o != nil){
+		obj2dir(d, o, Qobject, aux->mtime);
+		unref(aux->olslast);
+		aux->olslast = ref(o);
+		return 0;
+	}
+	return -1;
+}
+
+static void
+objread(Req *r, Gitaux *aux)
+{
+	Object *o;
+
+	o = aux->obj;
+	switch(o->type){
+	case GBlob:
+		readbuf(r, o->data, o->size);
+		break;
+	case GTag:
+		readbuf(r, o->data, o->size);
+		break;
+	case GTree:
+		dirread9p(r, gtreegen, aux);
+		break;
+	case GCommit:
+		dirread9p(r, gcommitgen, aux);
+		break;
+	default:
+		die("invalid object type %d", o->type);
+	}
+}
+
+static void
+readcommitparent(Req *r, Object *o)
+{
+	char *buf, *p;
+	int i, n;
+
+	n = o->commit->nparent * (40 + 2);
+	buf = emalloc(n);
+	p = buf;
+	for (i = 0; i < o->commit->nparent; i++)
+		p += sprint(p, "%H\n", o->commit->parent[i]);
+	readbuf(r, buf, n);
+	free(buf);
+}
+
+
+static void
+gitattach(Req *r)
+{
+	Gitaux *aux;
+	Dir *d;
+
+	if((d = dirstat(".git")) == nil)
+		sysfatal("git/fs: %r");
+	aux = emalloc(sizeof(Gitaux));
+	aux->path[0] = (Qid){Qroot, 0, QTDIR};
+	aux->opath[0] = nil;
+	aux->npath = 1;
+	aux->mtime = d->mtime;
+	r->ofcall.qid = (Qid){Qroot, 0, QTDIR};
+	r->fid->qid = r->ofcall.qid;
+	r->fid->aux = aux;
+	respond(r, nil);
+}
+
+static char *
+objwalk1(Qid *q, Gitaux *aux, char *name, vlong qdir)
+{
+	Object *o, *w;
+	char *e;
+	int i;
+
+	w = nil;
+	e = nil;
+	o = aux->obj;
+	if(!o)
+		return Eexist;
+	if(o->type == GTree){
+		q->type = 0;
+		for(i = 0; i < o->tree->nent; i++){
+			if(strcmp(o->tree->ent[i].name, name) != 0)
+				continue;
+			w = readobject(o->tree->ent[i].h);
+			if(!w && o->tree->ent[i].modref)
+				w = modrefobj(&o->tree->ent[i]);
+			if(!w)
+				die("could not read object for %s", name);
+			q->type = (w->type == GTree) ? QTDIR : 0;
+			q->path = QPATH(w->id, qdir);
+			aux->obj = w;
+		}
+		if(!w)
+			e = Eexist;
+	}else if(o->type == GCommit){
+		q->type = 0;
+		aux->mtime = o->commit->mtime;
+		assert(qdir == Qcommit || qdir == Qobject || qdir == Qcommittree || qdir == Qhead);
+		if(strcmp(name, "msg") == 0)
+			q->path = QPATH(o->id, Qcommitmsg);
+		else if(strcmp(name, "parent") == 0 && o->commit->nparent != 0)
+			q->path = QPATH(o->id, Qcommitparent);
+		else if(strcmp(name, "hash") == 0)
+			q->path = QPATH(o->id, Qcommithash);
+		else if(strcmp(name, "author") == 0)
+			q->path = QPATH(o->id, Qcommitauthor);
+		else if(strcmp(name, "tree") == 0){
+			q->type = QTDIR;
+			q->path = QPATH(o->id, Qcommittree);
+			aux->obj = readobject(o->commit->tree);
+		}
+		else
+			e = Eexist;
+	}else if(o->type == GTag){
+		e = "tag walk unimplemented";
+	}
+	return e;
+}
+
+static Object *
+readref(char *pathstr)
+{
+	char buf[128], path[128], *p, *e;
+	Hash h;
+	int n, f;
+
+	snprint(path, sizeof(path), "%s", pathstr);
+	while(1){
+		if((f = open(path, OREAD)) == -1)
+			return nil;
+		if((n = readn(f, buf, sizeof(buf) - 1)) == -1)
+			return nil;
+		close(f);
+		buf[n] = 0;
+		if(strncmp(buf, "ref:", 4) !=  0)
+			break;
+
+		p = buf + 4;
+		while(isspace(*p))
+			p++;
+		if((e = strchr(p, '\n')) != nil)
+			*e = 0;
+		snprint(path, sizeof(path), ".git/%s", p);
+	}
+
+	if(hparse(&h, buf) == -1){
+		print("failed to parse hash %s\n", buf);
+		return nil;
+	}
+
+	return readobject(h);
+}
+
+static char*
+gitwalk1(Fid *fid, char *name, Qid *q)
+{
+	char path[128];
+	Gitaux *aux;
+	Object *o;
+	char *e;
+	Dir *d;
+	Hash h;
+
+	e = nil;
+	aux = fid->aux;
+	q->vers = 0;
+
+	if(strcmp(name, "..") == 0){
+		if(aux->npath > 1)
+			aux->npath--;
+		*q = aux->path[aux->npath - 1];
+		o = ref(aux->opath[aux->npath - 1]);
+		if(aux->obj)
+			unref(aux->obj);
+		aux->obj = o;
+		fid->qid = *q;
+		return nil;
+	}
+	
+
+	switch(QDIR(&fid->qid)){
+	case Qroot:
+		if(strcmp(name, "HEAD") == 0){
+			*q = (Qid){Qhead, 0, QTDIR};
+			aux->obj = readref(".git/HEAD");
+		}else if(strcmp(name, "object") == 0){
+			*q = (Qid){Qobject, 0, QTDIR};
+		}else if(strcmp(name, "branch") == 0){
+			*q = (Qid){Qbranch, 0, QTDIR};
+			aux->refpath = estrdup(".git/refs/");
+		}else if(strcmp(name, "ctl") == 0){
+			*q = (Qid){Qctl, 0, 0};
+		}else{
+			e = Eexist;
+		}
+		break;
+	case Qbranch:
+		if(strcmp(aux->refpath, ".git/refs/heads") == 0 && strcmp(name, "HEAD") == 0)
+			snprint(path, sizeof(path), ".git/HEAD");
+		else
+			snprint(path, sizeof(path), "%s/%s", aux->refpath, name);
+		q->type = QTDIR;
+		d = dirstat(path);
+		if(d && d->qid.type == QTDIR)
+			q->path = QPATH(findbranch(aux, path), Qbranch);
+		else if(d && (aux->obj = readref(path)) != nil)
+			q->path = QPATH(aux->obj->id, Qcommit);
+		else
+			e = Eexist;
+		free(d);
+		break;
+	case Qobject:
+		if(aux->obj){
+			e = objwalk1(q, aux, name, Qobject);
+		}else{
+			if(hparse(&h, name) == -1)
+				return "invalid object name";
+			if((aux->obj = readobject(h)) == nil)
+				return "could not read object";
+			q->path = QPATH(aux->obj->id, Qobject);
+			q->type = (aux->obj->type == GBlob) ? 0 : QTDIR;
+			q->vers = 0;
+		}
+		break;
+	case Qhead:
+		e = objwalk1(q, aux, name, Qhead);
+		break;
+	case Qcommit:
+		e = objwalk1(q, aux, name, Qcommit);
+		break;
+	case Qcommittree:
+		e = objwalk1(q, aux, name, Qcommittree);
+		break;
+	case Qcommitparent:
+	case Qcommitmsg:
+	case Qcommitdata:
+	case Qcommithash:
+	case Qcommitauthor:
+	case Qctl:
+		return Enodir;
+	default:
+		die("walk: bad qid %Q", *q);
+	}
+	if(aux->npath >= Npath)
+		e = E2long;
+	if(!e && QDIR(q) >= Qmax){
+		print("npath: %d\n", aux->npath);
+		print("walking to %llx (name: %s)\n", q->path, name);
+		print("walking from %llx\n", fid->qid.path);
+		print("QDIR=%d\n", QDIR(&fid->qid));
+		if(aux->obj)
+			print("obj=%O\n", aux->obj);
+		abort();
+	}
+
+	aux->path[aux->npath] = *q;
+	if(aux->obj)
+		aux->opath[aux->npath] = ref(aux->obj);
+	aux->npath++;
+	fid->qid = *q;
+	return e;
+}
+
+static char*
+gitclone(Fid *o, Fid *n)
+{
+	Gitaux *aux, *oaux;
+	int i;
+
+	oaux = o->aux;
+	aux = emalloc(sizeof(Gitaux));
+	aux->npath = oaux->npath;
+	for(i = 0; i < aux->npath; i++){
+		aux->path[i] = oaux->path[i];
+		aux->opath[i] = oaux->opath[i];
+		if(aux->opath[i])
+			ref(aux->opath[i]);
+	}
+	if(oaux->refpath)
+		aux->refpath = strdup(oaux->refpath);
+	if(oaux->obj)
+		aux->obj = ref(oaux->obj);
+	aux->qdir = oaux->qdir;
+	aux->mtime = oaux->mtime;
+	n->aux = aux;
+	return nil;
+}
+
+static void
+gitdestroyfid(Fid *f)
+{
+	Gitaux *aux;
+	int i;
+
+	if((aux = f->aux) == nil)
+		return;
+	for(i = 0; i < aux->npath; i++)
+		unref(aux->opath[i]);
+	free(aux->refpath);
+	olsfree(aux->ols);
+	unref(aux->obj);
+	free(aux);
+}
+
+static char *
+readctl(Req *r)
+{
+	char data[512], buf[512], *p;
+	int fd, n;
+	if((fd = open(".git/HEAD", OREAD)) == -1)
+		return Erepo;
+	/* empty HEAD is invalid */
+	if((n = readn(fd, buf, sizeof(buf) - 1)) <= 0)
+		return Erepo;
+	close(fd);
+	p = buf;
+	buf[n] = 0;
+	if(strstr(p, "ref: ") == buf)
+		p += strlen("ref: ");
+	if(strstr(p, "refs/") == p)
+		p += strlen("refs/");
+	snprint(data, sizeof(data), "branch %s", p);
+	readstr(r, data);
+	return nil;
+}
+
+static void
+gitread(Req *r)
+{
+	char buf[64], *e;
+	Gitaux *aux;
+	Object *o;
+	Qid *q;
+
+	q = &r->fid->qid;
+	o = nil;
+	e = nil;
+	if(aux = r->fid->aux)
+		o = aux->obj;
+
+	switch(QDIR(q)){
+	case Qroot:
+		dirread9p(r, rootgen, aux);
+		break;
+	case Qbranch:
+		if(o)
+			objread(r, aux);
+		else
+			dirread9p(r, branchgen, aux);
+		break;
+	case Qobject:
+		if(o)
+			objread(r, aux);
+		else
+			dirread9p(r, objgen, aux);
+		break;
+	case Qcommitmsg:
+		readbuf(r, o->commit->msg, o->commit->nmsg);
+		break;
+	case Qcommitparent:
+		readcommitparent(r, o);
+		break;
+	case Qcommithash:
+		snprint(buf, sizeof(buf), "%H\n", o->hash);
+		readstr(r, buf);
+		break;
+	case Qcommitauthor:
+		readstr(r, o->commit->author);
+		break;
+	case Qctl:
+		e = readctl(r);
+		break;
+	case Qhead:
+		/* Empty repositories have no HEAD */
+		if(aux->obj == nil)
+			r->ofcall.count = 0;
+		else
+			objread(r, aux);
+		break;
+	case Qcommit:
+	case Qcommittree:
+	case Qcommitdata:
+		objread(r, aux);
+		break;
+	default:
+		die("read: bad qid %Q", *q);
+	}
+	respond(r, e);
+}
+
+static void
+gitstat(Req *r)
+{
+	Gitaux *aux;
+	Qid *q;
+
+	q = &r->fid->qid;
+	aux = r->fid->aux;
+	r->d.uid = estrdup9p(username);
+	r->d.gid = estrdup9p(username);
+	r->d.muid = estrdup9p(username);
+	r->d.mtime = aux->mtime;
+	r->d.atime = r->d.mtime;
+	r->d.qid = r->fid->qid;
+	r->d.mode = 0755 | DMDIR;
+	if(aux->obj){
+		obj2dir(&r->d, aux->obj, QDIR(q), aux->mtime);
+	} else {
+		switch(QDIR(q)){
+		case Qroot:
+			r->d.name = estrdup9p("/");
+			break;
+		case Qhead:
+			r->d.name = estrdup9p("HEAD");
+			break;
+		case Qbranch:
+			r->d.name = estrdup9p("branch");
+			break;
+		case Qobject:
+			r->d.name = estrdup9p("object");
+			break;
+		case Qctl:
+			r->d.name = estrdup9p("ctl");
+			r->d.mode = 0666;
+			break;
+		case Qcommit:
+			r->d.name = smprint("%H", aux->obj->hash);
+			break;
+		case Qcommitmsg:
+			r->d.name = estrdup9p("msg");
+			r->d.mode = 0644;
+			break;
+		case Qcommittree:
+			r->d.name = estrdup9p("tree");
+			break;
+		case Qcommitparent:
+			r->d.name = estrdup9p("info");
+			r->d.mode = 0644;
+			break;
+		case Qcommithash:
+			r->d.name = estrdup9p("hash");
+			r->d.mode = 0644;
+			break;
+		default:
+			die("stat: bad qid %Q", *q);
+		}
+	}
+
+	respond(r, nil);
+}
+
+Srv gitsrv = {
+	.attach=gitattach,
+	.walk1=gitwalk1,
+	.clone=gitclone,
+	.read=gitread,
+	.stat=gitstat,
+	.destroyfid=gitdestroyfid,
+};
+
+void
+usage(void)
+{
+	fprint(2, "usage: %s [-d]\n", argv0);
+	fprint(2, "\t-d:	debug\n");
+	exits("usage");
+}
+
+void
+main(int argc, char **argv)
+{
+	gitinit();
+	ARGBEGIN{
+	case 'd':	chatty9p++;	break;
+	default:	usage();	break;
+	}ARGEND;
+	if(argc != 0)
+		usage();
+
+	username = getuser();
+	branches = emalloc(sizeof(char*));
+	branches[0] = nil;
+	postmountsrv(&gitsrv, nil, "/mnt/git", MCREATE);
+	exits(nil);
+}
--- /dev/null
+++ b/git.1
@@ -1,0 +1,397 @@
+.TH GIT 1
+.SH NAME
+git, git/conf, git/fs, git/query, git/walk, git/clone, git/branch,
+git/commit, git/diff, git/init, git/log, git/merge, git/push, git/pull
+\- Manage git repositories.
+
+.SH SYNOPSIS
+.PP
+.B git/add
+[
+.B -r
+]
+.I path...
+.PP
+.B git/branch
+[
+.B -b
+.I base
+]
+.I newbranch
+.PP
+.B git/clone
+[
+.I remote
+[
+.I local
+]
+]
+.PP
+.B git/commit
+.PP
+.B git/conf
+[
+.B -r
+]
+[
+.B -f
+.I file
+]
+.I keys...
+.PP
+.B git/diff
+[
+.B -b
+.I branch
+]
+[
+.I file...
+]
+.PP
+.B git/fs
+[
+.B -d
+]
+.PP
+.B git/init
+[
+.B -b
+]
+[
+.I dir
+]
+[
+.B -u
+.I upstream
+]
+.PP
+.B git/log
+.PP
+.B git/merge
+.I theirs
+.PP
+.B git/pull
+[
+.B -f
+]
+[
+.B -u
+.I upstream
+]
+.PP
+.B git/push
+[
+.B -a
+]
+[
+.B -u
+.I upstream
+]
+[
+.B -b
+.I branch
+]
+[
+.B -r
+.I branch
+]
+.PP
+.B git/query
+[
+.B -p
+]
+.PP
+.B git/walk
+[
+.B -qc
+]
+[
+.B -b
+.I branch
+]
+[
+.B -f
+.I filters
+]
+
+.SH DESCRIPTION
+.PP
+Git is a distributed version control system.
+This means that each repository contains a full copy of the history.
+This history is then synced between computers as needed.
+
+.PP
+These programs provide tools to manage and interoperate with
+repositories hosted in git.
+
+.SH CONCEPTS
+
+Git stores snapshots of the working directory.
+Files can either be in a tracked or untracked state.
+Each commit takes the current version of all tracked files and
+adds them to a new commit.
+
+This history is stored in the
+.I .git
+directory.
+This suite of
+.I git
+tools provides a file interface to the
+.I .git
+directory mounted on
+.I /mnt/git.
+Modifications to the repository are done directly to the
+.I .git
+directory, and are reflected in the file system interface.
+This allows for easy scripting, without excessive complexity
+in the file API.
+
+.SH COMMANDS
+
+.PP
+.B Git/init
+is used to create a new git repository, with no code or commits.
+The repository is created in the current directory by default.
+Passing a directory name will cause the repository to be created
+there instead.
+Passing the
+.B -b
+option will cause the repository to be initialized as a bare repository.
+Passing the
+.B -u
+.I upstream
+option will cause the upstream to be configured to
+.I upstream.
+
+.PP
+.B Git/clone
+will take an existing repository, served over either the
+.I git://
+or
+.I ssh://
+protocols.
+The first argument is the repository to clone.
+The second argument, optionally, specifies the location to clone into.
+If not specified, the repository will be cloned into the last path component
+of the clone source, with the
+.I .git
+stripped off if present.
+
+.B Git/push
+is used to push the current changes to a remote repository.
+When no arguments are provided, the remote repository is taken from
+the origin configured in
+.I .git/config,
+and only the changes on the current branch are pushed.
+When passed the
+.I -a
+option, all branches are pushed.
+When passed the
+.I -u upstream
+option, the changed are pushed to
+.I upstream
+instead of the configured origin.
+When given the
+.I -r
+option, the branch is deleted from origin, instead of updated.
+
+.B Git/pull
+behaves in a similar manner to git/push, however it gets changes from
+the upstream repository.
+After fetching, it checks out the changes into the working directory.
+When passed the
+.I -f
+option, the update of the working copy is suppressed.
+When passed the
+.I -u upstream
+option, the changes are pulled from
+.I upstream
+instead of the configured origin.
+
+.B Git/fs
+provides a read-only file system mounted on /mnt/git.
+It provides a  view into the repository contents to allow convenient inspection of repository structure.
+Surrounding scripts and binaries manipulate the repository contents directly.
+These changes will be immediately mirrored in the file system.
+
+.PP
+Git/fs serves the following directories:
+
+.TP
+/mnt/git/object
+The objects in the repo.
+.TP
+/mnt/git/branch
+The branches and tags in the repo.
+.TP
+/mnt/git/ctl
+A file showing the status of the repo.
+Currently, it only shows the current branch.
+.TP
+/mnt/git/HEAD
+An alias for the currently checked out commit directory.
+
+.PP
+.B Git/add
+adds a file to the list of tracked files. When passed the
+.I -r
+flag, the file is removed from the list of tracked files.
+The copy of the file in the repository is left untouched.
+
+.PP
+.B Git/commit
+creates a new commit reflecting the state of the current repository.
+It takes no arguments: all files must be added or removed using git/add.
+If a file does not exist in the working directory, it is treated as though it was removed.
+
+.PP
+.B Git/branch
+is used to switch between branches.
+When passed the
+.I -c
+option, the branch will be created if it does not yet exist.
+When passed the
+.I -o original
+option, the branch created is based off of
+.I original
+instead of
+.I HEAD.
+When passed the
+.I -s
+option, the branch is created but the files are not checked out.
+
+.PP
+.B Git/log
+shows a history of the current branch.
+
+.PP
+.B Git/diff
+shows the differences between the currently checked out code and
+the
+.I HEAD
+commit.
+When passed the
+.I -b base
+option, the diff is computed against
+.I base
+instead of
+.I HEAD.
+
+.PP
+.B Git/merge
+takes two branches and merges them filewise using
+.I ape/diff3.
+The next commit made will be a merge commmit.
+
+.PP
+.B Git/conf
+is a tool for querying the git configuration.
+The configuration key is provided as a dotted string. Spaces
+are accepted. For example, to find the URL of the origin
+repository, one might pass
+.I 'remote "origin".url".
+When given the
+.I -r
+option, the root of the current repository is printed.
+
+.B Git/query
+takes an expression describing a commit, or set of commits,
+and resolves it to a list of commits. With the
+.I -p
+option, instead of printing the commit hashes, the full
+path to their
+.B git/fs
+path is printed.
+
+.PP
+.B Git/walk
+provides a tool for walking the list of tracked objects and printing their status.
+With no arguments, it prints a list of paths prefixed with the status character.
+When given the
+.I -c
+character, only the paths are printed.
+When given the
+.I -q
+option, all output is suppressed, and only the status is printed.
+When given the
+.I -f
+option, the output is filtered by status code, and only matching items are printed.
+
+.PP
+The status characters are as follows:
+.TP
+T
+Tracked, not modified since last commit.
+.TP
+M
+Modified since last commit.
+.TP
+R
+Removed from either working directory tracking list.
+.TP
+A
+Added, does not yet exist in a commit.
+
+.SH REF SYNTAX
+
+.PP
+Refs are specified with a simple query syntax.
+A bare hash always evaluates to itself.
+Ref names are resolved to their hashes.
+The
+.B a ^
+suffix operator finds the parent of a commit.
+The
+.B a b @
+suffix operator finds the common ancestor of the previous two commits.
+The
+.B a .. b
+or
+.B a : b
+operator finds all commits between
+.B a
+and
+.B b.
+Between is defined as the set of all commits which are ancestors of
+.B b
+and descendants of
+.B a.
+
+.SH EXAMPLES
+
+.PP
+In order to create a new repository, run
+.B git/init:
+
+.EX
+git/init myrepo
+.EE
+
+To clone an existing repository from a git server, run:
+.EX
+git/clone git://github.com/Harvey-OS/harvey
+cd harvey
+# edit files
+git/commit
+git/push
+.EE
+
+.SH SOURCE
+.B /sys/src/cmd/git
+
+.SH SEE ALSO
+.IR hg(1)
+.IR replica(1)
+.IR patch(1)
+.IR diff3
+
+.SH BUGS
+.PP
+Repositories with submodules are effectively read-only.
+
+.PP
+There are a number of missing commands, features, and tools. Notable
+missing features include
+.I http
+clones, history editing, and formatted patch management.
+
--- /dev/null
+++ b/git.h
@@ -1,0 +1,238 @@
+#include <bio.h>
+#include <mp.h>
+#include <libsec.h>
+#include <flate.h>
+#include <regexp.h>
+
+typedef struct Hash	Hash;
+typedef struct Cinfo	Cinfo;
+typedef struct Tinfo	Tinfo;
+typedef struct Object	Object;
+typedef struct Objset	Objset;
+typedef struct Pack	Pack;
+typedef struct Buf	Buf;
+typedef struct Dirent	Dirent;
+typedef struct Idxent	Idxent;
+typedef struct Ols	Ols;
+
+enum {
+	/* 5k objects should be enough */
+	Cachemax=5*1024,
+	Pathmax=512,
+	Hashsz=20,
+
+	Nproto	= 16,
+	Nport	= 16,
+	Nhost	= 256,
+	Npath	= 128,
+	Nrepo	= 64,
+	Nbranch	= 32,
+};
+
+typedef enum Type {
+	GNone	= 0,
+	GCommit	= 1,
+	GTree	= 2,
+	GBlob	= 3,
+	GTag	= 4,
+	GOdelta	= 6,
+	GRdelta	= 7,
+} Type;
+
+enum {
+	Cloaded	= 1 << 0,
+	Cidx	= 1 << 1,
+	Ccache	= 1 << 2,
+	Cexist	= 1 << 3,
+	Cparsed	= 1 << 5,
+};
+
+struct Ols {
+	int idx;
+
+	int fd;
+	int state;
+	int stage;
+
+	Dir *top;
+	int ntop;
+	int topidx;
+	Dir *loose;
+	int nloose;
+	int looseidx;
+	Dir *pack;
+	int npack;
+	int packidx;
+	int nent;
+	int entidx;
+};
+
+struct Hash {
+	uchar h[20];
+};
+
+struct Dirent {
+	char *name;
+	int modref;
+	int mode;
+	Hash h;
+};
+
+struct Object {
+	/* Git data */
+	Hash	hash;
+	Type	type;
+
+	/* Cache */
+	int	id;
+	int	flag;
+	int	refs;
+	Object	*next;
+	Object	*prev;
+
+	/* For indexing */
+	vlong	off;
+
+	/* Everything below here gets cleared */
+	char	*all;
+	char	*data;
+	/* size excludes header */
+	vlong	size;
+
+	union {
+		Cinfo *commit;
+		Tinfo *tree;
+	};
+};
+
+struct Tinfo {
+	/* Tree */
+	Dirent	*ent;
+	int	nent;
+};
+
+struct Cinfo {
+	/* Commit */
+	Hash	*parent;
+	int	nparent;
+	Hash	tree;
+	char	*author;
+	char	*committer;
+	char	*msg;
+	int	nmsg;
+	vlong	ctime;
+	vlong	mtime;
+};
+
+struct Objset {
+	Object	**obj;
+	int	nobj;
+	int	sz;
+};
+
+#define GETBE16(b)\
+		((((b)[0] & 0xFFul) <<  8) | \
+		 (((b)[1] & 0xFFul) <<  0))
+
+#define GETBE32(b)\
+		((((b)[0] & 0xFFul) << 24) | \
+		 (((b)[1] & 0xFFul) << 16) | \
+		 (((b)[2] & 0xFFul) <<  8) | \
+		 (((b)[3] & 0xFFul) <<  0))
+#define GETBE64(b)\
+		((((b)[0] & 0xFFull) << 56) | \
+		 (((b)[1] & 0xFFull) << 48) | \
+		 (((b)[2] & 0xFFull) << 40) | \
+		 (((b)[3] & 0xFFull) << 32) | \
+		 (((b)[4] & 0xFFull) << 24) | \
+		 (((b)[5] & 0xFFull) << 16) | \
+		 (((b)[6] & 0xFFull) <<  8) | \
+		 (((b)[7] & 0xFFull) <<  0))
+
+#define PUTBE16(b, n)\
+	do{ \
+		(b)[0] = (n) >> 8; \
+		(b)[1] = (n) >> 0; \
+	} while(0)
+
+#define PUTBE32(b, n)\
+	do{ \
+		(b)[0] = (n) >> 24; \
+		(b)[1] = (n) >> 16; \
+		(b)[2] = (n) >> 8; \
+		(b)[3] = (n) >> 0; \
+	} while(0)
+
+#define PUTBE64(b, n)\
+	do{ \
+		(b)[0] = (n) >> 56; \
+		(b)[1] = (n) >> 48; \
+		(b)[2] = (n) >> 40; \
+		(b)[3] = (n) >> 32; \
+		(b)[4] = (n) >> 24; \
+		(b)[5] = (n) >> 16; \
+		(b)[6] = (n) >> 8; \
+		(b)[7] = (n) >> 0; \
+	} while(0)
+
+#define QDIR(qid)	((int)(qid)->path & (0xff))
+#define QPATH(id, dt)	(((uvlong)(id) << 8) | ((dt) & 0x7f))
+#define isblank(c) \
+	(((c) != '\n') && isspace(c))
+
+extern Reprog *authorpat;
+extern Objset objcache;
+extern Hash Zhash;
+extern int chattygit;
+
+#pragma varargck type "H" Hash
+#pragma varargck type "T" Type
+#pragma varargck type "O" Object*
+#pragma varargck type "Q" Qid
+int Hfmt(Fmt*);
+int Tfmt(Fmt*);
+int Ofmt(Fmt*);
+int Qfmt(Fmt*);
+
+void gitinit(void);
+
+/* object io */
+int	resolverefs(Hash **, char *);
+int	resolveref(Hash *, char *);
+Object	*readobject(Hash);
+void	parseobject(Object *);
+int	indexpack(char *, char *, Hash);
+int	hasheq(Hash *, Hash *);
+Object	*ref(Object *);
+void	unref(Object *);
+void	cache(Object *);
+
+/* object sets */
+void	osinit(Objset *);
+void	osadd(Objset *, Object *);
+int	oshas(Objset *, Object *);
+Object	*osfind(Objset *, Hash);
+
+/* object listing */
+Ols	*mkols(void);
+int	olsnext(Ols *, Hash *);
+void	olsfree(Ols *);
+
+/* util functions */
+void	*emalloc(ulong);
+void	*erealloc(void *, ulong);
+char	*estrdup(char *);
+int	slurpdir(char *, Dir **);
+int	hparse(Hash *, char *);
+int	hassuffix(char *, char *);
+int	swapsuffix(char *, int, char *, char *, char *);
+char	*strip(char *);
+void	die(char *, ...);
+
+/* proto handling */
+int	readpkt(int, char*, int);
+int	writepkt(int, char*, int);
+int	flushpkt(int);
+int	parseuri(char *, char *, char *, char *, char *, char *);
+int	dialssh(char *, char *, char *, char *);
+int	dialgit(char *, char *, char *, char *);
--- /dev/null
+++ b/init
@@ -1,0 +1,41 @@
+#!/bin/rc
+
+rfork e
+
+fn usage{
+	echo git/init [-b] name >[1=2]
+	echo '	-b	init bare repository' >[1=2]
+	exit usage
+}
+
+sub='/.git'
+upstream=()
+while(~ $1 -*){
+	switch($1){
+	case '-b';
+		sub=''
+	case '-u';
+		shift
+		if(~ $#* 0)
+			usage
+		upstream=$1
+		shift
+	case *;
+		usage
+	}
+	shift
+}
+
+if (~ $#* 0)
+	dir=.
+if not if(~ $#* 1)
+	dir=$1
+if not
+	usage
+
+mkdir -p $dir$sub
+dircp /sys/lib/git/template $dir/$sub
+if(! ~ $#upstream 0){
+	echo '[remote "origin"]' >> $dir/$sub/config
+	echo '	url='$upstream >> $dir/$sub/config
+}
--- /dev/null
+++ b/log
@@ -1,0 +1,26 @@
+#!/bin/rc -e
+
+rfork en
+
+base=/mnt/git/object/
+branch=$1
+if(~ $1 '')
+	branch='master'
+if(! test -e /mnt/git/ctl)
+	git/fs
+
+commits=(`{git/query $branch})
+while(! ~$#commits 0){
+	c=$commits(1)
+	
+	echo 'Hash:	' `{cat $base/$c/hash}
+	echo 'Author:	' `{cat $base/$c/author}
+	cat $base/$c/msg | sed 's/^/	/g'
+	echo ''
+
+	commits=($commits(2-) `{cat $base/$c/parent >[2]/dev/null})
+	if(! ~ $#commits 0)
+		commits=`{mtime $base^$commits |
+			sort -rn | uniq |
+			awk -F/ '{print $NF}'}
+}
--- /dev/null
+++ b/merge
@@ -1,0 +1,37 @@
+#!/bin/rc -e
+
+rfork ne
+
+fn merge{
+	ourbr=$1/tree
+	basebr=$2/tree
+	theirbr=$3/tree
+
+	all=`{walk -f $ourbr $basebr $theirbr | \
+		 sed 's@^('$ourbr'|'$basebr'|'$theirbr')/*@@g' | sort | uniq}
+	for(f in $all){
+		if(! test -f $ourbr/$f)
+			ours=/dev/null
+		if(! test -f $basebr/$f)
+			base=/dev/null
+		if(! test -f $theirbr/$f)
+			theirs=/dev/null
+		if(! ape/diff3 -m $ourbr/$f $basebr/$f $theirbr/$f > $f)
+			echo merge needed: $f
+	}
+}
+
+fn usage{
+	echo usage: $argv0 theirs
+	exit usage
+}
+
+if(! ~ $#* 1)
+	usage
+
+git/fs
+theirs=`{git/query $1}
+ours=`{git/query HEAD}
+base=`{git/query $theirs ^ ' ' ^ $ours ^ '@'}
+
+merge /mnt/git/object/$ours /mnt/git/object/$base /mnt/git/object/$theirs
--- /dev/null
+++ b/mkfile
@@ -1,0 +1,65 @@
+</$objtype/mkfile
+
+BIN=/$objtype/bin/git
+TARG=\
+	conf\
+	fetch\
+	fs\
+	query\
+	save\
+	send\
+	walk\
+
+RC=\
+	add\
+	branch\
+	clone\
+	commit\
+	diff\
+	init\
+	log\
+	merge\
+	pull\
+	push\
+
+OFILES=\
+	objset.$O\
+	ols.$O\
+	pack.$O\
+	proto.$O\
+	util.$O\
+	ref.$O
+
+HFILES=git.h
+
+</sys/src/cmd/mkmany
+
+# Override install target to install rc.
+install:V:
+	mkdir -p $BIN
+	for (i in $TARG)
+		mk $MKFLAGS $i.install
+	for (i in $RC)
+		mk $MKFLAGS $i.rcinstall
+	cp git.1 /sys/man/1/git
+	mk $MKFLAGS /sys/lib/git/template
+
+uninstall:V:
+	rm -rf $BIN /sys/lib/git
+
+%.c %.h: %.y
+	$YACC $YFLAGS -D1 -d -s $stem $prereq
+	mv $stem.tab.c $stem.c
+	mv $stem.tab.h $stem.h
+
+%.c %.h: %.y
+	$YACC $YFLAGS -D1 -d -s $stem $prereq
+	mv $stem.tab.c $stem.c
+	mv $stem.tab.h $stem.h
+
+%.rcinstall:V:
+	cp $stem $BIN/$stem
+
+/sys/lib/git/template: template
+	mkdir -p /sys/lib/git/template
+	dircp template /sys/lib/git/template
--- /dev/null
+++ b/objset.c
@@ -1,0 +1,66 @@
+#include <u.h>
+#include <libc.h>
+#include <pool.h>
+
+#include "git.h"
+
+void
+osinit(Objset *s)
+{
+	s->sz = 16;
+	s->nobj = 0;
+	s->obj = emalloc(s->sz * sizeof(Hash));
+}
+
+void
+osfree(Objset *s)
+{
+	free(s->obj);
+}
+
+void
+osadd(Objset *s, Object *o)
+{
+	u32int probe;
+	Object **obj;
+	int i, sz;
+
+	probe = GETBE32(o->hash.h) % s->sz;
+	while(s->obj[probe]){
+		if(hasheq(&s->obj[probe]->hash, &o->hash))
+			return;
+		probe = (probe + 1) % s->sz;
+	}
+	assert(s->obj[probe] == nil);
+	s->obj[probe] = o;
+	s->nobj++;
+	if(s->sz < 2*s->nobj){
+		sz = s->sz;
+		obj = s->obj;
+
+		s->sz *= 2;
+		s->nobj = 0;
+		s->obj = emalloc(s->sz * sizeof(Hash));
+		for(i = 0; i < sz; i++)
+			if(obj[i])
+				osadd(s, obj[i]);
+		free(obj);
+	}
+}
+
+Object*
+osfind(Objset *s, Hash h)
+{
+	u32int probe;
+
+	for(probe = GETBE32(h.h) % s->sz; s->obj[probe]; probe = (probe + 1) % s->sz)
+		if(hasheq(&s->obj[probe]->hash, &h))
+			return s->obj[probe]; 
+	return 0;
+}
+
+int
+oshas(Objset *s, Object *o)
+{
+	return osfind(s, o->hash) != nil;
+}
--- /dev/null
+++ b/ols.c
@@ -1,0 +1,170 @@
+#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(Ols *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(Ols *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;
+}
+
+Ols*
+mkols(void)
+{
+	Ols *ols;
+
+	ols = emalloc(sizeof(Ols));
+	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(Ols *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(Ols *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;
+}
--- /dev/null
+++ b/pack.c
@@ -1,0 +1,981 @@
+#include <u.h>
+#include <libc.h>
+#include <ctype.h>
+
+#include "git.h"
+
+typedef struct Buf Buf;
+
+struct Buf {
+	int len;
+	int sz;
+	char *data;
+};
+
+static int	readpacked(Biobuf *, Object *, int);
+static Object	*readidxobject(Biobuf *, Hash, int);
+
+Objset objcache;
+Object *lruhead;
+Object *lrutail;
+int	ncache;
+
+static void
+clear(Object *o)
+{
+	if(!o)
+		return;
+
+	assert(o->refs == 0);
+	assert((o->flag & Ccache) == 0);
+	assert(o->flag & Cloaded);
+	switch(o->type){
+	case GCommit:
+		if(!o->commit)
+			break;
+		free(o->commit->parent);
+		free(o->commit->author);
+		free(o->commit->committer);
+		free(o->commit);
+		o->commit = nil;
+		break;
+	case GTree:
+		if(!o->tree)
+			break;
+		free(o->tree->ent);
+		free(o->tree);
+		o->tree = nil;
+		break;
+	default:
+		break;
+	}
+
+	free(o->all);
+	o->all = nil;
+	o->data = nil;
+	o->flag &= ~Cloaded;
+}
+
+void
+unref(Object *o)
+{
+	if(!o)
+		return;
+	o->refs--;
+	if(!o->refs)
+		clear(o);
+}
+
+Object*
+ref(Object *o)
+{
+	o->refs++;
+	return o;
+}
+
+void
+cache(Object *o)
+{
+	Object *p;
+
+	if(o == lruhead)
+		return;
+	if(o == lrutail)
+		lrutail = lrutail->prev;
+	if(!(o->flag & Cexist)){
+		osadd(&objcache, o);
+		o->id = objcache.nobj;
+		o->flag |= Cexist;
+	}
+	if(o->prev)
+		o->prev->next = o->next;
+	if(o->next)
+		o->next->prev = o->prev;
+	if(lrutail == o){
+		lrutail = o->prev;
+		lrutail->next = nil;
+	}else if(!lrutail)
+		lrutail = o;
+	if(lruhead)
+		lruhead->prev = o;
+	o->next = lruhead;
+	o->prev = nil;
+	lruhead = o;
+
+	if(!(o->flag & Ccache)){
+		o->flag |= Ccache;
+		ref(o);
+		ncache++;
+	}
+	while(ncache > Cachemax){
+		p = lrutail;
+		lrutail = p->prev;
+		lrutail->next = nil;
+		p->flag &= ~Ccache;
+		p->prev = nil;
+		p->next = nil;
+		unref(p);
+		ncache--;
+	}		
+}
+
+int
+bappend(void *p, void *src, int len)
+{
+	Buf *b = p;
+	char *n;
+
+	while(b->len + len >= b->sz){
+		b->sz = b->sz*2 + 64;
+		n = realloc(b->data, b->sz);
+		if(n == nil)
+			return -1;
+		b->data = n;
+	}
+	memmove(b->data + b->len, src, len);
+	b->len += len;
+	return len;
+}
+
+int
+breadc(void *p)
+{
+	return Bgetc(p);
+}
+
+int
+bdecompress(Buf *d, Biobuf *b, vlong *csz)
+{
+	vlong o;
+
+	o = Boffset(b);
+	if(inflatezlib(d, bappend, b, breadc) == -1){
+		free(d->data);
+		return -1;
+	}
+	if (csz)
+		*csz = Boffset(b) - o;
+	return d->len;
+}
+
+int
+decompress(void **p, Biobuf *b, vlong *csz)
+{
+	Buf d = {.len=0, .data=nil, .sz=0};
+
+	if(bdecompress(&d, b, csz) == -1){
+		free(d.data);
+		return -1;
+	}
+	*p = d.data;
+	return d.len;
+}
+
+static int
+preadbe32(Biobuf *b, int *v, vlong off)
+{
+	char buf[4];
+	
+	if(Bseek(b, off, 0) == -1)
+		return -1;
+	if(Bread(b, buf, sizeof(buf)) == -1)
+		return -1;
+	*v = GETBE32(buf);
+
+	return 0;
+}
+static int
+preadbe64(Biobuf *b, vlong *v, vlong off)
+{
+	char buf[8];
+	
+	if(Bseek(b, off, 0) == -1)
+		return -1;
+	if(Bread(b, buf, sizeof(buf)) == -1)
+		return -1;
+	*v = GETBE64(buf);
+	return 0;
+}
+
+int
+readvint(char *p, char **pp)
+{
+	int i, n, c;
+	
+	i = 0;
+	n = 0;
+	do {
+		c = *p++;
+		n |= (c & 0x7f) << i;
+		i += 7;
+	} while (c & 0x80);
+	*pp = p;
+
+	return n;
+}
+
+static int
+hashsearch(Hash *hlist, int nent, Hash h)
+{
+	int hi, lo, mid, d;
+
+	lo = 0;
+	hi = nent;
+	while(lo < hi){
+		mid = (lo + hi)/2;
+		d = memcmp(hlist[mid].h, h.h, sizeof h.h);
+		if(d < 0)
+			lo = mid + 1;
+		else if(d > 0)
+			hi = mid;
+		else
+			return mid;
+	}
+	return -1;
+}
+
+static int
+applydelta(Object *dst, Object *base, char *d, int nd)
+{
+	char *r, *b, *ed, *er;
+	int n, nr, c;
+	vlong o, l;
+
+	ed = d + nd;
+	b = base->data;
+	n = readvint(d, &d);
+	if(n != base->size){
+		werrstr("mismatched source size");
+		return -1;
+	}
+
+	nr = readvint(d, &d);
+	r = emalloc(nr + 64);
+	n = snprint(r, 64, "%T %d", base->type, nr) + 1;
+	dst->all = r;
+	dst->type = base->type;
+	dst->data = r + n;
+	dst->size = nr;
+	er = dst->data + nr;
+	r = dst->data;
+
+	while(1){
+		if(d == ed)
+			break;
+		c = *d++;
+		if(!c){
+			werrstr("bad delta encoding");
+			return -1;
+		}
+		/* copy from base */
+		if(c & 0x80){
+			o = 0;
+			l = 0;
+			/* Offset in base */
+			if(c & 0x01) o |= (*d++ <<  0) & 0x000000ff;
+			if(c & 0x02) o |= (*d++ <<  8) & 0x0000ff00;
+			if(c & 0x04) o |= (*d++ << 16) & 0x00ff0000;
+			if(c & 0x08) o |= (*d++ << 24) & 0xff000000;
+
+			/* Length to copy */
+			if(c & 0x10) l |= (*d++ <<  0) & 0x0000ff;
+			if(c & 0x20) l |= (*d++ <<  8) & 0x00ff00;
+			if(c & 0x40) l |= (*d++ << 16) & 0xff0000;
+			if(l == 0) l = 0x10000;
+
+			assert(o + l <= base->size);
+			memmove(r, b + o, l);
+			r += l;
+		/* inline data */
+		}else{
+			memmove(r, d, c);
+			d += c;
+			r += c;
+		}
+
+	}
+	if(r != er){
+		werrstr("truncated delta (%zd)", er - r);
+		return -1;
+	}
+
+	return nr;
+}
+
+static int
+readrdelta(Biobuf *f, Object *o, int nd, int flag)
+{
+	Object *b;
+	Hash h;
+	char *d;
+	int n;
+
+	d = nil;
+	if(Bread(f, h.h, sizeof(h.h)) != sizeof(h.h))
+		goto error;
+	if(hasheq(&o->hash, &h))
+		goto error;
+	if((n = decompress(&d, f, nil)) == -1)
+		goto error;
+	if(d == nil || n != nd)
+		goto error;
+	if((b = readidxobject(f, h, flag)) == nil)
+		goto error;
+	if(applydelta(o, b, d, n) == -1)
+		goto error;
+	free(d);
+	return 0;
+error:
+	free(d);
+	return -1;
+}
+
+static int
+readodelta(Biobuf *f, Object *o, vlong nd, vlong p, int flag)
+{
+	Object b;
+	char *d;
+	vlong r;
+	int c, n;
+
+	r = 0;
+	d = nil;
+	while(1){
+		if((c = Bgetc(f)) == -1)
+			goto error;
+		r |= c & 0x7f;
+		if (!(c & 0x80))
+			break;
+		r++;
+		r <<= 7;
+	}while(c & 0x80);
+
+	if(r > p){
+		werrstr("junk offset -%lld (from %lld)", r, p);
+		goto error;
+	}
+	if((n = decompress(&d, f, nil)) == -1)
+		goto error;
+	if(d == nil || n != nd)
+		goto error;
+	if(Bseek(f, p - r, 0) == -1)
+		goto error;
+	if(readpacked(f, &b, flag) == -1)
+		goto error;
+	if(applydelta(o, &b, d, nd) == -1)
+		goto error;
+	free(d);
+	return 0;
+error:
+	free(d);
+	return -1;
+}
+
+static int
+readpacked(Biobuf *f, Object *o, int flag)
+{
+	int c, s, n;
+	vlong l, p;
+	Type t;
+	Buf b;
+
+	p = Boffset(f);
+	c = Bgetc(f);
+	if(c == -1)
+		return -1;
+	l = c & 0xf;
+	s = 4;
+	t = (c >> 4) & 0x7;
+	if(!t){
+		werrstr("unknown type for byte %x", c);
+		return -1;
+	}
+	while(c & 0x80){
+		if((c = Bgetc(f)) == -1)
+			return -1;
+		l |= (c & 0x7f) << s;
+		s += 7;
+	}
+
+	switch(t){
+	default:
+		werrstr("invalid object at %lld", Boffset(f));
+		return -1;
+	case GCommit:
+	case GTree:
+	case GTag:
+	case GBlob:
+		b.sz = 64 + l;
+
+		b.data = emalloc(b.sz);
+		n = snprint(b.data, 64, "%T %lld", t, l) + 1;
+		b.len = n;
+		if(bdecompress(&b, f, nil) == -1){
+			free(b.data);
+			return -1;
+		}
+		o->type = t;
+		o->all = b.data;
+		o->data = b.data + n;
+		o->size = b.len - n;
+		break;
+	case GOdelta:
+		if(readodelta(f, o, l, p, flag) == -1)
+			return -1;
+		break;
+	case GRdelta:
+		if(readrdelta(f, o, l, flag) == -1)
+			return -1;
+		break;
+	}
+	o->flag |= Cloaded|flag;
+	return 0;
+}
+
+static int
+readloose(Biobuf *f, Object *o, int flag)
+{
+	struct { char *tag; int type; } *p, types[] = {
+		{"blob", GBlob},
+		{"tree", GTree},
+		{"commit", GCommit},
+		{"tag", GTag},
+		{nil},
+	};
+	char *d, *s, *e;
+	vlong sz, n;
+	int l;
+
+	n = decompress(&d, f, nil);
+	if(n == -1)
+		return -1;
+
+	s = d;
+	o->type = GNone;
+	for(p = types; p->tag; p++){
+		l = strlen(p->tag);
+		if(strncmp(s, p->tag, l) == 0){
+			s += l;
+			o->type = p->type;
+			while(!isspace(*s))
+				s++;
+			break;
+		}
+	}
+	if(o->type == GNone){
+		free(o->data);
+		return -1;
+	}
+	sz = strtol(s, &e, 0);
+	if(e == s || *e++ != 0){
+		werrstr("malformed object header");
+		goto error;
+	}
+	if(sz != n - (e - d)){
+		werrstr("mismatched sizes");
+		goto error;
+	}
+	o->size = sz;
+	o->data = e;
+	o->all = d;
+	o->flag |= Cloaded|flag;
+	return 0;
+
+error:
+	free(d);
+	return -1;
+}
+
+vlong
+searchindex(Biobuf *f, Hash h)
+{
+	int lo, hi, idx, i, nent;
+	vlong o, oo;
+	Hash hh;
+
+	o = 8;
+	/*
+	 * Read the fanout table. The fanout table
+	 * contains 256 entries, corresponsding to
+	 * the first byte of the hash. Each entry
+	 * is a 4 byte big endian integer, containing
+	 * the total number of entries with a leading
+	 * byte <= the table index, allowing us to
+	 * rapidly do a binary search on them.
+	 */
+	if (h.h[0] == 0){
+		lo = 0;
+		if(preadbe32(f, &hi, o) == -1)
+			goto err;
+	} else {
+		o += h.h[0]*4 - 4;
+		if(preadbe32(f, &lo, o + 0) == -1)
+			goto err;
+		if(preadbe32(f, &hi, o + 4) == -1)
+			goto err;
+	}
+	if(hi == lo)
+		goto notfound;
+	if(preadbe32(f, &nent, 8 + 255*4) == -1)
+		goto err;
+
+	/*
+	 * Now that we know the range of hashes that the
+	 * entry may exist in, read them in so we can do
+	 * a bsearch.
+	 */
+	idx = -1;
+	Bseek(f, Hashsz*lo + 8 + 256*4, 0);
+	for(i = 0; i < hi - lo; i++){
+		if(Bread(f, hh.h, sizeof(hh.h)) == -1)
+			goto err;
+		if(hasheq(&hh, &h))
+			idx = lo + i;
+	}
+	if(idx == -1)
+		goto notfound;
+
+
+	/*
+	 * We found the entry. If it's 32 bits, then we
+	 * can just return the oset, otherwise the 32
+	 * bit entry contains the oset to the 64 bit
+	 * entry.
+	 */
+	oo = 8;			/* Header */
+	oo += 256*4;		/* Fanout table */
+	oo += Hashsz*nent;	/* Hashes */
+	oo += 4*nent;		/* Checksums */
+	oo += 4*idx;		/* Offset offset */
+	if(preadbe32(f, &i, oo) == -1)
+		goto err;
+	o = i & 0xffffffff;
+	if(o & (1ull << 31)){
+		o &= 0x7fffffff;
+		if(preadbe64(f, &o, o) == -1)
+			goto err;
+	}
+	return o;
+
+err:
+	fprint(2, "unable to read packfile: %r\n");
+	return -1;
+notfound:
+	werrstr("not present: %H", h);
+	return -1;		
+}
+
+/*
+ * Scans for non-empty word, copying it into buf.
+ * Strips off word, leading, and trailing space
+ * from input.
+ * 
+ * Returns -1 on empty string or error, leaving
+ * input unmodified.
+ */
+static int
+scanword(char **str, int *nstr, char *buf, int nbuf)
+{
+	char *p;
+	int n, r;
+
+	r = -1;
+	p = *str;
+	n = *nstr;
+	while(n && isblank(*p)){
+		n--;
+		p++;
+	}
+
+	for(; n && *p && !isspace(*p); p++, n--){
+		r = 0;
+		*buf++ = *p;
+		nbuf--;
+		if(nbuf == 0)
+			return -1;
+	}
+	while(n && isblank(*p)){
+		n--;
+		p++;
+	}
+	*buf = 0;
+	*str = p;
+	*nstr = n;
+	return r;
+}
+
+static void
+nextline(char **str, int *nstr)
+{
+	char *s;
+
+	if((s = strchr(*str, '\n')) != nil){
+		*nstr -= s - *str + 1;
+		*str = s + 1;
+	}
+}
+
+static int
+parseauthor(char **str, int *nstr, char **name, vlong *time)
+{
+	char buf[128];
+	Resub m[4];
+	char *p;
+	int n, nm;
+
+	if((p = strchr(*str, '\n')) == nil)
+		sysfatal("malformed author line");
+	n = p - *str;
+	if(n >= sizeof(buf))
+		sysfatal("overlong author line");
+	memset(m, 0, sizeof(m));
+	snprint(buf, n + 1, *str);
+	*str = p;
+	*nstr -= n;
+	
+	if(!regexec(authorpat, buf, m, nelem(m)))
+		sysfatal("invalid author line %s", buf);
+	nm = m[1].ep - m[1].sp;
+	*name = emalloc(nm + 1);
+	memcpy(*name, m[1].sp, nm);
+	buf[nm] = 0;
+	
+	nm = m[2].ep - m[2].sp;
+	memcpy(buf, m[2].sp, nm);
+	buf[nm] = 0;
+	*time = atoll(buf);
+	return 0;
+}
+
+static void
+parsecommit(Object *o)
+{
+	char *p, *t, buf[128];
+	int np;
+
+	p = o->data;
+	np = o->size;
+	o->commit = emalloc(sizeof(Cinfo));
+	while(1){
+		if(scanword(&p, &np, buf, sizeof(buf)) == -1)
+			break;
+		if(strcmp(buf, "tree") == 0){
+			if(scanword(&p, &np, buf, sizeof(buf)) == -1)
+				sysfatal("invalid commit: tree missing");
+			if(hparse(&o->commit->tree, buf) == -1)
+				sysfatal("invalid commit: garbled tree");
+		}else if(strcmp(buf, "parent") == 0){
+			if(scanword(&p, &np, buf, sizeof(buf)) == -1)
+				sysfatal("invalid commit: missing parent");
+			o->commit->parent = realloc(o->commit->parent, ++o->commit->nparent * sizeof(Hash));
+			if(!o->commit->parent)
+				sysfatal("unable to malloc: %r");
+			if(hparse(&o->commit->parent[o->commit->nparent - 1], buf) == -1)
+				sysfatal("invalid commit: garbled parent");
+		}else if(strcmp(buf, "author") == 0){
+			parseauthor(&p, &np, &o->commit->author, &o->commit->mtime);
+		}else if(strcmp(buf, "committer") == 0){
+			parseauthor(&p, &np, &o->commit->committer, &o->commit->ctime);
+		}else if(strcmp(buf, "gpgsig") == 0){
+			/* just drop it */
+			if((t = strstr(p, "-----END PGP SIGNATURE-----")) == nil)
+				sysfatal("malformed gpg signature");
+			np -= t - p;
+			p = t;
+		}
+		nextline(&p, &np);
+	}
+	while (np && isspace(*p)) {
+		p++;
+		np--;
+	}
+	o->commit->msg = p;
+	o->commit->nmsg = np;
+}
+
+static void
+parsetree(Object *o)
+{
+	char *p, buf[256];
+	int np, nn, m;
+	Dirent *t;
+
+	p = o->data;
+	np = o->size;
+	o->tree = emalloc(sizeof(Tinfo));
+	while(np > 0){
+		if(scanword(&p, &np, buf, sizeof(buf)) == -1)
+			break;
+		o->tree->ent = erealloc(o->tree->ent, ++o->tree->nent * sizeof(Dirent));
+		t = &o->tree->ent[o->tree->nent - 1];
+		memset(t, 0, sizeof(Dirent));
+		m = strtol(buf, nil, 8);
+		/* FIXME: symlinks and other BS */
+		if(m == 0160000){
+			print("setting mode to link...\n");
+			t->mode |= DMDIR;
+			t->modref = 1;
+		}
+		t->mode = m & 0777;
+		if(m & 0040000)
+			t->mode |= DMDIR;
+		t->name = p;
+		nn = strlen(p) + 1;
+		p += nn;
+		np -= nn;
+		if(np < sizeof(t->h.h))
+			sysfatal("malformed tree %H, remaining %d (%s)", o->hash, np, p);
+		memcpy(t->h.h, p, sizeof(t->h.h));
+		p += sizeof(t->h.h);
+		np -= sizeof(t->h.h);
+	}
+}
+
+static void
+parsetag(Object *)
+{
+}
+
+void
+parseobject(Object *o)
+{
+	if(o->flag & Cparsed)
+		return;
+	switch(o->type){
+	case GTree:	parsetree(o);	break;
+	case GCommit:	parsecommit(o);	break;
+	case GTag:	parsetag(o);	break;
+	default:	break;
+	}
+	o->flag |= Cparsed;
+}
+
+static Object*
+readidxobject(Biobuf *idx, Hash h, int flag)
+{
+	char path[Pathmax];
+	char hbuf[41];
+	Biobuf *f;
+	Object *obj;
+	int l, i, n;
+	vlong o;
+	Dir *d;
+
+	USED(idx);
+	if((obj = osfind(&objcache, h)) != nil){
+		if(obj->flag & Cloaded)
+			return obj;
+		if(obj->flag & Cidx){
+			assert(idx != nil);
+			o = Boffset(idx);
+			if(Bseek(idx, obj->off, 0) == -1)
+				sysfatal("could not seek to object offset");
+			if(readpacked(idx, obj, flag) == -1)
+				sysfatal("could not reload object %H", obj->hash);
+			if(Bseek(idx, o, 0) == -1)
+				sysfatal("could not restore offset");
+			cache(obj);
+			return obj;
+		}
+	}
+
+	d = nil;
+	obj = emalloc(sizeof(Object));
+	obj->id = objcache.nobj + 1;
+	obj->hash = h;
+
+	snprint(hbuf, sizeof(hbuf), "%H", h);
+	snprint(path, sizeof(path), ".git/objects/%c%c/%s", hbuf[0], hbuf[1], hbuf + 2);
+	if((f = Bopen(path, OREAD)) != nil){
+		if(readloose(f, obj, flag) == -1)
+			goto error;
+		Bterm(f);
+		parseobject(obj);
+		cache(obj);
+		return obj;
+	}
+
+	if ((n = slurpdir(".git/objects/pack", &d)) == -1)
+		goto error;
+	o = -1;
+	for(i = 0; i < n; i++){
+		l = strlen(d[i].name);
+		if(l > 4 && strcmp(d[i].name + l - 4, ".idx") != 0)
+			continue;
+		snprint(path, sizeof(path), ".git/objects/pack/%s", d[i].name);
+		if((f = Bopen(path, OREAD)) == nil)
+			continue;
+		o = searchindex(f, h);
+		Bterm(f);
+		if(o == -1)
+			continue;
+		break;
+	}
+
+	if (o == -1)
+		goto error;
+
+	if((n = snprint(path, sizeof(path), "%s", path)) >= sizeof(path) - 4)
+		goto error;
+	memcpy(path + n - 4, ".pack", 6);
+	if((f = Bopen(path, OREAD)) == nil)
+		goto error;
+	if(Bseek(f, o, 0) == -1)
+		goto error;
+	if(readpacked(f, obj, flag) == -1)
+		goto error;
+	Bterm(f);
+	parseobject(obj);
+	cache(obj);
+	return obj;
+error:
+	free(d);
+	free(obj);
+	return nil;
+}
+
+Object*
+readobject(Hash h)
+{
+	Object *o;
+
+	o = readidxobject(nil, h, 0);
+	if(o)
+		ref(o);
+	return o;
+}
+
+int
+objcmp(void *pa, void *pb)
+{
+	Object *a, *b;
+
+	a = *(Object**)pa;
+	b = *(Object**)pb;
+	return memcmp(a->hash.h, b->hash.h, sizeof(a->hash.h));
+}
+
+static int
+hwrite(Biobuf *b, void *buf, int len, DigestState **st)
+{
+	*st = sha1(buf, len, nil, *st);
+	return Bwrite(b, buf, len);
+}
+
+int
+indexpack(char *pack, char *idx, Hash ph)
+{
+	char hdr[4*3], buf[8];
+	int nobj, nvalid, nbig, n, i, step;
+	Object *o, **objects;
+	DigestState *st;
+	char *valid;
+	Biobuf *f;
+	Hash h;
+	int c;
+
+	if((f = Bopen(pack, OREAD)) == nil)
+		return -1;
+	if(Bread(f, hdr, sizeof(hdr)) != sizeof(hdr)){
+		werrstr("short read on header");
+		return -1;
+	}
+	if(memcmp(hdr, "PACK\0\0\0\2", 8) != 0){
+		werrstr("invalid header");
+		return -1;
+	}
+
+	nvalid = 0;
+	nobj = GETBE32(hdr + 8);
+	objects = calloc(nobj, sizeof(Object*));
+	valid = calloc(nobj, sizeof(char));
+	step = nobj/100;
+	if(!step)
+		step++;
+	while(nvalid != nobj){
+		fprint(2, "indexing (%d/%d):", nvalid, nobj);
+		n = 0;
+		for(i = 0; i < nobj; i++){
+			if(valid[i]){
+				n++;
+				continue;
+			}
+			if(i % step == 0)
+				fprint(2, ".");
+			if(!objects[i]){
+				o = emalloc(sizeof(Object));
+				o->off = Boffset(f);
+				objects[i] = o;
+			}
+			o = objects[i];
+			Bseek(f, o->off, 0);
+			if (readpacked(f, o, Cidx) == 0){
+				sha1((uchar*)o->all, o->size + strlen(o->all) + 1, o->hash.h, nil);
+				cache(o);
+				valid[i] = 1;
+				n++;
+			}
+		}
+		fprint(2, "\n");
+		if(n == nvalid){
+			sysfatal("fix point reached too early: %d/%d: %r", nvalid, nobj);
+			goto error;
+		}
+		nvalid = n;
+	}
+	Bterm(f);
+
+	st = nil;
+	qsort(objects, nobj, sizeof(Object*), objcmp);
+	if((f = Bopen(idx, OWRITE)) == nil)
+		return -1;
+	if(Bwrite(f, "\xfftOc\x00\x00\x00\x02", 8) != 8)
+		goto error;
+	/* fanout table */
+	c = 0;
+	for(i = 0; i < 256; i++){
+		while(c < nobj && (objects[c]->hash.h[0] & 0xff) <= i)
+			c++;
+		PUTBE32(buf, c);
+		hwrite(f, buf, 4, &st);
+	}
+	for(i = 0; i < nobj; i++){
+		o = objects[i];
+		hwrite(f, o->hash.h, sizeof(o->hash.h), &st);
+	}
+
+	/* fuck it, pointless */
+	for(i = 0; i < nobj; i++){
+		PUTBE32(buf, 42);
+		hwrite(f, buf, 4, &st);
+	}
+
+	nbig = 0;
+	for(i = 0; i < nobj; i++){
+		if(objects[i]->off <= (1ull<<31))
+			PUTBE32(buf, objects[i]->off);
+		else
+			PUTBE32(buf, (1ull << 31) | nbig++);
+		hwrite(f, buf, 4, &st);
+	}
+	for(i = 0; i < nobj; i++){
+		if(objects[i]->off > (1ull<<31)){
+			PUTBE64(buf, objects[i]->off);
+			hwrite(f, buf, 8, &st);
+		}
+	}
+	hwrite(f, ph.h, sizeof(ph.h), &st);
+	sha1(nil, 0, h.h, st);
+	Bwrite(f, h.h, sizeof(h.h));
+
+	free(objects);
+	free(valid);
+	Bterm(f);
+	return 0;
+
+error:
+	free(objects);
+	free(valid);
+	Bterm(f);
+	return -1;
+}
--- /dev/null
+++ b/proto.c
@@ -1,0 +1,161 @@
+#include <u.h>
+#include <libc.h>
+
+#include "git.h"
+
+int chattygit;
+
+int
+readpkt(int fd, char *buf, int nbuf)
+{
+	char len[5];
+	char *e;
+	int n;
+
+	if(readn(fd, len, 4) == -1)
+		return -1;
+	len[4] = 0;
+	n = strtol(len, &e, 16);
+	if(n == 0){
+		if(chattygit)
+			fprint(2, "readpkt: 0000\n");
+		return 0;
+	}
+	if(e != len + 4 || n <= 4)
+		sysfatal("invalid packet line length");
+	n  -= 4;
+	if(n >= nbuf)
+		sysfatal("buffer too small");
+	if(readn(fd, buf, n) != n)
+		return -1;
+	buf[n] = 0;
+	if(chattygit)
+		fprint(2, "readpkt: %s:\t%.*s\n", len, nbuf, buf);
+	return n;
+}
+
+int
+writepkt(int fd, char *buf, int nbuf)
+{
+	char len[5];
+
+
+	snprint(len, sizeof(len), "%04x", nbuf + 4);
+	if(write(fd, len, 4) != 4)
+		return -1;
+	if(write(fd, buf, nbuf) != nbuf)
+		return -1;
+	if(chattygit){
+		fprint(2, "writepkt: %s:\t", len);
+		write(2, buf, nbuf);
+		write(2, "\n", 1);
+	}
+	return 0;
+}
+
+int
+flushpkt(int fd)
+{
+	if(chattygit)
+		fprint(2, "writepkt: 0000\n");
+	return write(fd, "0000", 4);
+}
+
+static void
+grab(char *dst, int n, char *p, char *e)
+{
+	int l;
+
+	l = e - p;
+	if(l >= n)
+		sysfatal("overlong component");
+	memcpy(dst, p, l);
+	dst[l + 1] = 0;
+
+}
+
+int
+parseuri(char *uri, char *proto, char *host, char *port, char *path, char *repo)
+{
+	char *s, *p, *q;
+	int n;
+
+	p = strstr(uri, "://");
+	if(!p){
+		werrstr("missing protocol");
+		return -1;
+	}
+	grab(proto, Nproto, uri, p);
+	s = p + 3;
+
+	p = strstr(s, "/");
+	if(!p || strlen(p) == 1){
+		werrstr("missing path");
+		return -1;
+	}
+	q = memchr(s, ':', p - s);
+	if(q){
+		grab(host, Nhost, s, q);
+		grab(port, Nport, q + 1, p);
+	}else{
+		grab(host, Nhost, s, p);
+		snprint(port, Nport, "9418");
+	}
+	
+	snprint(path, Npath, "%s", p);
+	p = strrchr(p, '/') + 1;
+	if(!p || strlen(p) == 0){
+		werrstr("missing repository in uri");
+		return -1;
+	}
+	n = strlen(p);
+	if(hassuffix(p, ".git"))
+		n -= 4;
+	grab(repo, Nrepo, p, p + n);
+	return 0;
+}
+
+int
+dialssh(char *host, char *, char *path, char *direction)
+{
+	int pid, pfd[2];
+	char cmd[64];
+
+	print("dialing via ssh %s...\n", host);
+	if(pipe(pfd) == -1)
+		sysfatal("unable to open pipe: %r");
+	pid = fork();
+	if(pid == -1)
+		sysfatal("unable to fork");
+	if(pid == 0){
+		close(pfd[1]);
+		dup(pfd[0], 0);
+		dup(pfd[0], 1);
+		snprint(cmd, sizeof(cmd), "git-%s-pack", direction);
+		execl("/bin/ssh", "ssh", host, cmd, path, nil);
+	}else{
+		close(pfd[0]);
+		return pfd[1];
+	}
+	return -1;
+}
+
+int
+dialgit(char *host, char *port, char *path, char *direction)
+{
+	char *ds, cmd[128];
+	int fd, l;
+
+	ds = netmkaddr(host, "tcp", port);
+	print("dialing %s...\n", ds);
+	fd = dial(ds, nil, nil, nil);
+	if(fd == -1)
+		return -1;
+	l = snprint(cmd, sizeof(cmd), "git-%s-pack %s\n", direction, path);
+	if(writepkt(fd, cmd, l + 1) == -1){
+		print("failed to write message\n");
+		close(fd);
+		return -1;
+	}
+	return fd;
+}
--- /dev/null
+++ b/pull
@@ -1,0 +1,96 @@
+#!/bin/rc -e
+
+rfork en
+
+nl='
+'
+
+fn update{
+	update=$1
+	branch=$2
+	upstream=$3
+	url=$4
+	dir=$5
+	
+	{git/fetch -b $branch -u $upstream $url >[2=3] | awk -v 'update='^$update '
+		function writeref(ref, hash)
+		{
+			outfile = ".git/"ref
+			system("mkdir -p `{basename -d "outfile"}")
+			print hash > outfile
+			close(outfile)
+		}
+
+		/^remote/{
+			if($2=="HEAD")
+				next
+
+			if(update)
+				writeref($2, $3)
+			gsub("^refs/heads", "refs/remotes/'$remote'", $2)
+			writeref($2, $3)
+		}
+	'} |[3] tr '\x0d' '\x0a'
+}
+
+fn usage{
+	echo 'usage: $argv0 [-a] [-u upstream] [-b branch]' >[1=2]
+	echo '	-u up:	pull from upstream "up" (default: origin)' >[1=2]
+	echo '	-f:	fetch without updating working copy' >[1=2]
+	exit usage
+}
+
+git/fs
+branch=`{awk '$1=="branch"{print $2}' < /mnt/git/ctl}
+remote=()
+update='true'
+upstream=origin
+while(~ $1 -*){
+	switch($1){
+	case -u;
+		shift
+		remote=$1
+		upstream=SOMEONE
+	case -b:
+		shift
+		branch=$1
+	case -f;
+		update=''
+	case *;
+		usage
+	}
+	shift
+}
+
+if(! ~ $#* 0)
+	usage
+if(~ $#remote 0)
+	remote=`{git/conf 'remote "'$upstream'".url'}
+if(~ $#remote 0){
+	echo 'no idea from where to pull'
+	exit upstream
+}
+
+if(! cd `{git/conf -r})
+	exit 'not in git repository'
+
+dir=/mnt/git/branch/$branch/tree
+if(! git/walk -q){
+	echo $status
+	echo 'repository is dirty: commit before pulling' >[1=2]
+	exit 'dirty'
+}
+oldfiles=`$nl{git/walk -cfT}
+update $update  $branch $upstream $remote $dir
+if(! ~ $update 0){
+	rm -f $oldfiles
+	tree=/mnt/git/HEAD/tree
+	@{builtin cd $tree && tar cif /fd/1 .} | @{tar xf /fd/0}
+	for(f in `$nl{walk -f $tree | sed 's@^'$tree'/*@@'}){
+		if(! ~ $#f 0){
+			idx=.git/index9/tracked/$f
+			mkdir -p `{basename -d $idx}
+			walk -eq $f > $idx
+		}
+	}
+}
--- /dev/null
+++ b/push
@@ -1,0 +1,53 @@
+#!/bin/rc
+
+rfork en
+
+if(! cd `{git/conf -r})
+	exit 'not in git repository'
+
+git/fs
+fn usage {
+	echo 'usage: git/push [-a] [-u upstream] [-b branch] [-r rmbranch]' >[1=2]
+	echo '	-a:		push all' >[1=2]
+	echo '	-u upstream:	push to repo "upstream" (default: origin)' >[1=2]
+	echo '	-b branch:	push branch "branch" (default: current branch)' >[1=2]
+	exit usage
+}
+
+remote=()
+sendall=''
+remove=()
+upstream='origin'
+branch=`{awk '$1=="branch"{print $2}' < /mnt/git/ctl}
+while(~ $1 -* && ! ~ $1 --){
+	switch($1){
+	case -u;
+		shift
+		upstream=$1
+	case -a;
+		sendall=true
+	case -b;
+		shift
+		branch=$1
+	case -r;
+		shift
+		remove=(-r$1 $remove);
+	case *; usage
+	}
+	shift
+}
+
+if(! ~ $#* 0)
+	usage
+if(~ $#remote 0)
+	remote=`{git/conf 'remote "'$upstream'".url'}
+if(~ $#remote 0)
+	remote=$upstream
+if(~ $#remote 0){
+	echo 'no idea where to push'
+	exit upstream
+}
+if(~ $sendall '')
+	git/send -b $branch  $remove $remote
+if not
+	git/send  $remove -a $remote
--- /dev/null
+++ b/query.c
@@ -1,0 +1,36 @@
+#include <u.h>
+#include <libc.h>
+
+#include "git.h"
+
+int fullpath;
+
+void
+usage(void)
+{
+	fprint(2, "usage: %s [-p]\n", argv0);
+	exits("usage");
+}
+
+void
+main(int argc, char **argv)
+{
+	int i, j, n;
+	Hash *h;
+
+	ARGBEGIN{
+	case 'p':	fullpath++;	break;
+	default:	usage();	break;
+	}ARGEND;
+
+	gitinit();
+	for(i = 0; i < argc; i++){
+		if((n = resolverefs(&h, argv[i])) == -1)
+			sysfatal("resolve %s: %r", argv[i]);
+		for(j = 0; j < n; j++)
+			if(fullpath)
+				print("/mnt/git/object/%H\n", h[j]);
+			else
+				print("%H\n", h[j]);
+	}
+}
--- /dev/null
+++ b/ref.c
@@ -1,0 +1,422 @@
+#include <u.h>
+#include <libc.h>
+#include <ctype.h>
+
+#include "git.h"
+
+typedef struct Eval	Eval;
+typedef struct XObject	XObject;
+
+struct Eval {
+	char	*str;
+	char	*p;
+	Object	**stk;
+	int	nstk;
+	int	stksz;
+};
+
+struct XObject {
+	Object	*obj;
+	Object	*mark;
+	XObject	*queue;
+	XObject	*next;
+};
+
+
+void
+eatspace(Eval *ev)
+{
+	while(isspace(ev->p[0]))
+		ev->p++;
+}
+
+int
+objdatecmp(void *pa, void *pb)
+{
+	Object *a, *b;
+
+	a = *(Object**)pa;
+	b = *(Object**)pb;
+	assert(a->type == GCommit && b->type == GCommit);
+	if(a->commit->mtime == b->commit->mtime)
+		return 0;
+	else if(a->commit->mtime < b->commit->mtime)
+		return -1;
+	else
+		return 1;
+}
+
+void
+push(Eval *ev, Object *o)
+{
+	if(ev->nstk == ev->stksz){
+		ev->stksz = 2*ev->stksz + 1;
+		ev->stk = erealloc(ev->stk, ev->stksz*sizeof(Object*));
+	}
+	ev->stk[ev->nstk++] = o;
+}
+
+Object*
+pop(Eval *ev)
+{
+	if(ev->nstk == 0)
+		sysfatal("stack underflow");
+	return ev->stk[--ev->nstk];
+}
+
+Object*
+peek(Eval *ev)
+{
+	if(ev->nstk == 0)
+		sysfatal("stack underflow");
+	return ev->stk[ev->nstk - 1];
+}
+
+int
+word(Eval *ev, char *b, int nb)
+{
+	char *p, *e;
+	int n;
+
+	p = ev->p;
+	for(e = p; isalnum(*e) || *e == '/'; e++)
+		/* nothing */;
+	/* 1 for nul terminator */
+	n = e - p + 1;
+	if(n >= nb)
+		n = nb;
+	snprint(b, n, "%s", p);
+	ev->p = e;
+	return n > 0;
+}
+
+int
+take(Eval *ev, char *m)
+{
+	int l;
+
+	l = strlen(m);
+	if(strncmp(ev->p, m, l) != 0)
+		return 0;
+	ev->p += l;
+	return 1;
+}
+
+XObject*
+hnode(XObject *ht[], Object *o)
+{
+	XObject *h;
+	int	hh;
+
+	hh = o->hash.h[0] & 0xff;
+	for(h = ht[hh]; h; h = h->next)
+		if(hasheq(&o->hash, &h->obj->hash))
+			return h;
+
+	h = malloc(sizeof(*h));
+	h->obj = o;
+	h->mark = nil;
+	h->queue = nil;
+	h->next = ht[hh];
+	ht[hh] = h;
+	return h;
+}
+
+int
+ancestor(Eval *ev)
+{
+	Object *a, *b, *o, *p;
+	XObject *ht[256];
+	XObject *h, *q, *q1, *q2;
+	int i, r;
+
+	if(ev->nstk < 2){
+		werrstr("ancestor needs 2 objects");
+		return -1;
+	}
+	a = pop(ev);
+	b = pop(ev);
+	if(a == b){
+		push(ev, a);
+		return 0;
+	}
+
+	r = -1;
+	memset(ht, 0, sizeof(ht));
+	q1 = nil;
+
+	h = hnode(ht, a);
+	h->mark = a;
+	h->queue = q1;
+	q1 = h;
+
+	h = hnode(ht, b);
+	h->mark = b;
+	h->queue = q1;
+	q1 = h;
+
+	while(1){
+		q2 = nil;
+		while(q = q1){
+			q1 = q->queue;
+			q->queue = nil;
+			o = q->obj;
+			for(i = 0; i < o->commit->nparent; i++){
+				p = readobject(o->commit->parent[i]);
+				h = hnode(ht, p);
+				if(h->mark != nil){
+					if(h->mark != q->mark){
+						push(ev, h->obj);
+						r = 0;
+						goto done;
+					}
+				} else {
+					h->mark = q->mark;
+					h->queue = q2;
+					q2 = h;
+				}
+			}
+		}
+		if(q2 == nil){
+			werrstr("no common ancestor");
+			break;
+		}
+		q1 = q2;
+	}
+done:
+	for(i=0; i<nelem(ht); i++){
+		while(h = ht[i]){
+			ht[i] = h->next;
+			free(h);
+		}
+	}
+	return r;
+}
+
+int
+parent(Eval *ev)
+{
+	Object *o, *p;
+
+	o = pop(ev);
+	/* Special case: first commit has no parent. */
+	if(o->commit->nparent == 0 || (p = readobject(o->commit->parent[0])) == nil){
+		werrstr("no parent for %H", o->hash);
+		return -1;
+	}
+	push(ev, p);
+	return 0;
+}
+
+int
+unwind(Eval *ev, Object **obj, int *idx, int nobj, Object **p, Objset *set, int keep)
+{
+	int i;
+
+	for(i = nobj; i >= 0; i--){
+		idx[i]++;
+		if(keep && !oshas(set, obj[i])){
+			push(ev, obj[i]);
+			osadd(set, obj[i]);
+		}else{
+			osadd(set, obj[i]);
+		}
+		if(idx[i] < obj[i]->commit->nparent){
+			*p = obj[i];
+			return i;
+		}
+	}
+	return -1;
+}
+
+int
+range(Eval *ev)
+{
+	Object *a, *b, *p, **all;
+	int nall, *idx, mark;
+	Objset keep, skip;
+
+	b = pop(ev);
+	a = pop(ev);
+	if(a->type != GCommit || b->type != GCommit){
+		werrstr("non-commit object in range");
+		return -1;
+	}
+
+	p = b;
+	all = nil;
+	idx = nil;
+	nall = 0;
+	mark = ev->nstk;
+	osinit(&keep);
+	osinit(&skip);
+	while(1){
+		all = erealloc(all, (nall + 1)*sizeof(Object*));
+		idx = erealloc(idx, (nall + 1)*sizeof(int));
+		all[nall] = p;
+		idx[nall] = 0;
+		if(p == a)
+			if((nall = unwind(ev, all, idx, nall, &p, &keep, 1)) == -1)
+				break;
+		else if(p->commit->nparent == 0)
+			if((nall = unwind(ev, all, idx, nall, &p, &skip, 0)) == -1)
+				break;
+		else if(oshas(&keep, p))
+			if((nall = unwind(ev, all, idx, nall, &p, &keep, 1)) == -1)
+				break;
+		else if(oshas(&skip, p))
+			if((nall = unwind(ev, all, idx, nall, &p, &skip, 0)) == -1)
+				break;
+
+		if((p = readobject(p->commit->parent[idx[nall]])) == nil)
+			sysfatal("bad commit %H", p->commit->parent[idx[nall]]);
+		nall++;
+	}
+	free(all);
+	qsort(ev->stk + mark, ev->nstk - mark, sizeof(Object*), objdatecmp);
+	return 0;
+}
+
+int
+readref(Hash *h, char *ref)
+{
+	static char *try[] = {"", "refs/", "refs/heads/", "refs/remotes/", "refs/tags/", nil};
+	char buf[256], s[256], **pfx;
+	int r, f, n;
+
+	/* TODO: support hash prefixes */
+	if((r = hparse(h, ref)) != -1)
+		return r;
+	if(strcmp(ref, "HEAD") == 0){
+		snprint(buf, sizeof(buf), ".git/HEAD");
+		if((f = open(buf, OREAD)) == -1)
+			return -1;
+		if((n = readn(f, s, sizeof(s) - 1))== -1)
+			return -1;
+		s[n] = 0;
+		strip(s);
+		r = hparse(h, s);
+		goto found;
+	}
+	for(pfx = try; *pfx; pfx++){
+		snprint(buf, sizeof(buf), ".git/%s%s", *pfx, ref);
+		if((f = open(buf, OREAD)) == -1)
+			continue;
+		if((n = readn(f, s, sizeof(s) - 1)) == -1)
+			continue;
+		s[n] = 0;
+		strip(s);
+		r = hparse(h, s);
+		close(f);
+		goto found;
+	}
+	return -1;
+
+found:
+	if(r == -1 && strstr(s, "ref: ") == s)
+		r = readref(h, s + strlen("ref: "));
+	return r;
+}
+
+int
+evalpostfix(Eval *ev)
+{
+	char name[256];
+	Object *o;
+	Hash h;
+
+	eatspace(ev);
+	if(!word(ev, name, sizeof(name))){
+		werrstr("expected name in expression");
+		return -1;
+	}
+	if(readref(&h, name) == -1){
+		werrstr("could not resolve ref %s", name);
+		return -1;
+	}else if((o = readobject(h)) == nil){
+		werrstr("invalid ref %s (hash %H)", name, h);
+		return -1;
+	}
+	push(ev, o);
+
+	while(1){
+		eatspace(ev);
+		switch(ev->p[0]){
+		case '^':
+			ev->p++;
+			if(parent(ev) == -1)
+				return -1;
+			break;
+		case '@':
+			ev->p++;
+			if(ancestor(ev) == -1)
+				return -1;
+			break;
+		default:
+			goto done;
+			break;
+		}	
+	}
+done:
+	return 0;
+}
+
+int
+evalexpr(Eval *ev, char *ref)
+{
+	memset(ev, 0, sizeof(*ev));
+	ev->str = ref;
+	ev->p = ref;
+
+	while(1){
+		if(evalpostfix(ev) == -1)
+			return -1;
+		if(ev->p[0] == '\0')
+			return 0;
+		else if(take(ev, ":") || take(ev, "..")){
+			if(evalpostfix(ev) == -1)
+				return -1;
+			if(ev->p[0] != '\0'){
+				werrstr("junk at end of expression");
+				return -1;
+			}
+			return range(ev);
+		}
+	}
+}
+
+int
+resolverefs(Hash **r, char *ref)
+{
+	Eval ev;
+	Hash *h;
+	int i;
+
+	if(evalexpr(&ev, ref) == -1){
+		free(ev.stk);
+		return -1;
+	}
+	h = emalloc(ev.nstk*sizeof(Hash));
+	for(i = 0; i < ev.nstk; i++)
+		h[i] = ev.stk[i]->hash;
+	*r = h;
+	return ev.nstk;
+}
+
+int
+resolveref(Hash *r, char *ref)
+{
+	Eval ev;
+
+	if(evalexpr(&ev, ref) == -1){
+		free(ev.stk);
+		return -1;
+	}
+	if(ev.nstk != 1){
+		werrstr("ambiguous ref expr");
+		free(ev.stk);
+		return -1;
+	}
+	*r = ev.stk[0]->hash;
+	return 0;
+}
--- /dev/null
+++ b/save.c
@@ -1,0 +1,273 @@
+#include <u.h>
+#include <libc.h>
+#include "git.h"
+
+typedef struct Objbuf Objbuf;
+struct Objbuf {
+	int off;
+	char *hdr;
+	int nhdr;
+	char *dat;
+	int ndat;
+};
+enum {
+	Maxparents = 16,
+};
+
+static int
+bwrite(void *p, void *buf, int nbuf)
+{
+	return Bwrite(p, buf, nbuf);
+}
+
+static int
+objbytes(void *p, void *buf, int nbuf)
+{
+	Objbuf *b;
+	int r, n, o;
+	char *s;
+
+	b = p;
+	n = 0;
+	if(b->off < b->nhdr){
+		r = b->nhdr - b->off;
+		r = (nbuf < r) ? nbuf : r;
+		memcpy(buf, b->hdr, r);
+		b->off += r;
+		nbuf -= r;
+		n += r;
+	}
+	if(b->off < b->ndat + b->nhdr){
+		s = buf;
+		o = b->off - b->nhdr;
+		r = b->ndat - o;
+		r = (nbuf < r) ? nbuf : r;
+		memcpy(s + n, b->dat + o, r);
+		b->off += r;
+		n += r;
+	}
+	return n;
+}
+
+void
+writeobj(Hash *h, char *hdr, int nhdr, char *dat, int ndat)
+{
+	Objbuf b = {.off=0, .hdr=hdr, .nhdr=nhdr, .dat=dat, .ndat=ndat};
+	char s[64], o[256];
+	SHA1state *st;
+	Biobuf *f;
+	int fd;
+
+	st = sha1((uchar*)hdr, nhdr, nil, nil);
+	st = sha1((uchar*)dat, ndat, nil, st);
+	sha1(nil, 0, h->h, st);
+	snprint(s, sizeof(s), "%H", *h);
+	fd = create(".git/objects", OREAD, DMDIR|0755);
+	close(fd);
+	snprint(o, sizeof(o), ".git/objects/%c%c", s[0], s[1]);
+	fd = create(o, OREAD, DMDIR | 0755);
+	close(fd);
+	snprint(o, sizeof(o), ".git/objects/%c%c/%s", s[0], s[1], s + 2);
+	if(readobject(*h) == nil){
+		if((f = Bopen(o, OWRITE)) == nil)
+			sysfatal("could not open %s: %r", o);
+		if(deflatezlib(f, bwrite, &b, objbytes, 9, 0) == -1)
+			sysfatal("could not write %s: %r", o);
+		Bterm(f);
+	}
+}
+
+int
+gitmode(int m)
+{
+	return (m & 0777) | ((m & DMDIR) ? 0040000 : 0100000);
+}
+
+void
+blobify(char *path, vlong size, Hash *bh)
+{
+	char h[64], *d;
+	int f, nh;
+
+	nh = snprint(h, sizeof(h), "%T %lld", GBlob, size) + 1;
+	if((f = open(path, OREAD)) == -1)
+		sysfatal("could not open %s: %r", path);
+	d = emalloc(size);
+	if(readn(f, d, size) != size)
+		sysfatal("could not read blob %s: %r", path);
+	writeobj(bh, h, nh, d, size);
+	close(f);
+	free(d);
+}
+
+int
+tracked(char *path)
+{
+	Dir *d;
+	char ipath[256];
+
+	/* Explicitly removed. */
+	snprint(ipath, sizeof(ipath), ".git/index9/removed/%s", path);
+	if(strstr(cleanname(ipath), ".git/index9/removed") != ipath)
+		sysfatal("path %s leaves index", ipath);
+	d = dirstat(ipath);
+	if(d != nil && d->qid.type != QTDIR){
+		free(d);
+		return 0;
+	}
+
+	/* Explicitly added. */
+	snprint(ipath, sizeof(ipath), ".git/index9/tracked/%s", path);
+	if(strstr(cleanname(ipath), ".git/index9/tracked") != ipath)
+		sysfatal("path %s leaves index", ipath);
+	if(access(ipath, AEXIST) == 0)
+		return 1;
+
+	return 0;
+}
+
+int
+dircmp(void *pa, void *pb)
+{
+	char aname[256], bname[256], c;
+	Dir *a, *b;
+
+	a = pa;
+	b = pb;
+	/*
+	 * If the files have the same name, they're equal.
+	 * Otherwise, If they're trees, they sort as thoug
+	 * there was a trailing slash.
+	 *
+	 * Wat.
+	 */
+	if(strcmp(a->name, b->name) == 0){
+		snprint(aname, sizeof(aname), "%s", a->name);
+		snprint(bname, sizeof(bname), "%s", b->name);
+	}else{
+		c = (a->qid.type & QTDIR) ? '/' : 0;
+		snprint(aname, sizeof(aname), "%s%c", a->name, c);
+		c = (b->qid.type & QTDIR) ? '/' : 0;
+		snprint(bname, sizeof(bname), "%s%c", b->name, c);
+	}
+
+	return strcmp(aname, bname);
+}
+
+int
+treeify(char *path, Hash *th)
+{
+	char *t, h[64], l[256], ep[256];
+	int nd, nl, nt, nh, i, s;
+	Hash eh;
+	Dir *d;
+		
+	if((nd = slurpdir(path, &d)) == -1)
+		sysfatal("could not read %s", path);
+	if(nd == 0)
+		return 0;
+
+	t = nil;
+	nt = 0;
+	qsort(d, nd, sizeof(Dir), dircmp);
+	for(i = 0; i < nd; i++){
+		snprint(ep, sizeof(ep), "%s/%s", path, d[i].name);
+		if(strcmp(d[i].name, ".git") == 0)
+			continue;
+		if(!tracked(ep))
+			continue;
+		if((d[i].qid.type & QTDIR) == 0)
+			blobify(ep, d[i].length, &eh);
+		else if(treeify(ep, &eh) == 0)
+			continue;
+
+		nl = snprint(l, sizeof(l), "%o %s", gitmode(d[i].mode), d[i].name);
+		s = nt + nl + sizeof(eh.h) + 1;
+		t = realloc(t, s);
+		memcpy(t + nt, l, nl + 1);
+		memcpy(t + nt + nl + 1, eh.h, sizeof(eh.h));
+		nt = s;
+	}
+	free(d);
+	nh = snprint(h, sizeof(h), "%T %d", GTree, nt) + 1;
+	if(nh >= sizeof(h))
+		sysfatal("overlong header");
+	writeobj(th, h, nh, t, nt);
+	free(t);
+	return nd;
+}
+
+
+void
+mkcommit(Hash *c, char *msg, char *name, char *email, Hash *parents, int nparents, Hash tree)
+{
+	char *s, h[64];
+	int ns, nh, i;
+	Fmt f;
+
+	fmtstrinit(&f);
+	fmtprint(&f, "tree %H\n", tree);
+	for(i = 0; i < nparents; i++)
+		fmtprint(&f, "parent %H\n", parents[i]);
+	fmtprint(&f, "author %s <%s> %lld +0000\n", name, email, (vlong)time(nil));
+	fmtprint(&f, "committer %s <%s> %lld +0000\n", name, email, (vlong)time(nil));
+	fmtprint(&f, "\n");
+	fmtprint(&f, "%s", msg);
+	s = fmtstrflush(&f);
+
+	ns = strlen(s);
+	nh = snprint(h, sizeof(h), "%T %d", GCommit, ns) + 1;
+	writeobj(c, h, nh, s, ns);
+	free(s);
+}
+
+void
+usage(void)
+{
+	fprint(2, "usage: git/commit -n name -e email -m message -d dir");
+	exits("usage");
+}
+
+void
+main(int argc, char **argv)
+{
+	Hash c, t, parents[Maxparents];
+	char *msg, *name, *email;
+	int r, nparents;
+
+
+	msg = nil;
+	name = nil;
+	email = nil;
+	nparents = 0;
+	gitinit();
+	ARGBEGIN{
+	case 'm':	msg = EARGF(usage());	break;
+	case 'n':	name = EARGF(usage());	break;
+	case 'e':	email = EARGF(usage());	break;
+	case 'p':
+		if(nparents >= Maxparents)
+			sysfatal("too many parents");
+		if(resolveref(&parents[nparents++], EARGF(usage())) == -1)
+			sysfatal("invalid parent: %r");
+		break;
+	}ARGEND;
+
+	if(!msg) sysfatal("missing message");
+	if(!name) sysfatal("missing name");
+	if(!email) sysfatal("missing email");
+	if(!msg || !name)
+		usage();
+
+	gitinit();
+	if(access(".git", AEXIST) != 0)
+		sysfatal("could not find git repo: %r");
+	r = treeify(".", &t);
+	if(r == -1)
+		sysfatal("could not commit: %r\n");
+	if(r == 0)
+		sysfatal("empty commit: aborting");
+	mkcommit(&c, msg, name, email, parents, nparents, t);
+	print("%H\n", c);
+	exits(nil);
+}
--- /dev/null
+++ b/send.c
@@ -1,0 +1,425 @@
+#include <u.h>
+#include <libc.h>
+#include <pool.h>
+
+#include "git.h"
+
+typedef struct Objq	Objq;
+typedef struct Buf	Buf;
+typedef struct Compout	Compout;
+typedef struct Update	Update;
+
+struct Buf {
+	int off;
+	int sz;
+	uchar *data;
+};
+
+struct Compout {
+	int fd;
+	DigestState *st;
+};
+
+struct Objq {
+	Objq *next;
+	Object *obj;
+};
+
+struct Update {
+	char	ref[128];
+	Hash	theirs;
+	Hash	ours;
+};
+
+int sendall;
+char *curbranch = "refs/heads/master";
+char *removed[128];
+int nremoved;
+
+static int
+hwrite(int fd, void *buf, int nbuf, DigestState **st)
+{
+	if(write(fd, buf, nbuf) != nbuf)
+		return -1;
+	*st = sha1(buf, nbuf, nil, *st);
+	return nbuf;
+}
+
+void
+pack(Objset *send, Objset *skip, Object *o)
+{
+	Dirent *e;
+	Object *s;
+	int i;
+
+	if(oshas(send, o) || oshas(skip, o))
+		return;
+	osadd(send, o);
+	switch(o->type){
+	case GCommit:
+		if((s = readobject(o->commit->tree)) == nil)
+			sysfatal("could not read tree %H: %r", o->hash);
+		pack(send, skip, s);
+		break;
+	case GTree:
+		for(i = 0; i < o->tree->nent; i++){
+			e = &o->tree->ent[i];
+			if(e->modref)
+				print("wtf, a link? %s\n", e->name);
+			if ((s = readobject(e->h)) == nil)
+				sysfatal("could not read entry %H: %r", e->h);
+			pack(send, skip, s);
+		}
+		break;
+	default:
+		break;
+	}
+}
+
+int
+compread(void *p, void *dst, int n)
+{
+	Buf *b;
+
+	b = p;
+	if(n > b->sz - b->off)
+		n = b->sz - b->off;
+	memcpy(dst, b->data + b->off, n);
+	b->off += n;
+	return n;
+}
+
+int
+compwrite(void *p, void *buf, int n)
+{
+	Compout *o;
+
+	o = p;
+	o->st = sha1(buf, n, nil, o->st);
+	return write(o->fd, buf, n);
+}
+
+int
+compress(int fd, void *buf, int sz, DigestState **st)
+{
+	int r;
+	Buf b ={
+		.off=0,
+		.data=buf,
+		.sz=sz,
+	};
+	Compout o = {
+		.fd = fd,
+		.st = *st,
+	};
+
+	r = deflatezlib(&o, compwrite, &b, compread, 6, 0);
+	*st = o.st;
+	return r;
+}
+
+int
+writeobject(int fd, Object *o, DigestState **st)
+{
+	char hdr[8];
+	uvlong sz;
+	int i;
+
+	i = 1;
+	sz = o->size;
+	hdr[0] = o->type << 4;
+	hdr[0] |= sz & 0xf;
+	if(sz >= (1 << 4)){
+		hdr[0] |= 0x80;
+		sz >>= 4;
+	
+		for(i = 1; i < sizeof(hdr); i++){
+			hdr[i] = sz & 0x7f;
+			if(sz <= 0x7f){
+				i++;
+				break;
+			}
+			hdr[i] |= 0x80;
+			sz >>= 7;
+		}
+	}
+
+	if(hwrite(fd, hdr, i, st) != i)
+		return -1;
+	if(compress(fd, o->data, o->size, st) == -1)
+		return -1;
+	return 0;
+}
+
+int
+writepack(int fd, Update *upd, int nupd)
+{
+	Objset send, skip;
+	Object *o, *p;
+	Objq *q, *n, *e;
+	DigestState *st;
+	Update *u;
+	char buf[4];
+	Hash h;
+	int i;
+
+	osinit(&send);
+	osinit(&skip);
+	for(i = 0; i < nupd; i++){
+		u = &upd[i];
+		if(hasheq(&u->theirs, &Zhash))
+			continue;
+		if((o = readobject(u->theirs)) == nil)
+			sysfatal("could not read %H", u->theirs);
+		pack(&skip, &skip, o);
+	}
+
+	q = nil;
+	e = nil;
+	for(i = 0; i < nupd; i++){
+		u = &upd[i];
+		if((o = readobject(u->ours)) == nil)
+			sysfatal("could not read object %H", u->ours);
+
+		n = emalloc(sizeof(Objq));
+		n->obj = o;
+		if(!q){
+			q = n;
+			e = n;
+		}else{
+			e->next = n;
+		}
+	}
+
+	for(n = q; n; n = n->next)
+		e = n;
+	for(; q; q = n){
+		o = q->obj;
+		if(oshas(&skip, o) || oshas(&send, o))
+			goto iter;
+		pack(&send, &skip, o);
+		for(i = 0; i < o->commit->nparent; i++){
+			if((p = readobject(o->commit->parent[i])) == nil)
+				sysfatal("could not read parent of %H", o->hash);
+			e->next = emalloc(sizeof(Objq));
+			e->next->obj = p;
+			e = e->next;
+		}
+iter:
+		n = q->next;
+		free(q);
+	}
+
+	st = nil;
+	PUTBE32(buf, send.nobj);
+	if(hwrite(fd, "PACK\0\0\0\02", 8, &st) != 8)
+		return -1;
+	if(hwrite(fd, buf, 4, &st) == -1)
+		return -1;
+	for(i = 0; i < send.sz; i++){
+		if(!send.obj[i])
+			continue;
+		if(writeobject(fd, send.obj[i], &st) == -1)
+			return -1;
+	}
+	sha1(nil, 0, h.h, st);
+	if(write(fd, h.h, sizeof(h.h)) == -1)
+		return -1;
+	return 0;
+}
+
+Update*
+findref(Update *u, int nu, char *ref)
+{
+	int i;
+
+	for(i = 0; i < nu; i++)
+		if(strcmp(u[i].ref, ref) == 0)
+			return &u[i];
+	return nil;
+}
+
+int
+readours(Update **ret)
+{
+	Update *u, *r;
+	Hash *h;
+	int nd, nu, i;
+	char *pfx;
+	Dir *d;
+
+	nu = 0;
+	if(!sendall){
+		u = emalloc((nremoved + 1)*sizeof(Update));
+		snprint(u[nu].ref, sizeof(u[nu].ref), "%s", curbranch);
+		if(resolveref(&u[nu].ours, curbranch) == -1)
+			sysfatal("broken branch %s", curbranch);
+		nu++;
+	}else{
+		if((nd = slurpdir(".git/refs/heads", &d)) == -1)
+			sysfatal("read branches: %r");
+		u = emalloc((nremoved + nd)*sizeof(Update));
+		for(i = 0; i < nd; i++){
+			snprint(u->ref, sizeof(u->ref), "refs/heads/%s", d[nu].name);
+			if(resolveref(&u[nu].ours, u[nu].ref) == -1)
+				continue;
+			nu++;
+		}
+	}
+	for(i = 0; i < nremoved; i++){
+		pfx = "refs/heads/";
+		if(strstr(removed[i], "heads/") == removed[i])
+			pfx = "refs/";
+		if(strstr(removed[i], "refs/heads/") == removed[i])
+			pfx = "";
+		snprint(u[nu].ref, sizeof(u[nu].ref), "%s%s", pfx, removed[i]);
+		h = &u[nu].ours;
+		if((r = findref(u, nu, u[nu].ref)) != nil)
+			h = &r->ours;
+		else
+			nu++;
+		memcpy(h, &Zhash, sizeof(Hash));
+	}
+
+	*ret = u;
+	return nu;	
+}
+
+int
+sendpack(int fd)
+{
+	char buf[65536];
+	char *sp[3], *p;
+	Update *upd, *u;
+	int i, n, nupd, nsp, updating;
+
+	if((nupd = readours(&upd)) == -1)
+		sysfatal("read refs: %r");
+	while(1){
+		n = readpkt(fd, buf, sizeof(buf));
+		if(n == -1)
+			return -1;
+		if(n == 0)
+			break;
+		if(strncmp(buf, "ERR ", 4) == 0)
+			sysfatal("%s", buf + 4);
+
+		if(getfields(buf, sp, nelem(sp), 1, " \t\n\r") != 2)
+			sysfatal("invalid ref line %.*s", utfnlen(buf, n), buf);
+		if((u = findref(upd, nupd, sp[1])) == nil)
+			continue;
+		if(hparse(&u->theirs, sp[0]) == -1)
+			sysfatal("invalid hash %s", sp[0]);
+		snprint(u->ref, sizeof(u->ref), sp[1]);
+	}
+
+	updating = 0;
+	for(i = 0; i < nupd; i++){
+		u = &upd[i];
+		if(!hasheq(&u->theirs, &Zhash) && readobject(u->theirs) == nil){
+			fprint(2, "remote has diverged: pull and try again\n");
+			updating = 0;
+			break;
+		}
+		if(hasheq(&u->ours, &Zhash)){
+			print("%s: deleting\n", u->ref);
+			continue;
+		}
+		if(hasheq(&u->theirs, &u->ours)){
+			print("%s: up to date\n", u->ref);
+			continue;
+		}
+		print("%s: %H => %H\n", u->ref, u->theirs, u->ours);
+		n = snprint(buf, sizeof(buf), "%H %H %s", u->theirs, u->ours, u->ref);
+
+		/*
+		 * Workaround for github.
+		 *
+		 * Github will accept the pack but fail to update the references
+		 * if we don't have capabilities advertised. Report-status seems
+		 * harmless to add, so we add it.
+		 *
+		 * Github doesn't advertise any capabilities, so we can't check
+		 * for compatibility. We just need to add it blindly.
+		 */
+		if(i == 0){
+			buf[n++] = '\0';
+			n += snprint(buf + n, sizeof(buf) - n, " report-status");
+		}
+		if(writepkt(fd, buf, n) == -1)
+			sysfatal("unable to send update pkt");
+		updating = 1;
+	}
+	flushpkt(fd);
+	if(updating){
+		if(writepack(fd, upd, nupd) == -1)
+			return -1;
+
+		/* We asked for a status report, may as well use it. */
+		while((n = readpkt(fd, buf, sizeof(buf))) > 0){
+ 			buf[n] = 0;
+			nsp = getfields(buf, sp, nelem(sp), 1, " \t\n\r");
+			if(nsp < 2) 
+				continue;
+			if(nsp < 3)
+				sp[2] = "";
+			if(strcmp(sp[0], "unpack") == 0 && strcmp(sp[1], "ok") != 0)
+				fprint(2, "unpack %s\n", sp[1]);
+			else if(strcmp(sp[0], "ok") == 0)
+				fprint(2, "%s: updated\n", sp[1]);
+			else if(strcmp(sp[0], "ng") == 0)
+				fprint(2, "failed update: %s\n", sp[1]);
+		}
+	}
+	return 0;
+}
+
+void
+usage(void)
+{
+	fprint(2, "usage: %s remote [reponame]\n", argv0);
+	exits("usage");
+}
+
+void
+main(int argc, char **argv)
+{
+	char proto[Nproto], host[Nhost], port[Nport];
+	char repo[Nrepo], path[Npath];
+	int fd;
+
+	ARGBEGIN{
+	default:	usage();	break;
+	case 'a':	sendall++;	break;
+	case 'd':	chattygit++;	break;
+	case 'r':
+		if(nremoved == nelem(removed))
+			sysfatal("too many deleted branches");
+		removed[nremoved++] = EARGF(usage());
+		break;
+	case 'b':
+		curbranch = smprint("refs/%s", EARGF(usage()));
+		break;
+	}ARGEND;
+
+	gitinit();
+	if(argc != 1)
+		usage();
+	fd = -1;
+	if(parseuri(argv[0], proto, host, port, path, repo) == -1)
+		sysfatal("bad uri %s", argv[0]);
+	if(strcmp(proto, "ssh") == 0 || strcmp(proto, "git+ssh") == 0)
+		fd = dialssh(host, port, path, "receive");
+	else if(strcmp(proto, "git") == 0)
+		fd = dialgit(host, port, path, "receive");
+	else if(strcmp(proto, "http") == 0 || strcmp(proto, "git+http") == 0)
+		sysfatal("http clone not implemented");
+	else
+		sysfatal("unknown protocol %s", proto);
+	
+	if(fd == -1)
+		sysfatal("could not dial %s:%s: %r", proto, host);
+	if(sendpack(fd) == -1)
+		sysfatal("send failed: %r");
+	exits(nil);
+}
--- /dev/null
+++ b/template/HEAD
@@ -1,0 +1,1 @@
+ref: refs/heads/master
--- /dev/null
+++ b/template/config
@@ -1,0 +1,5 @@
+[core]
+	repositoryformatversion = 0
+	filemode = true
+	bare = false
+	logallrefupdates = true
--- /dev/null
+++ b/template/description
@@ -1,0 +1,1 @@
+Unnamed repository; edit this file 'description' to name the repository.
--- /dev/null
+++ b/template/info/exclude
@@ -1,0 +1,6 @@
+# git ls-files --others --exclude-from=.git/info/exclude
+# Lines that start with '#' are comments.
+# For a project mostly in C, the following would be a good set of
+# exclude patterns (uncomment them if you want to use them):
+# *.[oa]
+# *~
--- /dev/null
+++ b/util.c
@@ -1,0 +1,235 @@
+#include <u.h>
+#include <libc.h>
+#include <ctype.h>
+
+#include "git.h"
+
+Reprog *authorpat;
+Hash Zhash;
+
+int
+hasheq(Hash *a, Hash *b)
+{
+	return memcmp(a->h, b->h, sizeof(a->h)) == 0;
+}
+
+static int
+charval(int c, int *err)
+{
+	if(c >= '0' && c <= '9')
+		return c - '0';
+	if(c >= 'a' && c <= 'f')
+		return c - 'a' + 10;
+	if(c >= 'A' && c <= 'F')
+		return c - 'A' + 10;
+	*err = 1;
+	return -1;
+}
+
+void *
+emalloc(ulong n)
+{
+	void *v;
+	
+	v = mallocz(n, 1);
+	if(v == nil)
+		sysfatal("malloc: %r");
+	setmalloctag(v, getcallerpc(&n));
+	return v;
+}
+
+void *
+erealloc(void *p, ulong n)
+{
+	void *v;
+	
+	v = realloc(p, n);
+	if(v == nil)
+		sysfatal("realloc: %r");
+	setmalloctag(v, getcallerpc(&n));
+	return v;
+}
+
+char*
+estrdup(char *s)
+{
+	s = strdup(s);
+	if(s == nil)
+		sysfatal("strdup: %r");
+	setmalloctag(s, getcallerpc(&s));
+	return s;
+}
+
+int
+Hfmt(Fmt *fmt)
+{
+	Hash h;
+	int i, n, l;
+	char c0, c1;
+
+	l = 0;
+	h = va_arg(fmt->args, Hash);
+	for(i = 0; i < sizeof h.h; i++){
+		n = (h.h[i] >> 4) & 0xf;
+		c0 = (n >= 10) ? n-10 + 'a' : n + '0';
+		n = h.h[i] & 0xf;
+		c1 = (n >= 10) ? n-10 + 'a' : n + '0';
+		l += fmtprint(fmt, "%c%c", c0, c1);
+	}
+	return l;
+}
+
+int
+Tfmt(Fmt *fmt)
+{
+	Type t;
+	int l;
+
+	t = va_arg(fmt->args, Type);
+	switch(t){
+	case GNone:	l = fmtprint(fmt, "none");	break;
+	case GCommit:	l = fmtprint(fmt, "commit");	break;
+	case GTree:	l = fmtprint(fmt, "tree");	break;
+	case GBlob:	l = fmtprint(fmt, "blob");	break;
+	case GTag:	l = fmtprint(fmt, "tag");	break;
+	case GOdelta:	l = fmtprint(fmt, "odelta");	break;
+	case GRdelta:	l = fmtprint(fmt, "gdelta");	break;
+	default:	l = fmtprint(fmt, "?%d?", t);	break;
+	}
+	return l;
+}
+
+int
+Ofmt(Fmt *fmt)
+{
+	Object *o;
+	int l;
+
+	o = va_arg(fmt->args, Object *);
+	print("== %H (%T) ==\n", o->hash, o->type);
+	switch(o->type){
+	case GTree:
+		l = fmtprint(fmt, "tree\n");
+		break;
+	case GBlob:
+		l = fmtprint(fmt, "blob %s\n", o->data);
+		break;
+	case GCommit:
+		l = fmtprint(fmt, "commit\n");
+		break;
+	case GTag:
+		l = fmtprint(fmt, "tag\n");
+		break;
+	default:
+		l = fmtprint(fmt, "invalid: %d\n", o->type);
+		break;
+	}
+	return l;
+}
+
+int
+Qfmt(Fmt *fmt)
+{
+	Qid q;
+
+	q = va_arg(fmt->args, Qid);
+	return fmtprint(fmt, "Qid{path=0x%llx(dir:%d,obj:%lld), vers=%ld, type=%d}",
+	    q.path, QDIR(&q), (q.path >> 8), q.vers, q.type);
+}
+
+void
+gitinit(void)
+{
+	fmtinstall('H', Hfmt);
+	fmtinstall('T', Tfmt);
+	fmtinstall('O', Ofmt);
+	fmtinstall('Q', Qfmt);
+	inflateinit();
+	deflateinit();
+	authorpat = regcomp("[\t ]*(.*)[\t ]+([0-9]+)[\t ]+([\\-+]?[0-9]+)");
+	osinit(&objcache);
+}
+
+int
+hparse(Hash *h, char *b)
+{
+	int i, err;
+
+	err = 0;
+	for(i = 0; i < sizeof(h->h); i++){
+		err = 0;
+		h->h[i] = 0;
+		h->h[i] |= ((charval(b[2*i], &err) & 0xf) << 4);
+		h->h[i] |= ((charval(b[2*i+1], &err)& 0xf) << 0);
+		if(err){
+			werrstr("invalid hash");
+			return -1;
+		}
+	}
+	return 0;
+}
+
+int
+slurpdir(char *p, Dir **d)
+{
+	int r, f;
+
+	if((f = open(p, OREAD)) == -1)
+		return -1;
+	r = dirreadall(f, d);
+	close(f);
+	return r;
+}	
+
+int
+hassuffix(char *base, char *suf)
+{
+	int nb, ns;
+
+	nb = strlen(base);
+	ns = strlen(suf);
+	if(ns <= nb && strcmp(base + (nb - ns), suf) == 0)
+		return 1;
+	return 0;
+}
+
+int
+swapsuffix(char *dst, int dstsz, char *base, char *oldsuf, char *suf)
+{
+	int bl, ol, sl, l;
+
+	bl = strlen(base);
+	ol = strlen(oldsuf);
+	sl = strlen(suf);
+	l = bl + sl - ol;
+	if(l + 1 > dstsz || ol > bl)
+		return -1;
+	memmove(dst, base, bl - ol);
+	memmove(dst + bl - ol, suf, sl);
+	dst[l] = 0;
+	return l;
+}
+
+char *
+strip(char *s)
+{
+	char *e;
+
+	while(isspace(*s))
+		s++;
+	e = s + strlen(s);
+	while(e > s && isspace(*--e))
+		*e = 0;
+	return s;
+}
+
+void
+die(char *fmt, ...)
+{
+	va_list ap;
+
+	va_start(ap, fmt);
+	vfprint(2, fmt, ap);
+	va_end(ap);
+	abort();
+}
--- /dev/null
+++ b/walk.c
@@ -1,0 +1,295 @@
+#include <u.h>
+#include <libc.h>
+#include "git.h"
+
+#define NCACHE 256
+#define TDIR ".git/index9/tracked"
+#define RDIR ".git/index9/removed"
+#define HDIR "/mnt/git/HEAD/tree"
+typedef struct Cache	Cache;
+typedef struct Wres	Wres;
+struct Cache {
+	Dir*	cache;
+	int	n;
+	int	max;
+};
+
+struct Wres {
+	char	**path;
+	int	npath;
+	int	pathsz;
+};
+
+enum {
+	Rflg	= 1 << 0,
+	Mflg	= 1 << 1,
+	Aflg	= 1 << 2,
+	Tflg	= 1 << 3,
+};
+
+Cache seencache[NCACHE];
+int quiet;
+int printflg;
+char *rstr = "R ";
+char *tstr = "T ";
+char *mstr = "M ";
+char *astr = "A ";
+
+int
+seen(Dir *dir)
+{
+	Dir *dp;
+	int i;
+	Cache *c;
+
+	c = &seencache[dir->qid.path&(NCACHE-1)];
+	dp = c->cache;
+	for(i=0; i<c->n; i++, dp++)
+		if(dir->qid.path == dp->qid.path &&
+		   dir->type == dp->type &&
+		   dir->dev == dp->dev)
+			return 1;
+	if(c->n == c->max){
+		if (c->max == 0)
+			c->max = 8;
+		else
+			c->max += c->max/2;
+		c->cache = realloc(c->cache, c->max*sizeof(Dir));
+		if(c->cache == nil)
+			sysfatal("realloc: %r");
+	}
+	c->cache[c->n++] = *dir;
+	return 0;
+}
+
+int
+readpaths(Wres *r, char *pfx, char *dir)
+{
+	char *f, *sub, *full, *sep;
+	Dir *d;
+	int fd, ret, i, n;
+
+	ret = -1;
+	sep = "";
+	if(dir[0] != 0)
+		sep = "/";
+	if((full = smprint("%s/%s", pfx, dir)) == nil)
+		sysfatal("smprint: %r");
+	if((fd = open(full, OREAD)) < 0)
+		goto error;
+	while((n = dirread(fd, &d)) > 0){
+		for(i = 0; i < n; i++){
+			if(seen(&d[i]))
+				continue;
+			if(d[i].qid.type & QTDIR){
+				if((sub = smprint("%s%s%s", dir, sep, d[i].name)) == nil)
+					sysfatal("smprint: %r");
+				if(readpaths(r, pfx, sub) == -1){
+					free(sub);
+					goto error;
+				}
+				free(sub);
+			}else{
+				if(r->npath == r->pathsz){
+					r->pathsz = 2*r->pathsz + 1;
+					r->path = erealloc(r->path, r->pathsz * sizeof(char*));
+				}
+				if((f = smprint("%s%s%s", dir, sep, d[i].name)) == nil)
+					sysfatal("smprint: %r");
+				r->path[r->npath++] = f;
+			}
+		}
+	}
+	ret = r->npath;
+error:
+	close(fd);
+	free(full);
+	free(d);
+	return ret;
+}
+
+int
+cmp(void *pa, void *pb)
+{
+	return strcmp(*(char **)pa, *(char **)pb);
+}
+
+void
+dedup(Wres *r)
+{
+	int i, o;
+
+	if(r->npath <= 1)
+		return;
+	o = 0;
+	qsort(r->path, r->npath, sizeof(r->path[0]), cmp);
+	for(i = 1; i < r->npath; i++)
+		if(strcmp(r->path[o], r->path[i]) != 0)
+			r->path[++o] = r->path[i];
+	r->npath = o + 1;
+}
+
+static void
+findroot(void)
+{
+	char path[256], buf[256], *p;
+
+	if(access("/mnt/git/ctl", AEXIST) != 0)
+		sysfatal("no running git/fs");
+	if((getwd(path, sizeof(path))) == nil)
+		sysfatal("could not get wd: %r");
+	while((p = strrchr(path, '/')) != nil){
+		snprint(buf, sizeof(buf), "%s/.git", path);
+		if(access(buf, AEXIST) == 0){
+			chdir(path);
+			return;
+		}
+		*p = '\0';
+	}
+	sysfatal("not a git repository");
+}
+
+int
+sameqid(char *f, char *qf)
+{
+	char indexqid[64], fileqid[64], *p;
+	Dir *d;
+	int fd, n;
+
+	if((fd = open(qf, OREAD)) == -1)
+		return -1;
+	if((n = readn(fd, indexqid, sizeof(indexqid) - 1)) == -1)
+		return -1;
+	indexqid[n] = 0;
+	close(fd);
+	if((p = strpbrk(indexqid, "  \t\n\r")) != nil)
+		*p = 0;
+
+	if((d = dirstat(f)) == nil)
+		return -1;
+	snprint(fileqid, sizeof(fileqid), "%ullx.%uld.%.2uhhx",
+	    d->qid.path, d->qid.vers, d->qid.type);
+	if(strcmp(indexqid, fileqid) == 0)
+		return 1;
+	return 0;
+}
+
+int
+samedata(char *pa, char *pb)
+{
+	char ba[32*1024], bb[32*1024];
+	int fa, fb, na, nb, same;
+
+	same = 0;
+	fa = open(pa, OREAD);
+	fb = open(pb, OREAD);
+	if(fa == -1 || fb == -1){
+		goto mismatch;
+	}
+	while(1){
+		if((na = readn(fa, ba, sizeof(ba))) == -1)
+			goto mismatch;
+		if((nb = readn(fb, bb, sizeof(bb))) == -1)
+			goto mismatch;
+		if(na != nb)
+			goto mismatch;
+		if(na == 0)
+			break;
+		if(memcmp(ba, bb, na) != 0)
+			goto mismatch;
+	}
+	same = 1;
+mismatch:
+	if(fa != -1)
+		close(fa);
+	if(fb != -1)
+		close(fb);
+	return same;
+}
+
+void
+usage(void)
+{
+	fprint(2, "usage: %s [-qbc] [-f filt]\n", argv0);
+	exits("usage");
+}
+
+void
+main(int argc, char **argv)
+{
+	char rmpath[256], tpath[256], bpath[256], buf[8];
+	char *p, *e;
+	int i, dirty;
+	Wres r;
+
+	ARGBEGIN{
+	case 'q':
+		quiet++;
+		break;
+	case 'c':
+		rstr = "";
+		tstr = "";
+		mstr = "";
+		astr = "";
+		break;
+	case 'f':
+		for(p = EARGF(usage()); *p; p++)
+			switch(*p){
+			case 'T':	printflg |= Tflg;	break;
+			case 'A':	printflg |= Aflg;	break;
+			case 'M':	printflg |= Mflg;	break;
+			case 'R':	printflg |= Rflg;	break;
+			default:	usage();		break;
+		}
+		break;
+	default:
+		usage();
+	}ARGEND
+
+	findroot();
+	dirty = 0;
+	r.path = nil;
+	r.npath = 0;
+	r.pathsz = 0;
+	if(access("/mnt/git/ctl", AEXIST) != 0)
+		sysfatal("git/fs does not seem to be running");
+	if(printflg == 0)
+		printflg = Tflg | Aflg | Mflg | Rflg;
+	if(access(TDIR, AEXIST) == 0 && readpaths(&r, TDIR, "") == -1)
+		sysfatal("read tracked: %r");
+	if(access(RDIR, AEXIST) == 0 && readpaths(&r, RDIR, "") == -1)
+		sysfatal("read removed: %r");
+	dedup(&r);
+
+	for(i = 0; i < r.npath; i++){
+		p = r.path[i];
+		snprint(rmpath, sizeof(rmpath), RDIR"/%s", p);
+		snprint(tpath, sizeof(tpath), TDIR"/%s", p);
+		snprint(bpath, sizeof(bpath), HDIR"/%s", p);
+		if(access(p, AEXIST) != 0 || access(rmpath, AEXIST) == 0){
+			dirty |= Mflg;
+			if(!quiet && (printflg & Rflg))
+				print("%s%s\n", rstr, p);
+		}else if(access(bpath, AEXIST) == -1) {
+			dirty |= Aflg;
+			if(!quiet && (printflg & Aflg))
+				print("%s%s\n", astr, p);
+		}else if(!sameqid(p, tpath) && !samedata(p, bpath)){
+			dirty |= Mflg;
+			if(!quiet && (printflg & Mflg))
+				print("%s%s\n", mstr, p);
+		}else{
+			if(!quiet && (printflg & Tflg))
+				print("%s%s\n", tstr, p);
+		}
+	}
+	if(!dirty)
+		exits(nil);
+
+	p = buf;
+	e = buf + sizeof(buf);
+	for(i = 0; (1 << i) != Tflg; i++)
+		if(dirty & (1 << i))
+			p = seprint(p, e, "%c", "DMAT"[i]);
+	exits(buf);
+}