| # Support routines for hand-crafting weird or malicious packs. | 
 | # | 
 | # You can make a complete pack like: | 
 | # | 
 | #   pack_header 2 >foo.pack && | 
 | #   pack_obj e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 >>foo.pack && | 
 | #   pack_obj e68fe8129b546b101aee9510c5328e7f21ca1d18 >>foo.pack && | 
 | #   pack_trailer foo.pack | 
 |  | 
 | # Print the big-endian 4-byte octal representation of $1 | 
 | uint32_octal () { | 
 | 	n=$1 | 
 | 	printf '\\%o' $(($n / 16777216)); n=$((n % 16777216)) | 
 | 	printf '\\%o' $(($n /    65536)); n=$((n %    65536)) | 
 | 	printf '\\%o' $(($n /      256)); n=$((n %      256)) | 
 | 	printf '\\%o' $(($n           )); | 
 | } | 
 |  | 
 | # Print the big-endian 4-byte binary representation of $1 | 
 | uint32_binary () { | 
 | 	printf "$(uint32_octal "$1")" | 
 | } | 
 |  | 
 | # Print a pack header, version 2, for a pack with $1 objects | 
 | pack_header () { | 
 | 	printf 'PACK' && | 
 | 	printf '\0\0\0\2' && | 
 | 	uint32_binary "$1" | 
 | } | 
 |  | 
 | # Print the pack data for object $1, as a delta against object $2 (or as a full | 
 | # object if $2 is missing or empty). The output is suitable for including | 
 | # directly in the packfile, and represents the entirety of the object entry. | 
 | # Doing this on the fly (especially picking your deltas) is quite tricky, so we | 
 | # have hardcoded some well-known objects. See the case statements below for the | 
 | # complete list. | 
 | pack_obj () { | 
 | 	case "$1" in | 
 | 	# empty blob | 
 | 	e69de29bb2d1d6434b8b29ae775ad8c2e48c5391) | 
 | 		case "$2" in | 
 | 		'') | 
 | 			printf '\060\170\234\003\0\0\0\0\1' | 
 | 			return | 
 | 			;; | 
 | 		esac | 
 | 		;; | 
 |  | 
 | 	# blob containing "\7\76" | 
 | 	e68fe8129b546b101aee9510c5328e7f21ca1d18) | 
 | 		case "$2" in | 
 | 		'') | 
 | 			printf '\062\170\234\143\267\3\0\0\116\0\106' | 
 | 			return | 
 | 			;; | 
 | 		01d7713666f4de822776c7622c10f1b07de280dc) | 
 | 			printf '\165\1\327\161\66\146\364\336\202\47\166' && | 
 | 			printf '\307\142\54\20\361\260\175\342\200\334\170' && | 
 | 			printf '\234\143\142\142\142\267\003\0\0\151\0\114' | 
 | 			return | 
 | 			;; | 
 | 		esac | 
 | 		;; | 
 |  | 
 | 	# blob containing "\7\0" | 
 | 	01d7713666f4de822776c7622c10f1b07de280dc) | 
 | 		case "$2" in | 
 | 		'') | 
 | 			printf '\062\170\234\143\147\0\0\0\20\0\10' | 
 | 			return | 
 | 			;; | 
 | 		e68fe8129b546b101aee9510c5328e7f21ca1d18) | 
 | 			printf '\165\346\217\350\22\233\124\153\20\32\356' && | 
 | 			printf '\225\20\305\62\216\177\41\312\35\30\170\234' && | 
 | 			printf '\143\142\142\142\147\0\0\0\53\0\16' | 
 | 			return | 
 | 			;; | 
 | 		esac | 
 | 		;; | 
 | 	esac | 
 |  | 
 | 	# If it's not a delta, we can convince pack-objects to generate a pack | 
 | 	# with just our entry, and then strip off the header (12 bytes) and | 
 | 	# trailer (20 bytes). | 
 | 	if test -z "$2" | 
 | 	then | 
 | 		echo "$1" | git pack-objects --stdout >pack_obj.tmp && | 
 | 		size=$(wc -c <pack_obj.tmp) && | 
 | 		dd if=pack_obj.tmp bs=1 count=$((size - 20 - 12)) skip=12 && | 
 | 		rm -f pack_obj.tmp | 
 | 		return | 
 | 	fi | 
 |  | 
 | 	echo >&2 "BUG: don't know how to print $1${2:+ (from $2)}" | 
 | 	return 1 | 
 | } | 
 |  | 
 | # Compute and append pack trailer to "$1" | 
 | pack_trailer () { | 
 | 	test-tool sha1 -b <"$1" >trailer.tmp && | 
 | 	cat trailer.tmp >>"$1" && | 
 | 	rm -f trailer.tmp | 
 | } | 
 |  | 
 | # Remove any existing packs to make sure that | 
 | # whatever we index next will be the pack that we | 
 | # actually use. | 
 | clear_packs () { | 
 | 	rm -f .git/objects/pack/* | 
 | } |