glibc/elf/tst-nodelete-opened.c
Siddhesh Poyarekar b632bdd3f7 Set NODELETE flag when opening already open objects with RTLD_NODELETE
The DF_1_NODELETE flag is set too late when opening a DSO, due to
which, if a DSO is already open, subsequently opening it with
RTLD_NODELETE fails to set the DF_1_NODELETE flag.  This patch fixes
this by setting the flag immediately after bumping the opencount.

Verified on x86_64.

	[BZ #18676]
	* elf/tst-nodelete-opened.c: New test case.
	* elf/tst-nodelete-opened-lib.c: New test case module.
	* elf/Makefile (tests, modules-names): Use them.
	* elf/dl-open.c (dl_open_worker): Set DF_1_NODELETE flag
	early.
2015-07-16 08:53:28 +05:30

68 lines
1.8 KiB
C

/* Verify that an already opened DSO opened agained with RTLD_NODELETE actually
sets the NODELETE flag.
Copyright (C) 2015 Free Software Foundation, Inc.
This file is part of the GNU C Library.
The GNU C Library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
The GNU C Library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with the GNU C Library; if not, see
<http://www.gnu.org/licenses/>. */
#include <dlfcn.h>
#include <stdio.h>
int
do_test (void)
{
void *h1 = dlopen ("$ORIGIN/tst-nodelete-opened-lib.so", RTLD_LAZY);
if (h1 == NULL)
{
printf ("h1: failed to open DSO: %s\n", dlerror ());
return 1;
}
void *h2 = dlopen ("$ORIGIN/tst-nodelete-opened-lib.so",
RTLD_LAZY | RTLD_NODELETE);
if (h2 == NULL)
{
printf ("h2: failed to open DSO: %s\n", dlerror ());
return 1;
}
int *foo = dlsym (h2, "foo_var");
if (foo == NULL)
{
printf ("failed to load symbol foo_var: %s\n", dlerror ());
return 1;
}
if (dlclose (h1) != 0)
{
printf ("h1: dlclose failed: %s\n", dlerror ());
return 1;
}
if (dlclose (h2) != 0)
{
printf ("h2: dlclose failed: %s\n", dlerror ());
return 1;
}
printf ("foo == %d\n", *foo);
return 0;
}
#define TEST_FUNCTION do_test ()
#include "../test-skeleton.c"