1*0e75aea0SChristian Brauner // SPDX-License-Identifier: GPL-2.0 2*0e75aea0SChristian Brauner // Copyright (c) 2026 Christian Brauner <brauner@kernel.org> 3*0e75aea0SChristian Brauner /* 4*0e75aea0SChristian Brauner * Test extended attributes on sockfs sockets. 5*0e75aea0SChristian Brauner * 6*0e75aea0SChristian Brauner * Sockets created via socket() have their inodes in sockfs, which supports 7*0e75aea0SChristian Brauner * user.* xattrs with per-inode limits: up to 128 xattrs and 128KB total 8*0e75aea0SChristian Brauner * value size. These tests verify xattr operations via fsetxattr/fgetxattr/ 9*0e75aea0SChristian Brauner * flistxattr/fremovexattr on the socket fd, as well as limit enforcement. 10*0e75aea0SChristian Brauner */ 11*0e75aea0SChristian Brauner 12*0e75aea0SChristian Brauner #define _GNU_SOURCE 13*0e75aea0SChristian Brauner #include <errno.h> 14*0e75aea0SChristian Brauner #include <stdio.h> 15*0e75aea0SChristian Brauner #include <stdlib.h> 16*0e75aea0SChristian Brauner #include <string.h> 17*0e75aea0SChristian Brauner #include <sys/socket.h> 18*0e75aea0SChristian Brauner #include <sys/types.h> 19*0e75aea0SChristian Brauner #include <sys/xattr.h> 20*0e75aea0SChristian Brauner #include <unistd.h> 21*0e75aea0SChristian Brauner 22*0e75aea0SChristian Brauner #include "../../kselftest_harness.h" 23*0e75aea0SChristian Brauner 24*0e75aea0SChristian Brauner #define TEST_XATTR_NAME "user.testattr" 25*0e75aea0SChristian Brauner #define TEST_XATTR_VALUE "testvalue" 26*0e75aea0SChristian Brauner #define TEST_XATTR_VALUE2 "newvalue" 27*0e75aea0SChristian Brauner 28*0e75aea0SChristian Brauner /* Per-inode limits for user.* xattrs on sockfs (from include/linux/xattr.h) */ 29*0e75aea0SChristian Brauner #define SIMPLE_XATTR_MAX_NR 128 30*0e75aea0SChristian Brauner #define SIMPLE_XATTR_MAX_SIZE (128 << 10) /* 128 KB */ 31*0e75aea0SChristian Brauner 32*0e75aea0SChristian Brauner #ifndef XATTR_SIZE_MAX 33*0e75aea0SChristian Brauner #define XATTR_SIZE_MAX 65536 34*0e75aea0SChristian Brauner #endif 35*0e75aea0SChristian Brauner 36*0e75aea0SChristian Brauner /* 37*0e75aea0SChristian Brauner * Fixture for sockfs socket xattr tests. 38*0e75aea0SChristian Brauner * Creates an AF_UNIX socket (lives in sockfs, not bound to any path). 39*0e75aea0SChristian Brauner */ 40*0e75aea0SChristian Brauner FIXTURE(xattr_sockfs) 41*0e75aea0SChristian Brauner { 42*0e75aea0SChristian Brauner int sockfd; 43*0e75aea0SChristian Brauner }; 44*0e75aea0SChristian Brauner 45*0e75aea0SChristian Brauner FIXTURE_SETUP(xattr_sockfs) 46*0e75aea0SChristian Brauner { 47*0e75aea0SChristian Brauner self->sockfd = socket(AF_UNIX, SOCK_STREAM, 0); 48*0e75aea0SChristian Brauner ASSERT_GE(self->sockfd, 0) { 49*0e75aea0SChristian Brauner TH_LOG("Failed to create socket: %s", strerror(errno)); 50*0e75aea0SChristian Brauner } 51*0e75aea0SChristian Brauner } 52*0e75aea0SChristian Brauner 53*0e75aea0SChristian Brauner FIXTURE_TEARDOWN(xattr_sockfs) 54*0e75aea0SChristian Brauner { 55*0e75aea0SChristian Brauner if (self->sockfd >= 0) 56*0e75aea0SChristian Brauner close(self->sockfd); 57*0e75aea0SChristian Brauner } 58*0e75aea0SChristian Brauner 59*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, set_get_user_xattr) 60*0e75aea0SChristian Brauner { 61*0e75aea0SChristian Brauner char buf[256]; 62*0e75aea0SChristian Brauner ssize_t ret; 63*0e75aea0SChristian Brauner 64*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, TEST_XATTR_NAME, 65*0e75aea0SChristian Brauner TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0); 66*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0) { 67*0e75aea0SChristian Brauner TH_LOG("fsetxattr failed: %s", strerror(errno)); 68*0e75aea0SChristian Brauner } 69*0e75aea0SChristian Brauner 70*0e75aea0SChristian Brauner memset(buf, 0, sizeof(buf)); 71*0e75aea0SChristian Brauner ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, buf, sizeof(buf)); 72*0e75aea0SChristian Brauner ASSERT_EQ(ret, (ssize_t)strlen(TEST_XATTR_VALUE)) { 73*0e75aea0SChristian Brauner TH_LOG("fgetxattr returned %zd: %s", ret, strerror(errno)); 74*0e75aea0SChristian Brauner } 75*0e75aea0SChristian Brauner ASSERT_STREQ(buf, TEST_XATTR_VALUE); 76*0e75aea0SChristian Brauner } 77*0e75aea0SChristian Brauner 78*0e75aea0SChristian Brauner /* 79*0e75aea0SChristian Brauner * Test listing xattrs on a sockfs socket. 80*0e75aea0SChristian Brauner * Should include user.* xattrs and system.sockprotoname. 81*0e75aea0SChristian Brauner */ 82*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, list_user_xattr) 83*0e75aea0SChristian Brauner { 84*0e75aea0SChristian Brauner char list[4096]; 85*0e75aea0SChristian Brauner ssize_t ret; 86*0e75aea0SChristian Brauner char *ptr; 87*0e75aea0SChristian Brauner bool found_user = false; 88*0e75aea0SChristian Brauner bool found_proto = false; 89*0e75aea0SChristian Brauner 90*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, TEST_XATTR_NAME, 91*0e75aea0SChristian Brauner TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0); 92*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0) { 93*0e75aea0SChristian Brauner TH_LOG("fsetxattr failed: %s", strerror(errno)); 94*0e75aea0SChristian Brauner } 95*0e75aea0SChristian Brauner 96*0e75aea0SChristian Brauner memset(list, 0, sizeof(list)); 97*0e75aea0SChristian Brauner ret = flistxattr(self->sockfd, list, sizeof(list)); 98*0e75aea0SChristian Brauner ASSERT_GT(ret, 0) { 99*0e75aea0SChristian Brauner TH_LOG("flistxattr failed: %s", strerror(errno)); 100*0e75aea0SChristian Brauner } 101*0e75aea0SChristian Brauner 102*0e75aea0SChristian Brauner for (ptr = list; ptr < list + ret; ptr += strlen(ptr) + 1) { 103*0e75aea0SChristian Brauner if (strcmp(ptr, TEST_XATTR_NAME) == 0) 104*0e75aea0SChristian Brauner found_user = true; 105*0e75aea0SChristian Brauner if (strcmp(ptr, "system.sockprotoname") == 0) 106*0e75aea0SChristian Brauner found_proto = true; 107*0e75aea0SChristian Brauner } 108*0e75aea0SChristian Brauner ASSERT_TRUE(found_user) { 109*0e75aea0SChristian Brauner TH_LOG("user xattr not found in list"); 110*0e75aea0SChristian Brauner } 111*0e75aea0SChristian Brauner ASSERT_TRUE(found_proto) { 112*0e75aea0SChristian Brauner TH_LOG("system.sockprotoname not found in list"); 113*0e75aea0SChristian Brauner } 114*0e75aea0SChristian Brauner } 115*0e75aea0SChristian Brauner 116*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, remove_user_xattr) 117*0e75aea0SChristian Brauner { 118*0e75aea0SChristian Brauner char buf[256]; 119*0e75aea0SChristian Brauner ssize_t ret; 120*0e75aea0SChristian Brauner 121*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, TEST_XATTR_NAME, 122*0e75aea0SChristian Brauner TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0); 123*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0); 124*0e75aea0SChristian Brauner 125*0e75aea0SChristian Brauner ret = fremovexattr(self->sockfd, TEST_XATTR_NAME); 126*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0) { 127*0e75aea0SChristian Brauner TH_LOG("fremovexattr failed: %s", strerror(errno)); 128*0e75aea0SChristian Brauner } 129*0e75aea0SChristian Brauner 130*0e75aea0SChristian Brauner ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, buf, sizeof(buf)); 131*0e75aea0SChristian Brauner ASSERT_EQ(ret, -1); 132*0e75aea0SChristian Brauner ASSERT_EQ(errno, ENODATA); 133*0e75aea0SChristian Brauner } 134*0e75aea0SChristian Brauner 135*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, update_user_xattr) 136*0e75aea0SChristian Brauner { 137*0e75aea0SChristian Brauner char buf[256]; 138*0e75aea0SChristian Brauner ssize_t ret; 139*0e75aea0SChristian Brauner 140*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, TEST_XATTR_NAME, 141*0e75aea0SChristian Brauner TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0); 142*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0); 143*0e75aea0SChristian Brauner 144*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, TEST_XATTR_NAME, 145*0e75aea0SChristian Brauner TEST_XATTR_VALUE2, strlen(TEST_XATTR_VALUE2), 0); 146*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0); 147*0e75aea0SChristian Brauner 148*0e75aea0SChristian Brauner memset(buf, 0, sizeof(buf)); 149*0e75aea0SChristian Brauner ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, buf, sizeof(buf)); 150*0e75aea0SChristian Brauner ASSERT_EQ(ret, (ssize_t)strlen(TEST_XATTR_VALUE2)); 151*0e75aea0SChristian Brauner ASSERT_STREQ(buf, TEST_XATTR_VALUE2); 152*0e75aea0SChristian Brauner } 153*0e75aea0SChristian Brauner 154*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, xattr_create_flag) 155*0e75aea0SChristian Brauner { 156*0e75aea0SChristian Brauner int ret; 157*0e75aea0SChristian Brauner 158*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, TEST_XATTR_NAME, 159*0e75aea0SChristian Brauner TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0); 160*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0); 161*0e75aea0SChristian Brauner 162*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, TEST_XATTR_NAME, 163*0e75aea0SChristian Brauner TEST_XATTR_VALUE2, strlen(TEST_XATTR_VALUE2), 164*0e75aea0SChristian Brauner XATTR_CREATE); 165*0e75aea0SChristian Brauner ASSERT_EQ(ret, -1); 166*0e75aea0SChristian Brauner ASSERT_EQ(errno, EEXIST); 167*0e75aea0SChristian Brauner } 168*0e75aea0SChristian Brauner 169*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, xattr_replace_flag) 170*0e75aea0SChristian Brauner { 171*0e75aea0SChristian Brauner int ret; 172*0e75aea0SChristian Brauner 173*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, TEST_XATTR_NAME, 174*0e75aea0SChristian Brauner TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 175*0e75aea0SChristian Brauner XATTR_REPLACE); 176*0e75aea0SChristian Brauner ASSERT_EQ(ret, -1); 177*0e75aea0SChristian Brauner ASSERT_EQ(errno, ENODATA); 178*0e75aea0SChristian Brauner } 179*0e75aea0SChristian Brauner 180*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, get_nonexistent) 181*0e75aea0SChristian Brauner { 182*0e75aea0SChristian Brauner char buf[256]; 183*0e75aea0SChristian Brauner ssize_t ret; 184*0e75aea0SChristian Brauner 185*0e75aea0SChristian Brauner ret = fgetxattr(self->sockfd, "user.nonexistent", buf, sizeof(buf)); 186*0e75aea0SChristian Brauner ASSERT_EQ(ret, -1); 187*0e75aea0SChristian Brauner ASSERT_EQ(errno, ENODATA); 188*0e75aea0SChristian Brauner } 189*0e75aea0SChristian Brauner 190*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, empty_value) 191*0e75aea0SChristian Brauner { 192*0e75aea0SChristian Brauner ssize_t ret; 193*0e75aea0SChristian Brauner 194*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, TEST_XATTR_NAME, "", 0, 0); 195*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0); 196*0e75aea0SChristian Brauner 197*0e75aea0SChristian Brauner ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, NULL, 0); 198*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0); 199*0e75aea0SChristian Brauner } 200*0e75aea0SChristian Brauner 201*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, get_size) 202*0e75aea0SChristian Brauner { 203*0e75aea0SChristian Brauner ssize_t ret; 204*0e75aea0SChristian Brauner 205*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, TEST_XATTR_NAME, 206*0e75aea0SChristian Brauner TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0); 207*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0); 208*0e75aea0SChristian Brauner 209*0e75aea0SChristian Brauner ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, NULL, 0); 210*0e75aea0SChristian Brauner ASSERT_EQ(ret, (ssize_t)strlen(TEST_XATTR_VALUE)); 211*0e75aea0SChristian Brauner } 212*0e75aea0SChristian Brauner 213*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, buffer_too_small) 214*0e75aea0SChristian Brauner { 215*0e75aea0SChristian Brauner char buf[2]; 216*0e75aea0SChristian Brauner ssize_t ret; 217*0e75aea0SChristian Brauner 218*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, TEST_XATTR_NAME, 219*0e75aea0SChristian Brauner TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0); 220*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0); 221*0e75aea0SChristian Brauner 222*0e75aea0SChristian Brauner ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, buf, sizeof(buf)); 223*0e75aea0SChristian Brauner ASSERT_EQ(ret, -1); 224*0e75aea0SChristian Brauner ASSERT_EQ(errno, ERANGE); 225*0e75aea0SChristian Brauner } 226*0e75aea0SChristian Brauner 227*0e75aea0SChristian Brauner /* 228*0e75aea0SChristian Brauner * Test maximum number of user.* xattrs per socket. 229*0e75aea0SChristian Brauner * The kernel enforces SIMPLE_XATTR_MAX_NR (128), so the 129th should 230*0e75aea0SChristian Brauner * fail with ENOSPC. 231*0e75aea0SChristian Brauner */ 232*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, max_nr_xattrs) 233*0e75aea0SChristian Brauner { 234*0e75aea0SChristian Brauner char name[32]; 235*0e75aea0SChristian Brauner int i, ret; 236*0e75aea0SChristian Brauner 237*0e75aea0SChristian Brauner for (i = 0; i < SIMPLE_XATTR_MAX_NR; i++) { 238*0e75aea0SChristian Brauner snprintf(name, sizeof(name), "user.test%03d", i); 239*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, name, "v", 1, 0); 240*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0) { 241*0e75aea0SChristian Brauner TH_LOG("fsetxattr %s failed at i=%d: %s", 242*0e75aea0SChristian Brauner name, i, strerror(errno)); 243*0e75aea0SChristian Brauner } 244*0e75aea0SChristian Brauner } 245*0e75aea0SChristian Brauner 246*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, "user.overflow", "v", 1, 0); 247*0e75aea0SChristian Brauner ASSERT_EQ(ret, -1); 248*0e75aea0SChristian Brauner ASSERT_EQ(errno, ENOSPC) { 249*0e75aea0SChristian Brauner TH_LOG("Expected ENOSPC for xattr %d, got %s", 250*0e75aea0SChristian Brauner SIMPLE_XATTR_MAX_NR + 1, strerror(errno)); 251*0e75aea0SChristian Brauner } 252*0e75aea0SChristian Brauner } 253*0e75aea0SChristian Brauner 254*0e75aea0SChristian Brauner /* 255*0e75aea0SChristian Brauner * Test maximum total value size for user.* xattrs. 256*0e75aea0SChristian Brauner * The kernel enforces SIMPLE_XATTR_MAX_SIZE (128KB). Individual xattr 257*0e75aea0SChristian Brauner * values are limited to XATTR_SIZE_MAX (64KB) by the VFS, so we need 258*0e75aea0SChristian Brauner * at least two xattrs to hit the total limit. 259*0e75aea0SChristian Brauner */ 260*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, max_xattr_size) 261*0e75aea0SChristian Brauner { 262*0e75aea0SChristian Brauner char *value; 263*0e75aea0SChristian Brauner int ret; 264*0e75aea0SChristian Brauner 265*0e75aea0SChristian Brauner value = malloc(XATTR_SIZE_MAX); 266*0e75aea0SChristian Brauner ASSERT_NE(value, NULL); 267*0e75aea0SChristian Brauner memset(value, 'A', XATTR_SIZE_MAX); 268*0e75aea0SChristian Brauner 269*0e75aea0SChristian Brauner /* First 64KB xattr - total = 64KB */ 270*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, "user.big1", value, XATTR_SIZE_MAX, 0); 271*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0) { 272*0e75aea0SChristian Brauner TH_LOG("first large xattr failed: %s", strerror(errno)); 273*0e75aea0SChristian Brauner } 274*0e75aea0SChristian Brauner 275*0e75aea0SChristian Brauner /* Second 64KB xattr - total = 128KB (exactly at limit) */ 276*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, "user.big2", value, XATTR_SIZE_MAX, 0); 277*0e75aea0SChristian Brauner free(value); 278*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0) { 279*0e75aea0SChristian Brauner TH_LOG("second large xattr failed: %s", strerror(errno)); 280*0e75aea0SChristian Brauner } 281*0e75aea0SChristian Brauner 282*0e75aea0SChristian Brauner /* Third xattr with 1 byte - total > 128KB, should fail */ 283*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, "user.big3", "v", 1, 0); 284*0e75aea0SChristian Brauner ASSERT_EQ(ret, -1); 285*0e75aea0SChristian Brauner ASSERT_EQ(errno, ENOSPC) { 286*0e75aea0SChristian Brauner TH_LOG("Expected ENOSPC when exceeding size limit, got %s", 287*0e75aea0SChristian Brauner strerror(errno)); 288*0e75aea0SChristian Brauner } 289*0e75aea0SChristian Brauner } 290*0e75aea0SChristian Brauner 291*0e75aea0SChristian Brauner /* 292*0e75aea0SChristian Brauner * Test that removing an xattr frees limit space, allowing re-addition. 293*0e75aea0SChristian Brauner */ 294*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, limit_remove_readd) 295*0e75aea0SChristian Brauner { 296*0e75aea0SChristian Brauner char name[32]; 297*0e75aea0SChristian Brauner int i, ret; 298*0e75aea0SChristian Brauner 299*0e75aea0SChristian Brauner /* Fill up to the maximum count */ 300*0e75aea0SChristian Brauner for (i = 0; i < SIMPLE_XATTR_MAX_NR; i++) { 301*0e75aea0SChristian Brauner snprintf(name, sizeof(name), "user.test%03d", i); 302*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, name, "v", 1, 0); 303*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0); 304*0e75aea0SChristian Brauner } 305*0e75aea0SChristian Brauner 306*0e75aea0SChristian Brauner /* Verify we're at the limit */ 307*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, "user.overflow", "v", 1, 0); 308*0e75aea0SChristian Brauner ASSERT_EQ(ret, -1); 309*0e75aea0SChristian Brauner ASSERT_EQ(errno, ENOSPC); 310*0e75aea0SChristian Brauner 311*0e75aea0SChristian Brauner /* Remove one xattr */ 312*0e75aea0SChristian Brauner ret = fremovexattr(self->sockfd, "user.test000"); 313*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0); 314*0e75aea0SChristian Brauner 315*0e75aea0SChristian Brauner /* Now we should be able to add one more */ 316*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, "user.newattr", "v", 1, 0); 317*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0) { 318*0e75aea0SChristian Brauner TH_LOG("re-add after remove failed: %s", strerror(errno)); 319*0e75aea0SChristian Brauner } 320*0e75aea0SChristian Brauner } 321*0e75aea0SChristian Brauner 322*0e75aea0SChristian Brauner /* 323*0e75aea0SChristian Brauner * Test that two different sockets have independent xattr limits. 324*0e75aea0SChristian Brauner */ 325*0e75aea0SChristian Brauner TEST_F(xattr_sockfs, limits_per_inode) 326*0e75aea0SChristian Brauner { 327*0e75aea0SChristian Brauner char buf[256]; 328*0e75aea0SChristian Brauner int sock2; 329*0e75aea0SChristian Brauner ssize_t ret; 330*0e75aea0SChristian Brauner 331*0e75aea0SChristian Brauner sock2 = socket(AF_UNIX, SOCK_STREAM, 0); 332*0e75aea0SChristian Brauner ASSERT_GE(sock2, 0); 333*0e75aea0SChristian Brauner 334*0e75aea0SChristian Brauner /* Set xattr on first socket */ 335*0e75aea0SChristian Brauner ret = fsetxattr(self->sockfd, TEST_XATTR_NAME, 336*0e75aea0SChristian Brauner TEST_XATTR_VALUE, strlen(TEST_XATTR_VALUE), 0); 337*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0); 338*0e75aea0SChristian Brauner 339*0e75aea0SChristian Brauner /* First socket's xattr should not be visible on second socket */ 340*0e75aea0SChristian Brauner ret = fgetxattr(sock2, TEST_XATTR_NAME, NULL, 0); 341*0e75aea0SChristian Brauner ASSERT_EQ(ret, -1); 342*0e75aea0SChristian Brauner ASSERT_EQ(errno, ENODATA); 343*0e75aea0SChristian Brauner 344*0e75aea0SChristian Brauner /* Second socket should independently accept xattrs */ 345*0e75aea0SChristian Brauner ret = fsetxattr(sock2, TEST_XATTR_NAME, 346*0e75aea0SChristian Brauner TEST_XATTR_VALUE2, strlen(TEST_XATTR_VALUE2), 0); 347*0e75aea0SChristian Brauner ASSERT_EQ(ret, 0); 348*0e75aea0SChristian Brauner 349*0e75aea0SChristian Brauner /* Verify each socket has its own value */ 350*0e75aea0SChristian Brauner memset(buf, 0, sizeof(buf)); 351*0e75aea0SChristian Brauner ret = fgetxattr(self->sockfd, TEST_XATTR_NAME, buf, sizeof(buf)); 352*0e75aea0SChristian Brauner ASSERT_EQ(ret, (ssize_t)strlen(TEST_XATTR_VALUE)); 353*0e75aea0SChristian Brauner ASSERT_STREQ(buf, TEST_XATTR_VALUE); 354*0e75aea0SChristian Brauner 355*0e75aea0SChristian Brauner memset(buf, 0, sizeof(buf)); 356*0e75aea0SChristian Brauner ret = fgetxattr(sock2, TEST_XATTR_NAME, buf, sizeof(buf)); 357*0e75aea0SChristian Brauner ASSERT_EQ(ret, (ssize_t)strlen(TEST_XATTR_VALUE2)); 358*0e75aea0SChristian Brauner ASSERT_STREQ(buf, TEST_XATTR_VALUE2); 359*0e75aea0SChristian Brauner 360*0e75aea0SChristian Brauner close(sock2); 361*0e75aea0SChristian Brauner } 362*0e75aea0SChristian Brauner 363*0e75aea0SChristian Brauner TEST_HARNESS_MAIN 364