blob: b461175c4781531ff9e3cfa9ca10102ca467fb11 [file] [log] [blame]
Alex Deymo2b19cfb2015-03-26 00:35:07 -07001#!/bin/bash
2
Alex Deymoaea4c1c2015-08-19 20:24:43 -07003#
4# Copyright (C) 2015 The Android Open Source Project
5#
6# Licensed under the Apache License, Version 2.0 (the "License");
7# you may not use this file except in compliance with the License.
8# You may obtain a copy of the License at
9#
10# http://www.apache.org/licenses/LICENSE-2.0
11#
12# Unless required by applicable law or agreed to in writing, software
13# distributed under the License is distributed on an "AS IS" BASIS,
14# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15# See the License for the specific language governing permissions and
16# limitations under the License.
17#
Alex Deymo2b19cfb2015-03-26 00:35:07 -070018
Alex Deymo04ef2072015-08-13 08:19:16 -070019# This script generates some sample images used in unittests and packages them
20# in the sample_images.tar.bz2 file. The list of generated images and their
21# options are described in the main() function. You need to manually run this
22# script to update the generated images whenever you modify this script.
23
Alex Deymo2b19cfb2015-03-26 00:35:07 -070024set -e
25
26# cleanup <path>
27# Unmount and remove the mountpoint <path>
28cleanup() {
29 if ! sudo umount "$1" 2>/dev/null; then
30 if mountpoint -q "$1"; then
31 sync && sudo umount "$1"
32 fi
33 fi
34 rmdir "$1"
35}
36
Alex Deymo2e9533b2015-06-26 20:57:06 -070037# add_files_default <mntdir> <block_size>
38# Add several test files to the image mounted in <mntdir>.
39add_files_default() {
40 local mntdir="$1"
41 local block_size="$2"
Alex Deymo2b19cfb2015-03-26 00:35:07 -070042
43 ### Generate the files used in unittest with descriptive names.
44 sudo touch "${mntdir}"/empty-file
45
46 # regular: Regular files.
47 echo "small file" | sudo dd of="${mntdir}"/regular-small status=none
48 dd if=/dev/zero bs=1024 count=16 status=none | tr '\0' '\141' |
49 sudo dd of="${mntdir}"/regular-16k status=none
50 sudo dd if=/dev/zero of="${mntdir}"/regular-32k-zeros bs=1024 count=16 \
51 status=none
52
53 echo "with net_cap" | sudo dd of="${mntdir}"/regular-with_net_cap status=none
54 sudo setcap cap_net_raw=ep "${mntdir}"/regular-with_net_cap
55
56 # sparse_empty: Files with no data blocks at all (only sparse holes).
57 sudo truncate --size=10240 "${mntdir}"/sparse_empty-10k
58 sudo truncate --size=$(( block_size * 2 )) "${mntdir}"/sparse_empty-2blocks
59
60 # sparse: Files with some data blocks but also sparse holes.
61 echo -n "foo" |
62 sudo dd of="${mntdir}"/sparse-16k-last_block bs=1 \
63 seek=$(( 16 * 1024 - 3)) status=none
64
65 # ext2 inodes have 12 direct blocks, one indirect, one double indirect and
66 # one triple indirect. 10000 should be enough to have an indirect and double
67 # indirect block.
68 echo -n "foo" |
69 sudo dd of="${mntdir}"/sparse-10000blocks bs=1 \
70 seek=$(( block_size * 10000 )) status=none
71
72 sudo truncate --size=16384 "${mntdir}"/sparse-16k-first_block
73 echo "first block" | sudo dd of="${mntdir}"/sparse-16k-first_block status=none
74
75 sudo truncate --size=16384 "${mntdir}"/sparse-16k-holes
76 echo "a" | sudo dd of="${mntdir}"/sparse-16k-holes bs=1 seek=100 status=none
77 echo "b" | sudo dd of="${mntdir}"/sparse-16k-holes bs=1 seek=10000 status=none
78
79 # link: symlinks and hardlinks.
80 sudo ln -s "broken-link" "${mntdir}"/link-short_symlink
81 sudo ln -s $(dd if=/dev/zero bs=256 count=1 status=none | tr '\0' '\141') \
82 "${mntdir}"/link-long_symlink
83 sudo ln "${mntdir}"/regular-16k "${mntdir}"/link-hard-regular-16k
84
85 # Directories.
86 sudo mkdir -p "${mntdir}"/dir1/dir2/dir1
87 echo "foo" | sudo tee "${mntdir}"/dir1/dir2/file >/dev/null
88 echo "bar" | sudo tee "${mntdir}"/dir1/file >/dev/null
89
90 # removed: removed files that should not be listed.
91 echo "We will remove this file so it's contents will be somewhere in the " \
92 "empty space data but it won't be all zeros." |
93 sudo dd of="${mntdir}"/removed conv=fsync status=none
94 sudo rm "${mntdir}"/removed
Alex Deymo2e9533b2015-06-26 20:57:06 -070095}
96
97# add_files_ue_settings <mntdir> <block_size>
98# Add the update_engine.conf settings file. This file contains the
99add_files_ue_settings() {
100 local mntdir="$1"
101
102 sudo mkdir -p "${mntdir}"/etc >/dev/null
103 sudo tee "${mntdir}"/etc/update_engine.conf >/dev/null <<EOF
104PAYLOAD_MINOR_VERSION=1234
105EOF
106 # Example of a real lsb-release file released on link stable.
107 sudo tee "${mntdir}"/etc/lsb-release >/dev/null <<EOF
108CHROMEOS_AUSERVER=https://tools.google.com/service/update2
109CHROMEOS_BOARD_APPID={F26D159B-52A3-491A-AE25-B23670A66B32}
110CHROMEOS_CANARY_APPID={90F229CE-83E2-4FAF-8479-E368A34938B1}
111CHROMEOS_DEVSERVER=
112CHROMEOS_RELEASE_APPID={F26D159B-52A3-491A-AE25-B23670A66B32}
113CHROMEOS_RELEASE_BOARD=link-signed-mp-v4keys
114CHROMEOS_RELEASE_BRANCH_NUMBER=63
115CHROMEOS_RELEASE_BUILD_NUMBER=6946
116CHROMEOS_RELEASE_BUILD_TYPE=Official Build
117CHROMEOS_RELEASE_CHROME_MILESTONE=43
118CHROMEOS_RELEASE_DESCRIPTION=6946.63.0 (Official Build) stable-channel link
119CHROMEOS_RELEASE_NAME=Chrome OS
120CHROMEOS_RELEASE_PATCH_NUMBER=0
121CHROMEOS_RELEASE_TRACK=stable-channel
122CHROMEOS_RELEASE_VERSION=6946.63.0
123GOOGLE_RELEASE=6946.63.0
124EOF
125}
126
Alex Deymocbc22742016-03-04 17:53:02 -0800127add_files_postinstall() {
128 local mntdir="$1"
129
130 sudo mkdir -p "${mntdir}"/bin >/dev/null
131
132 # A postinstall bash program.
133 sudo tee "${mntdir}"/bin/postinst_example >/dev/null <<EOF
134#!/etc/../bin/sh
135echo "I'm a postinstall program and I know how to write to stdout"
136echo "My call was $@"
137exit 0
138EOF
139
140 # A symlink to another program. This should also work.
141 sudo ln -s "postinst_example" "${mntdir}"/bin/postinst_link
142
143 sudo tee "${mntdir}"/bin/postinst_fail3 >/dev/null <<EOF
144#!/etc/../bin/sh
145exit 3
146EOF
147
148 sudo tee "${mntdir}"/bin/postinst_fail1 >/dev/null <<EOF
149#!/etc/../bin/sh
150exit 1
151EOF
152
Alex Deymod15c5462016-03-09 18:11:12 -0800153 # A program that succeeds if it is suspended during the first 5 minutes.
154 sudo tee "${mntdir}"/bin/postinst_suspend >/dev/null <<EOF
155#!/etc/../bin/sh
156trap "{ echo Got a SIGCONT; exit 0; }" CONT
157# Signal that we are ready to receive the signal by redirecting our stdin to
158# /dev/zero, the test can detect that.
159exec </dev/zero
160# Allow the signal handler to run every 100 ms.
161i=3000
162while [ \$i -ge 0 ]; do
163 sleep 0.1
164 i=\$((i-1))
165done
166exit 1
167EOF
168
Alex Deymocbc22742016-03-04 17:53:02 -0800169 # A postinstall bash program.
170 sudo tee "${mntdir}"/bin/self_check_context >/dev/null <<EOF
171#!/etc/../bin/sh
172echo "This is my context:"
173ls -lZ "\$0" | grep -F ' u:object_r:postinstall_file:s0 ' || exit 5
174exit 0
175EOF
176
177 sudo tee "${mntdir}"/postinst >/dev/null <<EOF
178#!/etc/../bin/sh
179echo "postinst"
180exit 0
181EOF
182
183 sudo chmod +x "${mntdir}"/postinst "${mntdir}"/bin/*
184}
185
Alex Deymo2e9533b2015-06-26 20:57:06 -0700186# generate_fs <filename> <kind> <size> [block_size] [block_groups]
187generate_fs() {
188 local filename="$1"
189 local kind="$2"
190 local size="$3"
191 local block_size="${4:-4096}"
192 local block_groups="${5:-}"
193
194 local mkfs_opts=( -q -F -b "${block_size}" -L "ROOT-TEST" -t ext2 )
195 if [[ -n "${block_groups}" ]]; then
196 mkfs_opts+=( -G "${block_groups}" )
197 fi
198
199 local mntdir=$(mktemp --tmpdir -d generate_ext2.XXXXXX)
200 trap 'cleanup "${mntdir}"; rm -f "${filename}"' INT TERM EXIT
201
202 # Cleanup old image.
203 if [[ -e "${filename}" ]]; then
204 rm -f "${filename}"
205 fi
206 truncate --size="${size}" "${filename}"
207
208 mkfs.ext2 "${mkfs_opts[@]}" "${filename}"
209 sudo mount "${filename}" "${mntdir}" -o loop
210
211 case "${kind}" in
Alex Deymod15c5462016-03-09 18:11:12 -0800212 unittest)
Alex Deymo2e9533b2015-06-26 20:57:06 -0700213 add_files_ue_settings "${mntdir}" "${block_size}"
Alex Deymocbc22742016-03-04 17:53:02 -0800214 add_files_postinstall "${mntdir}" "${block_size}"
Alex Deymo2e9533b2015-06-26 20:57:06 -0700215 ;;
216 default)
217 add_files_default "${mntdir}" "${block_size}"
218 ;;
Alex Deymoc90be632016-02-17 19:25:20 -0800219 empty)
220 ;;
Alex Deymo2e9533b2015-06-26 20:57:06 -0700221 esac
Alex Deymo2b19cfb2015-03-26 00:35:07 -0700222
223 cleanup "${mntdir}"
224 trap - INT TERM EXIT
225}
226
Alex Deymo04ef2072015-08-13 08:19:16 -0700227OUTPUT_DIR=$(dirname "$0")
228IMAGES=()
Alex Deymo2b19cfb2015-03-26 00:35:07 -0700229
Alex Deymo04ef2072015-08-13 08:19:16 -0700230# generate_image <image_name> [<image args> ...]
231generate_image() {
232 echo "Generating image $1.img"
233 IMAGES+=( "$1.img" )
234 generate_fs "${OUTPUT_DIR}/$1.img" "${@:2}"
235}
Alex Deymo2b19cfb2015-03-26 00:35:07 -0700236
Alex Deymo04ef2072015-08-13 08:19:16 -0700237main() {
238 # Add more sample images here.
239 generate_image disk_ext2_1k default 16777216 1024
240 generate_image disk_ext2_4k default 16777216 4096
Alex Deymoc90be632016-02-17 19:25:20 -0800241 generate_image disk_ext2_4k_empty empty $((1024 * 4096)) 4096
Alex Deymod15c5462016-03-09 18:11:12 -0800242 generate_image disk_ext2_unittest unittest $((1024 * 4096)) 4096
Alex Deymo04ef2072015-08-13 08:19:16 -0700243
244 # Generate the tarball and delete temporary images.
245 echo "Packing tar file sample_images.tar.bz2"
246 tar -jcf "${OUTPUT_DIR}/sample_images.tar.bz2" -C "${OUTPUT_DIR}" \
Alex Deymoa3553e42016-03-04 18:55:05 -0800247 --sparse "${IMAGES[@]}"
Alex Deymo04ef2072015-08-13 08:19:16 -0700248 cd "${OUTPUT_DIR}"
249 rm "${IMAGES[@]}"
250}
251
252main