Показать шестнадцатеричные числа файла - PullRequest
18 голосов
/ 05 января 2010

Я хочу создать программу bash, которая может читать файл, например * .bin, и печатать все его шестнадцатеричные числа, как это делают шестнадцатеричные редакторы. Где я могу начать?

Ответы [ 4 ]

26 голосов
/ 05 января 2010

Используйте команду od,

od -t x1  filename
24 голосов
/ 05 января 2010

Редактировать: Добавлена ​​функциональность "bytestream". Если имя сценария содержит слово «поток» (например, это символическая ссылка, такая как ln -s bash-hexdump bash-hexdump-stream и работает как ./bash-hexdump-stream), оно выведет непрерывный поток шестнадцатеричных символов, представляющих содержимое файла. В противном случае его вывод будет выглядеть как hexdump -C.

Требуется куча хитрости, так как Bash не очень хорош в двоичном коде:

#!/bin/bash
# bash-hexdump
# by Dennis Williamson - 2010-01-04
# in response to /1440865/pokazat-shestnadtsaterichnye-chisla-faila
# usage: bash-hexdump file

if [[ -z "$1" ]]
then
    exec 3<&0                           # read stdin
    [[ -p /dev/stdin ]] || tty="yes"    # no pipe
else
    exec 3<"$1"            # read file
fi

# if the script name contains "stream" then output will be continuous hex digits
# like hexdump -ve '1/1 "%.2x"'
[[ $0 =~ stream ]] && nostream=false || nostream=true

saveIFS="$IFS"
IFS=""                     # disables interpretation of \t, \n and space
saveLANG="$LANG"
LANG=C                     # allows characters > 0x7F
bytecount=0
valcount=0
$nostream && printf "%08x  " $bytecount
while read -s -u 3 -d '' -r -n 1 char    # -d '' allows newlines, -r allows \
do
    ((bytecount++))
    printf -v val "%02x" "'$char"    # see below for the ' trick
    [[ "$tty" == "yes" && "$val" == "04" ]] && break    # exit on ^D
    echo -n "$val"
    $nostream && echo -n " "
    ((valcount++))
    if [[ "$val" < 20 || "$val" > 7e ]]
    then
        string+="."                  # show unprintable characters as a dot
    else
        string+=$char
    fi
    if $nostream && (( bytecount % 8 == 0 ))      # add a space down the middle
    then
        echo -n " "
    fi
    if (( bytecount % 16 == 0 ))   # print 16 values per line
    then
        $nostream && echo "|$string|"
        string=''
        valcount=0
        $nostream && printf "%08x  " $bytecount
    fi
done

if [[ "$string" != "" ]]            # if the last line wasn't full, pad it out
then
    length=${#string}
    if (( length > 7 ))
    then
        ((length--))
    fi
    (( length += (16 - valcount) * 3 + 4))
    $nostream && printf "%${length}s\n" "|$string|"
    $nostream && printf "%08x  " $bytecount
fi
$nostream && echo

LANG="$saveLANG";
IFS="$saveIFS"

Трюк с апострофом задокументирован здесь . В соответствующей части написано:

Если ведущим персонажем является одинарные или двойные кавычки, значение должно быть числовым значением в основной кодовый набор символ, следующий за одинарной кавычкой или двойная кавычка.

Вот некоторые результаты из скрипта, показывающие первые несколько строк моего /bin/bash плюс еще несколько:

00000000  7f 45 4c 46 01 01 01 00  00 00 00 00 00 00 00 00  |.ELF............|
00000010  02 00 03 00 01 00 00 00  e0 1e 06 08 34 00 00 00  |............4...|
00000020  c4 57 0d 00 00 00 00 00  34 00 20 00 09 00 28 00  |.W......4. ...(.|
00000030  1d 00 1c 00 06 00 00 00  34 00 00 00 34 80 04 08  |........4...4...|
. . .
00000150  01 00 00 00 2f 6c 69 62  2f 6c 64 2d 6c 69 6e 75  |..../lib/ld-linu|
00000160  78 2e 73 6f 2e 32 00 00  04 00 00 00 10 00 00 00  |x.so.2..........|
00000170  01 00 00 00 47 4e 55 00  00 00 00 00 02 00 00 00  |....GNU.........|
3 голосов
/ 05 января 2010

Вы можете использовать од. "od -x file" Зачем изобретать это колесо?

1 голос
/ 05 января 2010

вы также можете использовать hexdump, если он у вас есть

hexdump -x /usr/bin/binaryfile
...