blob: fac20c6203b964232a483c579d402a286ae2a1ef [file] [log] [blame]
/*
* Copyright (C) 2014 Cyril Hrubis chrubis@suse.cz
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of version 2 of the GNU General Public License as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it would be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
* Further, this software is distributed without any warranty that it is
* free of the rightful claim of any third person regarding infringement
* or the like. Any license provided herein, whether implied or
* otherwise, applies only to this software file. Patent licenses, if
* any, provided herein do not apply to combinations of this program with
* other software, or any other product whatsoever.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <sys/mount.h>
#include <errno.h>
#include <unistd.h>
#include <stdlib.h>
#include <linux/loop.h>
#include "test.h"
#include "safe_macros.h"
#ifndef LOOP_CTL_GET_FREE
# define LOOP_CTL_GET_FREE 0x4C82
#endif
#define LOOP_CONTROL_FILE "/dev/loop-control"
#define DEV_FILE "test_dev.img"
static char dev_path[1024];
static int device_acquired;
static const char *dev_variants[] = {
"/dev/loop%i",
"/dev/loop/%i"
};
static int set_dev_path(int dev)
{
unsigned int i;
struct stat st;
for (i = 0; i < ARRAY_SIZE(dev_variants); i++) {
snprintf(dev_path, sizeof(dev_path), dev_variants[i], dev);
if (stat(dev_path, &st) == 0 && S_ISBLK(st.st_mode))
return 1;
}
return 0;
}
static int find_free_loopdev(void)
{
int ctl_fd, dev_fd, rc, i;
struct loop_info loopinfo;
/* since Linux 3.1 */
ctl_fd = open(LOOP_CONTROL_FILE, O_RDWR);
if (ctl_fd > 0) {
rc = ioctl(ctl_fd, LOOP_CTL_GET_FREE);
close(ctl_fd);
if (rc >= 0) {
set_dev_path(rc);
tst_resm(TINFO, "Found free device '%s'", dev_path);
return 0;
}
tst_resm(TINFO, "Couldn't find free loop device");
return 1;
}
switch (errno) {
case ENOENT:
break;
case EACCES:
tst_resm(TINFO | TERRNO,
"Not allowed to open " LOOP_CONTROL_FILE ". "
"Are you root?");
break;
default:
tst_resm(TBROK | TERRNO, "Failed to open " LOOP_CONTROL_FILE);
}
/*
* Older way is to iterate over /dev/loop%i and /dev/loop/%i and try
* LOOP_GET_STATUS ioctl() which fails for free loop devices.
*/
for (i = 0; i < 256; i++) {
if (!set_dev_path(i))
continue;
dev_fd = open(dev_path, O_RDONLY);
if (dev_fd < 0)
continue;
if (ioctl(dev_fd, LOOP_GET_STATUS, &loopinfo) == 0) {
tst_resm(TINFO, "Device '%s' in use", dev_path);
} else {
if (errno != ENXIO)
continue;
tst_resm(TINFO, "Found free device '%s'", dev_path);
close(dev_fd);
return 0;
}
close(dev_fd);
}
tst_resm(TINFO, "No free devices found");
return 1;
}
static void attach_device(void (*cleanup_fn)(void),
const char *dev, const char *file)
{
int dev_fd, file_fd, err;
dev_fd = SAFE_OPEN(cleanup_fn, dev, O_RDWR);
file_fd = SAFE_OPEN(cleanup_fn, file, O_RDWR);
if (ioctl(dev_fd, LOOP_SET_FD, file_fd) < 0) {
err = errno;
close(dev_fd);
close(file_fd);
tst_brkm(TBROK, cleanup_fn,
"ioctl(%s, LOOP_SET_FD, %s) failed: %s",
dev, file, tst_strerrno(err));
}
close(dev_fd);
close(file_fd);
}
static void detach_device(const char *dev)
{
int dev_fd, ret, i;
dev_fd = open(dev, O_RDONLY);
if (dev_fd < 0) {
tst_resm(TWARN | TERRNO, "open(%s) failed", dev);
return;
}
/* keep trying to clear LOOPDEV until we get ENXIO, a quick succession
* of attach/detach might not give udev enough time to complete */
for (i = 0; i < 40; i++) {
ret = ioctl(dev_fd, LOOP_CLR_FD, 0);
if (ret && (errno == ENXIO)) {
close(dev_fd);
return;
}
if (ret && (errno != EBUSY)) {
tst_resm(TWARN,
"ioctl(%s, LOOP_CLR_FD, 0) unexpectedly failed with: %s",
dev, tst_strerrno(errno));
}
usleep(50000);
}
close(dev_fd);
tst_resm(TWARN,
"ioctl(%s, LOOP_CLR_FD, 0) no ENXIO for too long", dev);
}
const char *tst_acquire_device(void (cleanup_fn)(void))
{
char *dev;
struct stat st;
if (device_acquired)
tst_brkm(TBROK, cleanup_fn, "Device allready acquired");
if (!tst_tmpdir_created()) {
tst_brkm(TBROK, cleanup_fn,
"Cannot acquire device without tmpdir() created");
}
dev = getenv("LTP_DEV");
if (dev) {
tst_resm(TINFO, "Using test device LTP_DEV='%s'", dev);
SAFE_STAT(cleanup_fn, dev, &st);
if (!S_ISBLK(st.st_mode)) {
tst_brkm(TBROK, cleanup_fn,
"%s is not a block device", dev);
}
if (tst_fill_file(dev, 0, 1024, 512)) {
tst_brkm(TBROK | TERRNO, cleanup_fn,
"Failed to clear the first 512k of %s", dev);
}
return dev;
}
if (tst_fill_file(DEV_FILE, 0, 1024, 102400)) {
tst_brkm(TBROK | TERRNO, cleanup_fn,
"Failed to create " DEV_FILE);
}
if (find_free_loopdev())
return NULL;
attach_device(cleanup_fn, dev_path, DEV_FILE);
device_acquired = 1;
return dev_path;
}
void tst_release_device(const char *dev)
{
if (getenv("LTP_DEV"))
return;
/*
* Loop device was created -> we need to deatch it.
*
* The file image is deleted in tst_rmdir();
*/
detach_device(dev);
device_acquired = 0;
}
int tst_umount(const char *path)
{
int err, ret, i;
for (i = 0; i < 50; i++) {
ret = umount(path);
err = errno;
if (!ret)
return 0;
tst_resm(TINFO, "umount('%s') failed with %s, try %2i...",
path, tst_strerrno(err), i+1);
if (i == 0 && err == EBUSY) {
tst_resm(TINFO, "Likely gvfsd-trash is probing newly "
"mounted fs, kill it to speed up tests.");
}
usleep(100000);
}
tst_resm(TWARN, "Failed to umount('%s') after 50 retries", path);
errno = err;
return -1;
}