1*71db0c75SAndroid Build Coastguard Worker //===-- Linux implementation of recvfrom ----------------------------------===// 2*71db0c75SAndroid Build Coastguard Worker // 3*71db0c75SAndroid Build Coastguard Worker // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4*71db0c75SAndroid Build Coastguard Worker // See https://llvm.org/LICENSE.txt for license information. 5*71db0c75SAndroid Build Coastguard Worker // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6*71db0c75SAndroid Build Coastguard Worker // 7*71db0c75SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===// 8*71db0c75SAndroid Build Coastguard Worker 9*71db0c75SAndroid Build Coastguard Worker #include "src/sys/socket/recvfrom.h" 10*71db0c75SAndroid Build Coastguard Worker 11*71db0c75SAndroid Build Coastguard Worker #include <linux/net.h> // For SYS_SOCKET socketcall number. 12*71db0c75SAndroid Build Coastguard Worker #include <sys/syscall.h> // For syscall numbers. 13*71db0c75SAndroid Build Coastguard Worker 14*71db0c75SAndroid Build Coastguard Worker #include "hdr/types/socklen_t.h" 15*71db0c75SAndroid Build Coastguard Worker #include "hdr/types/ssize_t.h" 16*71db0c75SAndroid Build Coastguard Worker #include "hdr/types/struct_sockaddr.h" 17*71db0c75SAndroid Build Coastguard Worker #include "src/__support/OSUtil/syscall.h" // For internal syscall function. 18*71db0c75SAndroid Build Coastguard Worker #include "src/__support/common.h" 19*71db0c75SAndroid Build Coastguard Worker #include "src/__support/macros/sanitizer.h" 20*71db0c75SAndroid Build Coastguard Worker #include "src/errno/libc_errno.h" 21*71db0c75SAndroid Build Coastguard Worker 22*71db0c75SAndroid Build Coastguard Worker namespace LIBC_NAMESPACE_DECL { 23*71db0c75SAndroid Build Coastguard Worker 24*71db0c75SAndroid Build Coastguard Worker LLVM_LIBC_FUNCTION(ssize_t, recvfrom, 25*71db0c75SAndroid Build Coastguard Worker (int sockfd, void *buf, size_t len, int flags, 26*71db0c75SAndroid Build Coastguard Worker sockaddr *__restrict src_addr, 27*71db0c75SAndroid Build Coastguard Worker socklen_t *__restrict addrlen)) { 28*71db0c75SAndroid Build Coastguard Worker // addrlen is a value-result argument. If it's not null, it passes the max 29*71db0c75SAndroid Build Coastguard Worker // size of the buffer src_addr to the syscall. After the syscall, it's updated 30*71db0c75SAndroid Build Coastguard Worker // to the actual size of the source address. This may be larger than the 31*71db0c75SAndroid Build Coastguard Worker // buffer, in which case the buffer contains a truncated result. 32*71db0c75SAndroid Build Coastguard Worker size_t srcaddr_sz; 33*71db0c75SAndroid Build Coastguard Worker if (src_addr) 34*71db0c75SAndroid Build Coastguard Worker srcaddr_sz = *addrlen; 35*71db0c75SAndroid Build Coastguard Worker (void)srcaddr_sz; // prevent "set but not used" warning 36*71db0c75SAndroid Build Coastguard Worker 37*71db0c75SAndroid Build Coastguard Worker #ifdef SYS_recvfrom 38*71db0c75SAndroid Build Coastguard Worker ssize_t ret = LIBC_NAMESPACE::syscall_impl<ssize_t>( 39*71db0c75SAndroid Build Coastguard Worker SYS_recvfrom, sockfd, buf, len, flags, src_addr, addrlen); 40*71db0c75SAndroid Build Coastguard Worker #elif defined(SYS_socketcall) 41*71db0c75SAndroid Build Coastguard Worker unsigned long sockcall_args[6] = {static_cast<unsigned long>(sockfd), 42*71db0c75SAndroid Build Coastguard Worker reinterpret_cast<unsigned long>(buf), 43*71db0c75SAndroid Build Coastguard Worker static_cast<unsigned long>(len), 44*71db0c75SAndroid Build Coastguard Worker static_cast<unsigned long>(flags), 45*71db0c75SAndroid Build Coastguard Worker reinterpret_cast<unsigned long>(src_addr), 46*71db0c75SAndroid Build Coastguard Worker static_cast<unsigned long>(addrlen)}; 47*71db0c75SAndroid Build Coastguard Worker ssize_t ret = LIBC_NAMESPACE::syscall_impl<ssize_t>( 48*71db0c75SAndroid Build Coastguard Worker SYS_socketcall, SYS_RECVFROM, sockcall_args); 49*71db0c75SAndroid Build Coastguard Worker #else 50*71db0c75SAndroid Build Coastguard Worker #error "socket and socketcall syscalls unavailable for this platform." 51*71db0c75SAndroid Build Coastguard Worker #endif 52*71db0c75SAndroid Build Coastguard Worker if (ret < 0) { 53*71db0c75SAndroid Build Coastguard Worker libc_errno = static_cast<int>(-ret); 54*71db0c75SAndroid Build Coastguard Worker return -1; 55*71db0c75SAndroid Build Coastguard Worker } 56*71db0c75SAndroid Build Coastguard Worker 57*71db0c75SAndroid Build Coastguard Worker MSAN_UNPOISON(buf, ret); 58*71db0c75SAndroid Build Coastguard Worker 59*71db0c75SAndroid Build Coastguard Worker if (src_addr) { 60*71db0c75SAndroid Build Coastguard Worker size_t min_src_addr_size = (*addrlen < srcaddr_sz) ? *addrlen : srcaddr_sz; 61*71db0c75SAndroid Build Coastguard Worker (void)min_src_addr_size; // prevent "set but not used" warning 62*71db0c75SAndroid Build Coastguard Worker 63*71db0c75SAndroid Build Coastguard Worker MSAN_UNPOISON(src_addr, min_src_addr_size); 64*71db0c75SAndroid Build Coastguard Worker } 65*71db0c75SAndroid Build Coastguard Worker return ret; 66*71db0c75SAndroid Build Coastguard Worker } 67*71db0c75SAndroid Build Coastguard Worker 68*71db0c75SAndroid Build Coastguard Worker } // namespace LIBC_NAMESPACE_DECL 69