#!/bin/sh # # 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 ;; esac ;; esac 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-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/* }