2011-11-26 05:14:03 +01:00
|
|
|
#!/bin/sh
|
2013-02-19 21:21:39 +01:00
|
|
|
# ragg2-cc : a shellcode compiler -- pancake - 2011-2013
|
|
|
|
#
|
|
|
|
# Supported operating systems:
|
|
|
|
# - GNU/Linux
|
|
|
|
# - OSX
|
|
|
|
# - BSD
|
|
|
|
# Supported compilers
|
|
|
|
# - gcc
|
|
|
|
# - clang
|
2011-11-26 05:14:03 +01:00
|
|
|
# TODO
|
2011-11-29 03:14:27 +01:00
|
|
|
# remove temporary files
|
2011-11-26 05:14:03 +01:00
|
|
|
# add support for arm
|
|
|
|
# add support for nested shellcodes
|
|
|
|
|
2014-10-09 21:50:40 +02:00
|
|
|
# Find which compiler is installed
|
2011-11-30 18:05:46 +01:00
|
|
|
if [ -z "${CC}" ]; then
|
2011-11-30 20:59:58 +01:00
|
|
|
for a in llvm-gcc clang gcc ; do
|
2011-11-30 18:05:46 +01:00
|
|
|
$a --version >/dev/null 2>&1
|
|
|
|
if [ $? = 0 ]; then
|
|
|
|
CC="$a"
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
if [ -z "${CC}" ]; then
|
|
|
|
echo "Cannot find CC" > /dev/stderr
|
|
|
|
exit 1
|
|
|
|
fi
|
|
|
|
fi
|
2014-10-09 21:50:40 +02:00
|
|
|
|
|
|
|
# Get path for sflib
|
2011-11-29 03:14:27 +01:00
|
|
|
SFLIBPATH=`ragg2 -v | cut -d ' ' -f 3-`
|
|
|
|
if [ ! -d "${SFLIBPATH}" ]; then
|
|
|
|
echo "Cannot find ${SFLIBPATH}"
|
|
|
|
exit 1
|
|
|
|
fi
|
2014-10-09 21:50:40 +02:00
|
|
|
|
|
|
|
# Get local architecture
|
2011-11-29 03:14:27 +01:00
|
|
|
case "`uname -m`" in
|
|
|
|
"x86_64")
|
|
|
|
B=64
|
|
|
|
;;
|
|
|
|
*)
|
|
|
|
B=32
|
|
|
|
;;
|
|
|
|
esac
|
2011-11-29 19:40:10 +01:00
|
|
|
|
|
|
|
|
|
|
|
dohelp() {
|
|
|
|
echo "Usage: ragg2-cc [-dabkoscxv] [file.c]"
|
|
|
|
echo " -d enable debug mode"
|
|
|
|
echo " -a x86 set arch (x86, arm)"
|
|
|
|
echo " -b 32 bits (32, 64)"
|
|
|
|
echo " -k linux set kernel (darwin, linux)"
|
|
|
|
echo " -o file set output file"
|
|
|
|
echo " -s generate assembly"
|
|
|
|
echo " -c generate compiled shellcode"
|
|
|
|
echo " -x show hexpair bytes"
|
|
|
|
echo " -v show version"
|
|
|
|
}
|
|
|
|
|
|
|
|
case "`uname`" in
|
|
|
|
Darwin)
|
|
|
|
K=darwin
|
|
|
|
;;
|
|
|
|
*)
|
|
|
|
K=linux
|
|
|
|
;;
|
|
|
|
esac
|
2014-10-09 21:50:40 +02:00
|
|
|
|
2011-11-29 03:14:27 +01:00
|
|
|
X=0
|
|
|
|
C=""
|
2011-11-26 05:14:03 +01:00
|
|
|
D=""
|
|
|
|
O=""
|
|
|
|
F=""
|
|
|
|
ASM=0
|
2014-10-09 21:50:40 +02:00
|
|
|
A=x86
|
2011-11-26 05:14:03 +01:00
|
|
|
while : ; do
|
|
|
|
[ -z "$1" ] && break
|
|
|
|
F=$1
|
|
|
|
case "$F" in
|
2014-10-09 21:50:40 +02:00
|
|
|
-a) # architecture (x86, mips, arm)
|
2011-11-29 03:14:27 +01:00
|
|
|
shift
|
|
|
|
A=$1
|
|
|
|
[ -z "$A" ] && { echo "Missing argument for -a" ; exit 1; }
|
|
|
|
;;
|
2014-10-09 21:50:40 +02:00
|
|
|
-b) # register size (32, 64, ...)
|
2011-11-29 03:14:27 +01:00
|
|
|
shift
|
|
|
|
B=$1
|
|
|
|
[ -z "$B" ] && { echo "Missing argument for -b" ; exit 1; }
|
|
|
|
;;
|
2014-10-09 21:50:40 +02:00
|
|
|
-k) # kernel
|
2011-11-29 03:14:27 +01:00
|
|
|
shift
|
|
|
|
K=$1
|
|
|
|
[ -z "$K" ] && { echo "Missing argument for -k" ; exit 1; }
|
|
|
|
;;
|
2014-10-09 21:50:40 +02:00
|
|
|
-x) # execute
|
2011-11-29 03:14:27 +01:00
|
|
|
X=1
|
|
|
|
;;
|
2014-10-09 21:50:40 +02:00
|
|
|
-c) # set configuration option
|
2011-11-29 03:14:27 +01:00
|
|
|
C=1
|
2011-11-26 05:14:03 +01:00
|
|
|
;;
|
2014-10-09 21:50:40 +02:00
|
|
|
-d) # patch dword (4 bytes) at given offset
|
2011-11-26 05:14:03 +01:00
|
|
|
D=1
|
|
|
|
;;
|
2014-10-09 21:50:40 +02:00
|
|
|
-s) # show assembler
|
2011-11-26 05:14:03 +01:00
|
|
|
ASM=1
|
|
|
|
;;
|
2014-10-09 21:50:40 +02:00
|
|
|
-o) # output file
|
2011-11-26 05:14:03 +01:00
|
|
|
shift
|
|
|
|
O=$1
|
|
|
|
if [ -z "$O" ]; then
|
|
|
|
echo "Missing argument for -o"
|
|
|
|
exit 1
|
|
|
|
fi
|
|
|
|
;;
|
2014-10-09 21:50:40 +02:00
|
|
|
-h) # help
|
2011-11-29 19:40:10 +01:00
|
|
|
dohelp
|
2011-11-26 05:14:03 +01:00
|
|
|
exit 0
|
|
|
|
;;
|
2014-10-09 21:50:40 +02:00
|
|
|
-v) # version
|
2011-11-29 03:14:27 +01:00
|
|
|
ragg2 -v | sed -e 's,2,2-cc,'
|
2011-11-26 05:14:03 +01:00
|
|
|
exit 0
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
shift
|
|
|
|
done
|
2014-10-09 21:50:40 +02:00
|
|
|
|
2011-11-26 05:14:03 +01:00
|
|
|
if [ -z "$F" ]; then
|
2011-11-29 19:40:10 +01:00
|
|
|
dohelp
|
2011-11-26 05:14:03 +01:00
|
|
|
exit 1
|
|
|
|
fi
|
2011-11-28 14:13:44 -05:00
|
|
|
|
2011-11-26 05:14:03 +01:00
|
|
|
if [ "`uname`" = Darwin ]; then
|
|
|
|
OBJCOPY=gobjcopy
|
2011-11-29 03:14:27 +01:00
|
|
|
case "$B" in
|
|
|
|
32)
|
|
|
|
CFLAGS="-arch i386 "
|
|
|
|
LDFLAGS="-arch i386 -shared -c"
|
|
|
|
ARCH=darwin-x86-32
|
|
|
|
;;
|
|
|
|
64)
|
|
|
|
CFLAGS="-arch x86_64"
|
|
|
|
LDFLAGS="-arch x86_64 -shared -c"
|
|
|
|
ARCH=darwin-x86-64
|
|
|
|
;;
|
|
|
|
esac
|
2011-11-26 05:14:03 +01:00
|
|
|
SHDR="
|
|
|
|
.text
|
|
|
|
jmp _main"
|
|
|
|
else
|
|
|
|
OBJCOPY=objcopy
|
2014-11-10 06:04:46 +01:00
|
|
|
CFLAGS="-fPIC -fPIE -pie -fpic"
|
|
|
|
LDFLAGS="-fPIC -fPIE -pie -fpic"
|
2011-11-26 05:14:03 +01:00
|
|
|
SHDR="
|
2011-11-28 14:13:44 -05:00
|
|
|
.section .text
|
2011-11-26 05:14:03 +01:00
|
|
|
.globl main
|
|
|
|
.type main, @function
|
|
|
|
jmp main
|
|
|
|
"
|
2011-11-29 03:14:27 +01:00
|
|
|
case "$B" in
|
|
|
|
64)
|
|
|
|
ARCH=linux-x86-64
|
2011-11-28 14:13:44 -05:00
|
|
|
;;
|
|
|
|
*)
|
2011-11-29 03:14:27 +01:00
|
|
|
ARCH=linux-x86-32
|
2011-11-28 14:13:44 -05:00
|
|
|
;;
|
|
|
|
esac
|
2011-11-26 05:14:03 +01:00
|
|
|
fi
|
2011-11-29 03:14:27 +01:00
|
|
|
|
|
|
|
[ "$A$K" ] && ARCH="$K-$A-$B"
|
|
|
|
|
2011-11-26 05:14:03 +01:00
|
|
|
OPT=-Os
|
2011-11-29 03:14:27 +01:00
|
|
|
CFLAGS="${CFLAGS} -nostdinc -include ${SFLIBPATH}/${ARCH}/sflib.h"
|
2014-10-09 21:50:40 +02:00
|
|
|
CFLAGS="${CFLAGS} -z execstack"
|
2011-11-26 05:14:03 +01:00
|
|
|
CFLAGS="${CFLAGS} -fomit-frame-pointer -finline-functions -fno-zero-initialized-in-bss"
|
|
|
|
LDFLAGS="${LDFLAGS} -nostdlib"
|
|
|
|
|
2011-11-29 19:40:10 +01:00
|
|
|
case "$K" in
|
|
|
|
darwin)
|
2012-09-21 02:47:07 +02:00
|
|
|
#TEXT="__TEXT.__text"
|
|
|
|
TEXT="0.__text"
|
2011-11-29 19:40:10 +01:00
|
|
|
;;
|
|
|
|
*|linux)
|
|
|
|
TEXT=".text"
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
|
2011-11-26 05:14:03 +01:00
|
|
|
rmtemps() {
|
2011-11-29 03:14:27 +01:00
|
|
|
[ -z "$D" ] && rm -f $F.tmp $F.text $F.s $F.o
|
2011-11-26 05:14:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
fail() {
|
|
|
|
rmtemps
|
|
|
|
exit 1
|
|
|
|
}
|
|
|
|
|
|
|
|
if [ "$D" ]; then
|
|
|
|
echo "==> Compile"
|
|
|
|
echo "${CC} ${CFLAGS} -o $F.tmp -S ${OPT} $F"
|
|
|
|
fi
|
|
|
|
rm -f $F.bin
|
2014-11-10 06:04:46 +01:00
|
|
|
# echo ${CC} ${CFLAGS} -o $F.tmp -S ${OPT} $F > /dev/stderr
|
2011-11-26 05:14:03 +01:00
|
|
|
${CC} ${CFLAGS} -o $F.tmp -S ${OPT} $F || fail
|
|
|
|
echo "${SHDR}" > $F.s
|
|
|
|
cat $F.tmp \
|
2012-09-03 18:49:29 +02:00
|
|
|
| sed -e s,rdata,text, -e s,rodata,text, -e 's,get_pc_thunk.bx,__getesp__,g' \
|
2011-11-26 05:14:03 +01:00
|
|
|
| grep -v .cstring | grep -v size | grep -v ___main | grep -v section \
|
|
|
|
| grep -v __alloca | grep -v zero | grep -v cfi >> $F.s
|
|
|
|
rm -f $F.tmp
|
|
|
|
if [ $ASM = 1 ]; then
|
|
|
|
echo $F.s
|
|
|
|
exit 0
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ "$D" ]; then
|
|
|
|
echo "==> Assemble"
|
|
|
|
echo "${CC} -c ${LDFLAGS} -Os -o $F.o $F.s"
|
|
|
|
fi
|
|
|
|
${CC} ${LDFLAGS} ${OPT} -o $F.o $F.s || fail
|
|
|
|
|
|
|
|
if [ "$D" ]; then
|
|
|
|
echo "==> Link"
|
2011-11-29 19:40:10 +01:00
|
|
|
#echo "${OBJCOPY} -j .text -O binary $F.o $.text"
|
2011-11-30 20:59:58 +01:00
|
|
|
echo "rabin2 -o '$F.text' -O d/S/${TEXT} $F.o"
|
2011-11-29 19:40:10 +01:00
|
|
|
fi
|
2011-11-30 20:59:58 +01:00
|
|
|
rabin2 -o "$F.text" -O d/S/${TEXT} $F.o
|
2011-11-29 19:40:10 +01:00
|
|
|
if [ "`du $F.text|awk '{print $1}'`" = 0 ]; then
|
|
|
|
# use objcopy as falback for rabin2
|
|
|
|
${OBJCOPY} -j .text -O binary $F.o $F.text || fail
|
2011-11-26 05:14:03 +01:00
|
|
|
fi
|
2011-11-29 19:40:10 +01:00
|
|
|
if [ "$C" = 1 ]; then
|
2011-11-29 03:14:27 +01:00
|
|
|
if [ "$O" ]; then
|
|
|
|
mv $F.text $O
|
|
|
|
else
|
|
|
|
O="$F.text"
|
|
|
|
fi
|
2011-11-29 19:40:10 +01:00
|
|
|
echo "$O"
|
2011-11-29 03:14:27 +01:00
|
|
|
exit 0
|
|
|
|
fi
|
2011-11-26 05:14:03 +01:00
|
|
|
|
2013-02-19 21:21:39 +01:00
|
|
|
[ "$X" = 1 ] && exec rax2 -S < $F.text
|
2011-11-26 05:14:03 +01:00
|
|
|
|
|
|
|
if [ "$D" ]; then
|
|
|
|
# hexdump -C $F.text
|
|
|
|
rax2 -S - < $F.text
|
|
|
|
ls -l $F.text
|
|
|
|
fi
|
2011-11-29 03:14:27 +01:00
|
|
|
[ -z "$O" ] && O=$F.bin
|
2014-03-28 15:57:50 +01:00
|
|
|
ragg2 -b $B -C $F.text -F -o $O || fail
|
2011-11-26 05:14:03 +01:00
|
|
|
echo $O
|
2011-11-29 03:14:27 +01:00
|
|
|
rmtemps
|
2011-11-26 05:14:03 +01:00
|
|
|
exit 0
|