bcachefs-tools/tools-util.h

212 lines
4.2 KiB
C
Raw Normal View History

2017-01-08 12:13:18 +03:00
#ifndef _TOOLS_UTIL_H
#define _TOOLS_UTIL_H
2017-02-02 06:16:42 +03:00
#include <errno.h>
2017-01-08 12:13:18 +03:00
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
2017-03-01 13:45:15 +03:00
#include <sys/stat.h>
2017-01-08 12:13:18 +03:00
#include <sys/types.h>
2017-02-02 06:16:42 +03:00
#include <unistd.h>
2017-01-08 12:13:18 +03:00
2017-03-01 13:45:15 +03:00
#include <linux/bug.h>
2017-01-08 12:13:18 +03:00
#include <linux/byteorder.h>
#include <linux/kernel.h>
#include <linux/log2.h>
#include <linux/string.h>
#include <linux/types.h>
2017-03-01 13:45:15 +03:00
#include "ccan/darray/darray.h"
2017-01-08 12:13:18 +03:00
2017-03-01 13:45:15 +03:00
#define die(arg, ...) \
do { \
fprintf(stderr, arg "\n", ##__VA_ARGS__); \
exit(EXIT_FAILURE); \
2017-01-08 12:13:18 +03:00
} while (0)
2017-03-01 13:45:15 +03:00
#define mprintf(...) \
({ \
char *_str; \
2017-04-10 11:33:58 +03:00
int ret = asprintf(&_str, __VA_ARGS__); \
if (ret < 0) \
die("insufficient memory"); \
2017-03-01 13:45:15 +03:00
_str; \
})
2017-02-02 06:16:42 +03:00
static inline void *xcalloc(size_t count, size_t size)
{
void *p = calloc(count, size);
if (!p)
die("insufficient memory");
return p;
}
static inline void *xmalloc(size_t size)
{
void *p = malloc(size);
if (!p)
die("insufficient memory");
memset(p, 0, size);
return p;
}
static inline void xpread(int fd, void *buf, size_t count, off_t offset)
{
ssize_t r = pread(fd, buf, count, offset);
if (r != count)
die("read error (ret %zi)", r);
}
static inline void xpwrite(int fd, const void *buf, size_t count, off_t offset)
{
ssize_t r = pwrite(fd, buf, count, offset);
if (r != count)
2017-04-15 07:40:50 +03:00
die("write error (ret %zi err %m)", r);
2017-02-02 06:16:42 +03:00
}
2017-03-01 13:45:15 +03:00
#define xopenat(_dirfd, _path, ...) \
({ \
int _fd = openat((_dirfd), (_path), __VA_ARGS__); \
if (_fd < 0) \
2017-04-15 07:40:50 +03:00
die("Error opening %s: %m", (_path)); \
2017-03-01 13:45:15 +03:00
_fd; \
})
#define xopen(...) xopenat(AT_FDCWD, __VA_ARGS__)
static inline struct stat xfstatat(int dirfd, const char *path, int flags)
{
struct stat stat;
if (fstatat(dirfd, path, &stat, flags))
2017-04-15 07:40:50 +03:00
die("stat error: %m");
2017-03-01 13:45:15 +03:00
return stat;
}
static inline struct stat xfstat(int fd)
{
struct stat stat;
if (fstat(fd, &stat))
2017-04-15 07:40:50 +03:00
die("stat error: %m");
2017-03-01 13:45:15 +03:00
return stat;
}
#define xioctl(_fd, _nr, ...) \
do { \
if (ioctl((_fd), (_nr), ##__VA_ARGS__)) \
2017-04-15 07:40:50 +03:00
die(#_nr " ioctl error: %m"); \
2017-03-01 13:45:15 +03:00
} while (0)
2017-01-08 12:13:18 +03:00
enum units {
BYTES,
SECTORS,
HUMAN_READABLE,
};
2017-02-02 06:16:42 +03:00
struct units_buf __pr_units(u64, enum units);
2017-01-08 12:13:18 +03:00
struct units_buf {
char b[20];
};
2017-02-02 06:16:42 +03:00
#define pr_units(_v, _u) __pr_units(_v, _u).b
2017-01-08 12:13:18 +03:00
char *read_file_str(int, const char *);
u64 read_file_u64(int, const char *);
ssize_t read_string_list_or_die(const char *, const char * const[],
const char *);
u64 get_size(const char *, int);
unsigned get_blocksize(const char *, int);
2017-03-09 21:13:45 +03:00
int open_for_format(const char *, bool);
2017-01-08 12:13:18 +03:00
int bcachectl_open(void);
struct bcache_handle {
int ioctl_fd;
int sysfs_fd;
};
struct bcache_handle bcache_fs_open(const char *);
2017-02-02 06:16:42 +03:00
bool ask_yn(void);
2017-01-08 12:13:18 +03:00
2017-03-01 13:45:15 +03:00
struct range {
u64 start;
u64 end;
};
typedef darray(struct range) ranges;
static inline void range_add(ranges *data, u64 offset, u64 size)
{
darray_append(*data, (struct range) {
.start = offset,
.end = offset + size
});
}
void ranges_sort_merge(ranges *);
void ranges_roundup(ranges *, unsigned);
void ranges_rounddown(ranges *, unsigned);
struct hole_iter {
ranges r;
size_t idx;
u64 end;
};
static inline struct range hole_iter_next(struct hole_iter *iter)
{
struct range r = {
.start = iter->idx ? iter->r.item[iter->idx - 1].end : 0,
.end = iter->idx < iter->r.size
? iter->r.item[iter->idx].start : iter->end,
};
BUG_ON(r.start > r.end);
iter->idx++;
return r;
}
#define for_each_hole(_iter, _ranges, _end, _i) \
for (_iter = (struct hole_iter) { .r = _ranges, .end = _end }; \
(_iter.idx <= _iter.r.size && \
(_i = hole_iter_next(&_iter), true));)
#include <linux/fiemap.h>
struct fiemap_iter {
struct fiemap f;
struct fiemap_extent fe[1024];
unsigned idx;
int fd;
};
static inline void fiemap_iter_init(struct fiemap_iter *iter, int fd)
{
memset(iter, 0, sizeof(*iter));
iter->f.fm_extent_count = ARRAY_SIZE(iter->fe);
iter->f.fm_length = FIEMAP_MAX_OFFSET;
iter->fd = fd;
}
struct fiemap_extent fiemap_iter_next(struct fiemap_iter *);
#define fiemap_for_each(fd, iter, extent) \
for (fiemap_iter_init(&iter, fd); \
(extent = fiemap_iter_next(&iter)).fe_length;)
const char *strcmp_prefix(const char *, const char *);
2017-03-09 21:13:45 +03:00
unsigned hatoi_validate(const char *, const char *);
2017-01-08 12:13:18 +03:00
#endif /* _TOOLS_UTIL_H */