xref: /aosp_15_r20/external/elfutils/tests/dwfl-report-offline-memory.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
1*7304104dSAndroid Build Coastguard Worker /* Test program for dwfl_report_offline_memory.
2*7304104dSAndroid Build Coastguard Worker    Copyright (C) 2022 Google LLC
3*7304104dSAndroid Build Coastguard Worker    This file is part of elfutils.
4*7304104dSAndroid Build Coastguard Worker 
5*7304104dSAndroid Build Coastguard Worker    This file is free software; you can redistribute it and/or modify
6*7304104dSAndroid Build Coastguard Worker    it under the terms of the GNU General Public License as published by
7*7304104dSAndroid Build Coastguard Worker    the Free Software Foundation; either version 3 of the License, or
8*7304104dSAndroid Build Coastguard Worker    (at your option) any later version.
9*7304104dSAndroid Build Coastguard Worker 
10*7304104dSAndroid Build Coastguard Worker    elfutils is distributed in the hope that it will be useful, but
11*7304104dSAndroid Build Coastguard Worker    WITHOUT ANY WARRANTY; without even the implied warranty of
12*7304104dSAndroid Build Coastguard Worker    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13*7304104dSAndroid Build Coastguard Worker    GNU General Public License for more details.
14*7304104dSAndroid Build Coastguard Worker 
15*7304104dSAndroid Build Coastguard Worker    You should have received a copy of the GNU General Public License
16*7304104dSAndroid Build Coastguard Worker    along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
17*7304104dSAndroid Build Coastguard Worker 
18*7304104dSAndroid Build Coastguard Worker #include <config.h>
19*7304104dSAndroid Build Coastguard Worker 
20*7304104dSAndroid Build Coastguard Worker #include <assert.h>
21*7304104dSAndroid Build Coastguard Worker #include <errno.h>
22*7304104dSAndroid Build Coastguard Worker #include <error.h>
23*7304104dSAndroid Build Coastguard Worker #include <fcntl.h>
24*7304104dSAndroid Build Coastguard Worker #include <locale.h>
25*7304104dSAndroid Build Coastguard Worker #include <stdio.h>
26*7304104dSAndroid Build Coastguard Worker #include <stdio_ext.h>
27*7304104dSAndroid Build Coastguard Worker #include <stdlib.h>
28*7304104dSAndroid Build Coastguard Worker #include <string.h>
29*7304104dSAndroid Build Coastguard Worker #include <unistd.h>
30*7304104dSAndroid Build Coastguard Worker 
31*7304104dSAndroid Build Coastguard Worker #include ELFUTILS_HEADER(dwfl)
32*7304104dSAndroid Build Coastguard Worker #include ELFUTILS_HEADER(elf)
33*7304104dSAndroid Build Coastguard Worker #include <gelf.h>
34*7304104dSAndroid Build Coastguard Worker 
35*7304104dSAndroid Build Coastguard Worker 
36*7304104dSAndroid Build Coastguard Worker static const Dwfl_Callbacks offline_callbacks =
37*7304104dSAndroid Build Coastguard Worker   {
38*7304104dSAndroid Build Coastguard Worker     .find_debuginfo = INTUSE(dwfl_standard_find_debuginfo),
39*7304104dSAndroid Build Coastguard Worker     .section_address = INTUSE(dwfl_offline_section_address),
40*7304104dSAndroid Build Coastguard Worker   };
41*7304104dSAndroid Build Coastguard Worker 
42*7304104dSAndroid Build Coastguard Worker static int
count_modules(Dwfl_Module * mod,void ** userdata,const char * name,Dwarf_Addr base,void * arg)43*7304104dSAndroid Build Coastguard Worker count_modules (Dwfl_Module *mod __attribute__ ((unused)),
44*7304104dSAndroid Build Coastguard Worker 	       void **userdata __attribute__ ((unused)),
45*7304104dSAndroid Build Coastguard Worker 	       const char *name __attribute__ ((unused)),
46*7304104dSAndroid Build Coastguard Worker 	       Dwarf_Addr base __attribute__ ((unused)), void *arg)
47*7304104dSAndroid Build Coastguard Worker {
48*7304104dSAndroid Build Coastguard Worker   unsigned long long *counter = arg;
49*7304104dSAndroid Build Coastguard Worker   ++(*counter);
50*7304104dSAndroid Build Coastguard Worker   return DWARF_CB_OK;
51*7304104dSAndroid Build Coastguard Worker }
52*7304104dSAndroid Build Coastguard Worker 
53*7304104dSAndroid Build Coastguard Worker static int
count_sections(Elf * elf)54*7304104dSAndroid Build Coastguard Worker count_sections (Elf *elf)
55*7304104dSAndroid Build Coastguard Worker {
56*7304104dSAndroid Build Coastguard Worker   int result = 0;
57*7304104dSAndroid Build Coastguard Worker   Elf_Scn *section = NULL;
58*7304104dSAndroid Build Coastguard Worker   GElf_Shdr header;
59*7304104dSAndroid Build Coastguard Worker   while ((section = elf_nextscn (elf, section)) != NULL)
60*7304104dSAndroid Build Coastguard Worker     {
61*7304104dSAndroid Build Coastguard Worker       assert (gelf_getshdr (section, &header) != NULL);
62*7304104dSAndroid Build Coastguard Worker       result += 1;
63*7304104dSAndroid Build Coastguard Worker     }
64*7304104dSAndroid Build Coastguard Worker   return result;
65*7304104dSAndroid Build Coastguard Worker }
66*7304104dSAndroid Build Coastguard Worker 
67*7304104dSAndroid Build Coastguard Worker int
main(int argc,char ** argv)68*7304104dSAndroid Build Coastguard Worker main (int argc, char **argv)
69*7304104dSAndroid Build Coastguard Worker {
70*7304104dSAndroid Build Coastguard Worker   /* We use no threads here which can interfere with handling a stream.  */
71*7304104dSAndroid Build Coastguard Worker   (void) __fsetlocking (stdout, FSETLOCKING_BYCALLER);
72*7304104dSAndroid Build Coastguard Worker 
73*7304104dSAndroid Build Coastguard Worker   /* Set locale.  */
74*7304104dSAndroid Build Coastguard Worker   (void) setlocale (LC_ALL, "");
75*7304104dSAndroid Build Coastguard Worker 
76*7304104dSAndroid Build Coastguard Worker   if (argc != 4)
77*7304104dSAndroid Build Coastguard Worker     error (-1, 0,
78*7304104dSAndroid Build Coastguard Worker 	   "usage: dwfl_report_offline_memory [filename] "
79*7304104dSAndroid Build Coastguard Worker 	   "[expected number of modules] "
80*7304104dSAndroid Build Coastguard Worker 	   "[expected number of sections]");
81*7304104dSAndroid Build Coastguard Worker 
82*7304104dSAndroid Build Coastguard Worker   const char *fname = argv[1];
83*7304104dSAndroid Build Coastguard Worker   int fd = open (fname, O_RDONLY);
84*7304104dSAndroid Build Coastguard Worker   if (fd < 0)
85*7304104dSAndroid Build Coastguard Worker     error (-1, 0, "can't open file %s: %s", fname, strerror (errno));
86*7304104dSAndroid Build Coastguard Worker   off_t size = lseek (fd, 0, SEEK_END);
87*7304104dSAndroid Build Coastguard Worker   if (size < 0)
88*7304104dSAndroid Build Coastguard Worker     error (-1, 0, "can't lseek file %s: %s", fname, strerror (errno));
89*7304104dSAndroid Build Coastguard Worker   lseek (fd, 0, SEEK_SET);
90*7304104dSAndroid Build Coastguard Worker   char *data = malloc (size);
91*7304104dSAndroid Build Coastguard Worker   if (data == NULL)
92*7304104dSAndroid Build Coastguard Worker     error (-1, 0, "can't malloc: %s", strerror (errno));
93*7304104dSAndroid Build Coastguard Worker   ssize_t bytes_read = read (fd, data, size);
94*7304104dSAndroid Build Coastguard Worker   assert (bytes_read == size);
95*7304104dSAndroid Build Coastguard Worker   close (fd);
96*7304104dSAndroid Build Coastguard Worker 
97*7304104dSAndroid Build Coastguard Worker   Dwfl *dwfl = dwfl_begin (&offline_callbacks);
98*7304104dSAndroid Build Coastguard Worker   assert (dwfl != NULL);
99*7304104dSAndroid Build Coastguard Worker 
100*7304104dSAndroid Build Coastguard Worker   Dwfl_Module *mod;
101*7304104dSAndroid Build Coastguard Worker 
102*7304104dSAndroid Build Coastguard Worker   /* Check error handling by suppling zero data */
103*7304104dSAndroid Build Coastguard Worker   mod = dwfl_report_offline_memory(dwfl, argv[1], argv[1], NULL, 0);
104*7304104dSAndroid Build Coastguard Worker   assert(mod == NULL);
105*7304104dSAndroid Build Coastguard Worker 
106*7304104dSAndroid Build Coastguard Worker   mod = dwfl_report_offline_memory(dwfl, argv[1], argv[1], data, size);
107*7304104dSAndroid Build Coastguard Worker   assert (mod != NULL);
108*7304104dSAndroid Build Coastguard Worker   dwfl_report_end (dwfl, NULL, NULL);
109*7304104dSAndroid Build Coastguard Worker 
110*7304104dSAndroid Build Coastguard Worker   unsigned long long number_of_modules = 0;
111*7304104dSAndroid Build Coastguard Worker   ptrdiff_t offset =
112*7304104dSAndroid Build Coastguard Worker       dwfl_getmodules (dwfl, &count_modules, &number_of_modules, 0);
113*7304104dSAndroid Build Coastguard Worker   if (offset < 0)
114*7304104dSAndroid Build Coastguard Worker     error (1, 0, "dwfl_getmodules: %s", dwfl_errmsg (-1));
115*7304104dSAndroid Build Coastguard Worker   assert (offset == 0);
116*7304104dSAndroid Build Coastguard Worker 
117*7304104dSAndroid Build Coastguard Worker   char *endptr;
118*7304104dSAndroid Build Coastguard Worker   unsigned long long expected_number_of_modules =
119*7304104dSAndroid Build Coastguard Worker       strtoull (argv[2], &endptr, 0);
120*7304104dSAndroid Build Coastguard Worker   assert (endptr && !*endptr);
121*7304104dSAndroid Build Coastguard Worker   assert (number_of_modules == expected_number_of_modules);
122*7304104dSAndroid Build Coastguard Worker 
123*7304104dSAndroid Build Coastguard Worker   GElf_Addr loadbase = 0;
124*7304104dSAndroid Build Coastguard Worker   Elf *elf = dwfl_module_getelf (mod, &loadbase);
125*7304104dSAndroid Build Coastguard Worker   int number_of_sections = count_sections (elf);
126*7304104dSAndroid Build Coastguard Worker   int expected_number_of_sections = atoi (argv[3]);
127*7304104dSAndroid Build Coastguard Worker   assert (number_of_sections == expected_number_of_sections);
128*7304104dSAndroid Build Coastguard Worker 
129*7304104dSAndroid Build Coastguard Worker   dwfl_end (dwfl);
130*7304104dSAndroid Build Coastguard Worker   free (data);
131*7304104dSAndroid Build Coastguard Worker 
132*7304104dSAndroid Build Coastguard Worker   return 0;
133*7304104dSAndroid Build Coastguard Worker }
134