2009-09-04 01:41:04 +00:00
|
|
|
#include <cstdlib>
|
|
|
|
#include <cstring>
|
|
|
|
#include <cstdio>
|
|
|
|
#include <cctype>
|
2009-05-05 01:04:47 +00:00
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include <sys/xattr.h>
|
|
|
|
|
2009-09-04 01:41:04 +00:00
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
#include <algorithm>
|
2009-05-05 01:04:47 +00:00
|
|
|
|
2009-09-04 20:23:20 +00:00
|
|
|
typedef std::vector<std::string>::iterator vsiter;
|
2009-09-04 01:41:04 +00:00
|
|
|
|
2009-09-04 01:49:55 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
// apple has additional parameter for position and options.
|
|
|
|
|
|
|
|
inline ssize_t getxattr(const char *path, const char *name, void *value, size_t size)
|
|
|
|
{
|
|
|
|
return getxattr(path, name, value, size, 0, XATTR_NOFOLLOW);
|
|
|
|
}
|
|
|
|
|
|
|
|
// apple has additional parameter for options.
|
|
|
|
inline ssize_t listxattr(const char *path, char *namebuf, size_t size)
|
|
|
|
{
|
|
|
|
return listxattr(path, namebuf, size, XATTR_NOFOLLOW);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2009-09-04 01:41:04 +00:00
|
|
|
void hexdump(const uint8_t *data, ssize_t size)
|
2009-05-05 01:04:47 +00:00
|
|
|
{
|
2009-09-04 01:41:04 +00:00
|
|
|
const char *HexMap = "0123456789abcdef";
|
2009-05-05 01:04:47 +00:00
|
|
|
|
2009-09-04 01:41:04 +00:00
|
|
|
char buffer1[16 * 3 + 1 + 1];
|
|
|
|
char buffer2[16 + 1];
|
|
|
|
ssize_t offset = 0;
|
|
|
|
unsigned i, j;
|
2009-05-05 01:04:47 +00:00
|
|
|
|
|
|
|
|
2009-09-04 01:41:04 +00:00
|
|
|
while(size > 0)
|
|
|
|
{
|
|
|
|
std::memset(buffer1, ' ', sizeof(buffer1));
|
|
|
|
std::memset(buffer2, ' ', sizeof(buffer2));
|
2009-05-05 01:04:47 +00:00
|
|
|
|
2009-09-04 20:23:20 +00:00
|
|
|
unsigned linelen = std::min((unsigned)size, (unsigned)16);
|
2009-09-04 01:41:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
for (i = 0, j = 0; i < linelen; i++)
|
|
|
|
{
|
|
|
|
unsigned x = data[i];
|
|
|
|
buffer1[j++] = HexMap[x >> 4];
|
|
|
|
buffer1[j++] = HexMap[x & 0x0f];
|
|
|
|
j++;
|
|
|
|
if (i == 7) j++;
|
2009-05-05 01:04:47 +00:00
|
|
|
|
2009-09-04 20:23:20 +00:00
|
|
|
// isascii not part of std:: and may be a macro.
|
|
|
|
buffer2[i] = isascii(x) && std::isprint(x) ? x : '.';
|
2009-05-05 01:04:47 +00:00
|
|
|
|
|
|
|
}
|
2009-09-04 01:41:04 +00:00
|
|
|
|
|
|
|
buffer1[sizeof(buffer1)-1] = 0;
|
|
|
|
buffer2[sizeof(buffer2)-1] = 0;
|
|
|
|
|
|
|
|
|
2009-09-04 20:23:20 +00:00
|
|
|
std::printf("%06x:\t%s\t%s\n", (unsigned)offset, buffer1, buffer2);
|
2009-09-04 01:41:04 +00:00
|
|
|
offset += 16;
|
|
|
|
data += 16;
|
|
|
|
size -= 16;
|
|
|
|
}
|
|
|
|
std::printf("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* build a list of attribute names for a file.
|
|
|
|
* returns -1 on failure
|
|
|
|
* otherwise, returns # of attributes.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
ssize_t get_attr_list(const char * fname, std::vector<std::string> &out)
|
|
|
|
{
|
2009-09-04 01:49:55 +00:00
|
|
|
ssize_t asize = listxattr(fname, NULL, 0);
|
2009-09-04 01:41:04 +00:00
|
|
|
if (asize < 0) return -1;
|
|
|
|
if (asize == 0) return 0;
|
|
|
|
|
|
|
|
char *buffer = new char[asize];
|
|
|
|
|
2009-09-04 01:49:55 +00:00
|
|
|
if ((asize = listxattr(fname, buffer, asize)) < 0)
|
2009-09-04 01:41:04 +00:00
|
|
|
{
|
|
|
|
delete []buffer;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
//hexdump((uint8_t *)buffer, asize);
|
|
|
|
|
|
|
|
char *cp = buffer;
|
|
|
|
// buffer is null-terminated utf-8 strings
|
|
|
|
// eg:
|
|
|
|
// c a t 0 d o g 0
|
|
|
|
// 0 1 2 3 4 5 6 7
|
|
|
|
while (asize > 0)
|
|
|
|
{
|
|
|
|
ssize_t len = std::strlen(cp);
|
|
|
|
|
|
|
|
out.push_back(std::string(cp, len));
|
|
|
|
++len;
|
|
|
|
asize -= len;
|
|
|
|
cp += len;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete []buffer;
|
2009-05-05 01:04:47 +00:00
|
|
|
|
2009-09-04 01:41:04 +00:00
|
|
|
return out.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* hexdump an individual attribute.
|
|
|
|
*/
|
|
|
|
void dumpxattr(const char *file, const char *attr)
|
2009-05-05 01:04:47 +00:00
|
|
|
{
|
|
|
|
ssize_t asize;
|
|
|
|
char *buffer;
|
|
|
|
|
2009-09-04 01:49:55 +00:00
|
|
|
asize = getxattr(file, attr, NULL, 0);
|
2009-09-04 01:41:04 +00:00
|
|
|
if (asize < 0)
|
|
|
|
{
|
|
|
|
std::perror(attr);
|
|
|
|
return;
|
|
|
|
}
|
2009-05-05 01:04:47 +00:00
|
|
|
|
2009-09-04 20:23:20 +00:00
|
|
|
std::printf("%s: %u\n", attr, (unsigned)asize);
|
2009-09-04 01:41:04 +00:00
|
|
|
|
|
|
|
buffer = new char[asize];
|
|
|
|
//if (!buffer) return;
|
|
|
|
|
2009-09-04 01:49:55 +00:00
|
|
|
asize = getxattr(file, attr, buffer, asize);
|
2009-09-04 01:41:04 +00:00
|
|
|
if (asize >= 0)
|
|
|
|
{
|
|
|
|
hexdump((uint8_t *)buffer, asize);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::perror(attr);
|
|
|
|
}
|
|
|
|
|
|
|
|
delete []buffer;
|
|
|
|
}
|
2009-05-05 01:04:47 +00:00
|
|
|
|
|
|
|
|
2009-09-04 01:41:04 +00:00
|
|
|
/*
|
|
|
|
* list a file's attributes (and size)
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
int list(int argc, char **argv)
|
|
|
|
{
|
|
|
|
|
|
|
|
const char *fname = *argv;
|
2009-05-05 01:04:47 +00:00
|
|
|
|
2009-09-04 01:41:04 +00:00
|
|
|
std::vector<std::string>attr;
|
2009-05-05 01:04:47 +00:00
|
|
|
|
2009-09-04 01:41:04 +00:00
|
|
|
if (argc == 1)
|
|
|
|
{
|
|
|
|
get_attr_list(fname, attr);
|
|
|
|
std::sort(attr.begin(), attr.end());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (int i = 1; i < argc; ++i)
|
2009-05-05 01:04:47 +00:00
|
|
|
{
|
2009-09-04 01:41:04 +00:00
|
|
|
attr.push_back(argv[i]);
|
2009-05-05 01:04:47 +00:00
|
|
|
}
|
|
|
|
}
|
2009-09-04 01:41:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
// find the max name length (not utf-8 happy)
|
2009-09-04 20:23:20 +00:00
|
|
|
unsigned maxname = 0;
|
|
|
|
for (vsiter i = attr.begin(); i != attr.end() ; ++i)
|
2009-09-04 01:41:04 +00:00
|
|
|
{
|
2009-09-04 20:23:20 +00:00
|
|
|
maxname = std::max(maxname, (unsigned)i->length());
|
2009-09-04 01:41:04 +00:00
|
|
|
}
|
|
|
|
maxname += 2;
|
2009-05-05 01:04:47 +00:00
|
|
|
|
2009-09-04 20:23:20 +00:00
|
|
|
for (vsiter i = attr.begin(); i != attr.end() ; ++i)
|
2009-09-04 01:41:04 +00:00
|
|
|
{
|
|
|
|
const char *attr_name = i->c_str();
|
|
|
|
|
2009-09-04 01:49:55 +00:00
|
|
|
ssize_t asize = getxattr(fname, attr_name, NULL, 0);
|
2009-09-04 01:41:04 +00:00
|
|
|
if (asize >= 0)
|
|
|
|
{
|
2009-09-04 20:23:20 +00:00
|
|
|
std::printf("%-*s %8u\n", maxname, attr_name, (unsigned)asize);
|
2009-09-04 01:41:04 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::perror(attr_name);
|
|
|
|
}
|
|
|
|
}
|
2009-05-05 01:04:47 +00:00
|
|
|
|
2009-09-04 01:41:04 +00:00
|
|
|
return 0;
|
2009-05-05 01:04:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-09-04 01:41:04 +00:00
|
|
|
/*
|
|
|
|
* hexdump a file's attributes.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
int dump(int argc, char **argv)
|
|
|
|
{
|
2009-05-05 01:04:47 +00:00
|
|
|
|
2009-09-04 01:41:04 +00:00
|
|
|
const char *fname = *argv;
|
|
|
|
|
|
|
|
std::vector<std::string>attr;
|
|
|
|
|
|
|
|
if (argc == 1)
|
|
|
|
{
|
|
|
|
get_attr_list(fname, attr);
|
|
|
|
std::sort(attr.begin(), attr.end());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (int i = 1; i < argc; ++i)
|
|
|
|
{
|
|
|
|
attr.push_back(argv[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-09-04 20:23:20 +00:00
|
|
|
for (vsiter i = attr.begin(); i != attr.end() ; ++i)
|
2009-05-05 01:04:47 +00:00
|
|
|
{
|
2009-09-04 20:23:20 +00:00
|
|
|
const char *attr_name = i->c_str();
|
2009-09-04 01:41:04 +00:00
|
|
|
|
|
|
|
dumpxattr(fname, attr_name);
|
2009-05-05 01:04:47 +00:00
|
|
|
}
|
2009-09-04 01:41:04 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-05-05 01:04:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-09-04 01:41:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void usage(const char *name)
|
|
|
|
{
|
|
|
|
std::printf("usage:\n");
|
|
|
|
std::printf("%s list file [attr ...]\n", name);
|
|
|
|
std::printf("%s dump file [attr ...]\n", name);
|
|
|
|
std::exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* xattr list file [xattr ...]
|
|
|
|
* xattr dump file [xattr ...]
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
if (argc < 3) usage(*argv);
|
|
|
|
|
|
|
|
if (std::strcmp(argv[1], "list") == 0) return list(argc - 2, argv + 2);
|
|
|
|
if (std::strcmp(argv[1], "dump") == 0) return dump(argc - 2, argv + 2);
|
|
|
|
|
|
|
|
usage(*argv);
|
|
|
|
|
|
|
|
return 1;
|
2009-05-05 01:04:47 +00:00
|
|
|
}
|