xref: /aosp_15_r20/external/swiftshader/src/System/Linux/MemFd.cpp (revision 03ce13f70fcc45d86ee91b7ee4cab1936a95046e)
1*03ce13f7SAndroid Build Coastguard Worker // Copyright 2019 The SwiftShader Authors. All Rights Reserved.
2*03ce13f7SAndroid Build Coastguard Worker //
3*03ce13f7SAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
4*03ce13f7SAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
5*03ce13f7SAndroid Build Coastguard Worker // You may obtain a copy of the License at
6*03ce13f7SAndroid Build Coastguard Worker //
7*03ce13f7SAndroid Build Coastguard Worker //    http://www.apache.org/licenses/LICENSE-2.0
8*03ce13f7SAndroid Build Coastguard Worker //
9*03ce13f7SAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*03ce13f7SAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
11*03ce13f7SAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*03ce13f7SAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
13*03ce13f7SAndroid Build Coastguard Worker // limitations under the License.
14*03ce13f7SAndroid Build Coastguard Worker 
15*03ce13f7SAndroid Build Coastguard Worker #include "MemFd.hpp"
16*03ce13f7SAndroid Build Coastguard Worker #include "../Debug.hpp"
17*03ce13f7SAndroid Build Coastguard Worker 
18*03ce13f7SAndroid Build Coastguard Worker #include <errno.h>
19*03ce13f7SAndroid Build Coastguard Worker #include <fcntl.h>
20*03ce13f7SAndroid Build Coastguard Worker #include <string.h>
21*03ce13f7SAndroid Build Coastguard Worker #include <sys/mman.h>
22*03ce13f7SAndroid Build Coastguard Worker #include <unistd.h>
23*03ce13f7SAndroid Build Coastguard Worker 
24*03ce13f7SAndroid Build Coastguard Worker #ifndef MFD_CLOEXEC
25*03ce13f7SAndroid Build Coastguard Worker #	define MFD_CLOEXEC 0x0001U
26*03ce13f7SAndroid Build Coastguard Worker #endif
27*03ce13f7SAndroid Build Coastguard Worker 
28*03ce13f7SAndroid Build Coastguard Worker #if __aarch64__
29*03ce13f7SAndroid Build Coastguard Worker #	define __NR_memfd_create 279
30*03ce13f7SAndroid Build Coastguard Worker #elif __arm__
31*03ce13f7SAndroid Build Coastguard Worker #	define __NR_memfd_create 279
32*03ce13f7SAndroid Build Coastguard Worker #elif __powerpc64__
33*03ce13f7SAndroid Build Coastguard Worker #	define __NR_memfd_create 360
34*03ce13f7SAndroid Build Coastguard Worker #elif __i386__
35*03ce13f7SAndroid Build Coastguard Worker #	define __NR_memfd_create 356
36*03ce13f7SAndroid Build Coastguard Worker #elif __x86_64__
37*03ce13f7SAndroid Build Coastguard Worker #	define __NR_memfd_create 319
38*03ce13f7SAndroid Build Coastguard Worker #endif /* __NR_memfd_create__ */
39*03ce13f7SAndroid Build Coastguard Worker 
~LinuxMemFd()40*03ce13f7SAndroid Build Coastguard Worker LinuxMemFd::~LinuxMemFd()
41*03ce13f7SAndroid Build Coastguard Worker {
42*03ce13f7SAndroid Build Coastguard Worker 	close();
43*03ce13f7SAndroid Build Coastguard Worker }
44*03ce13f7SAndroid Build Coastguard Worker 
importFd(int fd)45*03ce13f7SAndroid Build Coastguard Worker void LinuxMemFd::importFd(int fd)
46*03ce13f7SAndroid Build Coastguard Worker {
47*03ce13f7SAndroid Build Coastguard Worker 	close();
48*03ce13f7SAndroid Build Coastguard Worker 	fd_ = fd;
49*03ce13f7SAndroid Build Coastguard Worker }
50*03ce13f7SAndroid Build Coastguard Worker 
exportFd() const51*03ce13f7SAndroid Build Coastguard Worker int LinuxMemFd::exportFd() const
52*03ce13f7SAndroid Build Coastguard Worker {
53*03ce13f7SAndroid Build Coastguard Worker 	if(fd_ < 0)
54*03ce13f7SAndroid Build Coastguard Worker 	{
55*03ce13f7SAndroid Build Coastguard Worker 		return fd_;
56*03ce13f7SAndroid Build Coastguard Worker 	}
57*03ce13f7SAndroid Build Coastguard Worker 
58*03ce13f7SAndroid Build Coastguard Worker 	// Duplicate file descriptor while setting the clo-on-exec flag (Linux specific).
59*03ce13f7SAndroid Build Coastguard Worker 	return ::fcntl(fd_, F_DUPFD_CLOEXEC, 0);
60*03ce13f7SAndroid Build Coastguard Worker }
61*03ce13f7SAndroid Build Coastguard Worker 
allocate(const char * name,size_t size)62*03ce13f7SAndroid Build Coastguard Worker bool LinuxMemFd::allocate(const char *name, size_t size)
63*03ce13f7SAndroid Build Coastguard Worker {
64*03ce13f7SAndroid Build Coastguard Worker 	close();
65*03ce13f7SAndroid Build Coastguard Worker 
66*03ce13f7SAndroid Build Coastguard Worker #ifndef __NR_memfd_create
67*03ce13f7SAndroid Build Coastguard Worker 	TRACE("memfd_create() not supported on this system!");
68*03ce13f7SAndroid Build Coastguard Worker 	return false;
69*03ce13f7SAndroid Build Coastguard Worker #else
70*03ce13f7SAndroid Build Coastguard Worker 	// In the event of no system call this returns -1 with errno set
71*03ce13f7SAndroid Build Coastguard Worker 	// as ENOSYS.
72*03ce13f7SAndroid Build Coastguard Worker 	fd_ = syscall(__NR_memfd_create, name, MFD_CLOEXEC);
73*03ce13f7SAndroid Build Coastguard Worker 	if(fd_ < 0)
74*03ce13f7SAndroid Build Coastguard Worker 	{
75*03ce13f7SAndroid Build Coastguard Worker 		TRACE("memfd_create() returned %d: %s", errno, strerror(errno));
76*03ce13f7SAndroid Build Coastguard Worker 		return false;
77*03ce13f7SAndroid Build Coastguard Worker 	}
78*03ce13f7SAndroid Build Coastguard Worker 	// Ensure there is enough space.
79*03ce13f7SAndroid Build Coastguard Worker 	if(size > 0 && ::ftruncate(fd_, size) < 0)
80*03ce13f7SAndroid Build Coastguard Worker 	{
81*03ce13f7SAndroid Build Coastguard Worker 		TRACE("ftruncate() %lld returned %d: %s", (long long)size, errno, strerror(errno));
82*03ce13f7SAndroid Build Coastguard Worker 		close();
83*03ce13f7SAndroid Build Coastguard Worker 		return false;
84*03ce13f7SAndroid Build Coastguard Worker 	}
85*03ce13f7SAndroid Build Coastguard Worker #endif
86*03ce13f7SAndroid Build Coastguard Worker 	return true;
87*03ce13f7SAndroid Build Coastguard Worker }
88*03ce13f7SAndroid Build Coastguard Worker 
close()89*03ce13f7SAndroid Build Coastguard Worker void LinuxMemFd::close()
90*03ce13f7SAndroid Build Coastguard Worker {
91*03ce13f7SAndroid Build Coastguard Worker 	if(fd_ >= 0)
92*03ce13f7SAndroid Build Coastguard Worker 	{
93*03ce13f7SAndroid Build Coastguard Worker 		// WARNING: Never retry on close() failure, even with EINTR, see
94*03ce13f7SAndroid Build Coastguard Worker 		// https://lwn.net/Articles/576478/ for example.
95*03ce13f7SAndroid Build Coastguard Worker 		int ret = ::close(fd_);
96*03ce13f7SAndroid Build Coastguard Worker 		if(ret < 0)
97*03ce13f7SAndroid Build Coastguard Worker 		{
98*03ce13f7SAndroid Build Coastguard Worker 			TRACE("LinuxMemFd::close() failed with: %s", strerror(errno));
99*03ce13f7SAndroid Build Coastguard Worker 			assert(false);
100*03ce13f7SAndroid Build Coastguard Worker 		}
101*03ce13f7SAndroid Build Coastguard Worker 		fd_ = -1;
102*03ce13f7SAndroid Build Coastguard Worker 	}
103*03ce13f7SAndroid Build Coastguard Worker }
104*03ce13f7SAndroid Build Coastguard Worker 
mapReadWrite(size_t offset,size_t size)105*03ce13f7SAndroid Build Coastguard Worker void *LinuxMemFd::mapReadWrite(size_t offset, size_t size)
106*03ce13f7SAndroid Build Coastguard Worker {
107*03ce13f7SAndroid Build Coastguard Worker 	void *addr = ::mmap(nullptr, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd_,
108*03ce13f7SAndroid Build Coastguard Worker 	                    static_cast<off_t>(offset));
109*03ce13f7SAndroid Build Coastguard Worker 	return (addr == MAP_FAILED) ? nullptr : addr;
110*03ce13f7SAndroid Build Coastguard Worker }
111*03ce13f7SAndroid Build Coastguard Worker 
unmap(void * addr,size_t size)112*03ce13f7SAndroid Build Coastguard Worker bool LinuxMemFd::unmap(void *addr, size_t size)
113*03ce13f7SAndroid Build Coastguard Worker {
114*03ce13f7SAndroid Build Coastguard Worker 	return ::munmap(addr, size) == 0;
115*03ce13f7SAndroid Build Coastguard Worker }
116