|  | #!/bin/sh | 
|  | # Copyright (C) Martin Schlemmer <azarah@nosferatu.za.org> | 
|  | # Copyright (C) 2006 Sam Ravnborg <sam@ravnborg.org> | 
|  | # | 
|  | # Released under the terms of the GNU GPL | 
|  | # | 
|  | # Generate a cpio packed initramfs. It uses gen_init_cpio to generate | 
|  | # the cpio archive. | 
|  | # This script assumes that gen_init_cpio is located in usr/ directory | 
|  |  | 
|  | # error out on errors | 
|  | set -e | 
|  |  | 
|  | usage() { | 
|  | cat << EOF | 
|  | Usage: | 
|  | $0 [-o <file>] [-l <dep_list>] [-u <uid>] [-g <gid>] {-d | <cpio_source>} ... | 
|  | -o <file>      Create initramfs file named <file> by using gen_init_cpio | 
|  | -l <dep_list>  Create dependency list named <dep_list> | 
|  | -u <uid>       User ID to map to user ID 0 (root). | 
|  | <uid> is only meaningful if <cpio_source> is a | 
|  | directory.  "squash" forces all files to uid 0. | 
|  | -g <gid>       Group ID to map to group ID 0 (root). | 
|  | <gid> is only meaningful if <cpio_source> is a | 
|  | directory.  "squash" forces all files to gid 0. | 
|  | <cpio_source>  File list or directory for cpio archive. | 
|  | If <cpio_source> is a .cpio file it will be used | 
|  | as direct input to initramfs. | 
|  |  | 
|  | All options except -o and -l may be repeated and are interpreted | 
|  | sequentially and immediately.  -u and -g states are preserved across | 
|  | <cpio_source> options so an explicit "-u 0 -g 0" is required | 
|  | to reset the root/group mapping. | 
|  | EOF | 
|  | } | 
|  |  | 
|  | # awk style field access | 
|  | # $1 - field number; rest is argument string | 
|  | field() { | 
|  | shift $1 ; echo $1 | 
|  | } | 
|  |  | 
|  | filetype() { | 
|  | local argv1="$1" | 
|  |  | 
|  | # symlink test must come before file test | 
|  | if [ -L "${argv1}" ]; then | 
|  | echo "slink" | 
|  | elif [ -f "${argv1}" ]; then | 
|  | echo "file" | 
|  | elif [ -d "${argv1}" ]; then | 
|  | echo "dir" | 
|  | elif [ -b "${argv1}" -o -c "${argv1}" ]; then | 
|  | echo "nod" | 
|  | elif [ -p "${argv1}" ]; then | 
|  | echo "pipe" | 
|  | elif [ -S "${argv1}" ]; then | 
|  | echo "sock" | 
|  | else | 
|  | echo "invalid" | 
|  | fi | 
|  | return 0 | 
|  | } | 
|  |  | 
|  | print_mtime() { | 
|  | local my_mtime="0" | 
|  |  | 
|  | if [ -e "$1" ]; then | 
|  | my_mtime=$(find "$1" -printf "%T@\n" | sort -r | head -n 1) | 
|  | fi | 
|  |  | 
|  | echo "# Last modified: ${my_mtime}" >> $cpio_list | 
|  | echo "" >> $cpio_list | 
|  | } | 
|  |  | 
|  | list_parse() { | 
|  | if [ -z "$dep_list" -o -L "$1" ]; then | 
|  | return | 
|  | fi | 
|  | echo "$1" | sed 's/:/\\:/g; s/$/ \\/' >> $dep_list | 
|  | } | 
|  |  | 
|  | # for each file print a line in following format | 
|  | # <filetype> <name> <path to file> <octal mode> <uid> <gid> | 
|  | # for links, devices etc the format differs. See gen_init_cpio for details | 
|  | parse() { | 
|  | local location="$1" | 
|  | local name="/${location#${srcdir}}" | 
|  | # change '//' into '/' | 
|  | name=$(echo "$name" | sed -e 's://*:/:g') | 
|  | local mode="$2" | 
|  | local uid="$3" | 
|  | local gid="$4" | 
|  | local ftype=$(filetype "${location}") | 
|  | # remap uid/gid to 0 if necessary | 
|  | [ "$root_uid" = "squash" ] && uid=0 || [ "$uid" -eq "$root_uid" ] && uid=0 | 
|  | [ "$root_gid" = "squash" ] && gid=0 || [ "$gid" -eq "$root_gid" ] && gid=0 | 
|  | local str="${mode} ${uid} ${gid}" | 
|  |  | 
|  | [ "${ftype}" = "invalid" ] && return 0 | 
|  | [ "${location}" = "${srcdir}" ] && return 0 | 
|  |  | 
|  | case "${ftype}" in | 
|  | "file") | 
|  | str="${ftype} ${name} ${location} ${str}" | 
|  | ;; | 
|  | "nod") | 
|  | local dev="`LC_ALL=C ls -l "${location}"`" | 
|  | local maj=`field 5 ${dev}` | 
|  | local min=`field 6 ${dev}` | 
|  | maj=${maj%,} | 
|  |  | 
|  | [ -b "${location}" ] && dev="b" || dev="c" | 
|  |  | 
|  | str="${ftype} ${name} ${str} ${dev} ${maj} ${min}" | 
|  | ;; | 
|  | "slink") | 
|  | local target=`readlink "${location}"` | 
|  | str="${ftype} ${name} ${target} ${str}" | 
|  | ;; | 
|  | *) | 
|  | str="${ftype} ${name} ${str}" | 
|  | ;; | 
|  | esac | 
|  |  | 
|  | echo "${str}" >> $cpio_list | 
|  |  | 
|  | return 0 | 
|  | } | 
|  |  | 
|  | unknown_option() { | 
|  | printf "ERROR: unknown option \"$arg\"\n" >&2 | 
|  | printf "If the filename validly begins with '-', " >&2 | 
|  | printf "then it must be prefixed\n" >&2 | 
|  | printf "by './' so that it won't be interpreted as an option." >&2 | 
|  | printf "\n" >&2 | 
|  | usage >&2 | 
|  | exit 1 | 
|  | } | 
|  |  | 
|  | header() { | 
|  | printf "\n#####################\n# $1\n" >> $cpio_list | 
|  | } | 
|  |  | 
|  | # process one directory (incl sub-directories) | 
|  | dir_filelist() { | 
|  | header "$1" | 
|  |  | 
|  | srcdir=$(echo "$1" | sed -e 's://*:/:g') | 
|  | dirlist=$(find "${srcdir}" -printf "%p %m %U %G\n" | LANG=C sort) | 
|  |  | 
|  | # If $dirlist is only one line, then the directory is empty | 
|  | if [  "$(echo "${dirlist}" | wc -l)" -gt 1 ]; then | 
|  | print_mtime "$1" | 
|  |  | 
|  | echo "${dirlist}" | \ | 
|  | while read x; do | 
|  | list_parse $x | 
|  | parse $x | 
|  | done | 
|  | fi | 
|  | } | 
|  |  | 
|  | input_file() { | 
|  | source="$1" | 
|  | if [ -f "$1" ]; then | 
|  | # If a regular file is specified, assume it is in | 
|  | # gen_init_cpio format | 
|  | header "$1" | 
|  | print_mtime "$1" >> $cpio_list | 
|  | cat "$1"         >> $cpio_list | 
|  | if [ -n "$dep_list" ]; then | 
|  | echo "$1 \\"  >> $dep_list | 
|  | cat "$1" | while read type dir file perm ; do | 
|  | if [ "$type" = "file" ]; then | 
|  | echo "$file \\" >> $dep_list | 
|  | fi | 
|  | done | 
|  | fi | 
|  | elif [ -d "$1" ]; then | 
|  | # If a directory is specified then add all files in it to fs | 
|  | dir_filelist "$1" | 
|  | else | 
|  | echo "  ${prog}: Cannot open '$1'" >&2 | 
|  | exit 1 | 
|  | fi | 
|  | } | 
|  |  | 
|  | prog=$0 | 
|  | root_uid=0 | 
|  | root_gid=0 | 
|  | dep_list= | 
|  | cpio_list=$(mktemp ${TMPDIR:-/tmp}/cpiolist.XXXXXX) | 
|  | output="/dev/stdout" | 
|  |  | 
|  | trap "rm -f $cpio_list" EXIT | 
|  |  | 
|  | while [ $# -gt 0 ]; do | 
|  | arg="$1" | 
|  | shift | 
|  | case "$arg" in | 
|  | "-l")	# files included in initramfs - used by kbuild | 
|  | dep_list="$1" | 
|  | echo "deps_initramfs := \\" > $dep_list | 
|  | shift | 
|  | ;; | 
|  | "-o")	# generate cpio image named $1 | 
|  | output="$1" | 
|  | shift | 
|  | ;; | 
|  | "-u")	# map $1 to uid=0 (root) | 
|  | root_uid="$1" | 
|  | [ "$root_uid" = "-1" ] && root_uid=$(id -u || echo 0) | 
|  | shift | 
|  | ;; | 
|  | "-g")	# map $1 to gid=0 (root) | 
|  | root_gid="$1" | 
|  | [ "$root_gid" = "-1" ] && root_gid=$(id -g || echo 0) | 
|  | shift | 
|  | ;; | 
|  | "-h") | 
|  | usage | 
|  | exit 0 | 
|  | ;; | 
|  | *) | 
|  | case "$arg" in | 
|  | "-"*) | 
|  | unknown_option | 
|  | ;; | 
|  | *)	# input file/dir - process it | 
|  | input_file "$arg" | 
|  | ;; | 
|  | esac | 
|  | ;; | 
|  | esac | 
|  | done | 
|  |  | 
|  | # If output_file is set we will generate cpio archive | 
|  | # we are careful to delete tmp files | 
|  | timestamp= | 
|  | if test -n "$KBUILD_BUILD_TIMESTAMP"; then | 
|  | timestamp="$(date -d"$KBUILD_BUILD_TIMESTAMP" +%s || :)" | 
|  | if test -n "$timestamp"; then | 
|  | timestamp="-t $timestamp" | 
|  | fi | 
|  | fi | 
|  | usr/gen_init_cpio $timestamp $cpio_list > $output |