670 lines
14 KiB
C
670 lines
14 KiB
C
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||
|
|
/*
|
||
|
|
* Copyright (C) 2014 Cyril Hrubis chrubis@suse.cz
|
||
|
|
*/
|
||
|
|
|
||
|
|
#include <sys/types.h>
|
||
|
|
#include <sys/stat.h>
|
||
|
|
#include <sys/ioctl.h>
|
||
|
|
#include <sys/mount.h>
|
||
|
|
#include <mntent.h>
|
||
|
|
#include <errno.h>
|
||
|
|
#include <unistd.h>
|
||
|
|
#include <stdlib.h>
|
||
|
|
#include <linux/loop.h>
|
||
|
|
#include <stdint.h>
|
||
|
|
#include <inttypes.h>
|
||
|
|
#include <sys/sysmacros.h>
|
||
|
|
#include <linux/btrfs.h>
|
||
|
|
#include <linux/limits.h>
|
||
|
|
#include "lapi/syscalls.h"
|
||
|
|
#include "test.h"
|
||
|
|
#include "safe_macros.h"
|
||
|
|
#include "tst_device.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"
|
||
|
|
#define DEV_SIZE_MB 300u
|
||
|
|
#define UUID_STR_SZ 37
|
||
|
|
#define UUID_FMT "%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-%02x%02x%02x%02x%02x%02x"
|
||
|
|
|
||
|
|
static char dev_path[PATH_MAX];
|
||
|
|
static int device_acquired;
|
||
|
|
static unsigned long prev_dev_sec_write;
|
||
|
|
|
||
|
|
static const char * const dev_loop_variants[] = {
|
||
|
|
"/dev/loop%i",
|
||
|
|
"/dev/loop/%i",
|
||
|
|
"/dev/block/loop%i"
|
||
|
|
};
|
||
|
|
|
||
|
|
static const char * const dev_variants[] = {
|
||
|
|
"/dev/%s",
|
||
|
|
"/dev/block/%s"
|
||
|
|
};
|
||
|
|
|
||
|
|
static int set_dev_loop_path(int dev, char *path, size_t path_len)
|
||
|
|
{
|
||
|
|
unsigned int i;
|
||
|
|
struct stat st;
|
||
|
|
|
||
|
|
for (i = 0; i < ARRAY_SIZE(dev_loop_variants); i++) {
|
||
|
|
snprintf(path, path_len, dev_loop_variants[i], dev);
|
||
|
|
|
||
|
|
if (stat(path, &st) == 0 && S_ISBLK(st.st_mode))
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
return 1;
|
||
|
|
}
|
||
|
|
|
||
|
|
static int set_dev_path(char *dev, char *path, size_t path_len)
|
||
|
|
{
|
||
|
|
unsigned int i;
|
||
|
|
struct stat st;
|
||
|
|
|
||
|
|
for (i = 0; i < ARRAY_SIZE(dev_variants); i++) {
|
||
|
|
snprintf(path, path_len, dev_variants[i], dev);
|
||
|
|
|
||
|
|
if (stat(path, &st) == 0 && S_ISBLK(st.st_mode))
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
return 1;
|
||
|
|
}
|
||
|
|
|
||
|
|
int tst_find_free_loopdev(char *path, size_t path_len)
|
||
|
|
{
|
||
|
|
int ctl_fd, dev_fd, rc, i, path_set;
|
||
|
|
struct loop_info loopinfo;
|
||
|
|
char buf[PATH_MAX];
|
||
|
|
|
||
|
|
/* 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) {
|
||
|
|
if (path) {
|
||
|
|
// b/148978487 retry to allow time for device creation
|
||
|
|
for (i = 0; i < 50; i++) {
|
||
|
|
path_set = set_dev_loop_path(rc, path, path_len);
|
||
|
|
// set_dev_path returns 1 on success
|
||
|
|
if (!path_set)
|
||
|
|
break;
|
||
|
|
usleep(50000);
|
||
|
|
}
|
||
|
|
if (path_set)
|
||
|
|
tst_brkm(TBROK, NULL, "Could not stat loop device %i", rc);
|
||
|
|
}
|
||
|
|
tst_resm(TINFO, "Found free device %d '%s'",
|
||
|
|
rc, path ?: "");
|
||
|
|
return rc;
|
||
|
|
}
|
||
|
|
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_loop_path(i, buf, sizeof(buf)))
|
||
|
|
continue;
|
||
|
|
|
||
|
|
dev_fd = open(buf, O_RDONLY);
|
||
|
|
|
||
|
|
if (dev_fd < 0)
|
||
|
|
continue;
|
||
|
|
|
||
|
|
if (ioctl(dev_fd, LOOP_GET_STATUS, &loopinfo) == 0) {
|
||
|
|
tst_resm(TINFO, "Device '%s' in use", buf);
|
||
|
|
} else {
|
||
|
|
if (errno != ENXIO)
|
||
|
|
continue;
|
||
|
|
tst_resm(TINFO, "Found free device '%s'", buf);
|
||
|
|
close(dev_fd);
|
||
|
|
if (path != NULL) {
|
||
|
|
strncpy(path, buf, path_len);
|
||
|
|
path[path_len-1] = '\0';
|
||
|
|
}
|
||
|
|
return i;
|
||
|
|
}
|
||
|
|
|
||
|
|
close(dev_fd);
|
||
|
|
}
|
||
|
|
|
||
|
|
tst_resm(TINFO, "No free devices found");
|
||
|
|
|
||
|
|
return -1;
|
||
|
|
}
|
||
|
|
|
||
|
|
int tst_attach_device(const char *dev, const char *file)
|
||
|
|
{
|
||
|
|
int dev_fd, file_fd;
|
||
|
|
struct loop_info loopinfo;
|
||
|
|
|
||
|
|
// b/148978487 retry to allow time for device creation
|
||
|
|
int attach_tries = 20;
|
||
|
|
while (attach_tries--) {
|
||
|
|
dev_fd = open(dev, O_RDWR);
|
||
|
|
if (dev_fd >= 0)
|
||
|
|
break;
|
||
|
|
usleep(50000);
|
||
|
|
}
|
||
|
|
if (dev_fd < 0) {
|
||
|
|
tst_resm(TWARN | TERRNO, "open('%s', O_RDWR) failed", dev);
|
||
|
|
return 1;
|
||
|
|
}
|
||
|
|
|
||
|
|
file_fd = open(file, O_RDWR);
|
||
|
|
if (file_fd < 0) {
|
||
|
|
tst_resm(TWARN | TERRNO, "open('%s', O_RDWR) failed", file);
|
||
|
|
close(dev_fd);
|
||
|
|
return 1;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (ioctl(dev_fd, LOOP_SET_FD, file_fd) < 0) {
|
||
|
|
close(dev_fd);
|
||
|
|
close(file_fd);
|
||
|
|
tst_resm(TWARN | TERRNO, "ioctl(%s, LOOP_SET_FD, %s) failed",
|
||
|
|
dev, file);
|
||
|
|
return 1;
|
||
|
|
}
|
||
|
|
|
||
|
|
/* Old mkfs.btrfs use LOOP_GET_STATUS instead of backing_file to get
|
||
|
|
* associated filename, so we need to set up the device by calling
|
||
|
|
* LOOP_SET_FD and LOOP_SET_STATUS.
|
||
|
|
*/
|
||
|
|
memset(&loopinfo, 0, sizeof(loopinfo));
|
||
|
|
strcpy(loopinfo.lo_name, file);
|
||
|
|
|
||
|
|
if (ioctl(dev_fd, LOOP_SET_STATUS, &loopinfo)) {
|
||
|
|
close(dev_fd);
|
||
|
|
close(file_fd);
|
||
|
|
tst_resm(TWARN | TERRNO,
|
||
|
|
"ioctl(%s, LOOP_SET_STATUS, %s) failed", dev, file);
|
||
|
|
return 1;
|
||
|
|
}
|
||
|
|
|
||
|
|
close(dev_fd);
|
||
|
|
close(file_fd);
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
uint64_t tst_get_device_size(const char *dev_path)
|
||
|
|
{
|
||
|
|
int fd;
|
||
|
|
uint64_t size;
|
||
|
|
struct stat st;
|
||
|
|
|
||
|
|
if (!dev_path)
|
||
|
|
tst_brkm(TBROK, NULL, "No block device path");
|
||
|
|
|
||
|
|
if (stat(dev_path, &st)) {
|
||
|
|
tst_resm(TWARN | TERRNO, "stat() failed");
|
||
|
|
return -1;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (!S_ISBLK(st.st_mode)) {
|
||
|
|
tst_resm(TWARN, "%s is not a block device", dev_path);
|
||
|
|
return -1;
|
||
|
|
}
|
||
|
|
|
||
|
|
fd = open(dev_path, O_RDONLY);
|
||
|
|
if (fd < 0) {
|
||
|
|
tst_resm(TWARN | TERRNO,
|
||
|
|
"open(%s, O_RDONLY) failed", dev_path);
|
||
|
|
return -1;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (ioctl(fd, BLKGETSIZE64, &size)) {
|
||
|
|
tst_resm(TWARN | TERRNO,
|
||
|
|
"ioctl(fd, BLKGETSIZE64, ...) failed");
|
||
|
|
close(fd);
|
||
|
|
return -1;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (close(fd)) {
|
||
|
|
tst_resm(TWARN | TERRNO,
|
||
|
|
"close(fd) failed");
|
||
|
|
return -1;
|
||
|
|
}
|
||
|
|
|
||
|
|
return size/1024/1024;
|
||
|
|
}
|
||
|
|
|
||
|
|
int tst_detach_device_by_fd(const char *dev, int dev_fd)
|
||
|
|
{
|
||
|
|
int ret, i;
|
||
|
|
|
||
|
|
/* 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))
|
||
|
|
return 0;
|
||
|
|
|
||
|
|
if (ret && (errno != EBUSY)) {
|
||
|
|
tst_resm(TWARN,
|
||
|
|
"ioctl(%s, LOOP_CLR_FD, 0) unexpectedly failed with: %s",
|
||
|
|
dev, tst_strerrno(errno));
|
||
|
|
return 1;
|
||
|
|
}
|
||
|
|
|
||
|
|
usleep(50000);
|
||
|
|
}
|
||
|
|
|
||
|
|
tst_resm(TWARN,
|
||
|
|
"ioctl(%s, LOOP_CLR_FD, 0) no ENXIO for too long", dev);
|
||
|
|
return 1;
|
||
|
|
}
|
||
|
|
|
||
|
|
int tst_detach_device(const char *dev)
|
||
|
|
{
|
||
|
|
int dev_fd, ret;
|
||
|
|
|
||
|
|
dev_fd = open(dev, O_RDONLY);
|
||
|
|
if (dev_fd < 0) {
|
||
|
|
tst_resm(TWARN | TERRNO, "open(%s) failed", dev);
|
||
|
|
return 1;
|
||
|
|
}
|
||
|
|
|
||
|
|
ret = tst_detach_device_by_fd(dev, dev_fd);
|
||
|
|
close(dev_fd);
|
||
|
|
return ret;
|
||
|
|
}
|
||
|
|
|
||
|
|
int tst_dev_sync(int fd)
|
||
|
|
{
|
||
|
|
return syscall(__NR_syncfs, fd);
|
||
|
|
}
|
||
|
|
|
||
|
|
const char *tst_acquire_loop_device(unsigned int size, const char *filename)
|
||
|
|
{
|
||
|
|
unsigned int acq_dev_size = size ? size : DEV_SIZE_MB;
|
||
|
|
|
||
|
|
if (tst_prealloc_file(filename, 1024 * 1024, acq_dev_size)) {
|
||
|
|
tst_resm(TWARN | TERRNO, "Failed to create %s", filename);
|
||
|
|
return NULL;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (tst_find_free_loopdev(dev_path, sizeof(dev_path)) == -1)
|
||
|
|
return NULL;
|
||
|
|
|
||
|
|
if (tst_attach_device(dev_path, filename))
|
||
|
|
return NULL;
|
||
|
|
|
||
|
|
return dev_path;
|
||
|
|
}
|
||
|
|
|
||
|
|
const char *tst_acquire_device__(unsigned int size)
|
||
|
|
{
|
||
|
|
const char *dev;
|
||
|
|
unsigned int acq_dev_size;
|
||
|
|
uint64_t ltp_dev_size;
|
||
|
|
|
||
|
|
acq_dev_size = size ? size : DEV_SIZE_MB;
|
||
|
|
|
||
|
|
dev = getenv("LTP_DEV");
|
||
|
|
|
||
|
|
if (dev) {
|
||
|
|
tst_resm(TINFO, "Using test device LTP_DEV='%s'", dev);
|
||
|
|
|
||
|
|
ltp_dev_size = tst_get_device_size(dev);
|
||
|
|
|
||
|
|
if (acq_dev_size <= ltp_dev_size)
|
||
|
|
return dev;
|
||
|
|
|
||
|
|
tst_resm(TINFO, "Skipping $LTP_DEV size %"PRIu64"MB, requested size %uMB",
|
||
|
|
ltp_dev_size, acq_dev_size);
|
||
|
|
}
|
||
|
|
|
||
|
|
dev = tst_acquire_loop_device(acq_dev_size, DEV_FILE);
|
||
|
|
|
||
|
|
if (dev)
|
||
|
|
device_acquired = 1;
|
||
|
|
|
||
|
|
return dev;
|
||
|
|
}
|
||
|
|
|
||
|
|
const char *tst_acquire_device_(void (cleanup_fn)(void), unsigned int size)
|
||
|
|
{
|
||
|
|
const char *device;
|
||
|
|
|
||
|
|
if (device_acquired) {
|
||
|
|
tst_brkm(TBROK, cleanup_fn, "Device already acquired");
|
||
|
|
return NULL;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (!tst_tmpdir_created()) {
|
||
|
|
tst_brkm(TBROK, cleanup_fn,
|
||
|
|
"Cannot acquire device without tmpdir() created");
|
||
|
|
return NULL;
|
||
|
|
}
|
||
|
|
|
||
|
|
device = tst_acquire_device__(size);
|
||
|
|
|
||
|
|
if (!device) {
|
||
|
|
tst_brkm(TBROK, cleanup_fn, "Failed to acquire device");
|
||
|
|
return NULL;
|
||
|
|
}
|
||
|
|
|
||
|
|
return device;
|
||
|
|
}
|
||
|
|
|
||
|
|
int tst_release_device(const char *dev)
|
||
|
|
{
|
||
|
|
int ret;
|
||
|
|
|
||
|
|
if (!device_acquired)
|
||
|
|
return 0;
|
||
|
|
|
||
|
|
/*
|
||
|
|
* Loop device was created -> we need to detach it.
|
||
|
|
*
|
||
|
|
* The file image is deleted in tst_rmdir();
|
||
|
|
*/
|
||
|
|
ret = tst_detach_device(dev);
|
||
|
|
|
||
|
|
device_acquired = 0;
|
||
|
|
|
||
|
|
return ret;
|
||
|
|
}
|
||
|
|
|
||
|
|
int tst_clear_device(const char *dev)
|
||
|
|
{
|
||
|
|
if (tst_fill_file(dev, 0, 1024, 512)) {
|
||
|
|
tst_resm(TWARN, "Failed to clear 512k block on %s", dev);
|
||
|
|
return 1;
|
||
|
|
}
|
||
|
|
|
||
|
|
return 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;
|
||
|
|
|
||
|
|
if (err != EBUSY) {
|
||
|
|
tst_resm(TWARN, "umount('%s') failed with %s",
|
||
|
|
path, tst_strerrno(err));
|
||
|
|
errno = err;
|
||
|
|
return ret;
|
||
|
|
}
|
||
|
|
|
||
|
|
tst_resm(TINFO, "umount('%s') failed with %s, try %2i...",
|
||
|
|
path, tst_strerrno(err), i+1);
|
||
|
|
|
||
|
|
if (i == 0) {
|
||
|
|
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;
|
||
|
|
}
|
||
|
|
|
||
|
|
int tst_is_mounted(const char *path)
|
||
|
|
{
|
||
|
|
char line[PATH_MAX];
|
||
|
|
FILE *file;
|
||
|
|
int ret = 0;
|
||
|
|
|
||
|
|
file = SAFE_FOPEN(NULL, "/proc/mounts", "r");
|
||
|
|
|
||
|
|
while (fgets(line, sizeof(line), file)) {
|
||
|
|
if (strstr(line, path) != NULL) {
|
||
|
|
ret = 1;
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
SAFE_FCLOSE(NULL, file);
|
||
|
|
|
||
|
|
if (!ret)
|
||
|
|
tst_resm(TINFO, "No device is mounted at %s", path);
|
||
|
|
|
||
|
|
return ret;
|
||
|
|
}
|
||
|
|
|
||
|
|
int tst_is_mounted_at_tmpdir(const char *path)
|
||
|
|
{
|
||
|
|
char cdir[PATH_MAX], mpath[PATH_MAX];
|
||
|
|
int ret;
|
||
|
|
|
||
|
|
if (!getcwd(cdir, PATH_MAX)) {
|
||
|
|
tst_resm(TWARN | TERRNO, "Failed to find current directory");
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
ret = snprintf(mpath, PATH_MAX, "%s/%s", cdir, path);
|
||
|
|
if (ret < 0 || ret >= PATH_MAX) {
|
||
|
|
tst_resm(TWARN | TERRNO,
|
||
|
|
"snprintf() should have returned %d instead of %d",
|
||
|
|
PATH_MAX, ret);
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
return tst_is_mounted(mpath);
|
||
|
|
}
|
||
|
|
|
||
|
|
static int find_stat_file(const char *dev, char *path, size_t path_len)
|
||
|
|
{
|
||
|
|
const char *devname = strrchr(dev, '/') + 1;
|
||
|
|
|
||
|
|
snprintf(path, path_len, "/sys/block/%s/stat", devname);
|
||
|
|
|
||
|
|
if (!access(path, F_OK))
|
||
|
|
return 1;
|
||
|
|
|
||
|
|
DIR *dir = SAFE_OPENDIR(NULL, "/sys/block/");
|
||
|
|
struct dirent *ent;
|
||
|
|
|
||
|
|
while ((ent = readdir(dir))) {
|
||
|
|
snprintf(path, path_len, "/sys/block/%s/%s/stat", ent->d_name, devname);
|
||
|
|
|
||
|
|
if (!access(path, F_OK)) {
|
||
|
|
SAFE_CLOSEDIR(NULL, dir);
|
||
|
|
return 1;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
SAFE_CLOSEDIR(NULL, dir);
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
unsigned long tst_dev_bytes_written(const char *dev)
|
||
|
|
{
|
||
|
|
unsigned long dev_sec_write = 0, dev_bytes_written, io_ticks = 0;
|
||
|
|
char dev_stat_path[PATH_MAX];
|
||
|
|
|
||
|
|
if (!find_stat_file(dev, dev_stat_path, sizeof(dev_stat_path)))
|
||
|
|
tst_brkm(TCONF, NULL, "Test device stat file: %s not found",
|
||
|
|
dev_stat_path);
|
||
|
|
|
||
|
|
SAFE_FILE_SCANF(NULL, dev_stat_path,
|
||
|
|
"%*s %*s %*s %*s %*s %*s %lu %*s %*s %lu",
|
||
|
|
&dev_sec_write, &io_ticks);
|
||
|
|
|
||
|
|
if (!io_ticks)
|
||
|
|
tst_brkm(TCONF, NULL, "Test device stat file: %s broken",
|
||
|
|
dev_stat_path);
|
||
|
|
|
||
|
|
dev_bytes_written = (dev_sec_write - prev_dev_sec_write) * 512;
|
||
|
|
|
||
|
|
prev_dev_sec_write = dev_sec_write;
|
||
|
|
|
||
|
|
return dev_bytes_written;
|
||
|
|
}
|
||
|
|
|
||
|
|
__attribute__((nonnull))
|
||
|
|
void tst_find_backing_dev(const char *path, char *dev, size_t dev_size)
|
||
|
|
{
|
||
|
|
struct stat buf;
|
||
|
|
struct btrfs_ioctl_fs_info_args args = {0};
|
||
|
|
struct dirent *d;
|
||
|
|
char uevent_path[PATH_MAX+PATH_MAX+10]; //10 is for the static uevent path
|
||
|
|
char dev_name[NAME_MAX];
|
||
|
|
char bdev_path[PATH_MAX];
|
||
|
|
char tmp_path[PATH_MAX];
|
||
|
|
char btrfs_uuid_str[UUID_STR_SZ];
|
||
|
|
DIR *dir;
|
||
|
|
unsigned int dev_major, dev_minor;
|
||
|
|
int fd;
|
||
|
|
|
||
|
|
if (stat(path, &buf) < 0)
|
||
|
|
tst_brkm(TWARN | TERRNO, NULL, "stat() failed");
|
||
|
|
|
||
|
|
strncpy(tmp_path, path, PATH_MAX-1);
|
||
|
|
tmp_path[PATH_MAX-1] = '\0';
|
||
|
|
if (S_ISREG(buf.st_mode))
|
||
|
|
dirname(tmp_path);
|
||
|
|
|
||
|
|
dev_major = major(buf.st_dev);
|
||
|
|
dev_minor = minor(buf.st_dev);
|
||
|
|
*dev = '\0';
|
||
|
|
|
||
|
|
if (dev_major == 0) {
|
||
|
|
tst_resm(TINFO, "Use BTRFS specific strategy");
|
||
|
|
|
||
|
|
fd = SAFE_OPEN(NULL, tmp_path, O_DIRECTORY);
|
||
|
|
if (!ioctl(fd, BTRFS_IOC_FS_INFO, &args)) {
|
||
|
|
sprintf(btrfs_uuid_str,
|
||
|
|
UUID_FMT,
|
||
|
|
args.fsid[0], args.fsid[1],
|
||
|
|
args.fsid[2], args.fsid[3],
|
||
|
|
args.fsid[4], args.fsid[5],
|
||
|
|
args.fsid[6], args.fsid[7],
|
||
|
|
args.fsid[8], args.fsid[9],
|
||
|
|
args.fsid[10], args.fsid[11],
|
||
|
|
args.fsid[12], args.fsid[13],
|
||
|
|
args.fsid[14], args.fsid[15]);
|
||
|
|
sprintf(bdev_path,
|
||
|
|
"/sys/fs/btrfs/%s/devices", btrfs_uuid_str);
|
||
|
|
} else {
|
||
|
|
if (errno == ENOTTY)
|
||
|
|
tst_brkm(TBROK | TERRNO, NULL, "BTRFS ioctl failed. Is %s on a tmpfs?", path);
|
||
|
|
|
||
|
|
tst_brkm(TBROK | TERRNO, NULL, "BTRFS ioctl on %s failed.", tmp_path);
|
||
|
|
}
|
||
|
|
SAFE_CLOSE(NULL, fd);
|
||
|
|
|
||
|
|
dir = SAFE_OPENDIR(NULL, bdev_path);
|
||
|
|
while ((d = SAFE_READDIR(NULL, dir))) {
|
||
|
|
if (d->d_name[0] != '.')
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
|
||
|
|
uevent_path[0] = '\0';
|
||
|
|
|
||
|
|
if (d) {
|
||
|
|
sprintf(uevent_path, "%s/%s/uevent",
|
||
|
|
bdev_path, d->d_name);
|
||
|
|
} else {
|
||
|
|
tst_brkm(TBROK | TERRNO, NULL, "No backing device found while looking in %s.", bdev_path);
|
||
|
|
}
|
||
|
|
|
||
|
|
if (SAFE_READDIR(NULL, dir))
|
||
|
|
tst_resm(TINFO, "Warning: used first of multiple backing device.");
|
||
|
|
|
||
|
|
SAFE_CLOSEDIR(NULL, dir);
|
||
|
|
} else {
|
||
|
|
tst_resm(TINFO, "Use uevent strategy");
|
||
|
|
sprintf(uevent_path,
|
||
|
|
"/sys/dev/block/%d:%d/uevent", dev_major, dev_minor);
|
||
|
|
}
|
||
|
|
|
||
|
|
if (!access(uevent_path, R_OK)) {
|
||
|
|
FILE_LINES_SCANF(NULL, uevent_path, "DEVNAME=%s", dev_name);
|
||
|
|
|
||
|
|
if (!dev_name[0] || set_dev_path(dev_name, dev, dev_size))
|
||
|
|
tst_brkm(TBROK, NULL, "Could not stat backing device %s", dev);
|
||
|
|
|
||
|
|
} else {
|
||
|
|
tst_brkm(TBROK, NULL, "uevent file (%s) access failed", uevent_path);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
void tst_stat_mount_dev(const char *const mnt_path, struct stat *const st)
|
||
|
|
{
|
||
|
|
struct mntent *mnt;
|
||
|
|
FILE *mntf = setmntent("/proc/self/mounts", "r");
|
||
|
|
|
||
|
|
if (!mntf) {
|
||
|
|
tst_brkm(TBROK | TERRNO, NULL, "Can't open /proc/self/mounts");
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
|
||
|
|
mnt = getmntent(mntf);
|
||
|
|
if (!mnt) {
|
||
|
|
tst_brkm(TBROK | TERRNO, NULL, "Can't read mounts or no mounts?");
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
|
||
|
|
do {
|
||
|
|
if (strcmp(mnt->mnt_dir, mnt_path)) {
|
||
|
|
mnt = getmntent(mntf);
|
||
|
|
continue;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (stat(mnt->mnt_fsname, st)) {
|
||
|
|
tst_brkm(TBROK | TERRNO, NULL,
|
||
|
|
"Can't stat '%s', mounted at '%s'",
|
||
|
|
mnt->mnt_fsname, mnt_path);
|
||
|
|
}
|
||
|
|
|
||
|
|
return;
|
||
|
|
} while (mnt);
|
||
|
|
|
||
|
|
tst_brkm(TBROK, NULL, "Could not find mount device");
|
||
|
|
}
|
||
|
|
|
||
|
|
int tst_dev_block_size(const char *path)
|
||
|
|
{
|
||
|
|
int fd;
|
||
|
|
int size;
|
||
|
|
char dev_name[PATH_MAX];
|
||
|
|
|
||
|
|
tst_find_backing_dev(path, dev_name, sizeof(dev_name));
|
||
|
|
|
||
|
|
fd = SAFE_OPEN(NULL, dev_name, O_RDONLY);
|
||
|
|
SAFE_IOCTL(NULL, fd, BLKSSZGET, &size);
|
||
|
|
SAFE_CLOSE(NULL, fd);
|
||
|
|
|
||
|
|
return size;
|
||
|
|
}
|