You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
826 lines
26 KiB
826 lines
26 KiB
#!/bin/bash
|
|
|
|
show_help() {
|
|
cat << EOF
|
|
|
|
Mender conversion tool
|
|
|
|
A tool that takes an existing embedded image (Debian, Ubuntu, Raspbian, etc)
|
|
and converts it to a Mender image by restructuring partition table and adding
|
|
necessary files.
|
|
|
|
Usage: $0 COMMAND [options]
|
|
|
|
General commands:
|
|
|
|
from-raw-disk-image - composes fully functional Mender
|
|
image compliant with Mender
|
|
partition layout, having all
|
|
necessary files installed
|
|
mender-disk-image-to-artifact - creates Mender artifact file
|
|
from Mender image
|
|
|
|
Expert commands:
|
|
|
|
raw-disk-image-shrink-rootfs - shrinks existing embedded raw
|
|
disk image
|
|
raw-disk-image-create-partitions - converts raw disk image's
|
|
partition table to be compliant
|
|
with Mender partition layout
|
|
install-mender-to-mender-disk-image - installs Mender client related
|
|
files
|
|
install-bootloader-to-mender-disk-image - installs bootloader (U-Boot/GRUB)
|
|
related files
|
|
|
|
Options: [-r|--raw-disk-image | -m|--mender-disk-image | -s|--data-part-size-mb |
|
|
-d|--device-type | -p|--rootfs-partition-id | -i|--demo-host-ip |
|
|
-c|--server-cert | -u|--server-url | -t|--tenant-token |
|
|
-g|--mender-client -b|--bootloader-toolchain | -a|--artifact-name |
|
|
-k|--keep]
|
|
|
|
raw-disk-image - raw disk embedded Linux (Debian, Raspbian,
|
|
Ubuntu, etc.) image path
|
|
mender-disk-image - Mender disk image name where the script writes to
|
|
should have "sdimg" suffix
|
|
data-part-size-mb - data partition size in MB; default value 128MB
|
|
device-type - target device identification used to build
|
|
Mender image
|
|
rootfs-partition-id - selects root filesystem (primary|secondary)
|
|
as the source filesystem for an artifact
|
|
demo-host-ip - server demo ip used for testing purposes
|
|
server-cert - server certificate file
|
|
server-url - production server url
|
|
tenant-token - Mender tenant token
|
|
mender-client - Mender client binary
|
|
bootloader-toolchain - GNU Arm Embedded Toolchain
|
|
artifact-name - Mender artifact name
|
|
keep - keep intermediate files in output directory
|
|
|
|
Note: root filesystem size used in Mender image creation can be found as
|
|
an output from 'raw-disk-image-shrink-rootfs' command or, in case
|
|
of using unmodified embedded raw disk image, can be checked with
|
|
any partition manipulation program (e.g. parted, fdisk, etc.).
|
|
|
|
Examples:
|
|
|
|
To create fully functional Mender image from raw disk image in a single step:
|
|
|
|
./mender-convert from-raw-disk-image
|
|
--raw-disk-image <raw_disk_image_path>
|
|
[--mender-disk-image <mender_image_name>]
|
|
--device-type <beaglebone | raspberrypi3>
|
|
--mender-client <mender_binary_path>
|
|
--artifact-name release-1_1.5.0
|
|
--bootloader-toolchain arm-linux-gnueabihf
|
|
--demo-host-ip 192.168.10.2
|
|
--keep
|
|
|
|
Output:
|
|
- Mender image: ready to use image with client and bootloader installed
|
|
- Mender artifact: update file based on the already built Mender image
|
|
- Mender root filesystem: EXT4 image used to produce the Mender artifact
|
|
|
|
To create Mender artifact file from Mender image:
|
|
|
|
./mender-convert mender-disk-image-to-artifact
|
|
--mender-disk-image <mender_image_path>
|
|
--device-type <beaglebone | raspberrypi3>
|
|
--artifact-name release-1_1.5.0
|
|
--rootfs-partition-id <primary | secondary>
|
|
|
|
Note: artifact name format is: release-<release_no>_<mender_version>
|
|
|
|
Examples for expert actions:
|
|
|
|
To shrink the existing embedded raw disk image:
|
|
|
|
./mender-convert raw-disk-image-shrink-rootfs
|
|
--raw-disk-image <raw_disk_image_path>
|
|
|
|
Output: Root filesystem size (sectors): 4521984
|
|
|
|
To convert raw disk image's partition table to Mender layout:
|
|
|
|
./mender-convert raw-disk-image-create-partitions
|
|
--raw-disk-image <raw_disk_image_path>
|
|
[--mender-disk-image <mender_image_name>]
|
|
--artifact-name release-1_1.5.0
|
|
--device-type <beaglebone | raspberrypi3>
|
|
[--data-part-size-mb 128]
|
|
|
|
Output: repartitioned (respectively to Mender layout) raw disk image
|
|
|
|
To install Mender client related files:
|
|
|
|
./mender-convert install-mender-to-mender-disk-image
|
|
--mender-disk-image <mender_image_path>
|
|
--device-type <beaglebone | raspberrypi3>
|
|
--artifact-name release-1_1.5.0
|
|
--demo-host-ip 192.168.10.2
|
|
--mender-client <mender_binary_path>
|
|
|
|
Output: Mender image with Mender client related files installed
|
|
|
|
To install bootloader (U-Boot/GRUB) related files:
|
|
|
|
./mender-convert install-bootloader-to-mender-disk-image
|
|
--mender-disk-image <mender_image_path>
|
|
--device-type <beaglebone | raspberrypi3>
|
|
--bootloader-toolchain arm-linux-gnueabihf
|
|
|
|
Output: Mender image with appropriate bootloader (U-Boot/GRUB) installed
|
|
|
|
EOF
|
|
}
|
|
|
|
if [ $# -eq 0 ]; then
|
|
show_help
|
|
exit 1
|
|
fi
|
|
|
|
tool_dir="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )"
|
|
|
|
# Default sector size
|
|
sector_size=
|
|
# Boot partition start in sectors (512 bytes per sector).
|
|
pboot_start=
|
|
# Default 'boot' partition size in sectors: 16MB
|
|
# (i.e 16777216 bytes, equals to 'partition_alignment' * 2)
|
|
pboot_size=
|
|
# Default 'data' partition size in MiB.
|
|
data_part_size_mb=128
|
|
# Data partition size in sectors.
|
|
pdata_size=
|
|
# Exemplary values for Beaglebone: 9.3: 4521984 9.4: 4423680
|
|
prootfs_size=
|
|
|
|
mender_disk_image=
|
|
raw_disk_image=
|
|
device_type=
|
|
mender_disk_counts=
|
|
artifact_name=
|
|
rootfs_partition_id=
|
|
raw_disk_counts=
|
|
mender_client=
|
|
# Mender production certificate.
|
|
server_cert=
|
|
# Mender production server url.
|
|
server_url=
|
|
# Mender demo server IP address.
|
|
demo_host_ip=
|
|
# Mender hosted token.
|
|
tenant_token=
|
|
|
|
# Conversion progress.
|
|
declare -i step=1
|
|
declare -i total=
|
|
|
|
declare -a rootfs_partition_ids=("primary" "secondary")
|
|
declare -a mender_disk_mappings
|
|
declare -a raw_disk_mappings
|
|
#Supported devices
|
|
declare -a supported_devices=("beaglebone" "raspberrypi3")
|
|
|
|
do_raw_disk_image_shrink_rootfs() {
|
|
log "$step/$total Shrinking raw disk image root filesystem..."
|
|
((step++))
|
|
if [ -z "${raw_disk_image}" ]; then
|
|
log "Raw disk image not set. Aborting."
|
|
return 1
|
|
fi
|
|
|
|
local count=
|
|
local bootstart=
|
|
local bootsize=
|
|
local rootfsstart=
|
|
local rootfssize=
|
|
local bootflag=
|
|
|
|
# Gather information about raw disk image.
|
|
get_image_info $raw_disk_image count sector_size bootstart bootsize \
|
|
rootfsstart rootfssize bootflag
|
|
|
|
# Find first available loopback device.
|
|
loopdevice=($(losetup -f))
|
|
[ $? -ne 0 ] && { log "Error: inaccesible loopback device"; return 1; }
|
|
|
|
# Mount appropriate partition.
|
|
if [[ $count -eq 1 ]]; then
|
|
sudo losetup $loopdevice $raw_disk_image -o $(($bootstart * $sector_size))
|
|
elif [[ $count -eq 2 ]]; then
|
|
sudo losetup $loopdevice $raw_disk_image -o $(($rootfsstart * $sector_size))
|
|
else
|
|
log "Error: invalid/unsupported embedded raw disk image. Aborting."
|
|
return 1
|
|
fi
|
|
|
|
block_size=($(sudo dumpe2fs -h $loopdevice | grep 'Block size' | tr -s ' ' | cut -d ' ' -f3))
|
|
min_size_blocks=($(sudo resize2fs -P $loopdevice | awk '{print $NF}'))
|
|
|
|
new_size_sectors=$(( $min_size_blocks * $block_size / $sector_size ))
|
|
align_partition_size new_size_sectors $sector_size
|
|
|
|
log "\tRoot filesystem size:\
|
|
\n\tminimal: $(( $min_size_blocks * $block_size )) \
|
|
\n\taligned: $(( $new_size_sectors * $sector_size ))\
|
|
\n\tsectors: $new_size_sectors"
|
|
|
|
sudo e2fsck -y -f $loopdevice >> "$build_log" 2>&1
|
|
sudo resize2fs -p $loopdevice ${new_size_sectors}s >> "$build_log" 2>&1
|
|
sudo e2fsck -y -f $loopdevice >> "$build_log" 2>&1
|
|
|
|
sudo losetup -d $loopdevice
|
|
sudo losetup $loopdevice $raw_disk_image
|
|
|
|
if [[ $count -eq 1 ]]; then
|
|
create_single_disk_partition_table $loopdevice $bootstart $new_size_sectors
|
|
elif [[ $count -eq 2 ]]; then
|
|
create_double_disk_partition_table $loopdevice $rootfsstart $new_size_sectors
|
|
fi
|
|
|
|
sudo partprobe
|
|
endsector=($(sudo parted $loopdevice -ms unit s print | grep "^$count" | cut -f3 -d: | sed 's/[^0-9]*//g'))
|
|
|
|
sudo losetup -d $loopdevice
|
|
log "\tRaw disk image new endsector: $endsector"
|
|
truncate -s $((($endsector+1) * $sector_size)) $raw_disk_image
|
|
log "\tNew root filesystem size (sectors): $new_size_sectors"
|
|
|
|
return 0
|
|
}
|
|
|
|
do_raw_disk_image_create_partitions() {
|
|
log "$step/$total Repartitioning raw disk image..."
|
|
((step++))
|
|
|
|
if [ -z "$raw_disk_image" ] || [ -z "$device_type" ] || \
|
|
[ -z "$artifact_name" ]; then
|
|
show_help
|
|
return 1
|
|
fi
|
|
|
|
if [[ ! -f ${raw_disk_image} ]]; then
|
|
log "Raw disk image not found. Aborting."
|
|
return 1
|
|
fi
|
|
|
|
local supported=$(echo ${supported_devices[@]} | grep -o $device_type | wc -w)
|
|
|
|
[[ $supported -eq 0 ]] && \
|
|
{ log "Error: incorrect device type. Aborting."; return 1; }
|
|
|
|
# Change current directory to 'output' directory.
|
|
cd $output_dir
|
|
|
|
# Make sure the user's given Mender image name has a correct extension.
|
|
# If Mender image name is not provided, then use following syntax:
|
|
# mender-<device_name>-<artifact_name>.sdimg
|
|
if [ -n "${mender_disk_image}" ]; then
|
|
local mender_disk_basename=$(basename -- "$mender_disk_image")
|
|
if [[ $mender_disk_basename =~ \.sdimg$ ]]; then
|
|
mender_disk_image=$output_dir/$mender_disk_basename
|
|
else
|
|
local mender_disk_filename="${mender_disk_basename%.*}"
|
|
mender_disk_image=$output_dir/${mender_disk_filename}.sdimg
|
|
fi
|
|
else
|
|
mender_disk_image=$output_dir/mender-${device_type}-${artifact_name}.sdimg
|
|
fi
|
|
|
|
analyse_raw_disk_image ${raw_disk_image} pboot_start pboot_size prootfs_size \
|
|
sector_size raw_disk_counts
|
|
|
|
[ -z "${prootfs_size}" ] && \
|
|
{ log "root filesystem size not set. Aborting."; return 1; }
|
|
|
|
log "\tDetected raw disk image with $raw_disk_counts partition(s)."
|
|
|
|
local mender_disk_image_size=
|
|
calculate_mender_disk_size $pboot_start $pboot_size \
|
|
$prootfs_size $data_part_size_mb \
|
|
$sector_size pdata_size mender_disk_image_size
|
|
|
|
log "\tCreating Mender disk image:\
|
|
\n\t\timage size: ${mender_disk_image_size} bytes\
|
|
\n\t\troot filesystem size: $((${prootfs_size} * $sector_size)) bytes\
|
|
\n\t\tdata partition size: $(($pdata_size * $sector_size)) bytes"
|
|
|
|
create_test_config_file $device_type $partition_alignment $vfat_storage_offset \
|
|
$pboot_size $prootfs_size $pdata_size $mender_disk_image_size \
|
|
$sector_size
|
|
|
|
create_mender_disk $mender_disk_image $mender_disk_image_size
|
|
format_mender_disk $mender_disk_image $mender_disk_image_size $pboot_start \
|
|
$pboot_size $prootfs_size $pdata_size $sector_size
|
|
verify_mender_disk $mender_disk_image mender_disk_counts
|
|
|
|
create_device_maps $mender_disk_image mender_disk_mappings
|
|
|
|
log "$step/$total Formatting repartitioned raw disk image..."
|
|
((step++))
|
|
make_mender_disk_filesystem ${mender_disk_mappings[@]}
|
|
|
|
case "$device_type" in
|
|
"beaglebone")
|
|
do_make_sdimg_beaglebone
|
|
;;
|
|
"raspberrypi3")
|
|
do_make_sdimg_raspberrypi3
|
|
;;
|
|
esac
|
|
|
|
rc=$?
|
|
|
|
log "$step/$total Cleaning intermediate files..."
|
|
((step++))
|
|
# Clean and detach.
|
|
detach_device_maps ${mender_disk_mappings[@]}
|
|
detach_device_maps ${raw_disk_mappings[@]}
|
|
sync
|
|
rm -rf $embedded_base_dir
|
|
rm -rf $sdimg_base_dir
|
|
|
|
[ $rc -eq 0 ] || { log "\nRepartitioning of $raw_disk_image image failed."; }
|
|
|
|
return $rc
|
|
}
|
|
|
|
do_make_sdimg_beaglebone() {
|
|
local ret=0
|
|
|
|
create_device_maps $raw_disk_image raw_disk_mappings
|
|
|
|
mount_mender_disk ${mender_disk_mappings[@]}
|
|
mount_raw_disk ${raw_disk_mappings[@]}
|
|
|
|
log "$step/$total Setting boot partition..."
|
|
((step++))
|
|
stage_2_args="$sdimg_boot_dir $embedded_rootfs_dir"
|
|
${tool_dir}/bbb-convert-stage-2.sh ${stage_2_args} || ret=$?
|
|
[[ $ret -ne 0 ]] && { log "Aborting."; return $ret; }
|
|
|
|
log "$step/$total Setting root filesystem partition..."
|
|
((step++))
|
|
stage_3_args="$sdimg_primary_dir $embedded_rootfs_dir"
|
|
${tool_dir}/bbb-convert-stage-3.sh ${stage_3_args} || ret=$?
|
|
[[ $ret -ne 0 ]] && { log "Aborting."; return $ret; }
|
|
|
|
log "$step/$total Setting file system table..."
|
|
((step++))
|
|
set_fstab $device_type
|
|
|
|
return $ret
|
|
}
|
|
|
|
do_make_sdimg_raspberrypi3() {
|
|
log "$step/$total Setting boot partition..."
|
|
((step++))
|
|
image_boot_part=$(fdisk -l ${raw_disk_image} | grep FAT32)
|
|
|
|
boot_part_start=$(echo ${image_boot_part} | awk '{print $2}')
|
|
boot_part_end=$(echo ${image_boot_part} | awk '{print $3}')
|
|
boot_part_size=$(echo ${image_boot_part} | awk '{print $4}')
|
|
|
|
log "\tExtracting boot partition from raw disk image."
|
|
extract_file_from_image ${raw_disk_image} ${boot_part_start} \
|
|
${boot_part_size} "boot.vfat"
|
|
|
|
stage_2_args="$output_dir ${mender_disk_mappings[0]}"
|
|
${tool_dir}/rpi3-convert-stage-2.sh ${stage_2_args} || ret=$?
|
|
[[ $ret -ne 0 ]] && { return $ret; }
|
|
|
|
log "$step/$total Setting root filesystem partition..."
|
|
((step++))
|
|
image_rootfs_part=$(fdisk -l ${raw_disk_image} | grep Linux)
|
|
|
|
rootfs_part_start=$(echo ${image_rootfs_part} | awk '{print $2}')
|
|
rootfs_part_end=$(echo ${image_rootfs_part} | awk '{print $3}')
|
|
rootfs_part_size=$(echo ${image_rootfs_part} | awk '{print $4}')
|
|
|
|
log "\tExtracting root filesystem partition from raw disk image."
|
|
extract_file_from_image ${raw_disk_image} ${rootfs_part_start} \
|
|
${rootfs_part_size} "rootfs.img"
|
|
|
|
stage_3_args="$output_dir ${mender_disk_mappings[1]}"
|
|
${tool_dir}/rpi3-convert-stage-3.sh ${stage_3_args} || ret=$?
|
|
[[ $ret -ne 0 ]] && { return $ret; }
|
|
|
|
mount_mender_disk ${mender_disk_mappings[@]}
|
|
|
|
# Add mountpoints.
|
|
sudo install -d -m 755 ${sdimg_primary_dir}/uboot
|
|
sudo install -d -m 755 ${sdimg_primary_dir}/data
|
|
|
|
log "$step/$total Setting file system table..."
|
|
((step++))
|
|
set_fstab $device_type
|
|
|
|
[[ $keep != "-k" ]] && { rm -f $output_dir/boot.vfat\
|
|
$output_dir/cmdline.txt $output_dir/config.txt\
|
|
$output_dir/rootfs.img; }
|
|
|
|
return 0
|
|
}
|
|
|
|
do_install_mender_to_mender_disk_image() {
|
|
log "$step/$total Installing Mender to Mender disk image..."
|
|
((step++))
|
|
|
|
if [ -z "$mender_disk_image" ] || [ -z "$device_type" ] || \
|
|
[ -z "$mender_client" ] || [ -z "$artifact_name" ]; then
|
|
show_help
|
|
return 1
|
|
fi
|
|
|
|
local supported=$(echo ${supported_devices[@]} | grep -o $device_type | wc -w)
|
|
|
|
[[ $supported -eq 0 ]] && \
|
|
{ log "Error: incorrect device type. Aborting."; return 1; }
|
|
|
|
# mender-image-1.5.0
|
|
stage_4_args="-m $mender_disk_image -d $device_type -g ${mender_client} -a ${artifact_name}"
|
|
|
|
if [ -n "$demo_host_ip" ]; then
|
|
stage_4_args="${stage_4_args} -i ${demo_host_ip}"
|
|
fi
|
|
|
|
if [ -n "$server_cert" ]; then
|
|
stage_4_args="${stage_4_args} -c ${server_cert}"
|
|
fi
|
|
|
|
if [ -n "$server_url" ]; then
|
|
stage_4_args="${stage_4_args} -u ${server_url}"
|
|
fi
|
|
|
|
if [ -n "${tenant_token}" ]; then
|
|
stage_4_args="${stage_4_args} -t ${tenant_token}"
|
|
fi
|
|
|
|
if [ -n "${keep}" ]; then
|
|
stage_4_args="${stage_4_args} -k"
|
|
fi
|
|
|
|
eval set -- " ${stage_4_args}"
|
|
|
|
${tool_dir}/convert-stage-4.sh ${stage_4_args} || ret=$?
|
|
[[ $ret -ne 0 ]] && { log "\nInstalling Mender to Mender disk image failed."; return $ret; }
|
|
|
|
# Update test configuration file
|
|
update_test_config_file $device_type artifact-name $artifact_name
|
|
|
|
return 0
|
|
}
|
|
|
|
do_install_bootloader_to_mender_disk_image() {
|
|
log "$step/$total Installing Bootloader to Mender disk image..."
|
|
((step++))
|
|
|
|
if [ -z "$mender_disk_image" ] || [ -z "$device_type" ] || \
|
|
[ -z "$bootloader_toolchain" ]; then
|
|
show_help
|
|
return 1
|
|
fi
|
|
|
|
local supported=$(echo ${supported_devices[@]} | grep -o $device_type | wc -w)
|
|
|
|
[[ $supported -eq 0 ]] && \
|
|
{ log "Error: incorrect device type. Aborting."; return 1; }
|
|
|
|
case "$device_type" in
|
|
"beaglebone")
|
|
stage_5_args="-m $mender_disk_image -d $device_type -b ${bootloader_toolchain} $keep"
|
|
eval set -- " ${stage_5_args}"
|
|
${tool_dir}/bbb-convert-stage-5.sh ${stage_5_args}|| ret=$?
|
|
[[ $ret -ne 0 ]] && { log "\nInstalling Bootloader failed."; return $ret; }
|
|
|
|
# Update test configuration file
|
|
update_test_config_file $device_type distro-feature "mender-grub" \
|
|
mount-location "\/boot\/efi"
|
|
;;
|
|
"raspberrypi3")
|
|
stage_5_args="-m $mender_disk_image -d $device_type -b ${bootloader_toolchain} $keep"
|
|
eval set -- " ${stage_5_args}"
|
|
${tool_dir}/rpi3-convert-stage-5.sh ${stage_5_args}|| ret=$?
|
|
[[ $ret -ne 0 ]] && { log "\nInstalling Bootloader failed."; return $ret; }
|
|
|
|
# Update test configuration file
|
|
update_test_config_file $device_type mount-location "\/uboot"
|
|
;;
|
|
esac
|
|
|
|
return 0
|
|
}
|
|
|
|
do_mender_disk_image_to_artifact() {
|
|
log "$step/$total Creating Mender Artifact..."
|
|
((step++))
|
|
|
|
if [ -z "${mender_disk_image}" ]; then
|
|
log "Mender disk image not set. Aborting."
|
|
return 1
|
|
fi
|
|
|
|
if [ -z "${device_type}" ]; then
|
|
log "Target device_type name not set. Aborting."
|
|
return 1
|
|
fi
|
|
|
|
if [ -z "${artifact_name}" ]; then
|
|
log "Artifact name not set. Aborting."
|
|
return 1
|
|
fi
|
|
|
|
if [ -z "${rootfs_partition_id}" ]; then
|
|
log "\tRootfs partition id not set - 'primary' will be used by default."
|
|
rootfs_partition_id="primary"
|
|
fi
|
|
|
|
local supported=$(echo ${supported_devices[@]} | grep -o $device_type | wc -w)
|
|
|
|
[[ $supported -eq 0 ]] && \
|
|
{ log "Error: incorrect device type. Aborting."; return 1; }
|
|
|
|
inarray=$(echo ${rootfs_partition_ids[@]} | grep -o $rootfs_partition_id | wc -w)
|
|
|
|
[[ $inarray -eq 0 ]] && \
|
|
{ log "Error: invalid rootfs partition id provided. Aborting."; return 1; }
|
|
|
|
local count=
|
|
local bootstart=
|
|
local rootfs_a_start=
|
|
local rootfs_a_size=
|
|
local rootfs_b_start=
|
|
local rootfs_b_size=
|
|
local rootfs_path=
|
|
local sdimg_device_type=
|
|
|
|
get_mender_disk_info $mender_disk_image count sector_size rootfs_a_start \
|
|
rootfs_a_size rootfs_b_start rootfs_b_size
|
|
ret=$?
|
|
[[ $ret -ne 0 ]] && \
|
|
{ log "Error: cannot validate Mender disk image. Aborting."; return 1; }
|
|
|
|
# Check if device type matches.
|
|
create_device_maps $mender_disk_image mender_disk_mappings
|
|
mount_mender_disk ${mender_disk_mappings[@]}
|
|
|
|
# Find .sdimg file's dedicated device type.
|
|
mender_device_type=$( cat $sdimg_data_dir/mender/device_type | sed 's/[^=].*=//' )
|
|
|
|
if [ "$mender_device_type" != "$device_type" ]; then
|
|
log "Error: device types of Mender artifact & Mender not matching. Aborting."
|
|
ret=1
|
|
fi
|
|
|
|
if ! [ -x "$(command -v mender-artifact)" ]; then
|
|
log "Error: mender-artifact not found in PATH. Aborting."
|
|
ret=1
|
|
fi
|
|
|
|
if [ $ret -eq 0 ]; then
|
|
if [[ $rootfs_partition_id == "primary" ]]; then
|
|
prootfs_start=$rootfs_a_start
|
|
prootfs_size=$rootfs_a_size
|
|
rootfs_path=$sdimg_primary_dir
|
|
elif [[ $rootfs_partition_id == "secondary" ]]; then
|
|
prootfs_start=$rootfs_b_start
|
|
prootfs_size=$rootfs_b_size
|
|
rootfs_path=$sdimg_secondary_dir
|
|
fi
|
|
|
|
local mender_disk_basename=$(basename -- "$mender_disk_image")
|
|
local mender_disk_filename="${mender_disk_basename%.*}"
|
|
local mender_rootfs_basename=${mender_disk_filename}.ext4
|
|
mender_rootfs_image=${output_dir}/$mender_rootfs_basename
|
|
|
|
# Extract root filesystem ext4 image to use it to generate Mender artifact.
|
|
# Ext4 disk image will be also verified in acceptance tests.
|
|
extract_file_from_image $mender_disk_image $prootfs_start \
|
|
$prootfs_size $mender_rootfs_basename
|
|
|
|
fsck.ext4 -fp $mender_rootfs_image &> /dev/null || ret=$?
|
|
[[ $ret -ne 0 ]] && \
|
|
{ log "Error: checking $mender_rootfs_basename file system failed. Aborting."; }
|
|
|
|
# Find first available loopback device.
|
|
loopdevice=($(sudo losetup -f || ret=$?))
|
|
[[ $ret -ne 0 ]] && \
|
|
{ log "Error: cannot find an unused loop device. Aborting."; }
|
|
|
|
if [ $ret -eq 0 ]; then
|
|
#Mount extracted ext4 partition to verify 'artifact_info' file content.
|
|
sudo losetup $loopdevice ${mender_rootfs_image}
|
|
rootfs_mountpoint=${output_dir}/mnt/${rootfs_partition_id}
|
|
mkdir -p ${rootfs_mountpoint}
|
|
sudo mount $loopdevice ${rootfs_mountpoint}
|
|
|
|
# Set 'artifact name' as passed in the command line.
|
|
sudo sed -i '/^artifact/s/=.*$/='${artifact_name}'/' "${rootfs_mountpoint}/etc/mender/artifact_info"
|
|
|
|
sudo umount -l ${rootfs_mountpoint}
|
|
sudo losetup -d $loopdevice
|
|
rm -rf ${output_dir}/mnt
|
|
|
|
# Note: expected Mender Artifact name follows the scheme:
|
|
# mender-<device_name>-<artifact_name>.mender.
|
|
mender_artifact=${output_dir}/${mender_disk_filename}.mender
|
|
log "\tWriting Mender artifact to: ${mender_artifact}"
|
|
|
|
#Create Mender artifact
|
|
mender-artifact write rootfs-image \
|
|
--update ${mender_rootfs_image} \
|
|
--output-path ${mender_artifact} \
|
|
--artifact-name ${artifact_name} \
|
|
--device-type ${device_type}
|
|
|
|
ret=$?
|
|
[[ $ret -eq 0 ]] && \
|
|
{ log "\tCreating Mender Artifact succeeded."; } || \
|
|
{ log "\tCreating Mender Artifact failed."; }
|
|
fi
|
|
fi
|
|
|
|
# Clean and detach.
|
|
detach_device_maps ${mender_disk_mappings[@]}
|
|
|
|
rm -rf $sdimg_base_dir
|
|
return $ret
|
|
}
|
|
|
|
do_from_raw_disk_image() {
|
|
if [ -z "$raw_disk_image" ] || [ -z "$device_type" ] || \
|
|
[ -z "$artifact_name" ] || [ -z "$mender_client" ] || \
|
|
[ -z "$bootloader_toolchain" ]; then
|
|
show_help
|
|
return 1
|
|
fi
|
|
|
|
do_raw_disk_image_create_partitions || rc=$?
|
|
[[ $rc -ne 0 ]] && { return 1; }
|
|
|
|
do_install_mender_to_mender_disk_image || rc=$?
|
|
[[ $rc -ne 0 ]] && { return 1; }
|
|
|
|
do_install_bootloader_to_mender_disk_image || rc=$?
|
|
[[ $rc -ne 0 ]] && { return 1; }
|
|
|
|
do_mender_disk_image_to_artifact || rc=$?
|
|
[[ $rc -ne 0 ]] && { return 1; }
|
|
|
|
return 0
|
|
}
|
|
|
|
#read -s -p "Enter password for sudo: " sudoPW
|
|
#echo ""
|
|
|
|
PARAMS=""
|
|
|
|
# Load necessary functions.
|
|
source ${tool_dir}/mender-convert-functions.sh
|
|
|
|
export -f create_device_maps
|
|
export -f detach_device_maps
|
|
export -f mount_mender_disk
|
|
export -f log
|
|
export -f logsetup
|
|
|
|
# Before running any command first create output directory
|
|
# and configure where logs should be redirected.
|
|
mkdir -p $output_dir
|
|
logsetup
|
|
|
|
while (( "$#" )); do
|
|
case "$1" in
|
|
-p | --rootfs-partition-id)
|
|
rootfs_partition_id=$2
|
|
shift 2
|
|
;;
|
|
-m | --mender-disk-image)
|
|
mender_disk_image=$2
|
|
shift 2
|
|
;;
|
|
-r | --raw-disk-image)
|
|
raw_disk_image=$(get_path $2)
|
|
shift 2
|
|
;;
|
|
-s | --data-part-size-mb)
|
|
data_part_size_mb=$2
|
|
shift 2
|
|
;;
|
|
-d | --device-type)
|
|
device_type=$2
|
|
shift 2
|
|
;;
|
|
-a | --artifact-name)
|
|
artifact_name=$2
|
|
shift 2
|
|
;;
|
|
-g | --mender-client)
|
|
mender_client=$(get_path $2)
|
|
shift 2
|
|
;;
|
|
-b | --bootloader-toolchain)
|
|
bootloader_toolchain=$2
|
|
shift 2
|
|
;;
|
|
-i | --demo-host-ip)
|
|
demo_host_ip=$2
|
|
shift 2
|
|
;;
|
|
-c | --server-cert)
|
|
server_cert=$2
|
|
shift 2
|
|
;;
|
|
-u | --server-url)
|
|
server_url=$2
|
|
shift 2
|
|
;;
|
|
-t | --tenant-token)
|
|
tenant_token=$2
|
|
shift 2
|
|
;;
|
|
-k | --keep)
|
|
keep="-k"
|
|
shift 1
|
|
;;
|
|
-h | --help)
|
|
show_help
|
|
exit 0
|
|
;;
|
|
--)
|
|
shift
|
|
break
|
|
;;
|
|
-*)
|
|
log "Error: unsupported option $1"
|
|
exit 1
|
|
;;
|
|
*)
|
|
PARAMS="$PARAMS $1"
|
|
shift
|
|
;;
|
|
esac
|
|
done
|
|
|
|
[ -z "${data_part_size_mb}" ] && \
|
|
{ log "Default 'data' partition size set to 128MB"; data_part_size_mb=128; }
|
|
|
|
eval set -- "$PARAMS"
|
|
|
|
# Some commands expect elevated privileges.
|
|
sudo true
|
|
|
|
case "$1" in
|
|
raw-disk-image-shrink-rootfs)
|
|
total=1
|
|
do_raw_disk_image_shrink_rootfs || rc=$?
|
|
[[ $rc -ne 0 ]] && { log "Check $build_log for details."; exit 1; }
|
|
log "The rootfs partition in the raw disk image has been shrinked successfully!"
|
|
log "You can now convert the output raw disk image\n\t$raw_disk_image\
|
|
\nto a Mender disk image."
|
|
;;
|
|
raw-disk-image-create-partitions)
|
|
total=6
|
|
do_raw_disk_image_create_partitions || rc=$?
|
|
[[ $rc -ne 0 ]] && { log "Check $build_log for details."; exit 1; }
|
|
log "A new Mender disk image with partition layout to support Mender has been successfully created!"
|
|
log "You can find the output Mender disk image at:\n\t${mender_disk_image}."
|
|
;;
|
|
install-mender-to-mender-disk-image)
|
|
total=1
|
|
do_install_mender_to_mender_disk_image || rc=$?
|
|
[[ $rc -ne 0 ]] && { log "Check $build_log for details."; exit 1; }
|
|
log "The Mender client has been successfully installed to the Mender disk image."
|
|
log "You can find the output Mender disk image at:\n\t${mender_disk_image}."
|
|
;;
|
|
install-bootloader-to-mender-disk-image)
|
|
total=1
|
|
do_install_bootloader_to_mender_disk_image || rc=$?
|
|
[[ $rc -ne 0 ]] && { log "Check $build_log for details."; exit 1; }
|
|
log "A bootloader configuration supporting dual A/B rootfs updates has been installed to the Mender disk image!"
|
|
log "You can find the output Mender disk image at:\n\t${mender_disk_image}."
|
|
;;
|
|
mender-disk-image-to-artifact)
|
|
total=1
|
|
do_mender_disk_image_to_artifact || rc=$?
|
|
[[ $rc -ne 0 ]] && { log "Check $build_log for details."; exit 1; }
|
|
log "A Mender Artifact has been successfully extracted from your Mender disk image!"
|
|
log "You can find the Mender Artifact at:\n\t$mender_artifact\nand use it to deploy updates."
|
|
;;
|
|
from-raw-disk-image)
|
|
total=9
|
|
do_from_raw_disk_image || rc=$?
|
|
[[ $rc -ne 0 ]] && { log "Check $build_log for details."; exit 1; }
|
|
log "Conversion complete!"
|
|
log "The Mender disk image you can provision your device storage with is at:\
|
|
\n\t${mender_disk_image}."
|
|
log "The Mender root file system partition is at:\n\t${mender_rootfs_image}."
|
|
log "The Mender Artifact you can upload to your Mender server to deploy to your devices is at:\
|
|
\n\t${mender_artifact}."
|
|
;;
|
|
*)
|
|
show_help
|
|
;;
|
|
esac
|
|
|
|
exit 0
|
|
|