mirror of
https://sourceware.org/git/glibc.git
synced 2025-07-28 00:21:52 +03:00
Use IFUNC memmove/memset in x86-64 bcopy/bzero
Also add separate tests for bcopy and bzero.
This commit is contained in:
16
ChangeLog
16
ChangeLog
@ -1,3 +1,19 @@
|
|||||||
|
2012-10-11 H.J. Lu <hongjiu.lu@intel.com>
|
||||||
|
|
||||||
|
* string/Makefile (strop-tests): Add bcopy and bzero.
|
||||||
|
* string/test-bcopy.c: New file.
|
||||||
|
* string/test-bzero.c: Likewise.
|
||||||
|
* string/test-memmove.c: Support bcopy test if TEST_BCOPY is
|
||||||
|
defined.
|
||||||
|
* string/test-memset.c: Support bzero test if TEST_BZERO is
|
||||||
|
defined.
|
||||||
|
* sysdeps/x86_64/multiarch/bcopy.S (bcopy): Jump to
|
||||||
|
__libc_memmove.
|
||||||
|
* sysdeps/x86_64/multiarch/bzero.S (__bzero): Jump to
|
||||||
|
__libc_memset.
|
||||||
|
* sysdeps/x86_64/multiarch/memset.S (__libc_memset): New alias
|
||||||
|
of memset.
|
||||||
|
|
||||||
2012-10-10 Joseph Myers <joseph@codesourcery.com>
|
2012-10-10 Joseph Myers <joseph@codesourcery.com>
|
||||||
|
|
||||||
* configure.in: Run $CXX, not cc1plus, to locate C++ headers.
|
* configure.in: Run $CXX, not cc1plus, to locate C++ headers.
|
||||||
|
@ -49,7 +49,7 @@ strop-tests := memchr memcmp memcpy memmove mempcpy memset memccpy \
|
|||||||
stpcpy stpncpy strcat strchr strcmp strcpy strcspn \
|
stpcpy stpncpy strcat strchr strcmp strcpy strcspn \
|
||||||
strlen strncmp strncpy strpbrk strrchr strspn memmem \
|
strlen strncmp strncpy strpbrk strrchr strspn memmem \
|
||||||
strstr strcasestr strnlen strcasecmp strncasecmp \
|
strstr strcasestr strnlen strcasecmp strncasecmp \
|
||||||
strncat rawmemchr strchrnul
|
strncat rawmemchr strchrnul bcopy bzero
|
||||||
tests := tester inl-tester noinl-tester testcopy test-ffs \
|
tests := tester inl-tester noinl-tester testcopy test-ffs \
|
||||||
tst-strlen stratcliff tst-svc tst-inlcall \
|
tst-strlen stratcliff tst-svc tst-inlcall \
|
||||||
bug-strncat1 bug-strspn1 bug-strpbrk1 tst-bswap \
|
bug-strncat1 bug-strspn1 bug-strpbrk1 tst-bswap \
|
||||||
|
20
string/test-bcopy.c
Normal file
20
string/test-bcopy.c
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
/* Test and measure bcopy functions.
|
||||||
|
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/>. */
|
||||||
|
|
||||||
|
#define TEST_BCOPY
|
||||||
|
#include "test-memmove.c"
|
19
string/test-bzero.c
Normal file
19
string/test-bzero.c
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
/* Test and measure bzero functions.
|
||||||
|
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/>. */
|
||||||
|
#define TEST_BZERO
|
||||||
|
#include "test-memset.c"
|
@ -1,5 +1,5 @@
|
|||||||
/* Test and measure memmove functions.
|
/* Test and measure memmove functions.
|
||||||
Copyright (C) 1999, 2002, 2003 Free Software Foundation, Inc.
|
Copyright (C) 1999-2012 Free Software Foundation, Inc.
|
||||||
This file is part of the GNU C Library.
|
This file is part of the GNU C Library.
|
||||||
Written by Jakub Jelinek <jakub@redhat.com>, 1999.
|
Written by Jakub Jelinek <jakub@redhat.com>, 1999.
|
||||||
|
|
||||||
@ -20,11 +20,26 @@
|
|||||||
#define TEST_MAIN
|
#define TEST_MAIN
|
||||||
#include "test-string.h"
|
#include "test-string.h"
|
||||||
|
|
||||||
typedef char *(*proto_t) (char *, const char *, size_t);
|
|
||||||
char *simple_memmove (char *, const char *, size_t);
|
char *simple_memmove (char *, const char *, size_t);
|
||||||
|
|
||||||
|
#ifdef TEST_BCOPY
|
||||||
|
typedef void (*proto_t) (const char *, char *, size_t);
|
||||||
|
void simple_bcopy (const char *, char *, size_t);
|
||||||
|
|
||||||
|
IMPL (simple_bcopy, 0)
|
||||||
|
IMPL (bcopy, 1)
|
||||||
|
|
||||||
|
void
|
||||||
|
simple_bcopy (const char *src, char *dst, size_t n)
|
||||||
|
{
|
||||||
|
simple_memmove (dst, src, n);
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
typedef char *(*proto_t) (char *, const char *, size_t);
|
||||||
|
|
||||||
IMPL (simple_memmove, 0)
|
IMPL (simple_memmove, 0)
|
||||||
IMPL (memmove, 1)
|
IMPL (memmove, 1)
|
||||||
|
#endif
|
||||||
|
|
||||||
char *
|
char *
|
||||||
simple_memmove (char *dst, const char *src, size_t n)
|
simple_memmove (char *dst, const char *src, size_t n)
|
||||||
@ -47,9 +62,12 @@ static void
|
|||||||
do_one_test (impl_t *impl, char *dst, char *src, const char *orig_src,
|
do_one_test (impl_t *impl, char *dst, char *src, const char *orig_src,
|
||||||
size_t len)
|
size_t len)
|
||||||
{
|
{
|
||||||
|
memcpy (src, orig_src, len);
|
||||||
|
#ifdef TEST_BCOPY
|
||||||
|
CALL (impl, src, dst, len);
|
||||||
|
#else
|
||||||
char *res;
|
char *res;
|
||||||
|
|
||||||
memcpy (src, orig_src, len);
|
|
||||||
res = CALL (impl, dst, src, len);
|
res = CALL (impl, dst, src, len);
|
||||||
if (res != dst)
|
if (res != dst)
|
||||||
{
|
{
|
||||||
@ -58,6 +76,7 @@ do_one_test (impl_t *impl, char *dst, char *src, const char *orig_src,
|
|||||||
ret = 1;
|
ret = 1;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
if (memcmp (dst, orig_src, len) != 0)
|
if (memcmp (dst, orig_src, len) != 0)
|
||||||
{
|
{
|
||||||
@ -77,7 +96,11 @@ do_one_test (impl_t *impl, char *dst, char *src, const char *orig_src,
|
|||||||
for (i = 0; i < 32; ++i)
|
for (i = 0; i < 32; ++i)
|
||||||
{
|
{
|
||||||
HP_TIMING_NOW (start);
|
HP_TIMING_NOW (start);
|
||||||
|
#ifdef TEST_BCOPY
|
||||||
|
CALL (impl, src, dst, len);
|
||||||
|
#else
|
||||||
CALL (impl, dst, src, len);
|
CALL (impl, dst, src, len);
|
||||||
|
#endif
|
||||||
HP_TIMING_NOW (stop);
|
HP_TIMING_NOW (stop);
|
||||||
HP_TIMING_BEST (best_time, start, stop);
|
HP_TIMING_BEST (best_time, start, stop);
|
||||||
}
|
}
|
||||||
@ -123,7 +146,9 @@ do_random_tests (void)
|
|||||||
size_t srcstart, srcend, dststart, dstend;
|
size_t srcstart, srcend, dststart, dstend;
|
||||||
int c;
|
int c;
|
||||||
unsigned char *p1, *p2;
|
unsigned char *p1, *p2;
|
||||||
|
#ifndef TEST_BCOPY
|
||||||
unsigned char *res;
|
unsigned char *res;
|
||||||
|
#endif
|
||||||
|
|
||||||
for (n = 0; n < ITERATIONS; n++)
|
for (n = 0; n < ITERATIONS; n++)
|
||||||
{
|
{
|
||||||
@ -178,6 +203,9 @@ do_random_tests (void)
|
|||||||
{
|
{
|
||||||
memset (p2 + dststart, c, dstend - dststart);
|
memset (p2 + dststart, c, dstend - dststart);
|
||||||
memcpy (p2 + srcstart, p1 + srcstart, srcend - srcstart);
|
memcpy (p2 + srcstart, p1 + srcstart, srcend - srcstart);
|
||||||
|
#ifdef TEST_BCOPY
|
||||||
|
CALL (impl, (char *) (p2 + align1), (char *) (p2 + align2), len);
|
||||||
|
#else
|
||||||
res = (unsigned char *) CALL (impl,
|
res = (unsigned char *) CALL (impl,
|
||||||
(char *) (p2 + align2),
|
(char *) (p2 + align2),
|
||||||
(char *) (p2 + align1), len);
|
(char *) (p2 + align1), len);
|
||||||
@ -187,6 +215,7 @@ do_random_tests (void)
|
|||||||
n, impl->name, align1, align2, len, res, p2 + align2);
|
n, impl->name, align1, align2, len, res, p2 + align2);
|
||||||
ret = 1;
|
ret = 1;
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
if (memcmp (p1 + align1, p2 + align2, len))
|
if (memcmp (p1 + align1, p2 + align2, len))
|
||||||
{
|
{
|
||||||
error (0, 0, "Iteration %zd - different strings, %s (%zd, %zd, %zd)",
|
error (0, 0, "Iteration %zd - different strings, %s (%zd, %zd, %zd)",
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/* Test and measure memset functions.
|
/* Test and measure memset functions.
|
||||||
Copyright (C) 1999, 2002, 2003, 2005 Free Software Foundation, Inc.
|
Copyright (C) 1999-2012 Free Software Foundation, Inc.
|
||||||
This file is part of the GNU C Library.
|
This file is part of the GNU C Library.
|
||||||
Written by Jakub Jelinek <jakub@redhat.com>, 1999.
|
Written by Jakub Jelinek <jakub@redhat.com>, 1999.
|
||||||
|
|
||||||
@ -21,14 +21,43 @@
|
|||||||
#define MIN_PAGE_SIZE 131072
|
#define MIN_PAGE_SIZE 131072
|
||||||
#include "test-string.h"
|
#include "test-string.h"
|
||||||
|
|
||||||
typedef char *(*proto_t) (char *, int, size_t);
|
|
||||||
char *simple_memset (char *, int, size_t);
|
char *simple_memset (char *, int, size_t);
|
||||||
|
|
||||||
|
#ifdef TEST_BZERO
|
||||||
|
typedef void (*proto_t) (char *, size_t);
|
||||||
|
void simple_bzero (char *, size_t);
|
||||||
|
void builtin_bzero (char *, size_t);
|
||||||
|
|
||||||
|
IMPL (simple_bzero, 0)
|
||||||
|
IMPL (builtin_bzero, 0)
|
||||||
|
IMPL (bzero, 1)
|
||||||
|
|
||||||
|
void
|
||||||
|
simple_bzero (char *s, size_t n)
|
||||||
|
{
|
||||||
|
simple_memset (s, 0, n);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
builtin_bzero (char *s, size_t n)
|
||||||
|
{
|
||||||
|
__builtin_bzero (s, n);
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
typedef char *(*proto_t) (char *, int, size_t);
|
||||||
char *builtin_memset (char *, int, size_t);
|
char *builtin_memset (char *, int, size_t);
|
||||||
|
|
||||||
IMPL (simple_memset, 0)
|
IMPL (simple_memset, 0)
|
||||||
IMPL (builtin_memset, 0)
|
IMPL (builtin_memset, 0)
|
||||||
IMPL (memset, 1)
|
IMPL (memset, 1)
|
||||||
|
|
||||||
|
char *
|
||||||
|
builtin_memset (char *s, int c, size_t n)
|
||||||
|
{
|
||||||
|
return __builtin_memset (s, c, n);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
char *
|
char *
|
||||||
simple_memset (char *s, int c, size_t n)
|
simple_memset (char *s, int c, size_t n)
|
||||||
{
|
{
|
||||||
@ -38,20 +67,20 @@ simple_memset (char *s, int c, size_t n)
|
|||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
char *
|
|
||||||
builtin_memset (char *s, int c, size_t n)
|
|
||||||
{
|
|
||||||
return __builtin_memset (s, c, n);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
do_one_test (impl_t *impl, char *s, int c, size_t n)
|
do_one_test (impl_t *impl, char *s, int c __attribute ((unused)), size_t n)
|
||||||
{
|
{
|
||||||
char *res = CALL (impl, s, c, n);
|
|
||||||
char tstbuf[n];
|
char tstbuf[n];
|
||||||
|
#ifdef TEST_BZERO
|
||||||
|
simple_bzero (tstbuf, n);
|
||||||
|
CALL (impl, s, n);
|
||||||
|
if (memcmp (s, tstbuf, n) != 0)
|
||||||
|
#else
|
||||||
|
char *res = CALL (impl, s, c, n);
|
||||||
if (res != s
|
if (res != s
|
||||||
|| simple_memset (tstbuf, c, n) != tstbuf
|
|| simple_memset (tstbuf, c, n) != tstbuf
|
||||||
|| memcmp (s, tstbuf, n) != 0)
|
|| memcmp (s, tstbuf, n) != 0)
|
||||||
|
#endif
|
||||||
{
|
{
|
||||||
error (0, 0, "Wrong result in function %s", impl->name);
|
error (0, 0, "Wrong result in function %s", impl->name);
|
||||||
ret = 1;
|
ret = 1;
|
||||||
@ -68,7 +97,12 @@ do_one_test (impl_t *impl, char *s, int c, size_t n)
|
|||||||
for (i = 0; i < 32; ++i)
|
for (i = 0; i < 32; ++i)
|
||||||
{
|
{
|
||||||
HP_TIMING_NOW (start);
|
HP_TIMING_NOW (start);
|
||||||
|
#ifdef TEST_BZERO
|
||||||
|
CALL (impl, s, n);
|
||||||
|
#else
|
||||||
CALL (impl, s, c, n);
|
CALL (impl, s, c, n);
|
||||||
|
#endif
|
||||||
|
|
||||||
HP_TIMING_NOW (stop);
|
HP_TIMING_NOW (stop);
|
||||||
HP_TIMING_BEST (best_time, start, stop);
|
HP_TIMING_BEST (best_time, start, stop);
|
||||||
}
|
}
|
||||||
@ -94,6 +128,7 @@ do_test (size_t align, int c, size_t len)
|
|||||||
putchar ('\n');
|
putchar ('\n');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifndef TEST_BZERO
|
||||||
static void
|
static void
|
||||||
do_random_tests (void)
|
do_random_tests (void)
|
||||||
{
|
{
|
||||||
@ -178,12 +213,13 @@ do_random_tests (void)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
int
|
int
|
||||||
test_main (void)
|
test_main (void)
|
||||||
{
|
{
|
||||||
size_t i;
|
size_t i;
|
||||||
int c;
|
int c = 0;
|
||||||
|
|
||||||
test_init ();
|
test_init ();
|
||||||
|
|
||||||
@ -192,7 +228,9 @@ test_main (void)
|
|||||||
printf ("\t%s", impl->name);
|
printf ("\t%s", impl->name);
|
||||||
putchar ('\n');
|
putchar ('\n');
|
||||||
|
|
||||||
|
#ifndef TEST_BZERO
|
||||||
for (c = -65; c <= 130; c += 65)
|
for (c = -65; c <= 130; c += 65)
|
||||||
|
#endif
|
||||||
{
|
{
|
||||||
for (i = 0; i < 18; ++i)
|
for (i = 0; i < 18; ++i)
|
||||||
do_test (0, c, 1 << i);
|
do_test (0, c, 1 << i);
|
||||||
@ -208,7 +246,10 @@ test_main (void)
|
|||||||
do_test (2, c, 25);
|
do_test (2, c, 25);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifndef TEST_BZERO
|
||||||
do_random_tests ();
|
do_random_tests ();
|
||||||
|
#endif
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3,5 +3,5 @@
|
|||||||
.text
|
.text
|
||||||
ENTRY(bcopy)
|
ENTRY(bcopy)
|
||||||
xchg %rdi, %rsi
|
xchg %rdi, %rsi
|
||||||
jmp HIDDEN_BUILTIN_JUMPTARGET(memmove)
|
jmp __libc_memmove /* Branch to IFUNC memmove. */
|
||||||
END(bcopy)
|
END(bcopy)
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/* Multiple versions of bzero
|
/* bzero. x86-64 version.
|
||||||
Copyright (C) 2010 Free Software Foundation, Inc.
|
Copyright (C) 2010-2012 Free Software Foundation, Inc.
|
||||||
This file is part of the GNU C Library.
|
This file is part of the GNU C Library.
|
||||||
|
|
||||||
The GNU C Library is free software; you can redistribute it and/or
|
The GNU C Library is free software; you can redistribute it and/or
|
||||||
@ -21,35 +21,8 @@
|
|||||||
|
|
||||||
.text
|
.text
|
||||||
ENTRY(__bzero)
|
ENTRY(__bzero)
|
||||||
.type __bzero, @gnu_indirect_function
|
mov %rsi,%rdx /* Adjust parameter. */
|
||||||
cmpl $0, __cpu_features+KIND_OFFSET(%rip)
|
xorl %esi,%esi /* Fill with 0s. */
|
||||||
jne 1f
|
jmp __libc_memset /* Branch to IFUNC memset. */
|
||||||
call __init_cpu_features
|
|
||||||
1: leaq __bzero_x86_64(%rip), %rax
|
|
||||||
testl $bit_Prefer_SSE_for_memop, __cpu_features+FEATURE_OFFSET+index_Prefer_SSE_for_memop(%rip)
|
|
||||||
jz 2f
|
|
||||||
leaq __bzero_sse2(%rip), %rax
|
|
||||||
2: ret
|
|
||||||
END(__bzero)
|
END(__bzero)
|
||||||
|
|
||||||
.type __bzero_sse2, @function
|
|
||||||
__bzero_sse2:
|
|
||||||
cfi_startproc
|
|
||||||
CALL_MCOUNT
|
|
||||||
mov %rsi,%rdx /* Adjust parameter. */
|
|
||||||
xorl %esi,%esi /* Fill with 0s. */
|
|
||||||
jmp __memset_sse2
|
|
||||||
cfi_endproc
|
|
||||||
.size __bzero_sse2, .-__bzero_sse2
|
|
||||||
|
|
||||||
.type __bzero_x86_64, @function
|
|
||||||
__bzero_x86_64:
|
|
||||||
cfi_startproc
|
|
||||||
CALL_MCOUNT
|
|
||||||
mov %rsi,%rdx /* Adjust parameter. */
|
|
||||||
xorl %esi,%esi /* Fill with 0s. */
|
|
||||||
jmp __memset_x86_64
|
|
||||||
cfi_endproc
|
|
||||||
.size __bzero_x86_64, .-__bzero_x86_64
|
|
||||||
|
|
||||||
weak_alias (__bzero, bzero)
|
weak_alias (__bzero, bzero)
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/* Multiple versions of memset
|
/* Multiple versions of memset
|
||||||
Copyright (C) 2010 Free Software Foundation, Inc.
|
Copyright (C) 2010-2012 Free Software Foundation, Inc.
|
||||||
This file is part of the GNU C Library.
|
This file is part of the GNU C Library.
|
||||||
|
|
||||||
The GNU C Library is free software; you can redistribute it and/or
|
The GNU C Library is free software; you can redistribute it and/or
|
||||||
@ -33,6 +33,11 @@ ENTRY(memset)
|
|||||||
2: ret
|
2: ret
|
||||||
END(memset)
|
END(memset)
|
||||||
|
|
||||||
|
/* Define internal IFUNC memset for bzero. */
|
||||||
|
.globl __libc_memset
|
||||||
|
.hidden __libc_memset
|
||||||
|
__libc_memset = memset
|
||||||
|
|
||||||
# define USE_SSE2 1
|
# define USE_SSE2 1
|
||||||
|
|
||||||
# undef ENTRY
|
# undef ENTRY
|
||||||
|
Reference in New Issue
Block a user