|  | # Copyright (C) 2014 OpenWrt.org | 
|  | # | 
|  |  | 
|  | . /lib/functions.sh | 
|  |  | 
|  | # 'kernel' partition or UBI volume on NAND contains the kernel | 
|  | CI_KERNPART="${CI_KERNPART:-kernel}" | 
|  |  | 
|  | # 'ubi' partition on NAND contains UBI | 
|  | # There are also CI_KERN_UBIPART and CI_ROOT_UBIPART if kernel | 
|  | # and rootfs are on separated UBIs. | 
|  | CI_UBIPART="${CI_UBIPART:-ubi}" | 
|  |  | 
|  | # 'rootfs' UBI volume on NAND contains the rootfs | 
|  | CI_ROOTPART="${CI_ROOTPART:-rootfs}" | 
|  |  | 
|  | ubi_mknod() { | 
|  | local dir="$1" | 
|  | local dev="/dev/$(basename $dir)" | 
|  |  | 
|  | [ -e "$dev" ] && return 0 | 
|  |  | 
|  | local devid="$(cat $dir/dev)" | 
|  | local major="${devid%%:*}" | 
|  | local minor="${devid##*:}" | 
|  | mknod "$dev" c $major $minor | 
|  | } | 
|  |  | 
|  | nand_find_volume() { | 
|  | local ubidevdir ubivoldir | 
|  | ubidevdir="/sys/class/ubi/" | 
|  | [ ! -d "$ubidevdir" ] && return 1 | 
|  | for ubivoldir in $ubidevdir/${1}_*; do | 
|  | [ ! -d "$ubivoldir" ] && continue | 
|  | if [ "$( cat $ubivoldir/name )" = "$2" ]; then | 
|  | basename $ubivoldir | 
|  | ubi_mknod "$ubivoldir" | 
|  | return 0 | 
|  | fi | 
|  | done | 
|  | } | 
|  |  | 
|  | nand_find_ubi() { | 
|  | local ubidevdir ubidev mtdnum cmtdnum | 
|  | mtdnum="$( find_mtd_index $1 )" | 
|  | [ ! "$mtdnum" ] && return 1 | 
|  | for ubidevdir in /sys/class/ubi/ubi*; do | 
|  | [ ! -e "$ubidevdir/mtd_num" ] && continue | 
|  | cmtdnum="$( cat $ubidevdir/mtd_num )" | 
|  | if [ "$mtdnum" = "$cmtdnum" ]; then | 
|  | ubidev=$( basename $ubidevdir ) | 
|  | ubi_mknod "$ubidevdir" | 
|  | echo $ubidev | 
|  | return 0 | 
|  | fi | 
|  | done | 
|  | } | 
|  |  | 
|  | nand_get_magic_long() { | 
|  | ($2 < "$1" | dd bs=4 "skip=${3:-0}" count=1 | hexdump -v -n 4 -e '1/1 "%02x"') 2> /dev/null | 
|  | } | 
|  |  | 
|  | get_magic_long_tar() { | 
|  | ($2 < "$1" | tar xOf - "$3" | dd bs=4 count=1 | hexdump -v -n 4 -e '1/1 "%02x"') 2> /dev/null | 
|  | } | 
|  |  | 
|  | identify() { | 
|  | identify_magic_long $(nand_get_magic_long "$@") | 
|  | } | 
|  |  | 
|  | identify_tar() { | 
|  | identify_magic_long $(get_magic_long_tar "$@") | 
|  | } | 
|  |  | 
|  | identify_if_gzip() { | 
|  | if [ "$(identify "$1" "cat")" = gzip ]; then echo -n z; fi | 
|  | } | 
|  |  | 
|  | nand_restore_config() { | 
|  | local ubidev=$( nand_find_ubi "${CI_ROOT_UBIPART:-$CI_UBIPART}" ) | 
|  | local ubivol="$( nand_find_volume $ubidev rootfs_data )" | 
|  | if [ ! "$ubivol" ]; then | 
|  | ubivol="$( nand_find_volume $ubidev "$CI_ROOTPART" )" | 
|  | if [ ! "$ubivol" ]; then | 
|  | echo "cannot find ubifs data volume" | 
|  | return 1 | 
|  | fi | 
|  | fi | 
|  | mkdir /tmp/new_root | 
|  | if ! mount -t ubifs /dev/$ubivol /tmp/new_root; then | 
|  | echo "cannot mount ubifs volume $ubivol" | 
|  | rmdir /tmp/new_root | 
|  | return 1 | 
|  | fi | 
|  | if mv "$1" "/tmp/new_root/$BACKUP_FILE"; then | 
|  | if umount /tmp/new_root; then | 
|  | echo "configuration saved" | 
|  | rmdir /tmp/new_root | 
|  | return 0 | 
|  | fi | 
|  | else | 
|  | umount /tmp/new_root | 
|  | fi | 
|  | echo "could not save configuration to ubifs volume $ubivol" | 
|  | rmdir /tmp/new_root | 
|  | return 1 | 
|  | } | 
|  |  | 
|  | nand_remove_ubiblock() { | 
|  | local ubivol="$1" | 
|  |  | 
|  | local ubiblk="ubiblock${ubivol:3}" | 
|  | if [ -e "/dev/$ubiblk" ]; then | 
|  | umount "/dev/$ubiblk" 2>/dev/null && echo "unmounted /dev/$ubiblk" || : | 
|  | if ! ubiblock -r "/dev/$ubivol"; then | 
|  | echo "cannot remove $ubiblk" | 
|  | return 1 | 
|  | fi | 
|  | fi | 
|  | } | 
|  |  | 
|  | nand_attach_ubi() { | 
|  | local ubipart="$1" | 
|  | local has_env="${2:-0}" | 
|  |  | 
|  | local mtdnum="$( find_mtd_index "$ubipart" )" | 
|  | if [ ! "$mtdnum" ]; then | 
|  | >&2 echo "cannot find ubi mtd partition $ubipart" | 
|  | return 1 | 
|  | fi | 
|  |  | 
|  | local ubidev="$( nand_find_ubi "$ubipart" )" | 
|  | if [ ! "$ubidev" ]; then | 
|  | >&2 ubiattach -m "$mtdnum" | 
|  | ubidev="$( nand_find_ubi "$ubipart" )" | 
|  |  | 
|  | if [ ! "$ubidev" ]; then | 
|  | >&2 ubiformat /dev/mtd$mtdnum -y | 
|  | >&2 ubiattach -m "$mtdnum" | 
|  | ubidev="$( nand_find_ubi "$ubipart" )" | 
|  |  | 
|  | if [ ! "$ubidev" ]; then | 
|  | >&2 echo "cannot attach ubi mtd partition $ubipart" | 
|  | return 1 | 
|  | fi | 
|  |  | 
|  | if [ "$has_env" -gt 0 ]; then | 
|  | >&2 ubimkvol /dev/$ubidev -n 0 -N ubootenv -s 1MiB | 
|  | >&2 ubimkvol /dev/$ubidev -n 1 -N ubootenv2 -s 1MiB | 
|  | fi | 
|  | fi | 
|  | fi | 
|  |  | 
|  | echo "$ubidev" | 
|  | return 0 | 
|  | } | 
|  |  | 
|  | nand_detach_ubi() { | 
|  | local ubipart="$1" | 
|  |  | 
|  | local mtdnum="$( find_mtd_index "$ubipart" )" | 
|  | if [ ! "$mtdnum" ]; then | 
|  | echo "cannot find ubi mtd partition $ubipart" | 
|  | return 1 | 
|  | fi | 
|  |  | 
|  | local ubidev="$( nand_find_ubi "$ubipart" )" | 
|  | if [ "$ubidev" ]; then | 
|  | for ubivol in $(find /dev -name "${ubidev}_*" -maxdepth 1 | sort); do | 
|  | ubivol="${ubivol:5}" | 
|  | nand_remove_ubiblock "$ubivol" || : | 
|  | umount "/dev/$ubivol" && echo "unmounted /dev/$ubivol" || : | 
|  | done | 
|  | if ! ubidetach -m "$mtdnum"; then | 
|  | echo "cannot detach ubi mtd partition $ubipart" | 
|  | return 1 | 
|  | fi | 
|  | fi | 
|  | } | 
|  |  | 
|  | nand_upgrade_prepare_ubi() { | 
|  | local rootfs_length="$1" | 
|  | local rootfs_type="$2" | 
|  | local rootfs_data_max="$(fw_printenv -n rootfs_data_max 2> /dev/null)" | 
|  | [ -n "$rootfs_data_max" ] && rootfs_data_max=$((rootfs_data_max)) | 
|  |  | 
|  | local kernel_length="$3" | 
|  | local has_env="${4:-0}" | 
|  | local kern_ubidev | 
|  | local root_ubidev | 
|  |  | 
|  | [ -n "$rootfs_length" -o -n "$kernel_length" ] || return 1 | 
|  |  | 
|  | if [ -n "$CI_KERN_UBIPART" -a -n "$CI_ROOT_UBIPART" ]; then | 
|  | kern_ubidev="$( nand_attach_ubi "$CI_KERN_UBIPART" "$has_env" )" | 
|  | [ -n "$kern_ubidev" ] || return 1 | 
|  | root_ubidev="$( nand_attach_ubi "$CI_ROOT_UBIPART" )" | 
|  | [ -n "$root_ubidev" ] || return 1 | 
|  | else | 
|  | kern_ubidev="$( nand_attach_ubi "$CI_UBIPART" "$has_env" )" | 
|  | [ -n "$kern_ubidev" ] || return 1 | 
|  | root_ubidev="$kern_ubidev" | 
|  | fi | 
|  |  | 
|  | local kern_ubivol="$( nand_find_volume $kern_ubidev "$CI_KERNPART" )" | 
|  | local root_ubivol="$( nand_find_volume $root_ubidev "$CI_ROOTPART" )" | 
|  | local data_ubivol="$( nand_find_volume $root_ubidev rootfs_data )" | 
|  | [ "$root_ubivol" = "$kern_ubivol" ] && root_ubivol= | 
|  |  | 
|  | # remove ubiblocks | 
|  | [ "$kern_ubivol" ] && { nand_remove_ubiblock $kern_ubivol || return 1; } | 
|  | [ "$root_ubivol" ] && { nand_remove_ubiblock $root_ubivol || return 1; } | 
|  | [ "$data_ubivol" ] && { nand_remove_ubiblock $data_ubivol || return 1; } | 
|  |  | 
|  | # kill volumes | 
|  | [ "$kern_ubivol" ] && ubirmvol /dev/$kern_ubidev -N "$CI_KERNPART" || : | 
|  | [ "$root_ubivol" ] && ubirmvol /dev/$root_ubidev -N "$CI_ROOTPART" || : | 
|  | [ "$data_ubivol" ] && ubirmvol /dev/$root_ubidev -N rootfs_data || : | 
|  |  | 
|  | # create kernel vol | 
|  | if [ -n "$kernel_length" ]; then | 
|  | if ! ubimkvol /dev/$kern_ubidev -N "$CI_KERNPART" -s $kernel_length; then | 
|  | echo "cannot create kernel volume" | 
|  | return 1; | 
|  | fi | 
|  | fi | 
|  |  | 
|  | # create rootfs vol | 
|  | if [ -n "$rootfs_length" ]; then | 
|  | local rootfs_size_param | 
|  | if [ "$rootfs_type" = "ubifs" ]; then | 
|  | rootfs_size_param="-m" | 
|  | else | 
|  | rootfs_size_param="-s $rootfs_length" | 
|  | fi | 
|  | if ! ubimkvol /dev/$root_ubidev -N "$CI_ROOTPART" $rootfs_size_param; then | 
|  | echo "cannot create rootfs volume" | 
|  | return 1; | 
|  | fi | 
|  | fi | 
|  |  | 
|  | # create rootfs_data vol for non-ubifs rootfs | 
|  | if [ "$rootfs_type" != "ubifs" ]; then | 
|  | local rootfs_data_size_param="-m" | 
|  | if [ -n "$rootfs_data_max" ]; then | 
|  | rootfs_data_size_param="-s $rootfs_data_max" | 
|  | fi | 
|  | if ! ubimkvol /dev/$root_ubidev -N rootfs_data $rootfs_data_size_param; then | 
|  | if ! ubimkvol /dev/$root_ubidev -N rootfs_data -m; then | 
|  | echo "cannot initialize rootfs_data volume" | 
|  | return 1 | 
|  | fi | 
|  | fi | 
|  | fi | 
|  |  | 
|  | return 0 | 
|  | } | 
|  |  | 
|  | # Write the UBI image to MTD ubi partition | 
|  | nand_upgrade_ubinized() { | 
|  | local ubi_file="$1" | 
|  | local cmd="$2" | 
|  |  | 
|  | local ubi_length=$( ($cmd < "$ubi_file" | wc -c) 2> /dev/null) | 
|  |  | 
|  | nand_detach_ubi "$CI_UBIPART" || return 1 | 
|  |  | 
|  | local mtdnum="$( find_mtd_index "$CI_UBIPART" )" | 
|  | $cmd < "$ubi_file" | ubiformat "/dev/mtd$mtdnum" -S "$ubi_length" -y -f - && ubiattach -m "$mtdnum" | 
|  | } | 
|  |  | 
|  | # Write the UBIFS image to UBI rootfs volume | 
|  | nand_upgrade_ubifs() { | 
|  | local ubifs_file="$1" | 
|  | local cmd="$2" | 
|  |  | 
|  | local ubifs_length=$( ($cmd < "$ubifs_file" | wc -c) 2> /dev/null) | 
|  |  | 
|  | nand_upgrade_prepare_ubi "$ubifs_length" "ubifs" "" "" || return 1 | 
|  |  | 
|  | local ubidev="$( nand_find_ubi "$CI_UBIPART" )" | 
|  | local root_ubivol="$(nand_find_volume $ubidev "$CI_ROOTPART")" | 
|  | $cmd < "$ubifs_file" | ubiupdatevol /dev/$root_ubivol -s "$ubifs_length" - | 
|  | } | 
|  |  | 
|  | # Write the FIT image to UBI kernel volume | 
|  | nand_upgrade_fit() { | 
|  | local fit_file="$1" | 
|  | local cmd="$2" | 
|  |  | 
|  | local fit_length=$( ($cmd < "$fit_file" | wc -c) 2> /dev/null) | 
|  |  | 
|  | nand_upgrade_prepare_ubi "" "" "$fit_length" "1" || return 1 | 
|  |  | 
|  | local fit_ubidev="$(nand_find_ubi "$CI_UBIPART")" | 
|  | local fit_ubivol="$(nand_find_volume $fit_ubidev "$CI_KERNPART")" | 
|  | $cmd < "$fit_file" | ubiupdatevol /dev/$fit_ubivol -s "$fit_length" - | 
|  | } | 
|  |  | 
|  | # Write images in the TAR file to MTD partitions and/or UBI volumes as required | 
|  | nand_upgrade_tar() { | 
|  | local tar_file="$1" | 
|  | local cmd="${2:-cat}" | 
|  | local jffs2_markers="${CI_JFFS2_CLEAN_MARKERS:-0}" | 
|  |  | 
|  | # WARNING: This fails if tar contains more than one 'sysupgrade-*' directory. | 
|  | local board_dir="$($cmd < "$tar_file" | tar tf - | grep -m 1 '^sysupgrade-.*/$')" | 
|  | board_dir="${board_dir%/}" | 
|  |  | 
|  | local kernel_mtd kernel_length | 
|  | if [ "$CI_KERNPART" != "none" ]; then | 
|  | kernel_mtd="$(find_mtd_index "$CI_KERNPART")" | 
|  | kernel_length=$( ($cmd < "$tar_file" | tar xOf - "$board_dir/kernel" | wc -c) 2> /dev/null) | 
|  | [ "$kernel_length" = 0 ] && kernel_length= | 
|  | fi | 
|  | local rootfs_length=$( ($cmd < "$tar_file" | tar xOf - "$board_dir/root" | wc -c) 2> /dev/null) | 
|  | [ "$rootfs_length" = 0 ] && rootfs_length= | 
|  | local rootfs_type | 
|  | [ "$rootfs_length" ] && rootfs_type="$(identify_tar "$tar_file" "$cmd" "$board_dir/root")" | 
|  |  | 
|  | local ubi_kernel_length | 
|  | if [ "$kernel_length" ]; then | 
|  | if [ "$kernel_mtd" ]; then | 
|  | # On some devices, the raw kernel and ubi partitions overlap. | 
|  | # These devices brick if the kernel partition is erased. | 
|  | # Hence only invalidate kernel for now. | 
|  | dd if=/dev/zero bs=4096 count=1 2> /dev/null | \ | 
|  | mtd write - "$CI_KERNPART" | 
|  | else | 
|  | ubi_kernel_length="$kernel_length" | 
|  | fi | 
|  | fi | 
|  |  | 
|  | local has_env=0 | 
|  | nand_upgrade_prepare_ubi "$rootfs_length" "$rootfs_type" "$ubi_kernel_length" "$has_env" || return 1 | 
|  |  | 
|  | if [ "$rootfs_length" ]; then | 
|  | local ubidev="$( nand_find_ubi "${CI_ROOT_UBIPART:-$CI_UBIPART}" )" | 
|  | local root_ubivol="$( nand_find_volume $ubidev "$CI_ROOTPART" )" | 
|  | $cmd < "$tar_file" | tar xOf - "$board_dir/root" | \ | 
|  | ubiupdatevol /dev/$root_ubivol -s "$rootfs_length" - | 
|  | fi | 
|  | if [ "$kernel_length" ]; then | 
|  | if [ "$kernel_mtd" ]; then | 
|  | if [ "$jffs2_markers" = 1 ]; then | 
|  | flash_erase -j "/dev/mtd${kernel_mtd}" 0 0 | 
|  | $cmd < "$tar_file" | tar xOf - "$board_dir/kernel" | \ | 
|  | nandwrite "/dev/mtd${kernel_mtd}" - | 
|  | else | 
|  | $cmd < "$tar_file" | tar xOf - "$board_dir/kernel" | \ | 
|  | mtd write - "$CI_KERNPART" | 
|  | fi | 
|  | else | 
|  | local ubidev="$( nand_find_ubi "${CI_KERN_UBIPART:-$CI_UBIPART}" )" | 
|  | local kern_ubivol="$( nand_find_volume $ubidev "$CI_KERNPART" )" | 
|  | $cmd < "$tar_file" | tar xOf - "$board_dir/kernel" | \ | 
|  | ubiupdatevol /dev/$kern_ubivol -s "$kernel_length" - | 
|  | fi | 
|  | fi | 
|  |  | 
|  | return 0 | 
|  | } | 
|  |  | 
|  | nand_verify_if_gzip_file() { | 
|  | local file="$1" | 
|  | local cmd="$2" | 
|  |  | 
|  | if [ "$cmd" = zcat ]; then | 
|  | echo "verifying compressed sysupgrade file integrity" | 
|  | if ! gzip -t "$file"; then | 
|  | echo "corrupted compressed sysupgrade file" | 
|  | return 1 | 
|  | fi | 
|  | fi | 
|  | } | 
|  |  | 
|  | nand_verify_tar_file() { | 
|  | local file="$1" | 
|  | local cmd="$2" | 
|  |  | 
|  | echo "verifying sysupgrade tar file integrity" | 
|  | if ! $cmd < "$file" | tar xOf - > /dev/null; then | 
|  | echo "corrupted sysupgrade tar file" | 
|  | return 1 | 
|  | fi | 
|  | } | 
|  |  | 
|  | nand_do_flash_file() { | 
|  | local file="$1" | 
|  | local cmd="$2" | 
|  | local file_type | 
|  |  | 
|  | [ -z "$cmd" ] && cmd="$(identify_if_gzip "$file")cat" | 
|  | file_type="$(identify "$file" "$cmd" "")" | 
|  |  | 
|  | [ ! "$(find_mtd_index "$CI_UBIPART")" ] && CI_UBIPART=rootfs | 
|  |  | 
|  | case "$file_type" in | 
|  | "fit") | 
|  | nand_verify_if_gzip_file "$file" "$cmd" || return 1 | 
|  | nand_upgrade_fit "$file" "$cmd" | 
|  | ;; | 
|  | "ubi") | 
|  | nand_verify_if_gzip_file "$file" "$cmd" || return 1 | 
|  | nand_upgrade_ubinized "$file" "$cmd" | 
|  | ;; | 
|  | "ubifs") | 
|  | nand_verify_if_gzip_file "$file" "$cmd" || return 1 | 
|  | nand_upgrade_ubifs "$file" "$cmd" | 
|  | ;; | 
|  | *) | 
|  | nand_verify_tar_file "$file" "$cmd" || return 1 | 
|  | nand_upgrade_tar "$file" "$cmd" | 
|  | ;; | 
|  | esac | 
|  | } | 
|  |  | 
|  | nand_do_restore_config() { | 
|  | local conf_tar="/tmp/sysupgrade.tgz" | 
|  | [ ! -f "$conf_tar" ] || nand_restore_config "$conf_tar" | 
|  | } | 
|  |  | 
|  | # Recognize type of passed file and start the upgrade process | 
|  | # | 
|  | # Supported firmware containers: | 
|  | # 1. Raw file | 
|  | # 2. Gzip | 
|  | # 3. Custom (requires passing extracting command) | 
|  | # | 
|  | # Supported data formats: | 
|  | # 1. Tar with kernel/rootfs | 
|  | # 2. UBI image (built using "ubinized") | 
|  | # 3. UBIFS image (to update UBI volume with) | 
|  | # 4. FIT image (to update UBI volume with) | 
|  | # | 
|  | # $(1): firmware file path | 
|  | # $(2): (optional) pipe command to extract firmware | 
|  | nand_do_upgrade() { | 
|  | local file="$1" | 
|  | local cmd="$2" | 
|  |  | 
|  | sync | 
|  | nand_do_flash_file "$file" "$cmd" && nand_do_upgrade_success | 
|  | nand_do_upgrade_failed | 
|  | } | 
|  |  | 
|  | nand_do_upgrade_success() { | 
|  | if nand_do_restore_config && sync; then | 
|  | echo "sysupgrade successful" | 
|  | umount -a | 
|  | reboot -f | 
|  | fi | 
|  | nand_do_upgrade_failed | 
|  | } | 
|  |  | 
|  | nand_do_upgrade_failed() { | 
|  | sync | 
|  | echo "sysupgrade failed" | 
|  | # Should we reboot or bring up some failsafe mode instead? | 
|  | umount -a | 
|  | reboot -f | 
|  | } | 
|  |  | 
|  | # Check if passed file is a valid one for NAND sysupgrade. | 
|  | # Currently it accepts 4 types of files: | 
|  | # 1) UBI: a ubinized image containing required UBI volumes. | 
|  | # 2) UBIFS: a UBIFS rootfs volume image. | 
|  | # 3) FIT: a FIT image containing kernel and rootfs. | 
|  | # 4) TAR: an archive that includes directory "sysupgrade-${BOARD_NAME}" containing | 
|  | #         a non-empty "CONTROL" file and required partition and/or volume images. | 
|  | # | 
|  | # You usually want to call this function in platform_check_image. | 
|  | # | 
|  | # $(1): board name, used in case of passing TAR file | 
|  | # $(2): file to be checked | 
|  | nand_do_platform_check() { | 
|  | local board_name="$1" | 
|  | local file="$2" | 
|  |  | 
|  | local cmd="$(identify_if_gzip "$file")cat" | 
|  | local file_type="$(identify "$file" "$cmd" "")" | 
|  | local control_length=$( ($cmd < "$file" | tar xOf - "sysupgrade-${board_name//,/_}/CONTROL" | wc -c) 2> /dev/null) | 
|  |  | 
|  | if [ "$control_length" = 0 ]; then | 
|  | control_length=$( ($cmd < "$file" | tar xOf - "sysupgrade-${board_name//_/,}/CONTROL" | wc -c) 2> /dev/null) | 
|  | fi | 
|  |  | 
|  | if [ "$control_length" != 0 ]; then | 
|  | nand_verify_tar_file "$file" "$cmd" || return 1 | 
|  | else | 
|  | nand_verify_if_gzip_file "$file" "$cmd" || return 1 | 
|  | if [ "$file_type" != "fit" -a "$file_type" != "ubi" -a "$file_type" != "ubifs" ]; then | 
|  | echo "invalid sysupgrade file" | 
|  | return 1 | 
|  | fi | 
|  | fi | 
|  |  | 
|  | return 0 | 
|  | } |