[BACK]Return to fw_install.sh CVS log [TXT][DIR] Up to [local] / openbsd / fw_update

Diff for /openbsd/fw_update/fw_install.sh between version 1.39 and 1.51

version 1.39, 2021/12/01 02:19:00 version 1.51, 2021/12/07 02:42:01
Line 1 
Line 1 
 #!/bin/ksh  #!/bin/ksh
 #       $OpenBSD$  #       $OpenBSD$
 set -e  #
   
 # Copyright (c) 2021 Andrew Hewus Fresh <afresh1@openbsd.org>  # Copyright (c) 2021 Andrew Hewus Fresh <afresh1@openbsd.org>
 #  #
 # Permission to use, copy, modify, and distribute this software for any  # Permission to use, copy, modify, and distribute this software for any
Line 16 
Line 15 
 # ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF  # ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
 # OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.  # OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
   
 # Fake up some things from install.sub that we don't need to actually do  set -o errexit -o pipefail -o nounset
 prefetcharea_fs_list() {  
         echo "${DESTDIR}/tmp"  
 }  
   
 # tmpdir, do_as, unpriv, and unpriv2 are from install.sub  CFILE=SHA256.sig
   DESTDIR=${DESTDIR:-}
   FWPATTERNS="${DESTDIR}/usr/share/misc/firmware_patterns"
   
 # Create a temporary directory based on the supplied directory name prefix.  VNAME=${VNAME:-$(sysctl -n kern.osrelease)}
   VERSION=${VERSION:-"${VNAME%.*}${VNAME#*.}"}
   
   HTTP_FWDIR="$VNAME"
   VTYPE=$( sed -n "/^OpenBSD $VNAME\([^ ]*\).*$/s//\1/p" /var/run/dmesg.boot | sed '$!d' )
   [[ $VTYPE == -!(stable) ]] && HTTP_FWDIR=snapshots
   
   FWURL=http://firmware.openbsd.org/firmware/${HTTP_FWDIR}
   FWPUB_KEY=${DESTDIR}/etc/signify/openbsd-${VERSION}-fw.pub
   
 tmpdir() {  tmpdir() {
         local _i=1 _dir          local _i=1 _dir
   
         until _dir="${1?}.$_i.$RANDOM" && mkdir -- "$_dir" 2>/dev/null; do          # If we're not in the installer,
                 ((++_i < 10000)) || return 1          # we have mktemp and a more hostile environment
         done          if [ -x /usr/bin/mktemp ]; then
                   _dir=$( mktemp -d "${1}-XXXXXXXXX" )
           else
                   until _dir="${1}.$_i.$RANDOM" && mkdir -- "$_dir" 2>/dev/null; do
                       ((++_i < 10000)) || return 1
                   done
           fi
   
         echo "$_dir"          echo "$_dir"
 }  }
   
 # Run a command ($2+) as unprivileged user ($1).  fetch() {
 # Take extra care that after "cmd" no "user" processes exist.          local _file=$1 _user=_file _exit
 #  
 # Optionally:  
 #       - create "file" and chown it to "user"  
 #       - after "cmd", chown "file" back to root  
 #  
 # Usage: do_as user [-f file] cmd  
 do_as() {  
         (( $# >= 2 )) || return  
   
         local _file _rc _user=$1          >"$_file"
         shift          chown "$_user" "$_file"
   
         if [[ $1 == -f ]]; then          # If we're not in the installer, we have su(1)
                 _file=$2          # and doas(1) is unlikely to be configured.
                 shift 2          if [ -x /usr/bin/sh ]; then
                   /usr/bin/su -s /bin/ksh "$_user" -c \
                       "/usr/bin/ftp -D 'Get/Verify' -Vm \
                           -o '$_file' '${FWURL}/${_file}'"
                   _exit="$?"
           else
                   /usr/bin/doas -u "$_user" \
                       ftp -D 'Get/Verify' -Vm \
                           -o "$_file" "${FWURL}/${_file}"
                   _exit="$?"
         fi          fi
   
         if [[ -n $_file ]]; then          if [ "$_exit" -ne 0 ]; then
                 >$_file                  rm -f "$_file"
                 chown "$_user" "$_file"                  echo "Cannot fetch $_file" >&2
                   return 1
         fi          fi
   
         doas -u "$_user" "$@"          chown root "$_file"
         _rc=$?  }
   
         while doas -u "$_user" kill -9 -1 2>/dev/null; do  verify() {
                 echo "Processes still running for user $_user after: $@"          # On the installer we don't get sha256 -C, so fake it.
                 sleep 1          if ! fgrep -qx "SHA256 ($1) = $( /bin/sha256 -qb "$1" )" "$CFILE"; then
         done                  echo "Checksum test for $1 failed." >&2
                   return 1
           fi
   }
   
         [[ -n $_file ]] && chown root "$_file"  devices_needing_firmware() {
           local _d _m _grep _dmesgtail _last=''
   
         return $_rc          # When we're not in the installer, the dmesg.boot can
           # contain multiple boots, so only look in the last one
           _dmesgtail=$( sed -n 'H;/^OpenBSD/h;${g;p;}' /var/run/dmesg.boot )
   
           grep -v '^[[:space:]]*#' "$FWPATTERNS" |
               while read -r _d _m; do
                   _grep="grep"
                   [ "$_last" = "$_d" ] && continue
                   [ "$_m" ] || _m="^${_d}[0-9][0-9]* at "
                   [ "$_m" = "${_m#^}" ] && _grep="fgrep"
   
                   echo "$_dmesgtail" | $_grep -q "$_m" || continue
                   echo "$_d"
                   _last="$_d"
           done
 }  }
   
 unpriv() {  firmware_filename() {
         do_as _sndio "$@"          sed -n "s/.*(\($1-firmware-.*\.tgz\)).*/\1/p" "$CFILE" | sed '$!d'
 }  }
   
 unpriv2() {  installed_firmware() {
         do_as _file "$@"          for fw in "${DESTDIR}/var/db/pkg/$1-firmware"*; do
                   [ -e "$fw" ] || continue
                   echo "${fw##*/}"
           done
 }  }
   
 VNAME=${VNAME:-$(sysctl -n kern.osrelease)}  add_firmware () {
 VERSION=${VERSION:-"${VNAME%.*}${VNAME#*.}"}          local _f="$1" _pkgdir="${DESTDIR}/var/db/pkg"
 FWDIR=${FWDIR:-$VNAME}          ftp -D "Install" -Vmo- "file:${1}" |
 MODE=${MODE:-install}                  tar -s ",^\+,${_pkgdir}/${_f%.tgz}/+," \
                   -s ",^firmware,${DESTDIR}/etc/firmware," \
                   -C / -zxphf - "+*" "firmware/*"
   
 # TODO: We need the firmware for the system we just installed          # TODO: Should we mark these so real fw_update can -Drepair?
 #       not the one we booted from.  For example:          ed -s "${_pkgdir}/${_f%.tgz}/+CONTENTS" <<EOL
 #       * booting from a snapshot bsd.rd that thinks it is the 7.0 release  /^@comment pkgpath/ -1a
 #         will install the firmware from the 7.0 directory instead of  @option manual-installation
 #         from the snapshots dir.  @option firmware
 #       If they're using sysupgrade, then the installer kernel will be correct.  @comment install-script
 #       If we're doing this in the installer we can check what they picked  .
 #       for downloading sets and use that value.  w
 #       Otherwise, the fw_update after first boot will fix it up for us.  EOL
   }
   
 HTTP_FWDIR=$FWDIR  delete_firmware() {
 set -- sed -n "/^OpenBSD $VNAME\([^ ]*\).*$/s//\1/p" /var/run/dmesg.boot          local _cwd _pkg="$1" _pkgdir="${DESTDIR}/var/db/pkg"
 [[ $1 == -!(stable) ]] && HTTP_FWDIR=snapshots  
   
 FWURL=http://firmware.openbsd.org/firmware/${HTTP_FWDIR}          # TODO: Check hash for files before deleting
 FWPUB_KEY=${DESTDIR}/etc/signify/openbsd-${VERSION}-fw.pub          echo "Uninstalling $_pkg"
 FWPATTERNS="${DESTDIR}/usr/share/misc/firmware_patterns"          _cwd="${_pkgdir}/$_pkg"
   
 # TODO: support srclocal installation of firmware somehow          set -A _remove -- "${_cwd}/+CONTENTS" "${_cwd}"
 fw_install() {  
         local _src=$1 _tmpfs_list _tmpfs _tmpsrc \  
                 _t=Get _cfile="/tmp/SHA256" _pkgdir=${DESTDIR}/var/db/pkg \  
                 _f _r _remove _i _installed  
         local _srclocal=false _unpriv=unpriv  
   
         echo "Let's $MODE firmware!"          while read -r c g; do
         local _d _drivers=$(                  case $c in
                 last=''                  @cwd) _cwd="${DESTDIR}$g"
                 while read _d _m; do                    ;;
                         grep=grep                  @*) continue
                         [ "$last" = "$_d" ] && continue                    ;;
                         [ "$_m" ] || _m="^$_d[0-9][0-9]* at "                  *)  set -A _remove -- "$_cwd/$c" "${_remove[@]}"
                         [ "$_m" = "${_m#^}" ] && grep=fgrep                    ;;
                         $grep -q "$_m" /var/run/dmesg.boot || continue                  esac
                         echo $_d          done < "${_pkgdir}/${_pkg}/+CONTENTS"
                         last=$_d  
                 done < $FWPATTERNS  
         )  
   
         if [ -z "$_drivers" ]; then          # We specifically rm -f here because not removing files/dirs
                 echo "No devices found which need firmware files to be downloaded."          # is probably not worth failing over.
                 return          for _r in "${_remove[@]}" ; do
         fi                  if [ -d "$_r" ]; then
                           # Try hard not to actually remove recursively
         ! _tmpfs_list=$(prefetcharea_fs_list) &&                          # without rmdir on the install media.
                 echo "Cannot determine prefetch area" >&2 && return                          [ "$_r/*" = "$( echo "$_r"/* )" ] && rm -rf "$_r"
                   else
         for _tmpfs in $_tmpfs_list; do                          rm -f "$_r"
                 # Try to clean up from previous runs, assuming  
                 # the _tmpfs selection yields the same mount  
                 # point.  
                 for _tmpsrc in $_tmpfs/firmware.+([0-9]).+([0-9]); do  
                         [[ -d $_tmpsrc ]] && rm -r $_tmpsrc  
                 done  
   
                 # Create a download directory for the firmware and  
                 # check that the _sndio user can read files from  
                 # it. Otherwise cleanup and skip the filesystem.  
                 if _tmpsrc=$(tmpdir "$_tmpfs/firmware"); then  
                         (  
                         >$_tmpsrc/t &&  
                         $_unpriv cat $_tmpsrc/t  
                         ) >/dev/null 2>&1 && break ||  
                                 rm -r $_tmpsrc  
                 fi                  fi
         done          done
   }
   
         [[ ! -d $_tmpsrc ]] &&  set -A devices -- $( devices_needing_firmware )
                 echo "Cannot create prefetch area" >&2 && return 1  
   
         # Cleanup from previous runs.  if [ ! "${devices:-}" ]; then
         rm -f $_cfile $_cfile.sig          echo "No devices found which need firmware files to be downloaded."
           exit
   fi
   
         _t=Get/Verify  TMPDIR=$( tmpdir "${DESTDIR}/tmp/fw_install" )
   cd "$TMPDIR"
   
         ! $_unpriv ftp -D "$_t" -Vmo - "$_src/SHA256.sig" >"$_cfile.sig" &&  # To unpriv we need to let the unpriv user into this dir
             echo "Cannot fetch SHA256.sig" >&2 && return 1  chmod go+x .
   
         # Verify signature file with public keys.  fetch "$CFILE"
         ! $_unpriv -f "$_cfile" \  ! signify -Vep "$FWPUB_KEY" -x "$CFILE" -m "$CFILE" &&
             signify -Vep $FWPUB_KEY -x "$_cfile.sig" -m "$_cfile" &&      echo "Signature check of SHA256.sig failed" >&2 && exit 1
             echo "Signature check of SHA256.sig failed" >&2 && return 1  
   
         for _d in $_drivers; do  for d in "${devices[@]}"; do
                 _f=$( sed -n "s/.*(\($_d-firmware-.*\.tgz\)).*/\1/p" "$_cfile" | tail -1 )          f=$( firmware_filename "$d" )
                 _installed=$(          [ "$f" ] || continue
                 for fw in "${_pkgdir}/$_d-firmware"*; do          set -A installed -- $( installed_firmware "$d" )
                         [ -e "$fw" ] || continue  
                         echo ${fw##*/}  
                 done  
                 )  
   
                 for _i in $_installed; do          if [ "${installed:-}" ]; then
                         if [ "$_f" = "$_i.tgz" ]; then                  for i in "${installed[@]:-}"; do
                                 echo "$_i already installed"                          if [ "$f" = "$i.tgz" ]; then
                                   echo "$i already installed"
                                 continue 2                                  continue 2
                         fi                          fi
                 done                  done
           fi
   
                 rm -f /tmp/h /tmp/fail          fetch  "$f" || continue
           verify "$f" || continue
   
                 # Fetch firmware file and create a checksum by piping through          if [ "${installed:-}" ]; then
                 # sha256. Create a flag file in case ftp failed. Firmware                  for i in "${installed[@]}"; do
                 # from net is written to the prefetch area.                          delete_firmware "$i"
                 ( $_unpriv ftp -D "$_t" -Vmo - "$_src/$_f" || >/tmp/fail ) |                  done
                 ( $_srclocal && unpriv2 sha256 -b >/tmp/h ||          fi
                     unpriv2 -f /tmp/h sha256 -bph /tmp/h >"$_tmpsrc/$_f" )  
   
                 # Handle failed transfer.          add_firmware "$f"
                 if [[ -f /tmp/fail ]]; then  done
                         rm -f "$_tmpsrc/$_f"  
                         echo "Fetching of $_f failed!" >&2  
                         continue  
                 fi  
   
                 # Verify firmware by comparing its checksum with SHA256.  
                 if ! fgrep -qx "SHA256 ($_f) = $(</tmp/h)" "$_cfile"; then  
                         [[ -d "$_tmpsrc" ]] && rm -rf "$_tmpsrc"  
                         echo "Checksum test for $_f failed." >&2  
                         continue  
                 fi  
   
                 # TODO: Check hash for files before deleting  
                 if [ "$_installed" ] && [ -e "${_pkgdir}/$_installed/+CONTENTS" ]; then  
                         echo "Uninstalling $_installed"  
                         cwd=${_pkgdir}/$_installed  
   
                         set -A _remove -- "${cwd}/+CONTENTS" "${cwd}"  
   
                         while read c g; do  
                                 case $c in  
                                 @cwd) cwd="${DESTDIR}/$g"  
                                   ;;  
                                 @*) continue  
                                   ;;  
                                 *)  set -A _remove -- "$cwd/$c" "${_remove[@]}"  
                                   ;;  
                                 esac  
                         done < "${_pkgdir}/$_installed/+CONTENTS"  
   
                         # We specifically rm -f here because not removing files/dirs  
                         # is probably not worth failing over.  
                         for _r in "${_remove[@]}" ; do  
                                 if [ -d "$_r" ]; then  
                                         # Try hard not to actually remove recursively  
                                         # without rmdir on the install media.  
                                         [ "$_r/*" = $( echo "$_r"/* ) ] && rm -rf "$_r"  
                                 else  
                                         rm -f "$_r"  
                                 fi  
                         done  
                 fi  
   
                 # TODO: Should we mark these so real fw_update can -Drepair?  
                 ftp -D "Install" -Vmo- "file:$_tmpsrc/$_f" |  
                         tar -s ",^\+,${_pkgdir}/${_f%.tgz}/+," \  
                         -s ",^firmware,${DESTDIR}/etc/firmware," \  
                         -C / -zxphf - "+*" "firmware/*"  
   
                 ed -s "${_pkgdir}/${_f%.tgz}/+CONTENTS" <<EOL  
 /^@comment pkgpath/ -1a  
 @option manual-installation  
 @option firmware  
 @comment install-script  
 .  
 w  
 EOL  
         done  
 }  
   
 fw_install "$FWURL"  

Legend:
Removed from v.1.39  
changed lines
  Added in v.1.51

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>