wrapper 6.9 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
#!/bin/sh

# Copyright (C) 2006 Paul Mackerras, IBM Corporation <paulus@samba.org>
# This program may be used under the terms of version 2 of the GNU
# General Public License.

# This script takes a kernel binary and optionally an initrd image
# and/or a device-tree blob, and creates a bootable zImage for a
# given platform.

# Options:
# -o zImage	specify output file
# -p platform	specify platform (links in $platform.o)
# -i initrd	specify initrd file
# -d devtree	specify device-tree blob
# -s tree.dts	specify device-tree source file (needs dtc installed)
# -c		cache $kernel.strip.gz (use if present & newer, else make)
# -C prefix	specify command prefix for cross-building tools
#		(strip, objcopy, ld)
# -D dir	specify directory containing data files used by script
#		(default ./arch/powerpc/boot)
# -W dir	specify working directory for temporary files (default .)

24 25 26 27 28
# Allow for verbose output
if [ "$V" = 1 ]; then
    set -x
fi

29 30 31 32 33 34 35 36
# defaults
kernel=
ofile=zImage
platform=of
initrd=
dtb=
dts=
cacheit=
37
binary=
38
gzip=.gz
39 40 41 42 43 44 45 46 47 48 49 50 51

# cross-compilation prefix
CROSS=

# directory for object and other files used by this script
object=arch/powerpc/boot

# directory for working files
tmpdir=.

usage() {
    echo 'Usage: wrapper [-o output] [-p platform] [-i initrd]' >&2
    echo '       [-d devtree] [-s tree.dts] [-c] [-C cross-prefix]' >&2
52
    echo '       [-D datadir] [-W workingdir] [--no-gzip] [vmlinux]' >&2
53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100
    exit 1
}

while [ "$#" -gt 0 ]; do
    case "$1" in
    -o)
	shift
	[ "$#" -gt 0 ] || usage
	ofile="$1"
	;;
    -p)
	shift
	[ "$#" -gt 0 ] || usage
	platform="$1"
	;;
    -i)
	shift
	[ "$#" -gt 0 ] || usage
	initrd="$1"
	;;
    -d)
	shift
	[ "$#" -gt 0 ] || usage
	dtb="$1"
	;;
    -s)
	shift
	[ "$#" -gt 0 ] || usage
	dts="$1"
	;;
    -c)
	cacheit=y
	;;
    -C)
	shift
	[ "$#" -gt 0 ] || usage
	CROSS="$1"
	;;
    -D)
	shift
	[ "$#" -gt 0 ] || usage
	object="$1"
	;;
    -W)
	shift
	[ "$#" -gt 0 ] || usage
	tmpdir="$1"
	;;
101 102 103
    --no-gzip)
        gzip=
        ;;
104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137
    -?)
	usage
	;;
    *)
	[ -z "$kernel" ] || usage
	kernel="$1"
	;;
    esac
    shift
done

if [ -n "$dts" ]; then
    if [ -z "$dtb" ]; then
	dtb="$platform.dtb"
    fi
    dtc -O dtb -o "$dtb" -b 0 -V 16 "$dts" || exit 1
fi

if [ -z "$kernel" ]; then
    kernel=vmlinux
fi

platformo=$object/"$platform".o
lds=$object/zImage.lds
ext=strip
objflags=-S
tmp=$tmpdir/zImage.$$.o
ksection=.kernel:vmlinux.strip
isection=.kernel:initrd

case "$platform" in
pmac|pseries|chrp)
    platformo=$object/of.o
    ;;
138
coff)
139 140 141 142 143 144 145 146 147 148 149
    platformo=$object/of.o
    lds=$object/zImage.coff.lds
    ;;
miboot|uboot)
    # miboot and U-boot want just the bare bits, not an ELF binary
    ext=bin
    objflags="-O binary"
    tmp="$ofile"
    ksection=image
    isection=initrd
    ;;
150
cuboot*)
151
    binary=y
152 153
    gzip=
    ;;
154 155 156
ps3)
    platformo="$object/ps3-head.o $object/ps3-hvcall.o $object/ps3.o"
    lds=$object/zImage.ps3.lds
157
    binary=y
158 159 160 161 162 163
    gzip=
    ext=bin
    objflags="-O binary --set-section-flags=.bss=contents,alloc,load,data"
    ksection=.kernel:vmlinux.bin
    isection=.kernel:initrd
    ;;
164 165 166 167
ep88xc)
    platformo="$object/fixed-head.o $object/$platform.o"
    binary=y
    ;;
168 169 170
esac

vmz="$tmpdir/`basename \"$kernel\"`.$ext"
171
if [ -z "$cacheit" -o ! -f "$vmz$gzip" -o "$vmz$gzip" -ot "$kernel" ]; then
172
    ${CROSS}objcopy $objflags "$kernel" "$vmz.$$"
173 174 175 176 177

    if [ -n "$gzip" ]; then
        gzip -f -9 "$vmz.$$"
    fi

178
    if [ -n "$cacheit" ]; then
179
	mv -f "$vmz.$$$gzip" "$vmz$gzip"
180 181 182 183 184
    else
	vmz="$vmz.$$"
    fi
fi

185 186
vmz="$vmz$gzip"

187 188 189 190 191 192 193
# Extract kernel version information, some platforms want to include
# it in the image header
version=`${CROSS}strings "$kernel" | grep '^Linux version [-0-9.]' | \
    cut -d' ' -f3`
if [ -n "$version" ]; then
    uboot_version="-n Linux-$version"
fi
194 195 196 197

case "$platform" in
uboot)
    rm -f "$ofile"
198
    mkimage -A ppc -O linux -T kernel -C gzip -a 00000000 -e 00000000 \
199
	$uboot_version -d "$vmz" "$ofile"
200
    if [ -z "$cacheit" ]; then
201
	rm -f "$vmz"
202 203 204 205 206 207 208 209 210 211 212
    fi
    exit 0
    ;;
esac

addsec() {
    ${CROSS}objcopy $4 $1 \
	--add-section=$3="$2" \
	--set-section-flags=$3=contents,alloc,load,readonly,data
}

213
addsec $tmp "$vmz" $ksection $object/empty.o
214
if [ -z "$cacheit" ]; then
215
    rm -f "$vmz"
216 217 218
fi

if [ -n "$initrd" ]; then
219
    addsec $tmp "$initrd" $isection
220 221 222
fi

if [ -n "$dtb" ]; then
223
    addsec $tmp "$dtb" .kernel:dtb
224 225 226
    if [ -n "$dts" ]; then
	rm $dtb
    fi
227 228 229 230
fi

if [ "$platform" != "miboot" ]; then
    ${CROSS}ld -m elf32ppc -T $lds -o "$ofile" \
231
	$platformo $tmp $object/wrapper.a
232 233 234
    rm $tmp
fi

235 236 237 238
# Some platforms need the zImage's entry point and base address
base=0x`${CROSS}nm "$ofile" | grep ' _start$' | cut -d' ' -f1`
entry=`${CROSS}objdump -f "$ofile" | grep '^start address ' | cut -d' ' -f3`

239 240 241 242 243
if [ -n "$binary" ]; then
    mv "$ofile" "$ofile".elf
    ${CROSS}objcopy -O binary "$ofile".elf "$ofile".bin
fi

244 245 246 247 248
# post-processing needed for some platforms
case "$platform" in
pseries|chrp)
    $object/addnote "$ofile"
    ;;
249
coff)
250
    ${CROSS}objcopy -O aixcoff-rs6000 --set-start "$entry" "$ofile"
251 252
    $object/hack-coff "$ofile"
    ;;
253 254 255
cuboot*)
    gzip -f -9 "$ofile".bin
    mkimage -A ppc -O linux -T kernel -C gzip -a "$base" -e "$entry" \
256
            $uboot_version -d "$ofile".bin.gz "$ofile"
257
    ;;
258 259 260 261 262 263 264 265
treeboot*)
    mv "$ofile" "$ofile.elf"
    $object/mktree "$ofile.elf" "$ofile" "$base" "$entry"
    if [ -z "$cacheit" ]; then
	rm -f "$ofile.elf"
    fi
    exit 0
    ;;
266 267 268 269 270 271 272 273 274 275 276
ps3)
    # The ps3's loader supports loading gzipped binary images from flash
    # rom to addr zero. The loader enters the image at addr 0x100.  A
    # bootwrapper overlay is use to arrange for the kernel to be loaded
    # to addr zero and to have a suitable bootwrapper entry at 0x100.
    # To construct the rom image, 0x100 bytes from offset 0x100 in the
    # kernel is copied to the bootwrapper symbol __system_reset_kernel.
    # The 0x100 bytes at the bootwrapper symbol __system_reset_overlay is
    # then copied to offset 0x100.  At runtime the bootwrapper program
    # copies the 0x100 bytes at __system_reset_kernel to addr 0x100.

277
    system_reset_overlay=0x`${CROSS}nm "$ofile".elf \
278 279 280
        | grep ' __system_reset_overlay$'       \
        | cut -d' ' -f1`
    system_reset_overlay=`printf "%d" $system_reset_overlay`
281
    system_reset_kernel=0x`${CROSS}nm "$ofile".elf \
282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309
        | grep ' __system_reset_kernel$'       \
        | cut -d' ' -f1`
    system_reset_kernel=`printf "%d" $system_reset_kernel`
    overlay_dest="256"
    overlay_size="256"

    rm -f "$object/otheros.bld"

    msg=$(dd if="$ofile.bin" of="$ofile.bin" conv=notrunc \
        skip=$overlay_dest seek=$system_reset_kernel      \
        count=$overlay_size bs=1 2>&1)

    if [ $? -ne "0" ]; then
       echo $msg
       exit 1
    fi

    msg=$(dd if="$ofile.bin" of="$ofile.bin" conv=notrunc \
        skip=$system_reset_overlay seek=$overlay_dest     \
        count=$overlay_size bs=1 2>&1)

    if [ $? -ne "0" ]; then
       echo $msg
       exit 2
    fi

    gzip --force -9 --stdout "$ofile.bin" > "$object/otheros.bld"
    ;;
310
esac