|
|
00db10 |
From 111012ea9031349ebbd7cc992b71662bdd10397c Mon Sep 17 00:00:00 2001
|
|
|
00db10 |
From: Stefan Liebler <stli@linux.vnet.ibm.com>
|
|
|
00db10 |
Date: Thu, 8 Oct 2015 11:18:25 +0200
|
|
|
00db10 |
Subject: [PATCH 18/30] S390: Optimize strcmp and wcscmp.
|
|
|
00db10 |
|
|
|
00db10 |
upstream-commit-id: 63724a6db60f98e91da474d11d83a19aa10fc54e
|
|
|
00db10 |
https://www.sourceware.org/ml/libc-alpha/2015-07/msg00084.html
|
|
|
00db10 |
|
|
|
00db10 |
This patch provides optimized versions of strcmp and wcscmp with the z13
|
|
|
00db10 |
vector instructions.
|
|
|
00db10 |
|
|
|
00db10 |
The architecture specific string.h had a typo, which leads to ommiting the
|
|
|
00db10 |
inline version in this file if __USE_STRING_INLINES is defined.
|
|
|
00db10 |
Tested this inline version by tweaking test-strcmp.c.
|
|
|
00db10 |
|
|
|
00db10 |
ChangeLog:
|
|
|
00db10 |
|
|
|
00db10 |
* sysdeps/s390/multiarch/strcmp-vx.S: New File.
|
|
|
00db10 |
* sysdeps/s390/multiarch/strcmp.c: Likewise.
|
|
|
00db10 |
* sysdeps/s390/multiarch/wcscmp-c.c: Likewise.
|
|
|
00db10 |
* sysdeps/s390/multiarch/wcscmp-vx.S: Likewise.
|
|
|
00db10 |
* sysdeps/s390/multiarch/wcscmp.c: Likewise.
|
|
|
00db10 |
* sysdeps/s390/s390-32/multiarch/strcmp.c: Likewise.
|
|
|
00db10 |
* sysdeps/s390/s390-64/multiarch/strcmp.c: Likewise.
|
|
|
00db10 |
* sysdeps/s390/multiarch/Makefile (sysdep_routines): Add strcmp and
|
|
|
00db10 |
wcscmp functions.
|
|
|
00db10 |
* sysdeps/s390/multiarch/ifunc-impl-list.c
|
|
|
00db10 |
(__libc_ifunc_impl_list): Add ifunc test for strcmp, wcscmp.
|
|
|
00db10 |
* string/strcmp.c (STRCMP): Define and use macro.
|
|
|
00db10 |
* benchtests/bench-wcscmp.c: New File.
|
|
|
00db10 |
* benchtests/Makefile (wcsmbs-bench): Add wcscmp.
|
|
|
00db10 |
* sysdeps/s390/bits/string.h: Fix typo: _HAVE_STRING_ARCH_strcmp
|
|
|
00db10 |
instead of _HAVE_STRING_ARCH_memchr.
|
|
|
00db10 |
---
|
|
|
00db10 |
benchtests/Makefile | 3 +-
|
|
|
00db10 |
benchtests/bench-wcscmp.c | 20 +++++
|
|
|
00db10 |
string/strcmp.c | 6 +-
|
|
|
00db10 |
sysdeps/s390/bits/string.h | 4 +-
|
|
|
00db10 |
sysdeps/s390/multiarch/Makefile | 6 +-
|
|
|
00db10 |
sysdeps/s390/multiarch/ifunc-impl-list.c | 3 +
|
|
|
00db10 |
sysdeps/s390/multiarch/strcmp-vx.S | 116 +++++++++++++++++++++++++++
|
|
|
00db10 |
sysdeps/s390/multiarch/strcmp.c | 26 ++++++
|
|
|
00db10 |
sysdeps/s390/multiarch/wcscmp-c.c | 30 +++++++
|
|
|
00db10 |
sysdeps/s390/multiarch/wcscmp-vx.S | 131 +++++++++++++++++++++++++++++++
|
|
|
00db10 |
sysdeps/s390/multiarch/wcscmp.c | 27 +++++++
|
|
|
00db10 |
sysdeps/s390/s390-32/multiarch/strcmp.c | 21 +++++
|
|
|
00db10 |
sysdeps/s390/s390-64/multiarch/strcmp.c | 21 +++++
|
|
|
00db10 |
13 files changed, 408 insertions(+), 6 deletions(-)
|
|
|
00db10 |
create mode 100644 benchtests/bench-wcscmp.c
|
|
|
00db10 |
create mode 100644 sysdeps/s390/multiarch/strcmp-vx.S
|
|
|
00db10 |
create mode 100644 sysdeps/s390/multiarch/strcmp.c
|
|
|
00db10 |
create mode 100644 sysdeps/s390/multiarch/wcscmp-c.c
|
|
|
00db10 |
create mode 100644 sysdeps/s390/multiarch/wcscmp-vx.S
|
|
|
00db10 |
create mode 100644 sysdeps/s390/multiarch/wcscmp.c
|
|
|
00db10 |
create mode 100644 sysdeps/s390/s390-32/multiarch/strcmp.c
|
|
|
00db10 |
create mode 100644 sysdeps/s390/s390-64/multiarch/strcmp.c
|
|
|
00db10 |
|
|
|
00db10 |
diff --git a/benchtests/Makefile b/benchtests/Makefile
|
|
|
00db10 |
index fd8b7ee..f6333eb 100644
|
|
|
00db10 |
--- a/benchtests/Makefile
|
|
|
00db10 |
+++ b/benchtests/Makefile
|
|
|
00db10 |
@@ -38,7 +38,8 @@ string-bench := bcopy bzero memccpy memchr memcmp memcpy memmem memmove \
|
|
|
00db10 |
strcat strchr strchrnul strcmp strcpy strcspn strlen \
|
|
|
00db10 |
strncasecmp strncat strncmp strncpy strnlen strpbrk strrchr \
|
|
|
00db10 |
strspn strstr strcpy_chk stpcpy_chk memrchr strsep strtok
|
|
|
00db10 |
-wcsmbs-bench := wcslen wcsnlen wcscpy wcpcpy wcsncpy wcpncpy wcscat wcsncat
|
|
|
00db10 |
+wcsmbs-bench := wcslen wcsnlen wcscpy wcpcpy wcsncpy wcpncpy wcscat wcsncat \
|
|
|
00db10 |
+ wcsncmp
|
|
|
00db10 |
string-bench-all := $(string-bench) ${wcsmbs-bench}
|
|
|
00db10 |
|
|
|
00db10 |
stdlib-bench := strtod
|
|
|
00db10 |
diff --git a/benchtests/bench-wcscmp.c b/benchtests/bench-wcscmp.c
|
|
|
00db10 |
new file mode 100644
|
|
|
00db10 |
index 0000000..bd483a2
|
|
|
00db10 |
--- /dev/null
|
|
|
00db10 |
+++ b/benchtests/bench-wcscmp.c
|
|
|
00db10 |
@@ -0,0 +1,20 @@
|
|
|
00db10 |
+/* Measure wcscmp functions.
|
|
|
00db10 |
+ Copyright (C) 2015 Free Software Foundation, Inc.
|
|
|
00db10 |
+ This file is part of the GNU C Library.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is free software; you can redistribute it and/or
|
|
|
00db10 |
+ modify it under the terms of the GNU Lesser General Public
|
|
|
00db10 |
+ License as published by the Free Software Foundation; either
|
|
|
00db10 |
+ version 2.1 of the License, or (at your option) any later version.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is distributed in the hope that it will be useful,
|
|
|
00db10 |
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
00db10 |
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
00db10 |
+ Lesser General Public License for more details.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ You should have received a copy of the GNU Lesser General Public
|
|
|
00db10 |
+ License along with the GNU C Library; if not, see
|
|
|
00db10 |
+ <http://www.gnu.org/licenses/>. */
|
|
|
00db10 |
+
|
|
|
00db10 |
+#define WIDE 1
|
|
|
00db10 |
+#include "bench-strcmp.c"
|
|
|
00db10 |
diff --git a/string/strcmp.c b/string/strcmp.c
|
|
|
00db10 |
index 8229d7c..5384db9 100644
|
|
|
00db10 |
--- a/string/strcmp.c
|
|
|
00db10 |
+++ b/string/strcmp.c
|
|
|
00db10 |
@@ -20,11 +20,15 @@
|
|
|
00db10 |
|
|
|
00db10 |
#undef strcmp
|
|
|
00db10 |
|
|
|
00db10 |
+#ifndef STRCMP
|
|
|
00db10 |
+# define STRCMP strcmp
|
|
|
00db10 |
+#endif
|
|
|
00db10 |
+
|
|
|
00db10 |
/* Compare S1 and S2, returning less than, equal to or
|
|
|
00db10 |
greater than zero if S1 is lexicographically less than,
|
|
|
00db10 |
equal to or greater than S2. */
|
|
|
00db10 |
int
|
|
|
00db10 |
-strcmp (p1, p2)
|
|
|
00db10 |
+STRCMP (p1, p2)
|
|
|
00db10 |
const char *p1;
|
|
|
00db10 |
const char *p2;
|
|
|
00db10 |
{
|
|
|
00db10 |
diff --git a/sysdeps/s390/bits/string.h b/sysdeps/s390/bits/string.h
|
|
|
00db10 |
index f3070f1..037aa71 100644
|
|
|
00db10 |
--- a/sysdeps/s390/bits/string.h
|
|
|
00db10 |
+++ b/sysdeps/s390/bits/string.h
|
|
|
00db10 |
@@ -226,8 +226,8 @@ memchr (const void *__str, int __c, size_t __n)
|
|
|
00db10 |
}
|
|
|
00db10 |
#endif
|
|
|
00db10 |
|
|
|
00db10 |
-/* Search N bytes of S for C. */
|
|
|
00db10 |
-#define _HAVE_STRING_ARCH_memchr 1
|
|
|
00db10 |
+/* Compare S1 and S2. */
|
|
|
00db10 |
+#define _HAVE_STRING_ARCH_strchr 1
|
|
|
00db10 |
#ifndef _FORCE_INLINES
|
|
|
00db10 |
__STRING_INLINE int
|
|
|
00db10 |
strcmp (const char *__s1, const char *__s2)
|
|
|
00db10 |
diff --git a/sysdeps/s390/multiarch/Makefile b/sysdeps/s390/multiarch/Makefile
|
|
|
00db10 |
index 33c1398..d8fbd55 100644
|
|
|
00db10 |
--- a/sysdeps/s390/multiarch/Makefile
|
|
|
00db10 |
+++ b/sysdeps/s390/multiarch/Makefile
|
|
|
00db10 |
@@ -6,7 +6,8 @@ sysdep_routines += strlen strlen-vx strlen-c \
|
|
|
00db10 |
strncpy strncpy-vx \
|
|
|
00db10 |
stpncpy stpncpy-vx stpncpy-c \
|
|
|
00db10 |
strcat strcat-vx strcat-c \
|
|
|
00db10 |
- strncat strncat-vx strncat-c
|
|
|
00db10 |
+ strncat strncat-vx strncat-c \
|
|
|
00db10 |
+ strcmp strcmp-vx
|
|
|
00db10 |
endif
|
|
|
00db10 |
|
|
|
00db10 |
ifeq ($(subdir),wcsmbs)
|
|
|
00db10 |
@@ -17,5 +18,6 @@ sysdep_routines += wcslen wcslen-vx wcslen-c \
|
|
|
00db10 |
wcsncpy wcsncpy-vx wcsncpy-c \
|
|
|
00db10 |
wcpncpy wcpncpy-vx wcpncpy-c \
|
|
|
00db10 |
wcscat wcscat-vx wcscat-c \
|
|
|
00db10 |
- wcsncat wcsncat-vx wcsncat-c
|
|
|
00db10 |
+ wcsncat wcsncat-vx wcsncat-c \
|
|
|
00db10 |
+ wcscmp wcscmp-vx wcscmp-c
|
|
|
00db10 |
endif
|
|
|
00db10 |
diff --git a/sysdeps/s390/multiarch/ifunc-impl-list.c b/sysdeps/s390/multiarch/ifunc-impl-list.c
|
|
|
00db10 |
index 1e57c0e..196d3ec 100644
|
|
|
00db10 |
--- a/sysdeps/s390/multiarch/ifunc-impl-list.c
|
|
|
00db10 |
+++ b/sysdeps/s390/multiarch/ifunc-impl-list.c
|
|
|
00db10 |
@@ -103,6 +103,9 @@ __libc_ifunc_impl_list (const char *name, struct libc_ifunc_impl *array,
|
|
|
00db10 |
IFUNC_VX_IMPL (strncat);
|
|
|
00db10 |
IFUNC_VX_IMPL (wcsncat);
|
|
|
00db10 |
|
|
|
00db10 |
+ IFUNC_VX_IMPL (strcmp);
|
|
|
00db10 |
+ IFUNC_VX_IMPL (wcscmp);
|
|
|
00db10 |
+
|
|
|
00db10 |
#endif /* HAVE_S390_VX_ASM_SUPPORT */
|
|
|
00db10 |
|
|
|
00db10 |
return i;
|
|
|
00db10 |
diff --git a/sysdeps/s390/multiarch/strcmp-vx.S b/sysdeps/s390/multiarch/strcmp-vx.S
|
|
|
00db10 |
new file mode 100644
|
|
|
00db10 |
index 0000000..57fce75
|
|
|
00db10 |
--- /dev/null
|
|
|
00db10 |
+++ b/sysdeps/s390/multiarch/strcmp-vx.S
|
|
|
00db10 |
@@ -0,0 +1,116 @@
|
|
|
00db10 |
+/* Vector optimized 32/64 bit S/390 version of strcmp.
|
|
|
00db10 |
+ Copyright (C) 2015 Free Software Foundation, Inc.
|
|
|
00db10 |
+ This file is part of the GNU C Library.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is free software; you can redistribute it and/or
|
|
|
00db10 |
+ modify it under the terms of the GNU Lesser General Public
|
|
|
00db10 |
+ License as published by the Free Software Foundation; either
|
|
|
00db10 |
+ version 2.1 of the License, or (at your option) any later version.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is distributed in the hope that it will be useful,
|
|
|
00db10 |
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
00db10 |
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
00db10 |
+ Lesser General Public License for more details.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ You should have received a copy of the GNU Lesser General Public
|
|
|
00db10 |
+ License along with the GNU C Library; if not, see
|
|
|
00db10 |
+ <http://www.gnu.org/licenses/>. */
|
|
|
00db10 |
+
|
|
|
00db10 |
+#if defined HAVE_S390_VX_ASM_SUPPORT && !defined NOT_IN_libc
|
|
|
00db10 |
+
|
|
|
00db10 |
+# include "sysdep.h"
|
|
|
00db10 |
+# include "asm-syntax.h"
|
|
|
00db10 |
+
|
|
|
00db10 |
+ .text
|
|
|
00db10 |
+
|
|
|
00db10 |
+/* int strcmp (const char *s1, const char *s2)
|
|
|
00db10 |
+ Compare two strings
|
|
|
00db10 |
+
|
|
|
00db10 |
+ Register usage:
|
|
|
00db10 |
+ -r1=loaded byte count s1
|
|
|
00db10 |
+ -r2=s1
|
|
|
00db10 |
+ -r3=s2
|
|
|
00db10 |
+ -r4=loaded byte coutn s2, tmp
|
|
|
00db10 |
+ -r5=current_len
|
|
|
00db10 |
+ -v16=part of s1
|
|
|
00db10 |
+ -v17=part of s2
|
|
|
00db10 |
+ -v18=index of unequal
|
|
|
00db10 |
+*/
|
|
|
00db10 |
+ENTRY(__strcmp_vx)
|
|
|
00db10 |
+ .machine "z13"
|
|
|
00db10 |
+ .machinemode "zarch_nohighgprs"
|
|
|
00db10 |
+
|
|
|
00db10 |
+ lghi %r5,0 /* current_len = 0. */
|
|
|
00db10 |
+
|
|
|
00db10 |
+.Lloop:
|
|
|
00db10 |
+ vlbb %v16,0(%r5,%r2),6 /* Load s1 to block boundary. */
|
|
|
00db10 |
+ vlbb %v17,0(%r5,%r3),6 /* Load s2 to block boundary. */
|
|
|
00db10 |
+ lcbb %r1,0(%r5,%r2),6 /* Get loaded byte count of s1. */
|
|
|
00db10 |
+ jo .Llt16_1 /* Jump away if vr is not fully loaded. */
|
|
|
00db10 |
+ lcbb %r4,0(%r5,%r3),6
|
|
|
00db10 |
+ jo .Llt16_2 /* Jump away if vr is not fully loaded. */
|
|
|
00db10 |
+ /* Both vrs are fully loaded. */
|
|
|
00db10 |
+ aghi %r5,16
|
|
|
00db10 |
+ vfenezbs %v18,%v16,%v17 /* Compare not equal with zero search. */
|
|
|
00db10 |
+ jno .Lfound
|
|
|
00db10 |
+
|
|
|
00db10 |
+ vlbb %v16,0(%r5,%r2),6
|
|
|
00db10 |
+ vlbb %v17,0(%r5,%r3),6
|
|
|
00db10 |
+ lcbb %r1,0(%r5,%r2),6
|
|
|
00db10 |
+ jo .Llt16_1
|
|
|
00db10 |
+ lcbb %r4,0(%r5,%r3),6
|
|
|
00db10 |
+ jo .Llt16_2
|
|
|
00db10 |
+ aghi %r5,16
|
|
|
00db10 |
+ vfenezbs %v18,%v16,%v17
|
|
|
00db10 |
+ jno .Lfound
|
|
|
00db10 |
+
|
|
|
00db10 |
+ vlbb %v16,0(%r5,%r2),6
|
|
|
00db10 |
+ vlbb %v17,0(%r5,%r3),6
|
|
|
00db10 |
+ lcbb %r1,0(%r5,%r2),6
|
|
|
00db10 |
+ jo .Llt16_1
|
|
|
00db10 |
+ lcbb %r4,0(%r5,%r3),6
|
|
|
00db10 |
+ jo .Llt16_2
|
|
|
00db10 |
+ aghi %r5,16
|
|
|
00db10 |
+ vfenezbs %v18,%v16,%v17
|
|
|
00db10 |
+ jno .Lfound
|
|
|
00db10 |
+
|
|
|
00db10 |
+ vlbb %v16,0(%r5,%r2),6
|
|
|
00db10 |
+ vlbb %v17,0(%r5,%r3),6
|
|
|
00db10 |
+ lcbb %r1,0(%r5,%r2),6
|
|
|
00db10 |
+ jo .Llt16_1
|
|
|
00db10 |
+ lcbb %r4,0(%r5,%r3),6
|
|
|
00db10 |
+ jo .Llt16_2
|
|
|
00db10 |
+ aghi %r5,16
|
|
|
00db10 |
+ vfenezbs %v18,%v16,%v17
|
|
|
00db10 |
+ jno .Lfound
|
|
|
00db10 |
+ j .Lloop
|
|
|
00db10 |
+
|
|
|
00db10 |
+.Llt16_1:
|
|
|
00db10 |
+ lcbb %r4,0(%r5,%r3),6 /* Get loaded byte count of s2. */
|
|
|
00db10 |
+.Llt16_2:
|
|
|
00db10 |
+ clr %r1,%r4
|
|
|
00db10 |
+ locrh %r1,%r4 /* Get minimum of bytes loaded in s1/2. */
|
|
|
00db10 |
+ algfr %r5,%r1 /* Add smallest loaded bytes to current_len. */
|
|
|
00db10 |
+ vfenezbs %v18,%v16,%v17 /* Compare not equal with zero search. */
|
|
|
00db10 |
+ vlgvb %r4,%v18,7 /* Get not equal index or 16 if all equal. */
|
|
|
00db10 |
+ clrjl %r4,%r1,.Lfound /* Jump away if miscompare is within loaded
|
|
|
00db10 |
+ bytes. */
|
|
|
00db10 |
+ j .Lloop
|
|
|
00db10 |
+
|
|
|
00db10 |
+.Lfound:
|
|
|
00db10 |
+ je .Lend_equal
|
|
|
00db10 |
+ lghi %r2,1
|
|
|
00db10 |
+ lghi %r1,-1
|
|
|
00db10 |
+ locgrl %r2,%r1
|
|
|
00db10 |
+ br %r14
|
|
|
00db10 |
+.Lend_equal:
|
|
|
00db10 |
+ lghi %r2,0
|
|
|
00db10 |
+ br %r14
|
|
|
00db10 |
+END(__strcmp_vx)
|
|
|
00db10 |
+
|
|
|
00db10 |
+# define strcmp __strcmp_c
|
|
|
00db10 |
+# undef libc_hidden_builtin_def
|
|
|
00db10 |
+# define libc_hidden_builtin_def(name) strong_alias(__strcmp_c, __GI_strcmp)
|
|
|
00db10 |
+#endif /* HAVE_S390_VX_ASM_SUPPORT && !defined NOT_IN_libc */
|
|
|
00db10 |
+
|
|
|
00db10 |
+#include <strcmp.S>
|
|
|
00db10 |
diff --git a/sysdeps/s390/multiarch/strcmp.c b/sysdeps/s390/multiarch/strcmp.c
|
|
|
00db10 |
new file mode 100644
|
|
|
00db10 |
index 0000000..dd462a2
|
|
|
00db10 |
--- /dev/null
|
|
|
00db10 |
+++ b/sysdeps/s390/multiarch/strcmp.c
|
|
|
00db10 |
@@ -0,0 +1,26 @@
|
|
|
00db10 |
+/* Multiple versions of strcmp.
|
|
|
00db10 |
+ Copyright (C) 2015 Free Software Foundation, Inc.
|
|
|
00db10 |
+ This file is part of the GNU C Library.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is free software; you can redistribute it and/or
|
|
|
00db10 |
+ modify it under the terms of the GNU Lesser General Public
|
|
|
00db10 |
+ License as published by the Free Software Foundation; either
|
|
|
00db10 |
+ version 2.1 of the License, or (at your option) any later version.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is distributed in the hope that it will be useful,
|
|
|
00db10 |
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
00db10 |
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
00db10 |
+ Lesser General Public License for more details.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ You should have received a copy of the GNU Lesser General Public
|
|
|
00db10 |
+ License along with the GNU C Library; if not, see
|
|
|
00db10 |
+ <http://www.gnu.org/licenses/>. */
|
|
|
00db10 |
+
|
|
|
00db10 |
+#if defined HAVE_S390_VX_ASM_SUPPORT && !defined NOT_IN_libc
|
|
|
00db10 |
+# include <string.h>
|
|
|
00db10 |
+# include <ifunc-resolve.h>
|
|
|
00db10 |
+
|
|
|
00db10 |
+
|
|
|
00db10 |
+# undef strcmp
|
|
|
00db10 |
+s390_vx_libc_ifunc2 (__strcmp, strcmp)
|
|
|
00db10 |
+#endif
|
|
|
00db10 |
diff --git a/sysdeps/s390/multiarch/wcscmp-c.c b/sysdeps/s390/multiarch/wcscmp-c.c
|
|
|
00db10 |
new file mode 100644
|
|
|
00db10 |
index 0000000..8358e1f
|
|
|
00db10 |
--- /dev/null
|
|
|
00db10 |
+++ b/sysdeps/s390/multiarch/wcscmp-c.c
|
|
|
00db10 |
@@ -0,0 +1,30 @@
|
|
|
00db10 |
+/* Default wcscmp implementation for S/390.
|
|
|
00db10 |
+ Copyright (C) 2015 Free Software Foundation, Inc.
|
|
|
00db10 |
+ This file is part of the GNU C Library.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is free software; you can redistribute it and/or
|
|
|
00db10 |
+ modify it under the terms of the GNU Lesser General Public
|
|
|
00db10 |
+ License as published by the Free Software Foundation; either
|
|
|
00db10 |
+ version 2.1 of the License, or (at your option) any later version.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is distributed in the hope that it will be useful,
|
|
|
00db10 |
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
00db10 |
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
00db10 |
+ Lesser General Public License for more details.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ You should have received a copy of the GNU Lesser General Public
|
|
|
00db10 |
+ License along with the GNU C Library; if not, see
|
|
|
00db10 |
+ <http://www.gnu.org/licenses/>. */
|
|
|
00db10 |
+
|
|
|
00db10 |
+#if defined HAVE_S390_VX_ASM_SUPPORT && !defined NOT_IN_libc
|
|
|
00db10 |
+# define WCSCMP __wcscmp_c
|
|
|
00db10 |
+
|
|
|
00db10 |
+# include <wchar.h>
|
|
|
00db10 |
+extern __typeof (wcscmp) __wcscmp_c;
|
|
|
00db10 |
+# ifdef SHARED
|
|
|
00db10 |
+# undef libc_hidden_def
|
|
|
00db10 |
+# define libc_hidden_def(name) \
|
|
|
00db10 |
+ __hidden_ver1 (__wcscmp_c, __GI_wcscmp, __wcscmp_c);
|
|
|
00db10 |
+# endif /* SHARED */
|
|
|
00db10 |
+# include <wcsmbs/wcscmp.c>
|
|
|
00db10 |
+#endif /* HAVE_S390_VX_ASM_SUPPORT && !defined NOT_IN_libc */
|
|
|
00db10 |
diff --git a/sysdeps/s390/multiarch/wcscmp-vx.S b/sysdeps/s390/multiarch/wcscmp-vx.S
|
|
|
00db10 |
new file mode 100644
|
|
|
00db10 |
index 0000000..2c9e610
|
|
|
00db10 |
--- /dev/null
|
|
|
00db10 |
+++ b/sysdeps/s390/multiarch/wcscmp-vx.S
|
|
|
00db10 |
@@ -0,0 +1,131 @@
|
|
|
00db10 |
+/* Vector optimized 32/64 bit S/390 version of wcscmp.
|
|
|
00db10 |
+ Copyright (C) 2015 Free Software Foundation, Inc.
|
|
|
00db10 |
+ This file is part of the GNU C Library.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is free software; you can redistribute it and/or
|
|
|
00db10 |
+ modify it under the terms of the GNU Lesser General Public
|
|
|
00db10 |
+ License as published by the Free Software Foundation; either
|
|
|
00db10 |
+ version 2.1 of the License, or (at your option) any later version.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is distributed in the hope that it will be useful,
|
|
|
00db10 |
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
00db10 |
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
00db10 |
+ Lesser General Public License for more details.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ You should have received a copy of the GNU Lesser General Public
|
|
|
00db10 |
+ License along with the GNU C Library; if not, see
|
|
|
00db10 |
+ <http://www.gnu.org/licenses/>. */
|
|
|
00db10 |
+
|
|
|
00db10 |
+#if defined HAVE_S390_VX_ASM_SUPPORT && !defined NOT_IN_libc
|
|
|
00db10 |
+
|
|
|
00db10 |
+# include "sysdep.h"
|
|
|
00db10 |
+# include "asm-syntax.h"
|
|
|
00db10 |
+
|
|
|
00db10 |
+ .text
|
|
|
00db10 |
+
|
|
|
00db10 |
+/* int wcscmp (const wchar_t *s1, const wchar_t *s2)
|
|
|
00db10 |
+ Compare two strings
|
|
|
00db10 |
+
|
|
|
00db10 |
+ Register usage:
|
|
|
00db10 |
+ -r1=loaded byte count s1
|
|
|
00db10 |
+ -r2=s1
|
|
|
00db10 |
+ -r3=s2
|
|
|
00db10 |
+ -r4=loaded byte coutn s2, tmp
|
|
|
00db10 |
+ -r5=current_len
|
|
|
00db10 |
+ -v16=part of s1
|
|
|
00db10 |
+ -v17=part of s2
|
|
|
00db10 |
+ -v18=index of unequal
|
|
|
00db10 |
+*/
|
|
|
00db10 |
+ENTRY(__wcscmp_vx)
|
|
|
00db10 |
+ .machine "z13"
|
|
|
00db10 |
+ .machinemode "zarch_nohighgprs"
|
|
|
00db10 |
+
|
|
|
00db10 |
+ lghi %r5,0 /* current_len = 0. */
|
|
|
00db10 |
+
|
|
|
00db10 |
+.Lloop:
|
|
|
00db10 |
+ vlbb %v16,0(%r5,%r2),6 /* Load s1 to block boundary. */
|
|
|
00db10 |
+ vlbb %v17,0(%r5,%r3),6 /* Load s2 to block boundary. */
|
|
|
00db10 |
+ lcbb %r1,0(%r5,%r2),6 /* Get loaded byte count of s1. */
|
|
|
00db10 |
+ jo .Llt16_1 /* Jump away if vr is not fully loaded. */
|
|
|
00db10 |
+ lcbb %r4,0(%r5,%r3),6
|
|
|
00db10 |
+ jo .Llt16_2 /* Jump away if vr is not fully loaded. */
|
|
|
00db10 |
+ /* Both vrs are fully loaded. */
|
|
|
00db10 |
+ aghi %r5,16
|
|
|
00db10 |
+ vfenezfs %v18,%v16,%v17 /* Compare not equal with zero search. */
|
|
|
00db10 |
+ jno .Lfound
|
|
|
00db10 |
+
|
|
|
00db10 |
+ vlbb %v16,0(%r5,%r2),6
|
|
|
00db10 |
+ vlbb %v17,0(%r5,%r3),6
|
|
|
00db10 |
+ lcbb %r1,0(%r5,%r2),6
|
|
|
00db10 |
+ jo .Llt16_1
|
|
|
00db10 |
+ lcbb %r4,0(%r5,%r3),6
|
|
|
00db10 |
+ jo .Llt16_2
|
|
|
00db10 |
+ aghi %r5,16
|
|
|
00db10 |
+ vfenezfs %v18,%v16,%v17
|
|
|
00db10 |
+ jno .Lfound
|
|
|
00db10 |
+
|
|
|
00db10 |
+ vlbb %v16,0(%r5,%r2),6
|
|
|
00db10 |
+ vlbb %v17,0(%r5,%r3),6
|
|
|
00db10 |
+ lcbb %r1,0(%r5,%r2),6
|
|
|
00db10 |
+ jo .Llt16_1
|
|
|
00db10 |
+ lcbb %r4,0(%r5,%r3),6
|
|
|
00db10 |
+ jo .Llt16_2
|
|
|
00db10 |
+ aghi %r5,16
|
|
|
00db10 |
+ vfenezfs %v18,%v16,%v17
|
|
|
00db10 |
+ jno .Lfound
|
|
|
00db10 |
+
|
|
|
00db10 |
+ vlbb %v16,0(%r5,%r2),6
|
|
|
00db10 |
+ vlbb %v17,0(%r5,%r3),6
|
|
|
00db10 |
+ lcbb %r1,0(%r5,%r2),6
|
|
|
00db10 |
+ jo .Llt16_1
|
|
|
00db10 |
+ lcbb %r4,0(%r5,%r3),6
|
|
|
00db10 |
+ jo .Llt16_2
|
|
|
00db10 |
+ aghi %r5,16
|
|
|
00db10 |
+ vfenezfs %v18,%v16,%v17
|
|
|
00db10 |
+ jno .Lfound
|
|
|
00db10 |
+ j .Lloop
|
|
|
00db10 |
+
|
|
|
00db10 |
+.Lcmp_one_char:
|
|
|
00db10 |
+ /* At least one of both strings is not 4-byte aligned
|
|
|
00db10 |
+ and there is no full character before next block-boundary.
|
|
|
00db10 |
+ Compare one character to get over the boundary and
|
|
|
00db10 |
+ proceed with normal loop! */
|
|
|
00db10 |
+ vlef %v16,0(%r5,%r2),0 /* Load one character. */
|
|
|
00db10 |
+ vlef %v17,0(%r5,%r3),0
|
|
|
00db10 |
+ lghi %r1,4 /* Loaded byte count is 4. */
|
|
|
00db10 |
+ j .Llt_cmp /* Proceed with comparision. */
|
|
|
00db10 |
+
|
|
|
00db10 |
+.Llt16_1:
|
|
|
00db10 |
+ lcbb %r4,0(%r5,%r3),6 /* Get loaded byte count of s2. */
|
|
|
00db10 |
+.Llt16_2:
|
|
|
00db10 |
+ clr %r1,%r4
|
|
|
00db10 |
+ locrh %r1,%r4 /* Get minimum of bytes loaded in s1/2. */
|
|
|
00db10 |
+ nill %r1,65532 /* Align bytes loaded to full characters. */
|
|
|
00db10 |
+ jz .Lcmp_one_char /* Jump away if no full char is available. */
|
|
|
00db10 |
+.Llt_cmp:
|
|
|
00db10 |
+ algfr %r5,%r1 /* Add smallest loaded bytes to current_len. */
|
|
|
00db10 |
+ vfenezfs %v18,%v16,%v17 /* Compare not equal with zero search. */
|
|
|
00db10 |
+ vlgvb %r4,%v18,7 /* Get not equal index or 16 if all equal. */
|
|
|
00db10 |
+ clrjl %r4,%r1,.Lfound /* Jump away if miscompare is within loaded
|
|
|
00db10 |
+ bytes. */
|
|
|
00db10 |
+ j .Lloop
|
|
|
00db10 |
+
|
|
|
00db10 |
+.Lfound:
|
|
|
00db10 |
+ /* vfenezf found an unequal element or zero.
|
|
|
00db10 |
+ This instruction compares unsigned words, but wchar_t is signed.
|
|
|
00db10 |
+ Thus we have to compare the found element again. */
|
|
|
00db10 |
+ vlgvb %r4,%v18,7 /* Extract not equal byte-index, */
|
|
|
00db10 |
+ srl %r4,2 /* Convert it to character-index. */
|
|
|
00db10 |
+ vlgvf %r3,%v16,0(%r4) /* Load character-values. */
|
|
|
00db10 |
+ vlgvf %r4,%v17,0(%r4)
|
|
|
00db10 |
+ cr %r3,%r4
|
|
|
00db10 |
+ je .Lend_equal
|
|
|
00db10 |
+ lghi %r2,1
|
|
|
00db10 |
+ lghi %r1,-1
|
|
|
00db10 |
+ locgrl %r2,%r1
|
|
|
00db10 |
+ br %r14
|
|
|
00db10 |
+.Lend_equal:
|
|
|
00db10 |
+ lghi %r2,0
|
|
|
00db10 |
+ br %r14
|
|
|
00db10 |
+END(__wcscmp_vx)
|
|
|
00db10 |
+#endif /* HAVE_S390_VX_ASM_SUPPORT && !defined NOT_IN_libc */
|
|
|
00db10 |
diff --git a/sysdeps/s390/multiarch/wcscmp.c b/sysdeps/s390/multiarch/wcscmp.c
|
|
|
00db10 |
new file mode 100644
|
|
|
00db10 |
index 0000000..99bae4f
|
|
|
00db10 |
--- /dev/null
|
|
|
00db10 |
+++ b/sysdeps/s390/multiarch/wcscmp.c
|
|
|
00db10 |
@@ -0,0 +1,27 @@
|
|
|
00db10 |
+/* Multiple versions of wcscmp.
|
|
|
00db10 |
+ Copyright (C) 2015 Free Software Foundation, Inc.
|
|
|
00db10 |
+ This file is part of the GNU C Library.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is free software; you can redistribute it and/or
|
|
|
00db10 |
+ modify it under the terms of the GNU Lesser General Public
|
|
|
00db10 |
+ License as published by the Free Software Foundation; either
|
|
|
00db10 |
+ version 2.1 of the License, or (at your option) any later version.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is distributed in the hope that it will be useful,
|
|
|
00db10 |
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
00db10 |
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
00db10 |
+ Lesser General Public License for more details.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ You should have received a copy of the GNU Lesser General Public
|
|
|
00db10 |
+ License along with the GNU C Library; if not, see
|
|
|
00db10 |
+ <http://www.gnu.org/licenses/>. */
|
|
|
00db10 |
+
|
|
|
00db10 |
+#if defined HAVE_S390_VX_ASM_SUPPORT && !defined NOT_IN_libc
|
|
|
00db10 |
+# include <wchar.h>
|
|
|
00db10 |
+# include <ifunc-resolve.h>
|
|
|
00db10 |
+
|
|
|
00db10 |
+s390_vx_libc_ifunc2 (__wcscmp, wcscmp)
|
|
|
00db10 |
+
|
|
|
00db10 |
+#else
|
|
|
00db10 |
+# include <wcsmbs/wcscmp.c>
|
|
|
00db10 |
+#endif /* !(defined HAVE_S390_VX_ASM_SUPPORT && !defined NOT_IN_libc) */
|
|
|
00db10 |
diff --git a/sysdeps/s390/s390-32/multiarch/strcmp.c b/sysdeps/s390/s390-32/multiarch/strcmp.c
|
|
|
00db10 |
new file mode 100644
|
|
|
00db10 |
index 0000000..1598bbc
|
|
|
00db10 |
--- /dev/null
|
|
|
00db10 |
+++ b/sysdeps/s390/s390-32/multiarch/strcmp.c
|
|
|
00db10 |
@@ -0,0 +1,21 @@
|
|
|
00db10 |
+/* Multiple versions of strcmp.
|
|
|
00db10 |
+ Copyright (C) 2015 Free Software Foundation, Inc.
|
|
|
00db10 |
+ This file is part of the GNU C Library.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is free software; you can redistribute it and/or
|
|
|
00db10 |
+ modify it under the terms of the GNU Lesser General Public
|
|
|
00db10 |
+ License as published by the Free Software Foundation; either
|
|
|
00db10 |
+ version 2.1 of the License, or (at your option) any later version.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is distributed in the hope that it will be useful,
|
|
|
00db10 |
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
00db10 |
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
00db10 |
+ Lesser General Public License for more details.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ You should have received a copy of the GNU Lesser General Public
|
|
|
00db10 |
+ License along with the GNU C Library; if not, see
|
|
|
00db10 |
+ <http://www.gnu.org/licenses/>. */
|
|
|
00db10 |
+
|
|
|
00db10 |
+/* This wrapper-file is needed, because otherwise file
|
|
|
00db10 |
+ sysdeps/s390/s390-[32|64]/strcmp.S will be used. */
|
|
|
00db10 |
+#include <sysdeps/s390/multiarch/strcmp.c>
|
|
|
00db10 |
diff --git a/sysdeps/s390/s390-64/multiarch/strcmp.c b/sysdeps/s390/s390-64/multiarch/strcmp.c
|
|
|
00db10 |
new file mode 100644
|
|
|
00db10 |
index 0000000..1598bbc
|
|
|
00db10 |
--- /dev/null
|
|
|
00db10 |
+++ b/sysdeps/s390/s390-64/multiarch/strcmp.c
|
|
|
00db10 |
@@ -0,0 +1,21 @@
|
|
|
00db10 |
+/* Multiple versions of strcmp.
|
|
|
00db10 |
+ Copyright (C) 2015 Free Software Foundation, Inc.
|
|
|
00db10 |
+ This file is part of the GNU C Library.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is free software; you can redistribute it and/or
|
|
|
00db10 |
+ modify it under the terms of the GNU Lesser General Public
|
|
|
00db10 |
+ License as published by the Free Software Foundation; either
|
|
|
00db10 |
+ version 2.1 of the License, or (at your option) any later version.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ The GNU C Library is distributed in the hope that it will be useful,
|
|
|
00db10 |
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
00db10 |
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
00db10 |
+ Lesser General Public License for more details.
|
|
|
00db10 |
+
|
|
|
00db10 |
+ You should have received a copy of the GNU Lesser General Public
|
|
|
00db10 |
+ License along with the GNU C Library; if not, see
|
|
|
00db10 |
+ <http://www.gnu.org/licenses/>. */
|
|
|
00db10 |
+
|
|
|
00db10 |
+/* This wrapper-file is needed, because otherwise file
|
|
|
00db10 |
+ sysdeps/s390/s390-[32|64]/strcmp.S will be used. */
|
|
|
00db10 |
+#include <sysdeps/s390/multiarch/strcmp.c>
|
|
|
00db10 |
--
|
|
|
00db10 |
2.3.0
|
|
|
00db10 |
|