mirror of
https://sourceware.org/git/glibc.git
synced 2025-01-13 12:40:08 +00:00
ccf970c7a7
This patch follows commit 5554304f0
(posix: Allow glob to match dangling
symlinks [BZ #866]) by adding a compat symbol that follow previous
semantic of not following dangling symlinks and thus avoiding call
gl_lstat with GLOB_ALTDIRFUNC.
It avoids failure with old binaries that not set the alternate function
pointer for lstat (GNUmake for instance). The following scenario, for
instance, fails with current GNUmake because glibc will access unitialized
memory when calling gl_lstat:
$ cat src/t/t.c
int main ()
{
return 0;
}
$ cat Makefile
SRC = $(wildcard src/*/t.c)
OBJ = $(patsubst src/%.c, obj/%.o, $(SRC))
prog: $(OBJ)
$(CC) $(CFLAGS) $(LDFLAGS) $(LIBS) $(OBJ) -o prog
obj/%.o: src/%.c
$(CC) $(CFLAGS) -c $< -o $@
$ make
This works as expected with the patch applied. Since it is for generic
ABI, default compat symbols are added with override for Linux due LFS.
Now we have two compat symbols for glob on Linux:
1. sysdeps/unix/sysv/linux/oldglob.c which implements glob64 with
the old dirent layout. For this implementation I also set it to
not follow dangling symlinks (which is the safest path).
2. sysdeps/unix/sysv/linux/glob{64}-lstat-compat.c which implements
the compat symbol for dangling symlinks. As for generic glob,
the implementation uses XSTAT_IS_XSTAT64 to define whether
both __glob_lstat_compat and __glob64_lstat_compat should be
different implementations. For archictures that define
XSTAT_IS_XSTAT64, __glob_lstat_compat is aliased to
__glob64_lstat_compat.
3. sysdeps/unix/sysv/linux/alpha/oldglob.c with a different glob_t
layout. As for 1. this patch changes it to not follow dangling
symlinks.
The patch also bumps _GNU_GLOB_INTERFACE_VERSION to 2 to advertise the
new semantic. On GNUmake, for instance, it will force to it use its
internal glob implementation instead and avoiding triggering the same
failure on builds against newer GLIBCs.
Checked on x86_64-linux-gnu and i686-linux-gnu. I also checked
with a build against the major ABIs required to check for the abilist.
The changes should also work on gnulib (I run gnulib-tool.py check glob
and it shown no regressions).
[BZ #22183]
* include/gnu-versions.h (_GNU_GLOB_INTERFACE_VERSION): Increase
version to 2.
* posix/Makefile (routines): Add glob-lstat-compat and
glob64-lstat-compat.
* posix/Versions (GLIBC_2.27, glob, glob64): Add symbol version.
* posix/glob-lstat-compat.c: New file.
* posix/glob64-lstat-compat.c: Likewise.
* posix/tst-glob_lstat_compat.c: Likewise.
* sysdeps/unix/sysv/linux/glob-lstat-compat.c: Likewise.
* sysdeps/unix/sysv/linux/alpha/glob-lstat-compat.c: Likewise.
* sysdeps/unix/sysv/linux/glob64-lstat-compat.c: Likewise.
* sysdeps/unix/sysv/linux/alpha/glob.c: Remove file.
* posix/glob.c (glob_lstat): New function.
(glob): Rename to __glob and add versioned symbol to 2.27.
(glob_in_dir): Use glob_lstat.
* posix/glob64.c (glob64): Add GLOB_ATTRIBUTE.
* sysdeps/unix/sysv/linux/arm/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/glob.c (glob): Add versioned symbol for
2.27.
* sysdeps/unix/sysv/linux/glob64.c (glob64): Likewise.
* sysdeps/unix/sysv/linux/oldglob.c (GLOB_NO_LSTAT): Define.
* sysdeps/unix/sysv/linux/alpha/oldglob.c (__old_glob): Do not use
gl_lstat on glob call.
* sysdeps/unix/sysv/linux/aarch64/libc.abilist: Add GLIBC_2.27 glob
and glob64 symbols.
* sysdeps/unix/sysv/linux/alpha/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/hppa/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/i386/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/ia64/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/m68k/coldfire/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/m68k/m680x0/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/microblaze/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/mips/mips32/fpu/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/mips/mips32/nofpu/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/mips/mips64/n32/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/mips/mips64/n64/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/nios2/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/powerpc/powerpc32/fpu/libc.abilist:
Likewise.
* sysdeps/unix/linux/powerpc/powerpc32/nofpu/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/powerpc/powerpc64/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/s390/s390-32/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/s390/s390-64/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/sh/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/sparc/sparc32/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/sparc/sparc64/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/tile/tilegx/tilegx32/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/tile/tilegx/tilegx64/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/tile/tilepro/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/x86_64/64/libc.abilist: Likewise.
* sysdeps/unix/sysv/linux/x86_64/x32/libc.abilist: Likewise.
264 lines
5.3 KiB
C
264 lines
5.3 KiB
C
/* Test glob compat symbol which avoid call GLOB_ALTDIRFUNC/gl_lstat.
|
|
Copyright (C) 2017 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 <glob.h>
|
|
#include <errno.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
#include <dirent.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <shlib-compat.h>
|
|
#include <support/check.h>
|
|
#include <support/temp_file.h>
|
|
|
|
#if TEST_COMPAT (libc, GLIBC_2_0, GLIBC_2_27)
|
|
|
|
__typeof (glob) glob;
|
|
compat_symbol_reference (libc, glob, glob, GLIBC_2_0);
|
|
|
|
/* Compat glob should not call gl_lstat since for some old binaries it
|
|
might be unitialized (for instance GNUmake). Check if it is indeed
|
|
not called. */
|
|
static bool stat_called;
|
|
static bool lstat_called;
|
|
|
|
static struct
|
|
{
|
|
const char *name;
|
|
int level;
|
|
int type;
|
|
} filesystem[] =
|
|
{
|
|
{ ".", 1, DT_DIR },
|
|
{ "..", 1, DT_DIR },
|
|
{ "dir1lev1", 1, DT_UNKNOWN },
|
|
{ ".", 2, DT_DIR },
|
|
{ "..", 2, DT_DIR },
|
|
{ "file1lev2", 2, DT_REG },
|
|
{ "file2lev2", 2, DT_REG },
|
|
};
|
|
static const size_t nfiles = sizeof (filesystem) / sizeof (filesystem [0]);
|
|
|
|
typedef struct
|
|
{
|
|
int level;
|
|
int idx;
|
|
struct dirent d;
|
|
char room_for_dirent[NAME_MAX];
|
|
} my_DIR;
|
|
|
|
static long int
|
|
find_file (const char *s)
|
|
{
|
|
int level = 1;
|
|
long int idx = 0;
|
|
|
|
while (s[0] == '/')
|
|
{
|
|
if (s[1] == '\0')
|
|
{
|
|
s = ".";
|
|
break;
|
|
}
|
|
++s;
|
|
}
|
|
|
|
if (strcmp (s, ".") == 0)
|
|
return 0;
|
|
|
|
if (s[0] == '.' && s[1] == '/')
|
|
s += 2;
|
|
|
|
while (*s != '\0')
|
|
{
|
|
char *endp = strchrnul (s, '/');
|
|
|
|
while (idx < nfiles && filesystem[idx].level >= level)
|
|
{
|
|
if (filesystem[idx].level == level
|
|
&& memcmp (s, filesystem[idx].name, endp - s) == 0
|
|
&& filesystem[idx].name[endp - s] == '\0')
|
|
break;
|
|
++idx;
|
|
}
|
|
|
|
if (idx == nfiles || filesystem[idx].level < level)
|
|
{
|
|
errno = ENOENT;
|
|
return -1;
|
|
}
|
|
|
|
if (*endp == '\0')
|
|
return idx + 1;
|
|
|
|
if (filesystem[idx].type != DT_DIR
|
|
&& (idx + 1 >= nfiles
|
|
|| filesystem[idx].level >= filesystem[idx + 1].level))
|
|
{
|
|
errno = ENOTDIR;
|
|
return -1;
|
|
}
|
|
|
|
++idx;
|
|
|
|
s = endp + 1;
|
|
++level;
|
|
}
|
|
|
|
errno = ENOENT;
|
|
return -1;
|
|
}
|
|
|
|
static void *
|
|
my_opendir (const char *s)
|
|
{
|
|
long int idx = find_file (s);
|
|
if (idx == -1 || filesystem[idx].type != DT_DIR)
|
|
return NULL;
|
|
|
|
my_DIR *dir = malloc (sizeof (my_DIR));
|
|
if (dir == NULL)
|
|
FAIL_EXIT1 ("cannot allocate directory handle");
|
|
|
|
dir->level = filesystem[idx].level;
|
|
dir->idx = idx;
|
|
|
|
return dir;
|
|
}
|
|
|
|
static struct dirent *
|
|
my_readdir (void *gdir)
|
|
{
|
|
my_DIR *dir = gdir;
|
|
|
|
if (dir->idx == -1)
|
|
return NULL;
|
|
|
|
while (dir->idx < nfiles && filesystem[dir->idx].level > dir->level)
|
|
++dir->idx;
|
|
|
|
if (dir->idx == nfiles || filesystem[dir->idx].level < dir->level)
|
|
{
|
|
dir->idx = -1;
|
|
return NULL;
|
|
}
|
|
|
|
dir->d.d_ino = 1; /* glob should not skip this entry. */
|
|
|
|
#ifdef _DIRENT_HAVE_D_TYPE
|
|
dir->d.d_type = filesystem[dir->idx].type;
|
|
#endif
|
|
|
|
strcpy (dir->d.d_name, filesystem[dir->idx].name);
|
|
|
|
++dir->idx;
|
|
|
|
return &dir->d;
|
|
}
|
|
|
|
static void
|
|
my_closedir (void *dir)
|
|
{
|
|
free (dir);
|
|
}
|
|
|
|
static int
|
|
my_stat (const char *name, struct stat *st)
|
|
{
|
|
stat_called = true;
|
|
|
|
long int idx = find_file (name);
|
|
if (idx == -1)
|
|
return -1;
|
|
|
|
memset (st, '\0', sizeof (*st));
|
|
|
|
if (filesystem[idx].type == DT_UNKNOWN)
|
|
st->st_mode = DTTOIF (idx + 1 < nfiles
|
|
&& filesystem[idx].level < filesystem[idx + 1].level
|
|
? DT_DIR : DT_REG) | 0777;
|
|
else
|
|
st->st_mode = DTTOIF (filesystem[idx].type) | 0777;
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
my_lstat (const char *name, struct stat *st)
|
|
{
|
|
lstat_called = true;
|
|
|
|
long int idx = find_file (name);
|
|
if (idx == -1)
|
|
return -1;
|
|
|
|
memset (st, '\0', sizeof (*st));
|
|
|
|
if (filesystem[idx].type == DT_UNKNOWN)
|
|
st->st_mode = DTTOIF (idx + 1 < nfiles
|
|
&& filesystem[idx].level < filesystem[idx + 1].level
|
|
? DT_DIR : DT_REG) | 0777;
|
|
else
|
|
st->st_mode = DTTOIF (filesystem[idx].type) | 0777;
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
do_test (void)
|
|
{
|
|
glob_t gl;
|
|
|
|
memset (&gl, '\0', sizeof (gl));
|
|
|
|
gl.gl_closedir = my_closedir;
|
|
gl.gl_readdir = my_readdir;
|
|
gl.gl_opendir = my_opendir;
|
|
gl.gl_lstat = my_lstat;
|
|
gl.gl_stat = my_stat;
|
|
|
|
int flags = GLOB_ALTDIRFUNC;
|
|
|
|
stat_called = false;
|
|
lstat_called = false;
|
|
|
|
TEST_VERIFY_EXIT (glob ("*/file1lev2", flags, NULL, &gl) == 0);
|
|
TEST_VERIFY_EXIT (gl.gl_pathc == 1);
|
|
TEST_VERIFY_EXIT (strcmp (gl.gl_pathv[0], "dir1lev1/file1lev2") == 0);
|
|
|
|
TEST_VERIFY_EXIT (stat_called == true);
|
|
TEST_VERIFY_EXIT (lstat_called == false);
|
|
|
|
return 0;
|
|
}
|
|
|
|
#else /* TEST_COMPAT (libc, GLIBC_2_0, GLIBC_2_27) */
|
|
|
|
static int
|
|
do_test (void)
|
|
{
|
|
return 77;
|
|
}
|
|
#endif
|
|
|
|
#include <support/test-driver.c>
|