xref: /linux/tools/testing/selftests/filesystems/xattr/xattr_sockfs_test.c (revision 0fc8f6200d2313278fbf4539bbab74677c685531)
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