2017-06-14 01:52:12 +02:00
|
|
|
/*
|
|
|
|
* Copyright © 2017 Guillem Jover <guillem@hadrons.org>
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. The name of the author may not be used to endorse or promote products
|
|
|
|
* derived from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
|
|
|
|
* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
|
|
* AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
|
|
|
* THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
|
|
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
|
|
|
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
|
|
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
|
|
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <nlist.h>
|
|
|
|
|
test: nlist: make resilient against -Wl,--gc-sections
With linker sections GC enabled, we get a test failure in `nlist.c`:
```
nlist: nlist.c:72: main: Assertion `rc == 0' failed.
```
This turns out to be because several sections used by the test can be discarded:
```
ld: removing unused section '.text.func_pub' in file 'nlist.o'
ld: removing unused section '.bss.data_pub_uninit' in file 'nlist.o'
ld: removing unused section '.data.data_pub_init' in file 'nlist.o'
```
Reproduced with `CFLAGS="-Og -fdata-sections -ffunction-sections"` and
`LDFLAGS="-Wl,-O1 -Wl,--as-needed -Wl,--gc-sections -Wl,-z,start-stop-gc"`.
Additionally, `LDFLAGS="... -Wl,--print-gc-sections"` can help with diagnosing
which sections get removed.
We already mark these symbols as `used`, but we need `retain` [0] for them
to survive linker GC too.
[0] https://releases.llvm.org/18.1.0/tools/lld/docs/ELF/start-stop-gc.html#annotate-c-identifier-name-sections
Closes: !29
Signed-off-by: Sam James <sam@gentoo.org>
Signed-off-by: Guillem Jover <guillem@hadrons.org>
2024-03-13 23:34:55 +00:00
|
|
|
#if defined(__has_attribute)
|
|
|
|
#if __has_attribute(__retain__)
|
|
|
|
#define ATTRIBUTE_RETAIN __attribute__((__retain__))
|
|
|
|
#else
|
|
|
|
#define ATTRIBUTE_RETAIN
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2017-06-14 01:52:12 +02:00
|
|
|
static int data_prv_init = 50;
|
|
|
|
extern int data_pub_init;
|
|
|
|
extern int data_pub_uninit[2048];
|
|
|
|
extern int *data_pub_ptr;
|
|
|
|
|
test: nlist: make resilient against -Wl,--gc-sections
With linker sections GC enabled, we get a test failure in `nlist.c`:
```
nlist: nlist.c:72: main: Assertion `rc == 0' failed.
```
This turns out to be because several sections used by the test can be discarded:
```
ld: removing unused section '.text.func_pub' in file 'nlist.o'
ld: removing unused section '.bss.data_pub_uninit' in file 'nlist.o'
ld: removing unused section '.data.data_pub_init' in file 'nlist.o'
```
Reproduced with `CFLAGS="-Og -fdata-sections -ffunction-sections"` and
`LDFLAGS="-Wl,-O1 -Wl,--as-needed -Wl,--gc-sections -Wl,-z,start-stop-gc"`.
Additionally, `LDFLAGS="... -Wl,--print-gc-sections"` can help with diagnosing
which sections get removed.
We already mark these symbols as `used`, but we need `retain` [0] for them
to survive linker GC too.
[0] https://releases.llvm.org/18.1.0/tools/lld/docs/ELF/start-stop-gc.html#annotate-c-identifier-name-sections
Closes: !29
Signed-off-by: Sam James <sam@gentoo.org>
Signed-off-by: Guillem Jover <guillem@hadrons.org>
2024-03-13 23:34:55 +00:00
|
|
|
/* GCC's __has_attribute has odd behavior for retain: https://gcc.gnu.org/PR99587 */
|
|
|
|
#pragma GCC diagnostic push
|
|
|
|
#pragma GCC diagnostic ignored "-Wattributes"
|
2017-06-14 01:52:12 +02:00
|
|
|
int *data_pub_ptr = &data_prv_init;
|
test: nlist: make resilient against -Wl,--gc-sections
With linker sections GC enabled, we get a test failure in `nlist.c`:
```
nlist: nlist.c:72: main: Assertion `rc == 0' failed.
```
This turns out to be because several sections used by the test can be discarded:
```
ld: removing unused section '.text.func_pub' in file 'nlist.o'
ld: removing unused section '.bss.data_pub_uninit' in file 'nlist.o'
ld: removing unused section '.data.data_pub_init' in file 'nlist.o'
```
Reproduced with `CFLAGS="-Og -fdata-sections -ffunction-sections"` and
`LDFLAGS="-Wl,-O1 -Wl,--as-needed -Wl,--gc-sections -Wl,-z,start-stop-gc"`.
Additionally, `LDFLAGS="... -Wl,--print-gc-sections"` can help with diagnosing
which sections get removed.
We already mark these symbols as `used`, but we need `retain` [0] for them
to survive linker GC too.
[0] https://releases.llvm.org/18.1.0/tools/lld/docs/ELF/start-stop-gc.html#annotate-c-identifier-name-sections
Closes: !29
Signed-off-by: Sam James <sam@gentoo.org>
Signed-off-by: Guillem Jover <guillem@hadrons.org>
2024-03-13 23:34:55 +00:00
|
|
|
int data_pub_init ATTRIBUTE_RETAIN __attribute__((__used__)) = 10;
|
|
|
|
int data_pub_uninit[2048] ATTRIBUTE_RETAIN __attribute__((__used__));
|
2017-06-14 01:52:12 +02:00
|
|
|
|
|
|
|
extern int
|
test: nlist: make resilient against -Wl,--gc-sections
With linker sections GC enabled, we get a test failure in `nlist.c`:
```
nlist: nlist.c:72: main: Assertion `rc == 0' failed.
```
This turns out to be because several sections used by the test can be discarded:
```
ld: removing unused section '.text.func_pub' in file 'nlist.o'
ld: removing unused section '.bss.data_pub_uninit' in file 'nlist.o'
ld: removing unused section '.data.data_pub_init' in file 'nlist.o'
```
Reproduced with `CFLAGS="-Og -fdata-sections -ffunction-sections"` and
`LDFLAGS="-Wl,-O1 -Wl,--as-needed -Wl,--gc-sections -Wl,-z,start-stop-gc"`.
Additionally, `LDFLAGS="... -Wl,--print-gc-sections"` can help with diagnosing
which sections get removed.
We already mark these symbols as `used`, but we need `retain` [0] for them
to survive linker GC too.
[0] https://releases.llvm.org/18.1.0/tools/lld/docs/ELF/start-stop-gc.html#annotate-c-identifier-name-sections
Closes: !29
Signed-off-by: Sam James <sam@gentoo.org>
Signed-off-by: Guillem Jover <guillem@hadrons.org>
2024-03-13 23:34:55 +00:00
|
|
|
func_pub(void) ATTRIBUTE_RETAIN __attribute__((__used__));
|
|
|
|
#pragma GCC diagnostic pop
|
2017-06-14 01:52:12 +02:00
|
|
|
|
|
|
|
int
|
|
|
|
func_pub(void)
|
|
|
|
{
|
|
|
|
return 42;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
struct nlist nl[] = {
|
2021-03-02 00:48:31 +01:00
|
|
|
{ .n_un.n_name = (char *)"main" },
|
|
|
|
{ .n_un.n_name = (char *)"func_pub" },
|
|
|
|
{ .n_un.n_name = (char *)"data_pub_uninit" },
|
|
|
|
{ .n_un.n_name = (char *)"data_pub_init" },
|
|
|
|
{ .n_un.n_name = (char *)"data_prv_init" },
|
2017-06-14 01:52:12 +02:00
|
|
|
{ .n_un.n_name = NULL },
|
2017-06-19 21:41:13 +02:00
|
|
|
};
|
2017-06-14 01:52:12 +02:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
assert(*data_pub_ptr == 50);
|
|
|
|
|
2019-02-03 00:11:15 +00:00
|
|
|
assert(nl[0].n_name == nl[0].n_un.n_name);
|
|
|
|
|
2017-06-14 01:52:12 +02:00
|
|
|
rc = nlist(argv[0], nl);
|
|
|
|
assert(rc == 0);
|
|
|
|
|
2017-12-03 16:50:07 +01:00
|
|
|
#if defined(__powerpc64__) && _CALL_ELF == 1
|
|
|
|
/* On PowerPC 64-bit ELFv1, the functions are stored in
|
2017-06-25 01:51:30 +02:00
|
|
|
* the .text sections but they are accessed through a function
|
|
|
|
* descriptor stored in a data section, for example for PowerPC
|
|
|
|
* 64-bit that section is called .opd. */
|
|
|
|
assert(nl[0].n_type == (N_DATA | N_EXT));
|
|
|
|
assert(nl[1].n_type == (N_DATA | N_EXT));
|
|
|
|
#else
|
2017-06-14 01:52:12 +02:00
|
|
|
assert(nl[0].n_type == (N_TEXT | N_EXT));
|
|
|
|
assert(nl[1].n_type == (N_TEXT | N_EXT));
|
2017-06-25 01:51:30 +02:00
|
|
|
#endif
|
2017-06-14 01:52:12 +02:00
|
|
|
assert(nl[2].n_type == (N_BSS | N_EXT));
|
|
|
|
assert(nl[3].n_type == (N_DATA | N_EXT));
|
|
|
|
assert(nl[4].n_type == (N_DATA));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|