[libc] Add Linux implementation of GNU extension function sendfile.

Reviewed By: michaelrj

Differential Revision: https://reviews.llvm.org/D132721
This commit is contained in:
Siva Chandra Reddy 2022-08-25 22:30:30 +00:00
parent cb0644b1cc
commit b8be3dabde
17 changed files with 219 additions and 2 deletions

View File

@ -100,6 +100,9 @@ set(TARGET_LIBC_ENTRYPOINTS
libc.src.sys.mman.mmap libc.src.sys.mman.mmap
libc.src.sys.mman.munmap libc.src.sys.mman.munmap
# sys/sendfile entrypoints
libc.src.sys.sendfile.sendfile
# sys/stat.h entrypoints # sys/stat.h entrypoints
libc.src.sys.stat.chmod libc.src.sys.stat.chmod
libc.src.sys.stat.fchmod libc.src.sys.stat.fchmod

View File

@ -274,3 +274,7 @@ def UniStdAPI : PublicAPI<"unistd.h"> {
def SysStatAPI : PublicAPI<"sys/stat.h"> { def SysStatAPI : PublicAPI<"sys/stat.h"> {
let Types = ["mode_t"]; let Types = ["mode_t"];
} }
def SysSendfileAPI : PublicAPI<"sys/sendfile.h"> {
let Types = ["off_t", "size_t", "ssize_t"];
}

View File

@ -100,6 +100,9 @@ set(TARGET_LIBC_ENTRYPOINTS
libc.src.sys.mman.mmap libc.src.sys.mman.mmap
libc.src.sys.mman.munmap libc.src.sys.mman.munmap
# sys/sendfile entrypoints
libc.src.sys.sendfile.sendfile
# sys/stat.h entrypoints # sys/stat.h entrypoints
libc.src.sys.stat.chmod libc.src.sys.stat.chmod
libc.src.sys.stat.fchmod libc.src.sys.stat.fchmod

View File

@ -212,6 +212,16 @@ add_gen_header(
.llvm-libc-types.mode_t .llvm-libc-types.mode_t
) )
add_gen_header(
sys_sendfile
DEF_FILE sys/sendfile.h.def
GEN_HDR sys/sendfile.h
DEPENDS
.llvm_libc_common_h
.llvm-libc-types.off_t
.llvm-libc-types.ssize_t
)
add_gen_header( add_gen_header(
sys_syscall sys_syscall
DEF_FILE sys/syscall.h.def DEF_FILE sys/syscall.h.def

View File

@ -0,0 +1,16 @@
//===-- Linux sys/sendfile.h ----------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#ifndef LLVM_LIBC_SYS_SENDFILE_H
#define LLVM_LIBC_SYS_SENDFILE_H
#include <__llvm-libc-common.h>
%%public_api()
#endif // LLVM_LIBC_SYS_SENDFILE_H

View File

@ -130,11 +130,26 @@ def GnuExtensions : StandardSpec<"GNUExtensions"> {
] ]
>; >;
HeaderSpec SendFile = HeaderSpec<
"sys/sendfile.h",
[], // Macros
[OffTType, SizeTType, SSizeTType,], // Types
[], // Enumerations
[
FunctionSpec<
"sendfile",
RetValSpec<SSizeTType>,
[ArgSpec<IntType>, ArgSpec<IntType>, ArgSpec<OffTPtr>, ArgSpec<SizeTType>]
>,
]
>;
let Headers = [ let Headers = [
CType, CType,
FEnv, FEnv,
Math, Math,
PThread, PThread,
SendFile,
StdIO, StdIO,
String, String,
]; ];

View File

@ -32,8 +32,6 @@ def POSIX : StandardSpec<"POSIX"> {
ConstType ConstRestrictedCharPtr = ConstType<RestrictedCharPtr>; ConstType ConstRestrictedCharPtr = ConstType<RestrictedCharPtr>;
NamedType ModeTType = NamedType<"mode_t">; NamedType ModeTType = NamedType<"mode_t">;
NamedType OffTType = NamedType<"off_t">;
NamedType SSizeTType = NamedType<"ssize_t">;
NamedType PThreadAttrTType = NamedType<"pthread_attr_t">; NamedType PThreadAttrTType = NamedType<"pthread_attr_t">;
PtrType PThreadAttrTPtr = PtrType<PThreadAttrTType>; PtrType PThreadAttrTPtr = PtrType<PThreadAttrTType>;

View File

@ -63,6 +63,10 @@ def LongDoublePtr : PtrType<LongDoubleType>;
def IntMaxTType : NamedType<"intmax_t">; def IntMaxTType : NamedType<"intmax_t">;
def UIntMaxTType : NamedType<"uintmax_t">; def UIntMaxTType : NamedType<"uintmax_t">;
def OffTType : NamedType<"off_t">;
def OffTPtr : PtrType<OffTType>;
def SSizeTType : NamedType<"ssize_t">;
// _Noreturn is really not a type, but it is convenient to treat it as a type. // _Noreturn is really not a type, but it is convenient to treat it as a type.
def NoReturn : NamedType<"_Noreturn void">; def NoReturn : NamedType<"_Noreturn void">;

View File

@ -1,2 +1,3 @@
add_subdirectory(mman) add_subdirectory(mman)
add_subdirectory(sendfile)
add_subdirectory(stat) add_subdirectory(stat)

View File

@ -0,0 +1,11 @@
if(EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/${LIBC_TARGET_OS})
add_subdirectory(${CMAKE_CURRENT_SOURCE_DIR}/${LIBC_TARGET_OS})
endif()
add_entrypoint_object(
sendfile
ALIAS
DEPENDS
.${LIBC_TARGET_OS}.sendfile
)

View File

@ -0,0 +1,12 @@
add_entrypoint_object(
sendfile
SRCS
sendfile.cpp
HDRS
../sendfile.h
DEPENDS
libc.include.sys_sendfile
libc.include.sys_syscall
libc.src.__support.OSUtil.osutil
libc.src.errno.errno
)

View File

@ -0,0 +1,30 @@
//===-- Linux implementation of sendfile ----------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "src/sys/sendfile/sendfile.h"
#include "src/__support/OSUtil/syscall.h" // For internal syscall function.
#include "src/__support/common.h"
#include <errno.h>
#include <sys/sendfile.h>
#include <sys/syscall.h> // For syscall numbers.
namespace __llvm_libc {
LLVM_LIBC_FUNCTION(ssize_t, sendfile,
(int out_fd, int in_fd, off_t *offset, size_t count)) {
long ret = __llvm_libc::syscall(SYS_sendfile, in_fd, out_fd, offset, count);
if (ret < 0) {
errno = -ret;
return -1;
}
return ret;
}
} // namespace __llvm_libc

View File

@ -0,0 +1,20 @@
//===-- Implementation header for sendfile ----------------------*- C++ -*-===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#ifndef LLVM_LIBC_SRC_SYS_STAT_SENDFILE_H
#define LLVM_LIBC_SRC_SYS_STAT_SENDFILE_H
#include <sys/sendfile.h>
namespace __llvm_libc {
ssize_t sendfile(int, int, off_t *, size_t);
} // namespace __llvm_libc
#endif // LLVM_LIBC_SRC_SYS_STAT_SENDFILE_H

View File

@ -1,2 +1,3 @@
add_subdirectory(mman) add_subdirectory(mman)
add_subdirectory(sendfile)
add_subdirectory(stat) add_subdirectory(stat)

View File

@ -0,0 +1,21 @@
add_libc_testsuite(libc_sys_sendfile_unittests)
add_subdirectory(testdata)
add_libc_unittest(
sendfile_test
SUITE
libc_sys_sendfile_unittests
SRCS
sendfile_test.cpp
DEPENDS
libc.include.errno
libc.include.fcntl
libc.include.sys_stat
libc.src.fcntl.open
libc.src.sys.sendfile.sendfile
libc.src.unistd.close
libc.src.unistd.read
libc.src.unistd.unlink
libc.src.unistd.write
)

View File

@ -0,0 +1,67 @@
//===-- Unittests for sendfile --------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "src/__support/CPP/string_view.h"
#include "src/fcntl/open.h"
#include "src/sys/sendfile/sendfile.h"
#include "src/unistd/close.h"
#include "src/unistd/read.h"
#include "src/unistd/unlink.h"
#include "src/unistd/write.h"
#include "test/ErrnoSetterMatcher.h"
#include "utils/UnitTest/Test.h"
#include "utils/testutils/FDReader.h"
#include <errno.h>
#include <fcntl.h>
#include <sys/stat.h>
namespace cpp = __llvm_libc::cpp;
TEST(LlvmLibcSendfileTest, CreateAndTransfer) {
using __llvm_libc::testing::ErrnoSetterMatcher::Fails;
using __llvm_libc::testing::ErrnoSetterMatcher::Succeeds;
// The test strategy is to
// 1. Create a temporary file with known data.
// 2. Use sendfile to copy it to another file.
// 3. Make sure that the data was actually copied.
// 4. Clean up the temporary files.
constexpr const char *IN_FILE = "testdata/sendfile_in.test";
constexpr const char *OUT_FILE = "testdata/sendfile_out.test";
const char IN_DATA[] = "sendfile test";
constexpr ssize_t IN_SIZE = ssize_t(sizeof(IN_DATA));
errno = 0;
int in_fd = __llvm_libc::open(IN_FILE, O_CREAT | O_WRONLY, S_IRWXU);
ASSERT_GT(in_fd, 0);
ASSERT_EQ(errno, 0);
ASSERT_EQ(__llvm_libc::write(in_fd, IN_DATA, IN_SIZE), IN_SIZE);
ASSERT_THAT(__llvm_libc::close(in_fd), Succeeds(0));
in_fd = __llvm_libc::open(IN_FILE, O_RDONLY);
ASSERT_GT(in_fd, 0);
ASSERT_EQ(errno, 0);
int out_fd = __llvm_libc::open(OUT_FILE, O_CREAT | O_WRONLY, S_IRWXU);
ASSERT_GT(out_fd, 0);
ASSERT_EQ(errno, 0);
ssize_t size = __llvm_libc::sendfile(in_fd, out_fd, nullptr, IN_SIZE);
ASSERT_EQ(size, IN_SIZE);
ASSERT_THAT(__llvm_libc::close(in_fd), Succeeds(0));
ASSERT_THAT(__llvm_libc::close(out_fd), Succeeds(0));
out_fd = __llvm_libc::open(OUT_FILE, O_RDONLY);
ASSERT_GT(out_fd, 0);
ASSERT_EQ(errno, 0);
char buf[IN_SIZE];
ASSERT_EQ(IN_SIZE, __llvm_libc::read(out_fd, buf, IN_SIZE));
ASSERT_EQ(cpp::string_view(buf), cpp::string_view(IN_DATA));
ASSERT_THAT(__llvm_libc::unlink(IN_FILE), Succeeds(0));
ASSERT_THAT(__llvm_libc::unlink(OUT_FILE), Succeeds(0));
}

View File

@ -0,0 +1 @@
# Directory to hold temporary files created by tests.