Page MenuHomeFreeBSD

D56114.diff
No OneTemporary

D56114.diff

diff --git a/lib/libc/gen/memfd_create.c b/lib/libc/gen/memfd_create.c
--- a/lib/libc/gen/memfd_create.c
+++ b/lib/libc/gen/memfd_create.c
@@ -95,16 +95,25 @@
npgs = getpagesizes(pgs, nitems(pgs));
if (npgs == -1)
goto clean;
- pgsize = (size_t)1 << ((flags & MFD_HUGE_MASK) >> MFD_HUGE_SHIFT);
- for (pgidx = 0; pgidx < npgs; pgidx++) {
- if (pgsize == pgs[pgidx])
- break;
- }
- if (pgidx == npgs) {
- errno = EOPNOTSUPP;
+ else if (npgs == 1) {
+ errno = ENOSYS;
goto clean;
}
+ if ((flags & MFD_HUGE_MASK) == 0) {
+ pgidx = 1;
+ } else {
+ pgsize = 1UL << ((flags & MFD_HUGE_MASK) >> MFD_HUGE_SHIFT);
+ for (pgidx = 1; pgidx < npgs; pgidx++) {
+ if (pgsize == pgs[pgidx])
+ break;
+ }
+ if (pgidx == npgs) {
+ errno = EOPNOTSUPP;
+ goto clean;
+ }
+ }
+
memset(&slc, 0, sizeof(slc));
slc.psind = pgidx;
slc.alloc_policy = SHM_LARGEPAGE_ALLOC_DEFAULT;
diff --git a/lib/libsys/shm_open.2 b/lib/libsys/shm_open.2
--- a/lib/libsys/shm_open.2
+++ b/lib/libsys/shm_open.2
@@ -26,7 +26,7 @@
.\" OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
.\" SUCH DAMAGE.
.\"
-.Dd August 4, 2025
+.Dd March 26, 2025
.Dt SHM_OPEN 2
.Os
.Sh NAME
@@ -343,7 +343,25 @@
.Xr fcntl 2
command.
.It Dv MFD_HUGETLB
-This flag is currently unsupported.
+Create a memfd backed by a
+.Dq largepage
+object.
+One of the
+.Dv MFD_HUGE_*
+flags defined in
+.In sys/mman.h
+may be included to specify a fixed size.
+If a specific size is not requested, the smallest supported large page size is
+selected.
+.Pp
+The behavior documented above for the
+.Fn shm_create_largepage
+.Fa psind
+argument also applies to largepage objects created by
+.Fn memfd_create ,
+and the
+.Dv SHM_LARGEPAGE_ALLOC_DEFAULT
+policy will always be used.
.El
.Sh RETURN VALUES
If successful,
@@ -458,17 +476,22 @@
.Pp
An invalid or unsupported flag was included in
.Fa flags .
+.It Bq Er EINVAL
+A hugetlb mapping was requested, but
+.Dv MFD_HUGETLB
+was not specified in
+.Fa flags .
.It Bq Er EMFILE
The process has already reached its limit for open file descriptors.
.It Bq Er ENFILE
The system file table is full.
.It Bq Er ENOSYS
-In
-.Fa memfd_create ,
.Dv MFD_HUGETLB
was specified in
.Fa flags ,
and this system does not support forced hugetlb mappings.
+.It Bq Er EOPNOTSUPP
+This system does not support the requested hugetlb page size.
.El
.Pp
.Fn shm_open
diff --git a/tests/sys/posixshm/memfd_test.c b/tests/sys/posixshm/memfd_test.c
--- a/tests/sys/posixshm/memfd_test.c
+++ b/tests/sys/posixshm/memfd_test.c
@@ -34,6 +34,8 @@
#include <errno.h>
#include <unistd.h>
+#include "posixshm.h"
+
ATF_TC_WITHOUT_HEAD(basic);
ATF_TC_BODY(basic, tc)
{
@@ -277,6 +279,38 @@
close(fd);
}
+ATF_TC_WITHOUT_HEAD(hugetlb);
+ATF_TC_BODY(hugetlb, tc)
+{
+ size_t ps[MAXPAGESIZES], pgsize;
+ int fd, pscnt;
+
+ pscnt = pagesizes(ps, false);
+#define MFD_HUGE_SUPPORTED(sz) (sz <= (1 << 24))
+#define MFD_HUGE_FLAGS(sz) (((ffsl(sz) - 1U) << MFD_HUGE_SHIFT) & MFD_HUGE_MASK)
+ for (int psidx = 1; psidx < pscnt; psidx++) {
+ pgsize = ps[psidx];
+
+ if (!MFD_HUGE_SUPPORTED(pgsize))
+ continue;
+
+ ATF_REQUIRE_MSG((fd = memfd_create("...",
+ MFD_HUGETLB | MFD_HUGE_FLAGS(pgsize))) != -1,
+ "Creating a %zu-size hugetlb memfd", pgsize);
+ }
+
+ fd = memfd_create("...", MFD_HUGETLB);
+ if (pscnt == 1) {
+ ATF_REQUIRE_MSG(fd == -1,
+ "Creating an unspecified hugetlb memfd without large page support");
+ ATF_REQUIRE(errno == ENOSYS);
+ } else {
+ ATF_REQUIRE_MSG(fd != -1,
+ "Creating an unspecified hugetlb memfd with large page support");
+ close(fd);
+ }
+}
+
ATF_TP_ADD_TCS(tp)
{
@@ -289,5 +323,6 @@
ATF_TP_ADD_TC(tp, get_seals);
ATF_TP_ADD_TC(tp, dup_seals);
ATF_TP_ADD_TC(tp, immutable_seals);
+ ATF_TP_ADD_TC(tp, hugetlb);
return (atf_no_error());
}
diff --git a/tests/sys/posixshm/posixshm.h b/tests/sys/posixshm/posixshm.h
new file mode 100644
--- /dev/null
+++ b/tests/sys/posixshm/posixshm.h
@@ -0,0 +1,45 @@
+/*-
+ *
+ * Copyright (c) 2020 Klara, Inc.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+#include <sys/param.h>
+#include <sys/mman.h>
+
+#include <stdbool.h>
+
+static int
+pagesizes(size_t ps[MAXPAGESIZES], bool required)
+{
+ int pscnt;
+
+ pscnt = getpagesizes(ps, MAXPAGESIZES);
+ ATF_REQUIRE_MSG(pscnt != -1, "getpagesizes failed; errno=%d", errno);
+ ATF_REQUIRE_MSG(ps[0] != 0, "psind 0 is %zu", ps[0]);
+ ATF_REQUIRE_MSG(pscnt <= MAXPAGESIZES, "invalid pscnt %d", pscnt);
+ if (pscnt == 1 && required)
+ atf_tc_skip("no large page support");
+ return (pscnt);
+}
+
diff --git a/tests/sys/posixshm/posixshm_test.c b/tests/sys/posixshm/posixshm_test.c
--- a/tests/sys/posixshm/posixshm_test.c
+++ b/tests/sys/posixshm/posixshm_test.c
@@ -49,6 +49,8 @@
#include <atf-c.h>
+#include "posixshm.h"
+
#define TEST_PATH_LEN 256
static char test_path[TEST_PATH_LEN];
static char test_path2[TEST_PATH_LEN];
@@ -1239,20 +1241,6 @@
return (fd);
}
-static int
-pagesizes(size_t ps[MAXPAGESIZES])
-{
- int pscnt;
-
- pscnt = getpagesizes(ps, MAXPAGESIZES);
- ATF_REQUIRE_MSG(pscnt != -1, "getpagesizes failed; errno=%d", errno);
- ATF_REQUIRE_MSG(ps[0] != 0, "psind 0 is %zu", ps[0]);
- ATF_REQUIRE_MSG(pscnt <= MAXPAGESIZES, "invalid pscnt %d", pscnt);
- if (pscnt == 1)
- atf_tc_skip("no large page support");
- return (pscnt);
-}
-
ATF_TC_WITHOUT_HEAD(largepage_basic);
ATF_TC_BODY(largepage_basic, tc)
{
@@ -1261,7 +1249,7 @@
size_t ps[MAXPAGESIZES];
int error, fd, pscnt;
- pscnt = pagesizes(ps);
+ pscnt = pagesizes(ps, true);
zeroes = calloc(1, ps[0]);
ATF_REQUIRE(zeroes != NULL);
for (int i = 1; i < pscnt; i++) {
@@ -1317,7 +1305,7 @@
size_t ps[MAXPAGESIZES + 1]; /* silence warnings if MAXPAGESIZES == 1 */
int error, fd;
- (void)pagesizes(ps);
+ (void)pagesizes(ps, true);
fd = shm_open(SHM_ANON, O_CREAT | O_RDWR, 0);
ATF_REQUIRE_MSG(fd >= 0, "shm_open failed; error=%d", errno);
@@ -1379,7 +1367,7 @@
size_t ps[MAXPAGESIZES];
int fd, pscnt;
- pscnt = pagesizes(ps);
+ pscnt = pagesizes(ps, true);
for (int i = 1; i < pscnt; i++) {
fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
@@ -1475,7 +1463,7 @@
size_t ps[MAXPAGESIZES], ps1;
int fd, pscnt;
- pscnt = pagesizes(ps);
+ pscnt = pagesizes(ps, true);
for (int i = 1; i < pscnt; i++) {
fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
ps1 = ps[i - 1];
@@ -1526,7 +1514,7 @@
size_t ps[MAXPAGESIZES];
int fd, pscnt;
- pscnt = pagesizes(ps);
+ pscnt = pagesizes(ps, true);
for (int i = 1; i < pscnt; i++) {
fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
addr = mmap(NULL, ps[i], PROT_READ | PROT_WRITE, MAP_SHARED, fd,
@@ -1595,7 +1583,7 @@
"sysctlbyname(vm.stats.vm.v_user_wire_count) failed; error=%d",
errno);
- pscnt = pagesizes(ps);
+ pscnt = pagesizes(ps, true);
for (int i = 1; i < pscnt; i++) {
if (ps[i] / ps[0] > max_wired - wired) {
/* Cannot wire past the limit. */
@@ -1638,7 +1626,7 @@
size_t ps[MAXPAGESIZES];
int fd, pscnt;
- pscnt = pagesizes(ps);
+ pscnt = pagesizes(ps, true);
for (int i = 1; i < pscnt; i++) {
fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
addr = mmap(NULL, ps[i], PROT_READ | PROT_WRITE, MAP_SHARED, fd,
@@ -1697,7 +1685,7 @@
size_t ps[MAXPAGESIZES];
int fd, pscnt;
- pscnt = pagesizes(ps);
+ pscnt = pagesizes(ps, true);
for (int i = 1; i < pscnt; i++) {
/*
* Reserve a contiguous region in the address space to avoid
@@ -1767,7 +1755,7 @@
pid_t child;
int fd, pscnt, status;
- pscnt = pagesizes(ps);
+ pscnt = pagesizes(ps, true);
for (int i = 1; i < pscnt; i++) {
fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
addr = mmap(NULL, ps[i], PROT_READ | PROT_WRITE, MAP_SHARED, fd,
@@ -1855,7 +1843,7 @@
int fd, pfd[2], pscnt, status;
pid_t child;
- pscnt = pagesizes(ps);
+ pscnt = pagesizes(ps, true);
for (int i = 1; i < pscnt; i++) {
fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
@@ -1908,7 +1896,7 @@
size_t ps[MAXPAGESIZES];
int fd, psind;
- (void)pagesizes(ps);
+ (void)pagesizes(ps, true);
psind = 1;
gen_test_path();

File Metadata

Mime Type
text/plain
Expires
Wed, Apr 29, 10:41 AM (10 h, 24 m)
Storage Engine
blob
Storage Format
Raw Data
Storage Handle
32272946
Default Alt Text
D56114.diff (9 KB)

Event Timeline