159 lines
2.7 KiB
Bash
159 lines
2.7 KiB
Bash
#!/bin/bash
|
|
|
|
# objdiff - a small script for validating that a commit or series of commits
|
|
# didn't change object code.
|
|
#
|
|
# Copyright 2014, Jason Cooper <jason@lakedaemon.net>
|
|
#
|
|
# Licensed under the terms of the GNU GPL version 2
|
|
|
|
# usage example:
|
|
#
|
|
# $ git checkout COMMIT_A
|
|
# $ <your fancy build command here>
|
|
# $ ./scripts/objdiff record path/to/*.o
|
|
#
|
|
# $ git checkout COMMIT_B
|
|
# $ <your fancy build command here>
|
|
# $ ./scripts/objdiff record path/to/*.o
|
|
#
|
|
# $ ./scripts/objdiff diff COMMIT_A COMMIT_B
|
|
# $
|
|
|
|
# And to clean up (everything is in .tmp_objdiff/*)
|
|
# $ ./scripts/objdiff clean all
|
|
#
|
|
# Note: 'make mrproper' will also remove .tmp_objdiff
|
|
|
|
SRCTREE=$(cd $(git rev-parse --show-toplevel 2>/dev/null); pwd)
|
|
|
|
if [ -z "$SRCTREE" ]; then
|
|
echo >&2 "ERROR: Not a git repository."
|
|
exit 1
|
|
fi
|
|
|
|
TMPD=$SRCTREE/.tmp_objdiff
|
|
|
|
usage() {
|
|
echo >&2 "Usage: $0 <command> <args>"
|
|
echo >&2 " record <list of object files or directories>"
|
|
echo >&2 " diff <commitA> <commitB>"
|
|
echo >&2 " clean all | <commit>"
|
|
exit 1
|
|
}
|
|
|
|
get_output_dir() {
|
|
dir=${1%/*}
|
|
|
|
if [ "$dir" = "$1" ]; then
|
|
dir=.
|
|
fi
|
|
|
|
dir=$(cd $dir; pwd)
|
|
|
|
echo $TMPD/$CMT${dir#$SRCTREE}
|
|
}
|
|
|
|
do_objdump() {
|
|
dir=$(get_output_dir $1)
|
|
base=${1##*/}
|
|
dis=$dir/${base%.o}.dis
|
|
|
|
[ ! -d "$dir" ] && mkdir -p $dir
|
|
|
|
# remove addresses for a cleaner diff
|
|
# http://dummdida.tumblr.com/post/60924060451/binary-diff-between-libc-from-scientificlinux-and
|
|
$OBJDUMP -D $1 | sed "s/^[[:space:]]\+[0-9a-f]\+//" > $dis
|
|
}
|
|
|
|
dorecord() {
|
|
[ $# -eq 0 ] && usage
|
|
|
|
FILES="$*"
|
|
|
|
CMT="`git rev-parse --short HEAD`"
|
|
|
|
OBJDUMP="${CROSS_COMPILE}objdump"
|
|
|
|
for d in $FILES; do
|
|
if [ -d "$d" ]; then
|
|
for f in $(find $d -name '*.o')
|
|
do
|
|
do_objdump $f
|
|
done
|
|
else
|
|
do_objdump $d
|
|
fi
|
|
done
|
|
}
|
|
|
|
dodiff() {
|
|
[ $# -ne 2 ] && [ $# -ne 0 ] && usage
|
|
|
|
if [ $# -eq 0 ]; then
|
|
SRC="`git rev-parse --short HEAD^`"
|
|
DST="`git rev-parse --short HEAD`"
|
|
else
|
|
SRC="`git rev-parse --short $1`"
|
|
DST="`git rev-parse --short $2`"
|
|
fi
|
|
|
|
DIFF="`which colordiff`"
|
|
|
|
if [ ${#DIFF} -eq 0 ] || [ ! -x "$DIFF" ]; then
|
|
DIFF="`which diff`"
|
|
fi
|
|
|
|
SRCD="$TMPD/$SRC"
|
|
DSTD="$TMPD/$DST"
|
|
|
|
if [ ! -d "$SRCD" ]; then
|
|
echo >&2 "ERROR: $SRCD doesn't exist"
|
|
exit 1
|
|
fi
|
|
|
|
if [ ! -d "$DSTD" ]; then
|
|
echo >&2 "ERROR: $DSTD doesn't exist"
|
|
exit 1
|
|
fi
|
|
|
|
$DIFF -Nurd $SRCD $DSTD
|
|
}
|
|
|
|
doclean() {
|
|
[ $# -eq 0 ] && usage
|
|
[ $# -gt 1 ] && usage
|
|
|
|
if [ "x$1" = "xall" ]; then
|
|
rm -rf $TMPD/*
|
|
else
|
|
CMT="`git rev-parse --short $1`"
|
|
|
|
if [ -d "$TMPD/$CMT" ]; then
|
|
rm -rf $TMPD/$CMT
|
|
else
|
|
echo >&2 "$CMT not found"
|
|
fi
|
|
fi
|
|
}
|
|
|
|
[ $# -eq 0 ] && usage
|
|
|
|
case "$1" in
|
|
record)
|
|
shift
|
|
dorecord $*
|
|
;;
|
|
diff)
|
|
shift
|
|
dodiff $*
|
|
;;
|
|
clean)
|
|
shift
|
|
doclean $*
|
|
;;
|
|
*)
|
|
echo >&2 "Unrecognized command '$1'"
|
|
exit 1
|
|
;;
|
|
esac
|