1*8d67ca89SAndroid Build Coastguard Worker /* 2*8d67ca89SAndroid Build Coastguard Worker * Copyright (C) 2019 The Android Open Source Project 3*8d67ca89SAndroid Build Coastguard Worker * All rights reserved. 4*8d67ca89SAndroid Build Coastguard Worker * 5*8d67ca89SAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without 6*8d67ca89SAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions 7*8d67ca89SAndroid Build Coastguard Worker * are met: 8*8d67ca89SAndroid Build Coastguard Worker * * Redistributions of source code must retain the above copyright 9*8d67ca89SAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer. 10*8d67ca89SAndroid Build Coastguard Worker * * Redistributions in binary form must reproduce the above copyright 11*8d67ca89SAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer in 12*8d67ca89SAndroid Build Coastguard Worker * the documentation and/or other materials provided with the 13*8d67ca89SAndroid Build Coastguard Worker * distribution. 14*8d67ca89SAndroid Build Coastguard Worker * 15*8d67ca89SAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 16*8d67ca89SAndroid Build Coastguard Worker * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 17*8d67ca89SAndroid Build Coastguard Worker * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 18*8d67ca89SAndroid Build Coastguard Worker * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE 19*8d67ca89SAndroid Build Coastguard Worker * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 20*8d67ca89SAndroid Build Coastguard Worker * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 21*8d67ca89SAndroid Build Coastguard Worker * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 22*8d67ca89SAndroid Build Coastguard Worker * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 23*8d67ca89SAndroid Build Coastguard Worker * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 24*8d67ca89SAndroid Build Coastguard Worker * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT 25*8d67ca89SAndroid Build Coastguard Worker * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 26*8d67ca89SAndroid Build Coastguard Worker * SUCH DAMAGE. 27*8d67ca89SAndroid Build Coastguard Worker */ 28*8d67ca89SAndroid Build Coastguard Worker 29*8d67ca89SAndroid Build Coastguard Worker #pragma once 30*8d67ca89SAndroid Build Coastguard Worker 31*8d67ca89SAndroid Build Coastguard Worker #include <stdatomic.h> 32*8d67ca89SAndroid Build Coastguard Worker #include <sys/cdefs.h> 33*8d67ca89SAndroid Build Coastguard Worker 34*8d67ca89SAndroid Build Coastguard Worker #include "platform/bionic/fdtrack.h" 35*8d67ca89SAndroid Build Coastguard Worker 36*8d67ca89SAndroid Build Coastguard Worker #include "bionic/pthread_internal.h" 37*8d67ca89SAndroid Build Coastguard Worker #include "private/ErrnoRestorer.h" 38*8d67ca89SAndroid Build Coastguard Worker #include "private/bionic_tls.h" 39*8d67ca89SAndroid Build Coastguard Worker 40*8d67ca89SAndroid Build Coastguard Worker extern "C" _Atomic(android_fdtrack_hook_t) __android_fdtrack_hook; 41*8d67ca89SAndroid Build Coastguard Worker extern "C" bool __android_fdtrack_globally_disabled; 42*8d67ca89SAndroid Build Coastguard Worker 43*8d67ca89SAndroid Build Coastguard Worker // Macro to record file descriptor creation. 44*8d67ca89SAndroid Build Coastguard Worker // e.g.: 45*8d67ca89SAndroid Build Coastguard Worker // int socket(int domain, int type, int protocol) { 46*8d67ca89SAndroid Build Coastguard Worker // return FDTRACK_CREATE_NAME("socket", __socket(domain, type, protocol)); 47*8d67ca89SAndroid Build Coastguard Worker // } 48*8d67ca89SAndroid Build Coastguard Worker #define FDTRACK_CREATE_NAME(name, fd_value) \ 49*8d67ca89SAndroid Build Coastguard Worker ({ \ 50*8d67ca89SAndroid Build Coastguard Worker int __fd = (fd_value); \ 51*8d67ca89SAndroid Build Coastguard Worker if (__fd != -1 && __predict_false(__android_fdtrack_hook) && \ 52*8d67ca89SAndroid Build Coastguard Worker !__predict_false(__get_thread()->is_vforked())) { \ 53*8d67ca89SAndroid Build Coastguard Worker bionic_tls& tls = __get_bionic_tls(); \ 54*8d67ca89SAndroid Build Coastguard Worker /* fdtrack_disabled is only true during reentrant calls. */ \ 55*8d67ca89SAndroid Build Coastguard Worker if (!__predict_false(tls.fdtrack_disabled) && \ 56*8d67ca89SAndroid Build Coastguard Worker !__predict_false(__android_fdtrack_globally_disabled)) { \ 57*8d67ca89SAndroid Build Coastguard Worker ErrnoRestorer r; \ 58*8d67ca89SAndroid Build Coastguard Worker tls.fdtrack_disabled = true; \ 59*8d67ca89SAndroid Build Coastguard Worker android_fdtrack_event event; \ 60*8d67ca89SAndroid Build Coastguard Worker event.fd = __fd; \ 61*8d67ca89SAndroid Build Coastguard Worker event.type = ANDROID_FDTRACK_EVENT_TYPE_CREATE; \ 62*8d67ca89SAndroid Build Coastguard Worker event.data.create.function_name = name; \ 63*8d67ca89SAndroid Build Coastguard Worker atomic_load (&__android_fdtrack_hook)(&event); \ 64*8d67ca89SAndroid Build Coastguard Worker tls.fdtrack_disabled = false; \ 65*8d67ca89SAndroid Build Coastguard Worker } \ 66*8d67ca89SAndroid Build Coastguard Worker } \ 67*8d67ca89SAndroid Build Coastguard Worker __fd; \ 68*8d67ca89SAndroid Build Coastguard Worker }) 69*8d67ca89SAndroid Build Coastguard Worker 70*8d67ca89SAndroid Build Coastguard Worker // Macro to record file descriptor creation, with the current function's name. 71*8d67ca89SAndroid Build Coastguard Worker // e.g.: 72*8d67ca89SAndroid Build Coastguard Worker // int socket(int domain, int type, int protocol) { 73*8d67ca89SAndroid Build Coastguard Worker // return FDTRACK_CREATE_NAME(__socket(domain, type, protocol)); 74*8d67ca89SAndroid Build Coastguard Worker // } 75*8d67ca89SAndroid Build Coastguard Worker #define FDTRACK_CREATE(fd_value) FDTRACK_CREATE_NAME(__func__, (fd_value)) 76*8d67ca89SAndroid Build Coastguard Worker 77*8d67ca89SAndroid Build Coastguard Worker // Macro to record file descriptor closure. 78*8d67ca89SAndroid Build Coastguard Worker // Note that this does not actually close the file descriptor. 79*8d67ca89SAndroid Build Coastguard Worker #define FDTRACK_CLOSE(fd_value) \ 80*8d67ca89SAndroid Build Coastguard Worker ({ \ 81*8d67ca89SAndroid Build Coastguard Worker int __fd = (fd_value); \ 82*8d67ca89SAndroid Build Coastguard Worker if (__fd != -1 && __predict_false(__android_fdtrack_hook) && \ 83*8d67ca89SAndroid Build Coastguard Worker !__predict_false(__get_thread()->is_vforked())) { \ 84*8d67ca89SAndroid Build Coastguard Worker bionic_tls& tls = __get_bionic_tls(); \ 85*8d67ca89SAndroid Build Coastguard Worker if (!__predict_false(tls.fdtrack_disabled) && \ 86*8d67ca89SAndroid Build Coastguard Worker !__predict_false(__android_fdtrack_globally_disabled)) { \ 87*8d67ca89SAndroid Build Coastguard Worker int saved_errno = errno; \ 88*8d67ca89SAndroid Build Coastguard Worker tls.fdtrack_disabled = true; \ 89*8d67ca89SAndroid Build Coastguard Worker android_fdtrack_event event; \ 90*8d67ca89SAndroid Build Coastguard Worker event.fd = __fd; \ 91*8d67ca89SAndroid Build Coastguard Worker event.type = ANDROID_FDTRACK_EVENT_TYPE_CLOSE; \ 92*8d67ca89SAndroid Build Coastguard Worker atomic_load (&__android_fdtrack_hook)(&event); \ 93*8d67ca89SAndroid Build Coastguard Worker tls.fdtrack_disabled = false; \ 94*8d67ca89SAndroid Build Coastguard Worker errno = saved_errno; \ 95*8d67ca89SAndroid Build Coastguard Worker } \ 96*8d67ca89SAndroid Build Coastguard Worker } \ 97*8d67ca89SAndroid Build Coastguard Worker __fd; \ 98*8d67ca89SAndroid Build Coastguard Worker }) 99