#define _GNU_SOURCE
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <sys/xattr.h>
#include <unistd.h>
#include "../../kselftest_harness.h"
#define TEST_XATTR_NAME "user.testattr"
#define TEST_XATTR_VALUE "testvalue"
#define TEST_XATTR_VALUE2 "newvalue"
#define SIMPLE_XATTR_MAX_NR 128
#define SIMPLE_XATTR_MAX_SIZE (128 << 10)
#ifndef XATTR_SIZE_MAX
#define XATTR_SIZE_MAX 65536
#endif
FIXTURE(xattr_sockfs)
{
int sockfd;
};
FIXTURE_SETUP(xattr_sockfs)
{
self->sockfd = socket(AF_UNIX, SOCK_STREAM, 0);
ASSERT_GE(self->sockfd, 0) {
TH_LOG("Failed to create socket: %s", strerror(errno));
}
}
FIXTURE_TEARDOWN(xattr_sockfs)
{
if (self->sockfd >= 0)
close(self->sockfd);
}
TEST_F(xattr_sockfs, set_get_user_xattr)
{
char buf[256];
ssize_t ret;
ret = fsetxattr(self->sockfd, TEST_XATTR_NAME,
TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0);
ASSERT_EQ(ret, 0) {
TH_LOG("fsetxattr failed: %s", strerror(errno));
}
memset(buf, 0, sizeof(buf));
ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, buf, sizeof(buf));
ASSERT_EQ(ret, (ssize_t)strlen(TEST_XATTR_VALUE)) {
TH_LOG("fgetxattr returned %zd: %s", ret, strerror(errno));
}
ASSERT_STREQ(buf, TEST_XATTR_VALUE);
}
TEST_F(xattr_sockfs, list_user_xattr)
{
char list[4096];
ssize_t ret;
char *ptr;
bool found_user = false;
bool found_proto = false;
ret = fsetxattr(self->sockfd, TEST_XATTR_NAME,
TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0);
ASSERT_EQ(ret, 0) {
TH_LOG("fsetxattr failed: %s", strerror(errno));
}
memset(list, 0, sizeof(list));
ret = flistxattr(self->sockfd, list, sizeof(list));
ASSERT_GT(ret, 0) {
TH_LOG("flistxattr failed: %s", strerror(errno));
}
for (ptr = list; ptr < list + ret; ptr += strlen(ptr) + 1) {
if (strcmp(ptr, TEST_XATTR_NAME) == 0)
found_user = true;
if (strcmp(ptr, "system.sockprotoname") == 0)
found_proto = true;
}
ASSERT_TRUE(found_user) {
TH_LOG("user xattr not found in list");
}
ASSERT_TRUE(found_proto) {
TH_LOG("system.sockprotoname not found in list");
}
}
TEST_F(xattr_sockfs, remove_user_xattr)
{
char buf[256];
ssize_t ret;
ret = fsetxattr(self->sockfd, TEST_XATTR_NAME,
TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0);
ASSERT_EQ(ret, 0);
ret = fremovexattr(self->sockfd, TEST_XATTR_NAME);
ASSERT_EQ(ret, 0) {
TH_LOG("fremovexattr failed: %s", strerror(errno));
}
ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, buf, sizeof(buf));
ASSERT_EQ(ret, -1);
ASSERT_EQ(errno, ENODATA);
}
TEST_F(xattr_sockfs, update_user_xattr)
{
char buf[256];
ssize_t ret;
ret = fsetxattr(self->sockfd, TEST_XATTR_NAME,
TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0);
ASSERT_EQ(ret, 0);
ret = fsetxattr(self->sockfd, TEST_XATTR_NAME,
TEST_XATTR_VALUE2, strlen(TEST_XATTR_VALUE2), 0);
ASSERT_EQ(ret, 0);
memset(buf, 0, sizeof(buf));
ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, buf, sizeof(buf));
ASSERT_EQ(ret, (ssize_t)strlen(TEST_XATTR_VALUE2));
ASSERT_STREQ(buf, TEST_XATTR_VALUE2);
}
TEST_F(xattr_sockfs, xattr_create_flag)
{
int ret;
ret = fsetxattr(self->sockfd, TEST_XATTR_NAME,
TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0);
ASSERT_EQ(ret, 0);
ret = fsetxattr(self->sockfd, TEST_XATTR_NAME,
TEST_XATTR_VALUE2, strlen(TEST_XATTR_VALUE2),
XATTR_CREATE);
ASSERT_EQ(ret, -1);
ASSERT_EQ(errno, EEXIST);
}
TEST_F(xattr_sockfs, xattr_replace_flag)
{
int ret;
ret = fsetxattr(self->sockfd, TEST_XATTR_NAME,
TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE),
XATTR_REPLACE);
ASSERT_EQ(ret, -1);
ASSERT_EQ(errno, ENODATA);
}
TEST_F(xattr_sockfs, get_nonexistent)
{
char buf[256];
ssize_t ret;
ret = fgetxattr(self->sockfd, "user.nonexistent", buf, sizeof(buf));
ASSERT_EQ(ret, -1);
ASSERT_EQ(errno, ENODATA);
}
TEST_F(xattr_sockfs, empty_value)
{
ssize_t ret;
ret = fsetxattr(self->sockfd, TEST_XATTR_NAME, "", 0, 0);
ASSERT_EQ(ret, 0);
ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, NULL, 0);
ASSERT_EQ(ret, 0);
}
TEST_F(xattr_sockfs, get_size)
{
ssize_t ret;
ret = fsetxattr(self->sockfd, TEST_XATTR_NAME,
TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0);
ASSERT_EQ(ret, 0);
ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, NULL, 0);
ASSERT_EQ(ret, (ssize_t)strlen(TEST_XATTR_VALUE));
}
TEST_F(xattr_sockfs, buffer_too_small)
{
char buf[2];
ssize_t ret;
ret = fsetxattr(self->sockfd, TEST_XATTR_NAME,
TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0);
ASSERT_EQ(ret, 0);
ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, buf, sizeof(buf));
ASSERT_EQ(ret, -1);
ASSERT_EQ(errno, ERANGE);
}
TEST_F(xattr_sockfs, max_nr_xattrs)
{
char name[32];
int i, ret;
for (i = 0; i < SIMPLE_XATTR_MAX_NR; i++) {
snprintf(name, sizeof(name), "user.test%03d", i);
ret = fsetxattr(self->sockfd, name, "v", 1, 0);
ASSERT_EQ(ret, 0) {
TH_LOG("fsetxattr %s failed at i=%d: %s",
name, i, strerror(errno));
}
}
ret = fsetxattr(self->sockfd, "user.overflow", "v", 1, 0);
ASSERT_EQ(ret, -1);
ASSERT_EQ(errno, ENOSPC) {
TH_LOG("Expected ENOSPC for xattr %d, got %s",
SIMPLE_XATTR_MAX_NR + 1, strerror(errno));
}
}
TEST_F(xattr_sockfs, max_xattr_size)
{
char *value;
int ret;
value = malloc(XATTR_SIZE_MAX);
ASSERT_NE(value, NULL);
memset(value, 'A', XATTR_SIZE_MAX);
ret = fsetxattr(self->sockfd, "user.big1", value, XATTR_SIZE_MAX, 0);
ASSERT_EQ(ret, 0) {
TH_LOG("first large xattr failed: %s", strerror(errno));
}
ret = fsetxattr(self->sockfd, "user.big2", value, XATTR_SIZE_MAX, 0);
free(value);
ASSERT_EQ(ret, 0) {
TH_LOG("second large xattr failed: %s", strerror(errno));
}
ret = fsetxattr(self->sockfd, "user.big3", "v", 1, 0);
ASSERT_EQ(ret, -1);
ASSERT_EQ(errno, ENOSPC) {
TH_LOG("Expected ENOSPC when exceeding size limit, got %s",
strerror(errno));
}
}
TEST_F(xattr_sockfs, limit_remove_readd)
{
char name[32];
int i, ret;
for (i = 0; i < SIMPLE_XATTR_MAX_NR; i++) {
snprintf(name, sizeof(name), "user.test%03d", i);
ret = fsetxattr(self->sockfd, name, "v", 1, 0);
ASSERT_EQ(ret, 0);
}
ret = fsetxattr(self->sockfd, "user.overflow", "v", 1, 0);
ASSERT_EQ(ret, -1);
ASSERT_EQ(errno, ENOSPC);
ret = fremovexattr(self->sockfd, "user.test000");
ASSERT_EQ(ret, 0);
ret = fsetxattr(self->sockfd, "user.newattr", "v", 1, 0);
ASSERT_EQ(ret, 0) {
TH_LOG("re-add after remove failed: %s", strerror(errno));
}
}
TEST_F(xattr_sockfs, limits_per_inode)
{
char buf[256];
int sock2;
ssize_t ret;
sock2 = socket(AF_UNIX, SOCK_STREAM, 0);
ASSERT_GE(sock2, 0);
ret = fsetxattr(self->sockfd, TEST_XATTR_NAME,
TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0);
ASSERT_EQ(ret, 0);
ret = fgetxattr(sock2, TEST_XATTR_NAME, NULL, 0);
ASSERT_EQ(ret, -1);
ASSERT_EQ(errno, ENODATA);
ret = fsetxattr(sock2, TEST_XATTR_NAME,
TEST_XATTR_VALUE2, strlen(TEST_XATTR_VALUE2), 0);
ASSERT_EQ(ret, 0);
memset(buf, 0, sizeof(buf));
ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, buf, sizeof(buf));
ASSERT_EQ(ret, (ssize_t)strlen(TEST_XATTR_VALUE));
ASSERT_STREQ(buf, TEST_XATTR_VALUE);
memset(buf, 0, sizeof(buf));
ret = fgetxattr(sock2, TEST_XATTR_NAME, buf, sizeof(buf));
ASSERT_EQ(ret, (ssize_t)strlen(TEST_XATTR_VALUE2));
ASSERT_STREQ(buf, TEST_XATTR_VALUE2);
close(sock2);
}
TEST_HARNESS_MAIN