cookutils annotate cross @ rev 416
cross: more clean-tools and add gen-prebuilt (for testing)
author | Christophe Lincoln <pankso@slitaz.org> |
---|---|
date | Mon May 14 14:59:53 2012 +0200 (2012-05-14) |
parents | 0951a386e1e6 |
children | 0f2a3fcd0e24 |
rev | line source |
---|---|
pankso@361 | 1 #!/bin/sh |
pankso@361 | 2 # |
pankso@361 | 3 # Cross - Help build a cross toolchain on SliTaz. |
pankso@361 | 4 # |
pankso@361 | 5 # Copyright 2012 (C) SliTaz GNU/Linux - BSD License |
pankso@361 | 6 # Author: Christophe Lincoln <pankso@slitaz.org> |
pankso@361 | 7 # |
pankso@361 | 8 . /lib/libtaz.sh |
pankso@370 | 9 |
pankso@370 | 10 [ -f "/etc/slitaz/cross.conf" ] && . /etc/slitaz/cross.conf |
pankso@370 | 11 [ -f "cross.conf" ] && . ./cross.conf |
pankso@361 | 12 |
pankso@374 | 13 # Handle --config=/path/to/cross.conf |
pankso@374 | 14 [ "$config" ] && . $config |
pankso@412 | 15 source=$WORK/source |
pankso@412 | 16 logdir=$WORK/log |
pankso@412 | 17 install=$WORK/install |
pankso@374 | 18 |
pankso@361 | 19 # Help and usage. |
pankso@361 | 20 usage() { |
pankso@361 | 21 cat << EOT |
pankso@361 | 22 |
pankso@361 | 23 Usage: $(basename $0) command --option |
pankso@361 | 24 |
pankso@361 | 25 Commands: |
pankso@361 | 26 howto Man alike and howto |
paul@382 | 27 info Display cross-tools info |
pankso@361 | 28 testsuite Execute a small testsuite |
pankso@361 | 29 check-env Check build host tools |
pankso@361 | 30 download Download necessary sources |
pankso@371 | 31 clean Clean-up build environment |
pankso@371 | 32 clean-tools Clean: $PREFIX |
pankso@361 | 33 show-log Show a compile log |
pankso@361 | 34 binutils Compile Binutils |
pankso@361 | 35 gcc-static Compile GCC static |
pankso@361 | 36 linux-headers Install Kernel headers |
pankso@361 | 37 glibc Compile GNU Glibc |
pankso@361 | 38 gcc-final Compile final GCC |
pankso@361 | 39 busybox Cross compile Busybox |
pankso@361 | 40 compile Compile everything at once |
pankso@361 | 41 |
pankso@361 | 42 EOT |
pankso@361 | 43 } |
pankso@361 | 44 |
pankso@361 | 45 # Make sure we have all directories. |
pankso@361 | 46 init_compile() { |
pankso@361 | 47 export LC_ALL=POSIX LANG=POSIX |
pankso@403 | 48 [ "$SYSROOT" ] || export PATH=$PATH:$PREFIX/bin |
pankso@400 | 49 export CROSS_COMPILE=${TARGET}- |
pankso@361 | 50 mkdir -p $source $logdir $install |
pankso@361 | 51 cd $source |
pankso@361 | 52 } |
pankso@361 | 53 |
pankso@361 | 54 # Get source if not yet in $SRC. |
pankso@361 | 55 download_src() { |
pankso@361 | 56 mkdir -p $SRC && cd $SRC |
pankso@361 | 57 [ -f "binutils-$BINUTILS_VERSION.tar.bz2" ] || wget $BINUTILS_WGET |
pankso@361 | 58 [ -f "linux-$LINUX_VERSION.tar.bz2" ] || wget $LINUX_WGET |
pankso@361 | 59 [ -f "glibc-$GLIBC_VERSION.tar.bz2" ] || wget $GLIBC_WGET |
pankso@361 | 60 [ -f "gcc-$GCC_VERSION.tar.bz2" ] || wget $GCC_WGET |
pankso@361 | 61 [ -f "busybox-$BUSYBOX_VERSION.tar.bz2" ] || wget $BUSYBOX_WGET |
pankso@361 | 62 } |
pankso@361 | 63 |
pankso@403 | 64 # Use sysroot or not ? |
pankso@403 | 65 check_sysroot() { |
pankso@403 | 66 if [ "$SYSROOT" ]; then |
pankso@403 | 67 PREFIX=/usr |
pankso@403 | 68 HDR_PATH=$SYSROOT/usr |
pankso@403 | 69 sysroot="--with-sysroot=$SYSROOT" |
pankso@411 | 70 echo "Configure : $sysroot" |
pankso@403 | 71 else |
pankso@403 | 72 HDR_PATH=$PREFIX |
pankso@403 | 73 fi |
pankso@403 | 74 } |
pankso@403 | 75 |
pankso@361 | 76 # 1. Binutils |
pankso@361 | 77 binutils() { |
pankso@408 | 78 init_compile |
pankso@411 | 79 rm -rf binutils-$BINUTILS_VERSION |
pankso@361 | 80 echo "Extracting: binutils-$BINUTILS_VERSION.tar.bz2" |
pankso@361 | 81 tar xjf $SRC/binutils-$BINUTILS_VERSION.tar.bz2 |
pankso@400 | 82 : ${BINUTILS_ARGS=--enable-shared} |
pankso@411 | 83 echo "Configure : $BINUTILS_ARGS" |
pankso@403 | 84 check_sysroot |
pankso@361 | 85 cd binutils-$BINUTILS_VERSION |
pankso@361 | 86 ./configure \ |
pankso@361 | 87 --prefix=$PREFIX \ |
pankso@361 | 88 --target=$TARGET \ |
pankso@361 | 89 --enable-targets=$BUILD_SYSTEM \ |
pankso@403 | 90 $BINUTILS_ARGS $sysroot |
pankso@361 | 91 make || exit 1 |
pankso@361 | 92 make install |
pankso@412 | 93 #echo "cross: binutils compiled on: $(date)" |
pankso@361 | 94 } |
pankso@361 | 95 |
pankso@403 | 96 # 2. GCC static (first pass) |
pankso@361 | 97 gcc_static() { |
pankso@408 | 98 init_compile |
pankso@361 | 99 echo "Extracting: gcc-$GCC_VERSION.tar.bz2" |
pankso@361 | 100 tar xjf $SRC/gcc-$GCC_VERSION.tar.bz2 |
pankso@411 | 101 echo "Configure : $GCC_STATIC_ARGS" |
pankso@403 | 102 check_sysroot |
pankso@403 | 103 # Arch fixes and work around |
pankso@403 | 104 case "$ARCH" in |
pankso@403 | 105 x86_64) |
pankso@403 | 106 # GCC wants Glib headers in cross environment (not tested |
pankso@403 | 107 # with sysroot) Should we install glibc-headers before ? |
pankso@403 | 108 [ "$SYSROOT" ] || \ |
pankso@409 | 109 ln -s /usr/include $PREFIX/$TARGET/include ;; |
pankso@403 | 110 esac |
pankso@361 | 111 rm -rf gcc-static |
pankso@361 | 112 mkdir gcc-static && cd gcc-static |
pankso@361 | 113 ../gcc-$GCC_VERSION/configure \ |
pankso@361 | 114 --prefix=$PREFIX \ |
pankso@374 | 115 --libexec=$PREFIX/lib \ |
pankso@361 | 116 --target=$TARGET \ |
pankso@361 | 117 --disable-shared \ |
pankso@361 | 118 --disable-threads \ |
pankso@361 | 119 --without-headers \ |
pankso@361 | 120 --with-newlib \ |
pankso@403 | 121 $GCC_STATIC_ARGS $sysroot |
pankso@361 | 122 make all-gcc all-target-libgcc || exit 1 |
pankso@361 | 123 make install-gcc install-target-libgcc |
pankso@361 | 124 cd $PREFIX/lib/gcc/$TARGET/$GCC_VERSION |
paul@382 | 125 echo "Creating symlink for static libgcc: libgcc_eh.a" |
pankso@366 | 126 rm -f libgcc_eh.a && ln -s libgcc.a libgcc_eh.a |
pankso@361 | 127 } |
pankso@361 | 128 |
pankso@403 | 129 # 3. Kernel headers use static GCC |
pankso@403 | 130 linux_headers() { |
pankso@408 | 131 init_compile |
pankso@403 | 132 echo "Extracting: linux-$LINUX_VERSION.tar.bz2" |
pankso@403 | 133 tar xjf $SRC/linux-$LINUX_VERSION.tar.bz2 |
pankso@403 | 134 check_sysroot |
pankso@403 | 135 cd linux-$LINUX_VERSION |
pankso@403 | 136 make mrproper |
pankso@403 | 137 make ARCH=$ARCH headers_check |
pankso@403 | 138 make ARCH=$ARCH headers_install \ |
pankso@403 | 139 INSTALL_HDR_PATH=$HDR_PATH |
pankso@403 | 140 } |
pankso@403 | 141 |
pankso@361 | 142 # 4. GNU Glibc |
pankso@361 | 143 glibc() { |
pankso@408 | 144 init_compile |
pankso@361 | 145 echo "Extracting: glibc-$GLIBC_VERSION.tar.bz2" |
pankso@361 | 146 tar xjf $SRC/glibc-$GLIBC_VERSION.tar.bz2 |
pankso@411 | 147 echo "Configure : $GLIBC_ARGS" |
pankso@361 | 148 [ "$continue" ] || rm -rf glibc-build |
pankso@400 | 149 # Some arch may need glibc-ports and custom CFLAGS |
pankso@403 | 150 case "$ARCH" in |
pankso@361 | 151 arm) |
pankso@401 | 152 #export CFLAGS="-march=armv6 -mtune=generic -g -O2" |
pankso@362 | 153 [ -f "$SRC/glibc-ports-$GLIBC_VERSION.tar.bz2" ] || wget \ |
pankso@362 | 154 http://ftp.gnu.org/gnu/libc/glibc-ports-$GLIBC_VERSION.tar.bz2 \ |
pankso@362 | 155 -O $SRC/glibc-ports-$GLIBC_VERSION.tar.bz2 || exit 1 |
pankso@361 | 156 echo "Extracting: glibc-ports-$GLIBC_VERSION.tar.bz2" |
pankso@361 | 157 rm -rf glibc-$GLIBC_VERSION/ports |
pankso@361 | 158 tar xjf $SRC/glibc-ports-$GLIBC_VERSION.tar.bz2 |
pankso@361 | 159 mv glibc-ports-$GLIBC_VERSION glibc-$GLIBC_VERSION/ports ;; |
pankso@361 | 160 esac |
pankso@404 | 161 #echo "CFLAGS: $CFLAGS" |
pankso@361 | 162 mkdir -p glibc-build && cd glibc-build |
pankso@365 | 163 BUILD_CC="gcc" \ |
pankso@361 | 164 CC="$PREFIX/bin/$TARGET-gcc" \ |
pankso@361 | 165 libc_cv_forced_unwind=yes \ |
pankso@361 | 166 libc_cv_c_cleanup=yes \ |
pankso@361 | 167 ../glibc-$GLIBC_VERSION/configure \ |
pankso@361 | 168 --prefix=$PREFIX \ |
pankso@361 | 169 --host=$TARGET \ |
pankso@361 | 170 --with-headers=$PREFIX/include \ |
pankso@361 | 171 --with-binutils=$PREFIX/bin \ |
pankso@410 | 172 $GLIBC_ARGS |
pankso@361 | 173 make || exit 1 |
pankso@361 | 174 make install |
pankso@403 | 175 # Work around to let GCC find Glibc headers. |
pankso@404 | 176 if [ "$SYSROOT" ]; then |
pankso@404 | 177 cd $SYSROOT |
pankso@404 | 178 ln -s usr/include sys-include |
pankso@404 | 179 else |
pankso@403 | 180 cd $PREFIX/$TARGET |
pankso@403 | 181 rm -rf lib include |
pankso@403 | 182 ln -s ../lib lib |
pankso@403 | 183 ln -s ../include include |
pankso@403 | 184 fi |
pankso@403 | 185 #unset CFLAGS |
pankso@361 | 186 } |
pankso@361 | 187 |
pankso@361 | 188 # 5. GCC final |
pankso@361 | 189 gcc_final() { |
pankso@408 | 190 init_compile |
pankso@361 | 191 if [ ! -d "gcc-$GCC_VERSION" ]; then |
pankso@361 | 192 echo "Extracting: gcc-$GCC_VERSION.tar.bz2" |
pankso@361 | 193 tar xjf $SRC/gcc-$GCC_VERSION.tar.bz2 |
pankso@361 | 194 fi |
pankso@411 | 195 echo "Configure : $GCC_FINAL_ARGS" |
pankso@403 | 196 check_sysroot |
pankso@361 | 197 rm -rf gcc-build |
pankso@361 | 198 mkdir gcc-build && cd gcc-build |
pankso@361 | 199 ../gcc-$GCC_VERSION/configure \ |
pankso@361 | 200 --prefix=$PREFIX \ |
pankso@374 | 201 --libexec=$PREFIX/lib \ |
pankso@361 | 202 --target=$TARGET \ |
pankso@361 | 203 --enable-shared \ |
pankso@361 | 204 --enable-c99 \ |
pankso@361 | 205 --enable-long-long \ |
pankso@361 | 206 --enable-__cxa_atexit \ |
pankso@400 | 207 --with-pkgversion="SliTaz" \ |
pankso@403 | 208 $GCC_FINAL_ARGS $sysroot |
pankso@361 | 209 make || exit 1 |
pankso@361 | 210 make install |
pankso@361 | 211 } |
pankso@361 | 212 |
paul@382 | 213 # Build Busybox to we can create prebuilt tiny rootfs image and boot |
pankso@361 | 214 # from NFS ? |
pankso@361 | 215 cross_busybox() { |
pankso@408 | 216 init_compile |
pankso@361 | 217 echo "Extracting: busybox-$BUSYBOX_VERSION.tar.bz2" |
pankso@361 | 218 tar xjf $SRC/busybox-$BUSYBOX_VERSION.tar.bz2 |
pankso@361 | 219 cd busybox-$BUSYBOX_VERSION |
pankso@400 | 220 # CROSS_COMPILE is exported via init_compile. |
pankso@400 | 221 make defconfig |
pankso@400 | 222 make || exit 1 |
pankso@400 | 223 make install |
pankso@361 | 224 chmod 4755 _install/bin/busybox |
pankso@361 | 225 readelf -h _install/bin/busybox |
pankso@403 | 226 echo "Busybox install path: $(pwd)/_install" |
pankso@361 | 227 } |
pankso@361 | 228 |
pankso@361 | 229 # |
pankso@361 | 230 # Commands |
pankso@361 | 231 # |
pankso@361 | 232 |
pankso@361 | 233 case "$1" in |
pankso@361 | 234 howto|man) |
pankso@361 | 235 doc=/usr/share/doc/cookutils/cross.txt |
pankso@361 | 236 [ -f "$doc" ] && less -E $doc ;; |
pankso@361 | 237 info) |
pankso@361 | 238 init_compile |
pankso@361 | 239 CC=${TARGET}-gcc |
paul@382 | 240 echo -e "\nCross Toolchain information" && separator |
pankso@374 | 241 [ "$config" ] && echo "Config file : $config" |
pankso@361 | 242 cat << EOT |
pankso@374 | 243 Target arch : $ARCH |
pankso@374 | 244 C Compiler : $CC |
pankso@374 | 245 Build directory : $WORK |
pankso@361 | 246 EOT |
pankso@403 | 247 if [ "$SYSROOT" ]; then |
pankso@404 | 248 PREFIX=/usr |
pankso@403 | 249 echo "Arch sysroot : $SYSROOT" |
pankso@403 | 250 else |
pankso@403 | 251 echo "Additional path : $PREFIX/bin" |
pankso@403 | 252 fi |
pankso@361 | 253 separator && echo "" |
pankso@361 | 254 echo "GCC version" && separator |
pankso@404 | 255 if [ -x "$PREFIX/bin/$CC" ]; then |
pankso@374 | 256 $CC -v |
pankso@374 | 257 else |
pankso@374 | 258 echo "No C compiler. To build a toolchain run: cross compile" |
pankso@374 | 259 fi |
pankso@361 | 260 separator && echo "" ;; |
pankso@361 | 261 testsuite) |
pankso@361 | 262 init_compile |
pankso@361 | 263 echo "[COMPILING] $TARGET-gcc -v -Wall -o test.out test.c" \ |
pankso@361 | 264 | tee $logdir/testsuite.log |
pankso@361 | 265 echo 'int main() { return 0; }' > test.c |
pankso@361 | 266 $TARGET-gcc -v -Wall -o test.out test.c 2>&1 | tee -a $logdir/testsuite.log |
pankso@361 | 267 if [ -x /usr/bin/file ]; then |
pankso@361 | 268 echo -e "\n[CHECKING] file test.out" | tee -a $logdir/testsuite.log |
pankso@361 | 269 file test.out | tee -a $logdir/testsuite.log |
pankso@361 | 270 fi |
pankso@361 | 271 echo -e "\n[CHECKING] readelf -h test.out" | tee -a $logdir/testsuite.log |
pankso@361 | 272 readelf -h test.out | tee -a $logdir/testsuite.log ;; |
pankso@365 | 273 check-env) |
pankso@361 | 274 for pkg in mpfr mpfr-dev gmp gmp-dev mpc-library gawk autoconf |
pankso@361 | 275 do |
pankso@361 | 276 if [ ! -d "/var/lib/tazpkg/installed/$pkg" ]; then |
pankso@361 | 277 echo "Missing packages: $pkg" |
pankso@361 | 278 [ "$install" ] && tazpkg -gi $pkg |
pankso@361 | 279 fi |
pankso@361 | 280 done ;; |
pankso@361 | 281 download) |
pankso@361 | 282 download_src ;; |
pankso@361 | 283 show-log) |
pankso@361 | 284 pkg=$2 |
pankso@371 | 285 log=$logdir/$pkg.log |
pankso@371 | 286 if [ ! -f "$log" ]; then |
pankso@371 | 287 echo "No log file found for: $pkg" && exit 1 |
pankso@371 | 288 fi |
pankso@371 | 289 less -E $log ;; |
pankso@361 | 290 binutils) |
pankso@412 | 291 rm -f $logdir/binutils.log |
pankso@361 | 292 binutils 2>&1 | tee $logdir/binutils.log ;; |
pankso@403 | 293 gcc-static) |
pankso@403 | 294 gcc_static 2>&1 | tee $logdir/gcc-static.log ;; |
pankso@361 | 295 linux-headers) |
pankso@363 | 296 linux_headers 2>&1 | tee $logdir/linux-headers.log ;; |
pankso@361 | 297 glibc) |
pankso@361 | 298 glibc 2>&1 | tee $logdir/glibc.log ;; |
pankso@361 | 299 gcc-final) |
pankso@361 | 300 gcc_final 2>&1 | tee $logdir/gcc-final.log ;; |
pankso@361 | 301 busybox) |
pankso@361 | 302 cross_busybox 2>&1 | tee $logdir/busybox.log ;; |
pankso@361 | 303 compile) |
pankso@400 | 304 # Compile the full toolchain. |
pankso@400 | 305 time=$(date +%s) |
pankso@361 | 306 init_compile |
pankso@361 | 307 echo "Compile start: $(date)" | tee $logdir/compile.log |
pankso@361 | 308 download_src |
pankso@361 | 309 binutils 2>&1 | tee $logdir/binutils.log |
pankso@403 | 310 gcc_static 2>&1 | tee $logdir/gcc-static.log |
pankso@361 | 311 linux_headers 2>&1 | tee $logdir/linux-headers.log |
pankso@361 | 312 glibc 2>&1 | tee $logdir/glibc.log |
pankso@361 | 313 gcc_final 2>&1 | tee $logdir/gcc-final.log |
pankso@361 | 314 echo "" |
pankso@365 | 315 echo "Compile end : $(date)" | tee -a $logdir/compile.log |
pankso@400 | 316 time=$(($(date +%s) - $time)) |
pankso@400 | 317 sec=$time |
pankso@400 | 318 div=$(( ($time + 30) / 60)) |
pankso@400 | 319 [ "$div" != 0 ] && min="~ ${div}m" |
pankso@401 | 320 echo "Build time : ${sec}s $min" | tee -a $logdir/compile.log |
pankso@361 | 321 echo "" ;; |
pankso@416 | 322 clean) |
pankso@416 | 323 echo -n "Removing all source files..." |
pankso@416 | 324 rm -rf $WORK/source/* && status |
pankso@416 | 325 [ "$log" ] && rm -f $WORK/log/*.log |
pankso@416 | 326 echo "To clean chroot: rm -rf $PREFIX" ;; |
pankso@371 | 327 clean-tools) |
pankso@371 | 328 # Remove crap :-) |
pankso@371 | 329 init_compile |
pankso@416 | 330 echo "Cleaning : $PREFIX ($(du -sh $PREFIX | awk '{print $1}'))" |
pankso@371 | 331 for dir in info man locale |
pankso@371 | 332 do |
pankso@416 | 333 echo -n "Removing : $dir" |
pankso@371 | 334 rm -rf $PREFIX/share && status |
pankso@371 | 335 done |
pankso@416 | 336 rm -f $PREFIX/lib/*-gdb.py |
pankso@416 | 337 #echo -n "Stripping : shared libs" |
pankso@416 | 338 #${TARGET}-strip -s $PREFIX/lib/*.so.* |
pankso@416 | 339 #status |
pankso@416 | 340 echo -n "Stripping : GCC libs" |
pankso@416 | 341 ${TARGET}-strip -s $PREFIX/$TARGET/lib/gcc/$TARGET/*/cc1* |
pankso@416 | 342 ${TARGET}-strip -s $PREFIX/$TARGET/lib/gcc/$TARGET/*/lto* |
pankso@416 | 343 sleep 1 && status |
pankso@416 | 344 echo -n "Stripping : binaries" |
pankso@371 | 345 for bin in $PREFIX/bin/${TARGET}-* |
pankso@371 | 346 do |
pankso@371 | 347 [ "$bin" == "$PREFIX/bin/${TARGET}-strip" ] && continue |
pankso@371 | 348 if [ -x "$bin" ]; then |
pankso@371 | 349 ${TARGET}-strip -s $bin 2>/dev/null |
pankso@371 | 350 fi |
pankso@371 | 351 done && status |
pankso@416 | 352 echo -n "Tools size : " && du -sh $PREFIX | awk '{print $1}' ;; |
pankso@366 | 353 gen-rootfs) |
pankso@366 | 354 # |
pankso@366 | 355 # TESTING |
pankso@366 | 356 # |
pankso@366 | 357 # Create a bootable rootfs ? dd for an HD image ? |
pankso@366 | 358 init_compile |
pankso@366 | 359 rootfs=/tmp/cross/rootfs |
pankso@366 | 360 tarball="rootfs.tar.bz2" |
pankso@366 | 361 rm -rf $rootfs && mkdir -p $rootfs |
pankso@366 | 362 cd /tmp/cross |
pankso@366 | 363 echo -n "Installing SliTaz base files..." |
pankso@366 | 364 tar xzf $SRC/slitaz-base-files-5.2.tar.gz |
pankso@366 | 365 cp -a slitaz-base-files-*/rootfs/* $rootfs |
pankso@366 | 366 status |
pankso@366 | 367 echo -n "Installing Busybox..." |
pankso@366 | 368 cp -a $source/busybox-$BUSYBOX_VERSION/_install/* $rootfs |
pankso@366 | 369 status |
pankso@366 | 370 echo -n "Creating tarball: $tarball" |
pankso@366 | 371 tar cjf $tarball rootfs |
pankso@366 | 372 status |
pankso@366 | 373 echo -n "Moving rootfs to: $WORK" |
pankso@366 | 374 mv $tarball $WORK |
pankso@366 | 375 status |
pankso@366 | 376 du -sh $WORK/$tarball |
pankso@366 | 377 rm -rf /tmp/cross ;; |
pankso@416 | 378 gen-prebuilt) |
pankso@416 | 379 # |
pankso@416 | 380 # TESTING |
pankso@416 | 381 # |
pankso@416 | 382 # Create a prebuilt cross toolchain tarball (or package ?) |
pankso@416 | 383 init_compile |
pankso@416 | 384 cd /usr/cross |
pankso@416 | 385 tarball="slitaz-cross-$ARCH-toolchain.tar.bz2" |
pankso@416 | 386 echo -n "Creating prebuilt $ARCH toolchain tarball..." |
pankso@416 | 387 tar cjf $tarball $ARCH |
pankso@416 | 388 status |
pankso@416 | 389 mv -f $tarball $WORK |
pankso@416 | 390 #echo "Tarball size: $(du -sh $WORK/$tarball | awk '{print $1}')" |
pankso@416 | 391 du -sh $WORK/$tarball ;; |
pankso@361 | 392 *) |
pankso@361 | 393 usage ;; |
pankso@361 | 394 esac |
pankso@361 | 395 |