blob: 890cb302b5c77cdb232e44351e2816af8d9635aa [file] [log] [blame]
/* vi: set sw=4 ts=4: */
/*
* See README for additional information
*
* This file can be redistributed under the terms of the GNU Library General
* Public License
*/
#include "libbb.h"
#include "e2fs_lib.h"
#define HAVE_EXT2_IOCTLS 1
#if INT_MAX == LONG_MAX
#define IF_LONG_IS_SAME(...) __VA_ARGS__
#define IF_LONG_IS_WIDER(...)
#else
#define IF_LONG_IS_SAME(...)
#define IF_LONG_IS_WIDER(...) __VA_ARGS__
#endif
static void close_silently(int fd)
{
int e = errno;
close(fd);
errno = e;
}
/* Iterate a function on each entry of a directory */
int iterate_on_dir(const char *dir_name,
int (*func)(const char *, struct dirent *, void *),
void * private)
{
DIR *dir;
struct dirent *de, *dep;
int max_len, len;
max_len = PATH_MAX + sizeof(struct dirent);
de = xmalloc(max_len+1);
memset(de, 0, max_len+1);
dir = opendir(dir_name);
if (dir == NULL) {
free(de);
return -1;
}
while ((dep = readdir(dir))) {
len = sizeof(struct dirent);
if (len < dep->d_reclen)
len = dep->d_reclen;
if (len > max_len)
len = max_len;
memcpy(de, dep, len);
func(dir_name, de, private);
}
closedir(dir);
free(de);
return 0;
}
/* Get/set a file version on an ext2 file system */
int fgetsetversion(const char *name, unsigned long *get_version, unsigned long set_version)
{
#if HAVE_EXT2_IOCTLS
int fd, r;
IF_LONG_IS_WIDER(int ver;)
fd = open(name, O_NONBLOCK);
if (fd == -1)
return -1;
if (!get_version) {
IF_LONG_IS_WIDER(
ver = (int) set_version;
r = ioctl(fd, EXT2_IOC_SETVERSION, &ver);
)
IF_LONG_IS_SAME(
r = ioctl(fd, EXT2_IOC_SETVERSION, (void*)&set_version);
)
} else {
IF_LONG_IS_WIDER(
r = ioctl(fd, EXT2_IOC_GETVERSION, &ver);
*get_version = ver;
)
IF_LONG_IS_SAME(
r = ioctl(fd, EXT2_IOC_GETVERSION, (void*)get_version);
)
}
close_silently(fd);
return r;
#else /* ! HAVE_EXT2_IOCTLS */
errno = EOPNOTSUPP;
return -1;
#endif /* ! HAVE_EXT2_IOCTLS */
}
/* Get/set a file flags on an ext2 file system */
int fgetsetflags(const char *name, unsigned long *get_flags, unsigned long set_flags)
{
#if HAVE_EXT2_IOCTLS
struct stat buf;
int fd, r;
IF_LONG_IS_WIDER(int f;)
if (stat(name, &buf) == 0 /* stat is ok */
&& !S_ISREG(buf.st_mode) && !S_ISDIR(buf.st_mode)
) {
goto notsupp;
}
fd = open(name, O_NONBLOCK); /* neither read nor write asked for */
if (fd == -1)
return -1;
if (!get_flags) {
IF_LONG_IS_WIDER(
f = (int) set_flags;
r = ioctl(fd, EXT2_IOC_SETFLAGS, &f);
)
IF_LONG_IS_SAME(
r = ioctl(fd, EXT2_IOC_SETFLAGS, (void*)&set_flags);
)
} else {
IF_LONG_IS_WIDER(
r = ioctl(fd, EXT2_IOC_GETFLAGS, &f);
*get_flags = f;
)
IF_LONG_IS_SAME(
r = ioctl(fd, EXT2_IOC_GETFLAGS, (void*)get_flags);
)
}
close_silently(fd);
return r;
notsupp:
#endif /* HAVE_EXT2_IOCTLS */
errno = EOPNOTSUPP;
return -1;
}
/* Print file attributes on an ext2 file system */
struct flags_name {
unsigned long flag;
char short_name;
const char *long_name;
};
/* TODO: apart from I and (disabled) COMPRESSION flags, this
* is a duplicate of a table from chattr. Merge? */
static const struct flags_name flags_array[] = {
{ EXT2_SECRM_FL, 's', "Secure_Deletion" },
{ EXT2_UNRM_FL, 'u' , "Undelete" },
{ EXT2_SYNC_FL, 'S', "Synchronous_Updates" },
{ EXT2_DIRSYNC_FL, 'D', "Synchronous_Directory_Updates" },
{ EXT2_IMMUTABLE_FL, 'i', "Immutable" },
{ EXT2_APPEND_FL, 'a', "Append_Only" },
{ EXT2_NODUMP_FL, 'd', "No_Dump" },
{ EXT2_NOATIME_FL, 'A', "No_Atime" },
{ EXT2_COMPR_FL, 'c', "Compression_Requested" },
#ifdef ENABLE_COMPRESSION
{ EXT2_COMPRBLK_FL, 'B', "Compressed_File" },
{ EXT2_DIRTY_FL, 'Z', "Compressed_Dirty_File" },
{ EXT2_NOCOMPR_FL, 'X', "Compression_Raw_Access" },
{ EXT2_ECOMPR_FL, 'E', "Compression_Error" },
#endif
{ EXT3_JOURNAL_DATA_FL, 'j', "Journaled_Data" },
{ EXT2_INDEX_FL, 'I', "Indexed_directory" },
{ EXT2_NOTAIL_FL, 't', "No_Tailmerging" },
{ EXT2_TOPDIR_FL, 'T', "Top_of_Directory_Hierarchies" },
{ 0, '\0', NULL }
};
void print_flags(FILE *f, unsigned long flags, unsigned options)
{
const struct flags_name *fp;
if (options & PFOPT_LONG) {
int first = 1;
for (fp = flags_array; fp->short_name; fp++) {
if (flags & fp->flag) {
if (!first)
fputs(", ", f);
fputs(fp->long_name, f);
first = 0;
}
}
if (first)
fputs("---", f);
} else {
for (fp = flags_array; fp->short_name; fp++) {
char c = '-';
if (flags & fp->flag)
c = fp->short_name;
fputc(c, f);
}
}
}