mirror of
https://sourceware.org/git/glibc.git
synced 2024-11-24 14:00:30 +00:00
Framework to test IFUNC implementations on target
This commit is contained in:
parent
9a387d1f78
commit
11dd4af68c
14
ChangeLog
14
ChangeLog
@ -1,5 +1,19 @@
|
|||||||
2012-10-11 H.J. Lu <hongjiu.lu@intel.com>
|
2012-10-11 H.J. Lu <hongjiu.lu@intel.com>
|
||||||
|
|
||||||
|
* Rules [$(multi-arch) = no] (tests): Filter out $(tests-ifunc).
|
||||||
|
[$(multi-arch) = no] (xtests): Filter out $(xtests-ifunc).
|
||||||
|
* include/ifunc-impl-list.h: New file.
|
||||||
|
* misc/ifunc-impl-list.c: Likewise.
|
||||||
|
* misc/Makefile (routines): Add ifunc-impl-list.
|
||||||
|
* misc/Versions (GLIBC_PRIVATE): Add __libc_ifunc_impl_list.
|
||||||
|
* string/test-string.h: Include <ifunc-impl-list.h>.
|
||||||
|
[TEST_IFUNC && TEST_NAME] (func_list, func_count, impl_count,
|
||||||
|
impl_array): New variables.
|
||||||
|
(FOR_EACH_IMPL): Support func_list if TEST_IFUNC and TEST_NAME
|
||||||
|
are defined.
|
||||||
|
(test_init): Call __libc_ifunc_impl_list to initialize
|
||||||
|
func_list if TEST_IFUNC and TEST_NAME are defined.
|
||||||
|
|
||||||
* string/Makefile (strop-tests): Add bcopy and bzero.
|
* string/Makefile (strop-tests): Add bcopy and bzero.
|
||||||
* string/test-bcopy.c: New file.
|
* string/test-bcopy.c: New file.
|
||||||
* string/test-bzero.c: Likewise.
|
* string/test-bzero.c: Likewise.
|
||||||
|
5
Rules
5
Rules
@ -84,6 +84,11 @@ common-generated += dummy.o dummy.c
|
|||||||
# This makes all the auxiliary and test programs.
|
# This makes all the auxiliary and test programs.
|
||||||
|
|
||||||
.PHONY: others tests
|
.PHONY: others tests
|
||||||
|
ifeq ($(multi-arch),no)
|
||||||
|
tests := $(filter-out $(tests-ifunc), $(tests))
|
||||||
|
xtests := $(filter-out $(xtests-ifunc), $(xtests))
|
||||||
|
endif
|
||||||
|
|
||||||
ifeq ($(build-programs),yes)
|
ifeq ($(build-programs),yes)
|
||||||
others: $(addprefix $(objpfx),$(others) $(sysdep-others) $(extra-objs))
|
others: $(addprefix $(objpfx),$(others) $(sysdep-others) $(extra-objs))
|
||||||
else
|
else
|
||||||
|
56
include/ifunc-impl-list.h
Normal file
56
include/ifunc-impl-list.h
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
/* Internal header file for __libc_supported_implementations.
|
||||||
|
Copyright (C) 2012 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/>. */
|
||||||
|
|
||||||
|
#ifndef _IFUNC_IMPL_LIST_H
|
||||||
|
#define _IFUNC_IMPL_LIST_H 1
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <stddef.h>
|
||||||
|
|
||||||
|
struct libc_ifunc_impl
|
||||||
|
{
|
||||||
|
/* The name of function to be tested. */
|
||||||
|
const char *name;
|
||||||
|
/* The address of function to be tested. */
|
||||||
|
void (*fn) (void);
|
||||||
|
/* True if this implementation is usable on this machine. */
|
||||||
|
bool usable;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Add an IFUNC implementation, IMPL, for function FUNC, to ARRAY with
|
||||||
|
USABLE at index I and advance I by one. */
|
||||||
|
#define IFUNC_IMPL_ADD(array, i, func, usable, impl) \
|
||||||
|
extern __typeof (func) impl attribute_hidden; \
|
||||||
|
(array)[i++] = (struct libc_ifunc_impl) { #impl, (void (*) (void)) impl, (usable) };
|
||||||
|
|
||||||
|
/* Return the number of IFUNC implementations, N, for function FUNC if
|
||||||
|
string NAME matches FUNC. */
|
||||||
|
#define IFUNC_IMPL(n, name, func, ...) \
|
||||||
|
if (strcmp (name, #func) == 0) \
|
||||||
|
{ \
|
||||||
|
__VA_ARGS__; \
|
||||||
|
return n; \
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Fill ARRAY of MAX elements with IFUNC implementations for function
|
||||||
|
NAME and return the number of valid entries. */
|
||||||
|
extern size_t __libc_ifunc_impl_list (const char *name,
|
||||||
|
struct libc_ifunc_impl *array,
|
||||||
|
size_t max);
|
||||||
|
|
||||||
|
#endif /* ifunc-impl-list.h */
|
@ -65,7 +65,7 @@ routines := brk sbrk sstk ioctl \
|
|||||||
getloadavg getclktck \
|
getloadavg getclktck \
|
||||||
fgetxattr flistxattr fremovexattr fsetxattr getxattr \
|
fgetxattr flistxattr fremovexattr fsetxattr getxattr \
|
||||||
listxattr lgetxattr llistxattr lremovexattr lsetxattr \
|
listxattr lgetxattr llistxattr lremovexattr lsetxattr \
|
||||||
removexattr setxattr getauxval
|
removexattr setxattr getauxval ifunc-impl-list
|
||||||
|
|
||||||
generated := tst-error1.mtrace tst-error1-mem
|
generated := tst-error1.mtrace tst-error1-mem
|
||||||
|
|
||||||
|
@ -151,5 +151,6 @@ libc {
|
|||||||
}
|
}
|
||||||
GLIBC_PRIVATE {
|
GLIBC_PRIVATE {
|
||||||
__madvise;
|
__madvise;
|
||||||
|
__libc_ifunc_impl_list;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
32
misc/ifunc-impl-list.c
Normal file
32
misc/ifunc-impl-list.c
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
/* Enumerate available IFUNC implementations of a function. Stub version.
|
||||||
|
Copyright (C) 2012 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 <ifunc-impl-list.h>
|
||||||
|
|
||||||
|
/* Fill ARRAY of MAX elements with IFUNC implementations for function
|
||||||
|
NAME supported on target machine and return the number of valid
|
||||||
|
entries. */
|
||||||
|
|
||||||
|
size_t
|
||||||
|
__libc_ifunc_impl_list
|
||||||
|
(const char *name __attribute__ ((unused)),
|
||||||
|
struct libc_ifunc_impl *array __attribute__ ((unused)),
|
||||||
|
size_t max __attribute__ ((unused)))
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
@ -50,6 +50,7 @@ extern impl_t __start_impls[], __stop_impls[];
|
|||||||
#include <error.h>
|
#include <error.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
|
#include <ifunc-impl-list.h>
|
||||||
#define GL(x) _##x
|
#define GL(x) _##x
|
||||||
#define GLRO(x) _##x
|
#define GLRO(x) _##x
|
||||||
#include <hp-timing.h>
|
#include <hp-timing.h>
|
||||||
@ -106,9 +107,57 @@ size_t iterations = 100000;
|
|||||||
#define CALL(impl, ...) \
|
#define CALL(impl, ...) \
|
||||||
(* (proto_t) (impl)->fn) (__VA_ARGS__)
|
(* (proto_t) (impl)->fn) (__VA_ARGS__)
|
||||||
|
|
||||||
#define FOR_EACH_IMPL(impl, notall) \
|
#if defined TEST_IFUNC && defined TEST_NAME
|
||||||
|
/* Increase size of FUNC_LIST if assert is triggered at run-time. */
|
||||||
|
static struct libc_ifunc_impl func_list[32];
|
||||||
|
static int func_count;
|
||||||
|
static int impl_count = -1;
|
||||||
|
static impl_t *impl_array;
|
||||||
|
|
||||||
|
# define FOR_EACH_IMPL(impl, notall) \
|
||||||
|
impl_t *impl; \
|
||||||
|
int count; \
|
||||||
|
if (impl_count == -1) \
|
||||||
|
{ \
|
||||||
|
impl_count = 0; \
|
||||||
|
if (func_count != 0) \
|
||||||
|
{ \
|
||||||
|
int f; \
|
||||||
|
impl_t *skip = NULL, *a; \
|
||||||
|
for (impl = __start_impls; impl < __stop_impls; ++impl) \
|
||||||
|
if (strcmp (impl->name, TEST_NAME) == 0) \
|
||||||
|
skip = impl; \
|
||||||
|
else \
|
||||||
|
impl_count++; \
|
||||||
|
a = impl_array = malloc ((impl_count + func_count) * \
|
||||||
|
sizeof (impl_t)); \
|
||||||
|
for (impl = __start_impls; impl < __stop_impls; ++impl) \
|
||||||
|
if (impl != skip) \
|
||||||
|
*a++ = *impl; \
|
||||||
|
for (f = 0; f < func_count; f++) \
|
||||||
|
if (func_list[f].usable) \
|
||||||
|
{ \
|
||||||
|
a->name = func_list[f].name; \
|
||||||
|
a->fn = func_list[f].fn; \
|
||||||
|
a->test = 1; \
|
||||||
|
a++; \
|
||||||
|
} \
|
||||||
|
impl_count = a - impl_array; \
|
||||||
|
} \
|
||||||
|
else \
|
||||||
|
{ \
|
||||||
|
impl_count = __stop_impls - __start_impls; \
|
||||||
|
impl_array = __start_impls; \
|
||||||
|
} \
|
||||||
|
} \
|
||||||
|
impl = impl_array; \
|
||||||
|
for (count = 0; count < impl_count; ++count, ++impl) \
|
||||||
|
if (!notall || impl->test)
|
||||||
|
#else
|
||||||
|
# define FOR_EACH_IMPL(impl, notall) \
|
||||||
for (impl_t *impl = __start_impls; impl < __stop_impls; ++impl) \
|
for (impl_t *impl = __start_impls; impl < __stop_impls; ++impl) \
|
||||||
if (!notall || impl->test)
|
if (!notall || impl->test)
|
||||||
|
#endif
|
||||||
|
|
||||||
#define HP_TIMING_BEST(best_time, start, end) \
|
#define HP_TIMING_BEST(best_time, start, end) \
|
||||||
do \
|
do \
|
||||||
@ -127,6 +176,12 @@ size_t iterations = 100000;
|
|||||||
static void
|
static void
|
||||||
test_init (void)
|
test_init (void)
|
||||||
{
|
{
|
||||||
|
#if defined TEST_IFUNC && defined TEST_NAME
|
||||||
|
func_count = __libc_ifunc_impl_list (TEST_NAME, func_list,
|
||||||
|
(sizeof func_list
|
||||||
|
/ sizeof func_list[0]));
|
||||||
|
#endif
|
||||||
|
|
||||||
page_size = 2 * getpagesize ();
|
page_size = 2 * getpagesize ();
|
||||||
#ifdef MIN_PAGE_SIZE
|
#ifdef MIN_PAGE_SIZE
|
||||||
if (page_size < MIN_PAGE_SIZE)
|
if (page_size < MIN_PAGE_SIZE)
|
||||||
|
Loading…
Reference in New Issue
Block a user