]> git.apps.os.sepia.ceph.com Git - ceph.git/commitdiff
qa: add rbd/concurrent workunit
authorAlex Elder <elder@inktank.com>
Tue, 29 Jan 2013 21:51:13 +0000 (15:51 -0600)
committerAlex Elder <elder@inktank.com>
Tue, 29 Jan 2013 22:42:44 +0000 (16:42 -0600)
This defines a new workunit shell script that performs a bunch of
rbd operations concurrently in order to exercise code paths and
catch reference count and bad pointer problems.

Signed-off-by: Alex Elder <elder@inktank.com>
qa/workunits/rbd/concurrent.sh [new file with mode: 0755]

diff --git a/qa/workunits/rbd/concurrent.sh b/qa/workunits/rbd/concurrent.sh
new file mode 100755 (executable)
index 0000000..70fc11d
--- /dev/null
@@ -0,0 +1,311 @@
+#!/bin/bash
+
+# Copyright (C) 2013 Inktank Storage, Inc.
+#
+# This is free software; see the source for copying conditions.
+# There is NO warranty; not even for MERCHANTABILITY or FITNESS FOR
+# A PARTICULAR PURPOSE.
+#
+# This is free software; you can redistribute it and/or modify it
+# under the terms of the GNU General Public License as
+# published by the Free Software Foundation version 2.
+
+# Alex Elder <elder@inktank.com>
+# January 29, 2013
+
+################################################################
+
+# The purpose of this test is to exercise paths through the rbd
+# code, making sure no bad pointer references or invalid reference
+# count operations occur in the face of concurrent activity.
+#
+# Each pass of the test creates an rbd image, maps it, and writes
+# some data into the image.  It also reads some data from all of the
+# other images that exist at the time the pass executes.  Finally,
+# the image is unmapped and removed.  The image removal completes in
+# the background.
+#
+# An iteration of the test consists of performing some number of
+# passes, initating each pass as a background job, and finally
+# sleeping for a variable delay.  The delay is initially a specified
+# value, but each iteration shortens that proportionally, such that
+# the last iteration will not delay at all.
+#
+# The result exercises concurrent creates and deletes of rbd images,
+# writes to new images, reads from both written and unwritten image
+# data (including reads concurrent with writes), and attempts to
+# unmap images being read.
+
+# Usage: concurrent [-i <iter>] [-c <count>] [-d <delay>]
+#
+# Exit status:
+#     0:  success
+#     1:  usage error
+#     2:  other runtime error
+#    99:  argument count error (programming error)
+#   100:  getopt error (internal error)
+
+################################################################
+
+set -x
+
+CEPH_SECRET_FILE=${CEPH_SECRET_FILE:-}
+CEPH_ID=${CEPH_ID:-admin}
+SECRET_ARGS=""
+if [ "${CEPH_SECRET_FILE}" ]; then
+       SECRET_ARGS="--secret $CEPH_SECRET_FILE"
+fi
+
+# Default flag values; RBD_CONCURRENT_ITER names are intended
+# to be used in yaml scripts to pass in alternate values, e.g.:
+#    env:
+#        RBD_CONCURRENT_ITER: 20
+#        RBD_CONCURRENT_COUNT: 5
+#        RBD_CONCURRENT_DELAY: 3
+ITER_DEFAULT=${RBD_CONCURRENT_ITER:-10}
+COUNT_DEFAULT=${RBD_CONCURRENT_COUNT:-2}
+DELAY_DEFAULT=${RBD_CONCURRENT_DELAY:-5}               # seconds
+
+################################################################
+
+# List of rbd id's *not* created by this script
+export INITIAL_RBD_IDS=$(ls /sys/bus/rbd/devices)
+
+function setup() {
+       sudo chown ubuntu /sys/bus/rbd/add /sys/bus/rbd/remove
+
+       # Set up some environment for normal teuthology test setup.
+       # This really should not be necessary but I found it was.
+       TOP="/tmp/cephtest"
+       export CEPH_ARGS="--conf ${TOP}/ceph.conf"
+       export CEPH_ARGS="${CEPH_ARGS} --keyring ${TOP}/data/client.0.keyring"
+       export CEPH_ARGS="${CEPH_ARGS} --name client.0"
+
+       export LD_LIBRARY_PATH="${TOP}/binary/usr/local/lib:${LD_LIBRARY_PATH}"
+       export PATH="${TOP}/binary/usr/local/bin:${PATH}"
+       export PATH="${TOP}/binary/usr/local/sbin:${PATH}"
+}
+
+function cleanup() {
+       [ ! "${NAMESDIR}" ] && return
+       local id
+       local image
+
+       # Unmap mapped devices
+       for id in $(rbd_ids); do
+               image=$(cat "/sys/bus/rbd/devices/${id}/name")
+               rbd_unmap_image "${id}"
+               rbd_destroy_image "${image}"
+       done
+       # Get any leftover images
+       for image in $(rbd ls 2>/dev/null); do
+               rbd_destroy_image "${image}"
+       done
+       wait
+       sync
+       rmdir "${NAMESDIR}"
+       sudo chown root /sys/bus/rbd/add /sys/bus/rbd/remove
+}
+trap cleanup HUP INT QUIT
+
+NAMESDIR=$(mktemp -d /tmp/image_names.XXXXXX)
+
+# print a usage message and quit
+#
+# if a message is supplied, print that first, and then exit
+# with non-zero status
+function usage() {
+       if [ $# -gt 0 ]; then
+               echo "" >&2
+               echo "$@" >&2
+       fi
+
+       echo "" >&2
+       echo "Usage: ${PROGNAME} <options> <tests>" >&2
+       echo "" >&2
+       echo "    options:" >&2
+       echo "        -h or --help" >&2
+       echo "            show this message" >&2
+       echo "        -i or --iterations" >&2
+       echo "            iteration count (1 or more)" >&2
+       echo "        -c or --count" >&2
+       echo "            images created per iteration (1 or more)" >&2
+       echo "        -d or --delay" >&2
+       echo "            maximum delay between iterations" >&2
+       echo "" >&2
+       echo "    defaults:" >&2
+       echo "        iterations: ${ITER_DEFAULT}"
+       echo "        count: ${COUNT_DEFAULT}"
+       echo "        delay: ${DELAY_DEFAULT} (seconds)"
+       echo "" >&2
+
+       [ $# -gt 0 ] && exit 1
+
+       exit 0          # This is used for a --help
+}
+
+# parse command line arguments
+function parseargs() {
+       ITER="${ITER_DEFAULT}"
+       COUNT="${COUNT_DEFAULT}"
+       DELAY="${DELAY_DEFAULT}"
+
+       # Short option flags
+       SHORT_OPTS=""
+       SHORT_OPTS="${SHORT_OPTS},h"
+       SHORT_OPTS="${SHORT_OPTS},i:"
+       SHORT_OPTS="${SHORT_OPTS},c:"
+       SHORT_OPTS="${SHORT_OPTS},d:"
+
+       # Short option flags
+       LONG_OPTS=""
+       LONG_OPTS="${LONG_OPTS},help"
+       LONG_OPTS="${LONG_OPTS},iterations:"
+       LONG_OPTS="${LONG_OPTS},count:"
+       LONG_OPTS="${LONG_OPTS},delay:"
+
+       TEMP=$(getopt --name "${PROGNAME}" \
+               --options "${SHORT_OPTS}" \
+               --longoptions "${LONG_OPTS}" \
+               -- "$@")
+       eval set -- "$TEMP"
+
+       while [ "$1" != "--" ]; do
+               case "$1" in
+                       -h|--help)
+                               usage
+                               ;;
+                       -i|--iterations)
+                               ITER="$2"
+                               [ "${ITER}" -lt 1 ] &&
+                                       usage "bad iterations value"
+                               shift
+                               ;;
+                       -c|--count)
+                               COUNT="$2"
+                               [ "${COUNT}" -lt 1 ] &&
+                                       usage "bad count value"
+                               shift
+                               ;;
+                       -d|--delay)
+                               DELAY="$2"
+                               shift
+                               ;;
+                       *)
+                               exit 100        # Internal error
+                               ;;
+               esac
+               shift
+       done
+       shift
+}
+
+function rbd_ids() {
+       [ $# -eq 0 ] || exit 99
+       local ids
+       local i
+
+       [ -d /sys/bus/rbd ] || return
+       ids=" $(echo $(ls /sys/bus/rbd/devices)) "
+       for i in ${INITIAL_RBD_IDS}; do
+               ids=${ids/ ${i} / }
+       done
+       echo ${ids}
+}
+
+function rbd_create_image() {
+       [ $# -eq 0 ] || exit 99
+       local image=$(basename $(mktemp "${NAMESDIR}/image.XXXXXX"))
+
+       rbd create "${image}" --size=1024
+       echo "${image}"
+}
+
+function rbd_image_id() {
+       [ $# -eq 1 ] || exit 99
+       local image="$1"
+
+       grep -l "${image}" /sys/bus/rbd/devices/*/name 2>/dev/null |
+               cut -d / -f 6
+}
+
+function rbd_map_image() {
+       [ $# -eq 1 ] || exit 99
+       local image="$1"
+       local id
+
+       rbd map "${image}" --user "${CEPH_ID}" ${SECRET_ARGS}
+
+       udevadm settle
+       id=$(rbd_image_id "${image}")
+       echo "${id}"
+}
+
+function rbd_write_image() {
+       [ $# -eq 1 ] || exit 99
+       local id="$1"
+
+       dd if="/bin/busybox" of="/dev/rbd${id}" bs=65536 count=16 seek=16 \
+               > /dev/null 2>&1
+}
+
+function rbd_read_image() {
+       [ $# -eq 1 ] || exit 99
+       local id="$1"
+
+       dd if="/dev/rbd${id}" of=/dev/null bs=65536 count=1 > /dev/null 2>&1
+}
+
+function rbd_unmap_image() {
+       [ $# -eq 1 ] || exit 99
+       local id="$1"
+
+       rbd unmap "/dev/rbd${id}" > /dev/null 2>&1
+       udevadm settle
+}
+
+function rbd_destroy_image() {
+       [ $# -eq 1 ] || exit 99
+       local image="$1"
+
+       # Don't wait for it to complete, to increase concurrency
+       rbd rm "${image}" >/dev/null 2>&1 &
+       rm -f "${NAMESDIR}/${image}"
+}
+
+function one_pass() {
+       [ $# -eq 0 ] || exit 99
+       local image
+       local id
+       local i
+
+       image=$(rbd_create_image)
+       id=$(rbd_map_image "${image}")
+       for i in $(rbd_ids); do
+               if [ "${i}" -eq "${id}" ]; then
+                       rbd_write_image "${i}"
+               else
+                       rbd_read_image "${i}"
+               fi
+       done
+       rbd_unmap_image "${id}"
+       rbd_destroy_image "${image}"
+}
+
+################################################################
+
+parseargs "$@"
+
+setup
+
+for iter in $(seq 1 "${ITER}"); do
+       for count in $(seq 1 "${COUNT}"); do
+               one_pass &
+       done
+       # Sleep longer at first, overlap iterations more later
+       sleep $(expr "${DELAY}" - "${DELAY}" \* "${iter}" / "${ITER}")
+done
+wait
+cleanup
+
+exit 0