1 /*
2  * Copyright (C) 2019 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include <errno.h>
18 #include <pthread.h>
19 #include <signal.h>
20 #include <stdlib.h>
21 #include <string.h>
22 #include <libgen.h> // For POSIX basename().
23 
24 // Use _system_properties.h to use __system_property_wait_any()
25 #define _REALLY_INCLUDE_SYS__SYSTEM_PROPERTIES_H_
26 #include <sys/_system_properties.h>
27 
28 #include "profile-extras.h"
29 
30 extern "C" {
31 
32 void __gcov_dump(void);
33 void __gcov_reset(void);
34 
35 // storing SIG_ERR helps us detect (unlikely) looping.
36 static sighandler_t chained_gcov_signal_handler = SIG_ERR;
37 
gcov_signal_handler(int signum)38 static void gcov_signal_handler(int signum) {
39   __gcov_dump();
40   __gcov_reset();
41   if (chained_gcov_signal_handler != SIG_ERR &&
42       chained_gcov_signal_handler != SIG_IGN &&
43       chained_gcov_signal_handler != SIG_DFL) {
44     (chained_gcov_signal_handler)(signum);
45   }
46 }
47 
48 __attribute__((weak)) int init_profile_extras_once = 0;
49 
50 // Initialize libprofile-extras:
51 // - Install a signal handler that triggers __gcov_flush on <COVERAGE_FLUSH_SIGNAL>.
52 //
53 // We want this initiazlier to run during load time.
54 //
55 // Just marking init_profile_extras() with __attribute__((constructor)) isn't
56 // enough since the linker drops it from its output since no other symbol from
57 // this static library is referenced.
58 //
59 // We force the linker to include init_profile_extras() by passing
60 // '-uinit_profile_extras' to the linker (in build/soong).
init_profile_extras(void)61 __attribute__((constructor)) int init_profile_extras(void) {
62   if (init_profile_extras_once)
63     return 0;
64   init_profile_extras_once = 1;
65 
66   // is this instance already registered?
67   if (chained_gcov_signal_handler != SIG_ERR) {
68     return -1;
69   }
70   sighandler_t ret1 = signal(COVERAGE_FLUSH_SIGNAL, gcov_signal_handler);
71   if (ret1 == SIG_ERR) {
72     return -1;
73   }
74   chained_gcov_signal_handler = ret1;
75 
76   return 0;
77 }
78 }
79