00db10
From c806cab89b52a644b5c563b8f1c8ae59abfc2c13 Mon Sep 17 00:00:00 2001
00db10
From: Stefan Liebler <stli@linux.vnet.ibm.com>
00db10
Date: Mon, 7 Nov 2016 16:22:17 +0100
00db10
Subject: [PATCH 12/17] S390: Optimize utf16-utf32 module.
00db10
00db10
Upstream commit 6896776c3c9c32fd22324e6de6737dd69ae73213
00db10
00db10
This patch reworks the s390 specific module to convert between utf16 and utf32.
00db10
Now ifunc is used to choose either the c or etf3eh (with convert utf
00db10
instruction) variants at runtime.
00db10
Furthermore a new vector variant for z13 is introduced which will be build
00db10
and chosen if vector support is available at build / runtime.
00db10
00db10
In case of converting utf 32 to utf16, the vector variant optimizes input of
00db10
2byte utf16 characters. The convert utf instruction is used if an utf16
00db10
surrogate is found.
00db10
00db10
For the other direction utf16 to utf32, the cu24 instruction can't be re-
00db10
enabled, because it does not report an error, if the input-stream consists of
00db10
a single low surrogate utf16 char (e.g. 0xdc00). This applies to the newest z13,
00db10
too. Thus there is only the c or the new vector variant, which can handle utf16
00db10
surrogate characters.
00db10
00db10
This patch also fixes some whitespace errors. Furthermore, the etf3eh variant is
00db10
handling the "UTF-xx//IGNORE" case now. Before they ignored the ignore-case and
00db10
always stopped at an error.
00db10
00db10
ChangeLog:
00db10
00db10
	* sysdeps/s390/s390-64/utf16-utf32-z9.c: Use ifunc to select c,
00db10
	etf3eh or new vector loop-variant.
00db10
---
00db10
 sysdeps/s390/s390-64/utf16-utf32-z9.c | 471 +++++++++++++++++++++++++++-------
00db10
 1 file changed, 379 insertions(+), 92 deletions(-)
00db10
00db10
diff --git a/sysdeps/s390/s390-64/utf16-utf32-z9.c b/sysdeps/s390/s390-64/utf16-utf32-z9.c
00db10
index e6a033d..33594f1 100644
00db10
--- a/sysdeps/s390/s390-64/utf16-utf32-z9.c
00db10
+++ b/sysdeps/s390/s390-64/utf16-utf32-z9.c
00db10
@@ -30,47 +30,27 @@
00db10
 #include <dl-procinfo.h>
00db10
 #include <gconv.h>
00db10
 
00db10
+#if defined HAVE_S390_VX_GCC_SUPPORT
00db10
+# define ASM_CLOBBER_VR(NR) , NR
00db10
+#else
00db10
+# define ASM_CLOBBER_VR(NR)
00db10
+#endif
00db10
+
00db10
 /* UTF-32 big endian byte order mark.  */
00db10
 #define BOM_UTF32               0x0000feffu
00db10
 
00db10
 /* UTF-16 big endian byte order mark.  */
00db10
-#define BOM_UTF16	        0xfeff
00db10
+#define BOM_UTF16               0xfeff
00db10
 
00db10
 #define DEFINE_INIT		0
00db10
 #define DEFINE_FINI		0
00db10
 #define MIN_NEEDED_FROM		2
00db10
 #define MAX_NEEDED_FROM		4
00db10
 #define MIN_NEEDED_TO		4
00db10
-#define FROM_LOOP		from_utf16_loop
00db10
-#define TO_LOOP			to_utf16_loop
00db10
+#define FROM_LOOP		__from_utf16_loop
00db10
+#define TO_LOOP			__to_utf16_loop
00db10
 #define FROM_DIRECTION		(dir == from_utf16)
00db10
 #define ONE_DIRECTION           0
00db10
-#define PREPARE_LOOP							\
00db10
-  enum direction dir = ((struct utf16_data *) step->__data)->dir;	\
00db10
-  int emit_bom = ((struct utf16_data *) step->__data)->emit_bom;	\
00db10
-									\
00db10
-  if (emit_bom && !data->__internal_use					\
00db10
-      && data->__invocation_counter == 0)				\
00db10
-    {									\
00db10
-      if (dir == to_utf16)						\
00db10
-	{								\
00db10
-          /* Emit the UTF-16 Byte Order Mark.  */			\
00db10
-          if (__glibc_unlikely (outbuf + 2 > outend))			      \
00db10
-	    return __GCONV_FULL_OUTPUT;					\
00db10
-									\
00db10
-	  put16u (outbuf, BOM_UTF16);					\
00db10
-	  outbuf += 2;							\
00db10
-	}								\
00db10
-      else								\
00db10
-	{								\
00db10
-          /* Emit the UTF-32 Byte Order Mark.  */			\
00db10
-	  if (__glibc_unlikely (outbuf + 4 > outend))			      \
00db10
-	    return __GCONV_FULL_OUTPUT;					\
00db10
-									\
00db10
-	  put32u (outbuf, BOM_UTF32);					\
00db10
-	  outbuf += 4;							\
00db10
-	}								\
00db10
-    }
00db10
 
00db10
 /* Direction of the transformation.  */
00db10
 enum direction
00db10
@@ -169,16 +149,16 @@ gconv_end (struct __gconv_step *data)
00db10
     register unsigned long long outlen __asm__("11") = outend - outptr;	\
00db10
     uint64_t cc = 0;							\
00db10
 									\
00db10
-    __asm__ volatile (".machine push       \n\t"			\
00db10
-		      ".machine \"z9-109\" \n\t"			\
00db10
-		      "0: " INSTRUCTION "  \n\t"			\
00db10
-		      ".machine pop        \n\t"			\
00db10
-		      "   jo     0b        \n\t"			\
00db10
-		      "   ipm    %2        \n"				\
00db10
-		      : "+a" (pOutput), "+a" (pInput), "+d" (cc),	\
00db10
-		      "+d" (outlen), "+d" (inlen)			\
00db10
-		      :							\
00db10
-		      : "cc", "memory");				\
00db10
+    __asm__ __volatile__ (".machine push       \n\t"			\
00db10
+			  ".machine \"z9-109\" \n\t"			\
00db10
+			  "0: " INSTRUCTION "  \n\t"			\
00db10
+			  ".machine pop        \n\t"			\
00db10
+			  "   jo     0b        \n\t"			\
00db10
+			  "   ipm    %2        \n"			\
00db10
+			  : "+a" (pOutput), "+a" (pInput), "+d" (cc),	\
00db10
+			    "+d" (outlen), "+d" (inlen)			\
00db10
+			  :						\
00db10
+			  : "cc", "memory");				\
00db10
 									\
00db10
     inptr = pInput;							\
00db10
     outptr = pOutput;							\
00db10
@@ -187,44 +167,46 @@ gconv_end (struct __gconv_step *data)
00db10
     if (cc == 1)							\
00db10
       {									\
00db10
 	result = __GCONV_FULL_OUTPUT;					\
00db10
-	break;								\
00db10
       }									\
00db10
     else if (cc == 2)							\
00db10
       {									\
00db10
 	result = __GCONV_ILLEGAL_INPUT;					\
00db10
-	break;								\
00db10
       }									\
00db10
   }
00db10
 
00db10
+#define PREPARE_LOOP							\
00db10
+  enum direction dir = ((struct utf16_data *) step->__data)->dir;	\
00db10
+  int emit_bom = ((struct utf16_data *) step->__data)->emit_bom;	\
00db10
+									\
00db10
+  if (emit_bom && !data->__internal_use					\
00db10
+      && data->__invocation_counter == 0)				\
00db10
+    {									\
00db10
+      if (dir == to_utf16)						\
00db10
+	{								\
00db10
+	  /* Emit the UTF-16 Byte Order Mark.  */			\
00db10
+	  if (__glibc_unlikely (outbuf + 2 > outend))			\
00db10
+	    return __GCONV_FULL_OUTPUT;					\
00db10
+									\
00db10
+	  put16u (outbuf, BOM_UTF16);					\
00db10
+	  outbuf += 2;							\
00db10
+	}								\
00db10
+      else								\
00db10
+	{								\
00db10
+	  /* Emit the UTF-32 Byte Order Mark.  */			\
00db10
+	  if (__glibc_unlikely (outbuf + 4 > outend))			\
00db10
+	    return __GCONV_FULL_OUTPUT;					\
00db10
+									\
00db10
+	  put32u (outbuf, BOM_UTF32);					\
00db10
+	  outbuf += 4;							\
00db10
+	}								\
00db10
+    }
00db10
+
00db10
 /* Conversion function from UTF-16 to UTF-32 internal/BE.  */
00db10
 
00db10
-#define MIN_NEEDED_INPUT	MIN_NEEDED_FROM
00db10
-#define MAX_NEEDED_INPUT	MAX_NEEDED_FROM
00db10
-#define MIN_NEEDED_OUTPUT	MIN_NEEDED_TO
00db10
-#define LOOPFCT			FROM_LOOP
00db10
 /* The software routine is copied from utf-16.c (minus bytes
00db10
    swapping).  */
00db10
-#define BODY								\
00db10
+#define BODY_FROM_C							\
00db10
   {									\
00db10
-    /* The hardware instruction currently fails to report an error for	\
00db10
-       isolated low surrogates so we have to disable the instruction	\
00db10
-       until this gets resolved.  */					\
00db10
-    if (0) /* (GLRO (dl_hwcap) & HWCAP_S390_ETF3EH) */			\
00db10
-      {									\
00db10
-	HARDWARE_CONVERT ("cu24 %0, %1, 1");				\
00db10
-	if (inptr != inend)						\
00db10
-	  {								\
00db10
-	    /* Check if the third byte is				\
00db10
-	       a valid start of a UTF-16 surrogate.  */			\
00db10
-	    if (inend - inptr == 3 && (inptr[3] & 0xfc) != 0xdc)	\
00db10
-	      STANDARD_FROM_LOOP_ERR_HANDLER (3);			\
00db10
-									\
00db10
-	    result = __GCONV_INCOMPLETE_INPUT;				\
00db10
-	    break;							\
00db10
-	  }								\
00db10
-	continue;							\
00db10
-      }									\
00db10
-									\
00db10
     uint16_t u1 = get16 (inptr);					\
00db10
 									\
00db10
     if (__builtin_expect (u1 < 0xd800, 1) || u1 > 0xdfff)		\
00db10
@@ -235,15 +217,15 @@ gconv_end (struct __gconv_step *data)
00db10
       }									\
00db10
     else								\
00db10
       {									\
00db10
-        /* An isolated low-surrogate was found.  This has to be         \
00db10
+	/* An isolated low-surrogate was found.  This has to be         \
00db10
 	   considered ill-formed.  */					\
00db10
-        if (__glibc_unlikely (u1 >= 0xdc00))				      \
00db10
+	if (__glibc_unlikely (u1 >= 0xdc00))				\
00db10
 	  {								\
00db10
 	    STANDARD_FROM_LOOP_ERR_HANDLER (2);				\
00db10
 	  }								\
00db10
 	/* It's a surrogate character.  At least the first word says	\
00db10
 	   it is.  */							\
00db10
-	if (__glibc_unlikely (inptr + 4 > inend))			      \
00db10
+	if (__glibc_unlikely (inptr + 4 > inend))			\
00db10
 	  {								\
00db10
 	    /* We don't have enough input for another complete input	\
00db10
 	       character.  */						\
00db10
@@ -266,48 +248,200 @@ gconv_end (struct __gconv_step *data)
00db10
       }									\
00db10
     outptr += 4;							\
00db10
   }
00db10
-#define LOOP_NEED_FLAGS
00db10
-#include <iconv/loop.c>
00db10
+
00db10
+#define BODY_FROM_VX							\
00db10
+  {									\
00db10
+    size_t inlen = inend - inptr;					\
00db10
+    size_t outlen = outend - outptr;					\
00db10
+    unsigned long tmp, tmp2, tmp3;					\
00db10
+    asm volatile (".machine push\n\t"					\
00db10
+		  ".machine \"z13\"\n\t"				\
00db10
+		  ".machinemode \"zarch_nohighgprs\"\n\t"		\
00db10
+		  /* Setup to check for surrogates.  */			\
00db10
+		  "    larl %[R_TMP],9f\n\t"				\
00db10
+		  "    vlm %%v30,%%v31,0(%[R_TMP])\n\t"			\
00db10
+		  /* Loop which handles UTF-16 chars <0xd800, >0xdfff.  */ \
00db10
+		  "0:  clgijl %[R_INLEN],16,2f\n\t"			\
00db10
+		  "    clgijl %[R_OUTLEN],32,2f\n\t"			\
00db10
+		  "1:  vl %%v16,0(%[R_IN])\n\t"				\
00db10
+		  /* Check for surrogate chars.  */			\
00db10
+		  "    vstrchs %%v19,%%v16,%%v30,%%v31\n\t"		\
00db10
+		  "    jno 10f\n\t"					\
00db10
+		  /* Enlarge to UTF-32.  */				\
00db10
+		  "    vuplhh %%v17,%%v16\n\t"				\
00db10
+		  "    la %[R_IN],16(%[R_IN])\n\t"			\
00db10
+		  "    vupllh %%v18,%%v16\n\t"				\
00db10
+		  "    aghi %[R_INLEN],-16\n\t"				\
00db10
+		  /* Store 32 bytes to buf_out.  */			\
00db10
+		  "    vstm %%v17,%%v18,0(%[R_OUT])\n\t"		\
00db10
+		  "    aghi %[R_OUTLEN],-32\n\t"			\
00db10
+		  "    la %[R_OUT],32(%[R_OUT])\n\t"			\
00db10
+		  "    clgijl %[R_INLEN],16,2f\n\t"			\
00db10
+		  "    clgijl %[R_OUTLEN],32,2f\n\t"			\
00db10
+		  "    j 1b\n\t"					\
00db10
+		  /* Setup to check for ch >= 0xd800 && ch <= 0xdfff. (v30, v31)  */ \
00db10
+		  "9:  .short 0xd800,0xdfff,0x0,0x0,0x0,0x0,0x0,0x0\n\t" \
00db10
+		  "    .short 0xa000,0xc000,0x0,0x0,0x0,0x0,0x0,0x0\n\t" \
00db10
+		  /* At least on uint16_t is in range of surrogates.	\
00db10
+		     Store the preceding chars.  */			\
00db10
+		  "10: vlgvb %[R_TMP],%%v19,7\n\t"			\
00db10
+		  "    vuplhh %%v17,%%v16\n\t"				\
00db10
+		  "    sllg %[R_TMP3],%[R_TMP],1\n\t" /* Number of out bytes.  */ \
00db10
+		  "    ahik %[R_TMP2],%[R_TMP3],-1\n\t" /* Highest index to store.  */ \
00db10
+		  "    jl 12f\n\t"					\
00db10
+		  "    vstl %%v17,%[R_TMP2],0(%[R_OUT])\n\t"		\
00db10
+		  "    vupllh %%v18,%%v16\n\t"				\
00db10
+		  "    ahi %[R_TMP2],-16\n\t"				\
00db10
+		  "    jl 11f\n\t"					\
00db10
+		  "    vstl %%v18,%[R_TMP2],16(%[R_OUT])\n\t"		\
00db10
+		  "11: \n\t" /* Update pointers.  */			\
00db10
+		  "    la %[R_IN],0(%[R_TMP],%[R_IN])\n\t"		\
00db10
+		  "    slgr %[R_INLEN],%[R_TMP]\n\t"			\
00db10
+		  "    la %[R_OUT],0(%[R_TMP3],%[R_OUT])\n\t"		\
00db10
+		  "    slgr %[R_OUTLEN],%[R_TMP3]\n\t"			\
00db10
+		  /* Calculate remaining uint16_t values in loaded vrs.  */ \
00db10
+		  "12: lghi %[R_TMP2],16\n\t"				\
00db10
+		  "    sgr %[R_TMP2],%[R_TMP]\n\t"			\
00db10
+		  "    srl %[R_TMP2],1\n\t"				\
00db10
+		  "    llh %[R_TMP],0(%[R_IN])\n\t"			\
00db10
+		  "    aghi %[R_OUTLEN],-4\n\t"				\
00db10
+		  "    j 16f\n\t"					\
00db10
+		  /* Handle remaining bytes.  */			\
00db10
+		  "2:  \n\t"						\
00db10
+		  /* Zero, one or more bytes available?  */		\
00db10
+		  "    clgfi %[R_INLEN],1\n\t"				\
00db10
+		  "    je 97f\n\t" /* Only one byte available.  */	\
00db10
+		  "    jl 99f\n\t" /* End if no bytes available.  */	\
00db10
+		  /* Calculate remaining uint16_t values in inptr.  */	\
00db10
+		  "    srlg %[R_TMP2],%[R_INLEN],1\n\t"			\
00db10
+		  /* Handle remaining uint16_t values.  */		\
00db10
+		  "13: llh %[R_TMP],0(%[R_IN])\n\t"			\
00db10
+		  "    slgfi %[R_OUTLEN],4\n\t"				\
00db10
+		  "    jl 96f \n\t"					\
00db10
+		  "    clfi %[R_TMP],0xd800\n\t"			\
00db10
+		  "    jhe 15f\n\t"					\
00db10
+		  "14: st %[R_TMP],0(%[R_OUT])\n\t"			\
00db10
+		  "    la %[R_IN],2(%[R_IN])\n\t"			\
00db10
+		  "    aghi %[R_INLEN],-2\n\t"				\
00db10
+		  "    la %[R_OUT],4(%[R_OUT])\n\t"			\
00db10
+		  "    brctg %[R_TMP2],13b\n\t"				\
00db10
+		  "    j 0b\n\t" /* Switch to vx-loop.  */		\
00db10
+		  /* Handle UTF-16 surrogate pair.  */			\
00db10
+		  "15: clfi %[R_TMP],0xdfff\n\t"			\
00db10
+		  "    jh 14b\n\t" /* Jump away if ch > 0xdfff.  */	\
00db10
+		  "16: clfi %[R_TMP],0xdc00\n\t"			\
00db10
+		  "    jhe 98f\n\t" /* Jump away in case of low-surrogate.  */ \
00db10
+		  "    slgfi %[R_INLEN],4\n\t"				\
00db10
+		  "    jl 97f\n\t" /* Big enough input?  */		\
00db10
+		  "    llh %[R_TMP3],2(%[R_IN])\n\t" /* Load low surrogate.  */ \
00db10
+		  "    slfi %[R_TMP],0xd7c0\n\t"			\
00db10
+		  "    sll %[R_TMP],10\n\t"				\
00db10
+		  "    risbgn %[R_TMP],%[R_TMP3],54,63,0\n\t" /* Insert klmnopqrst.  */ \
00db10
+		  "    nilf %[R_TMP3],0xfc00\n\t"			\
00db10
+		  "    clfi %[R_TMP3],0xdc00\n\t" /* Check if it starts with 0xdc00.  */ \
00db10
+		  "    jne 98f\n\t"					\
00db10
+		  "    st %[R_TMP],0(%[R_OUT])\n\t"			\
00db10
+		  "    la %[R_IN],4(%[R_IN])\n\t"			\
00db10
+		  "    la %[R_OUT],4(%[R_OUT])\n\t"			\
00db10
+		  "    aghi %[R_TMP2],-2\n\t"				\
00db10
+		  "    jh 13b\n\t" /* Handle remaining uint16_t values.  */ \
00db10
+		  "    j 0b\n\t" /* Switch to vx-loop.  */		\
00db10
+		  "96: \n\t" /* Return full output.  */			\
00db10
+		  "    lghi %[R_RES],%[RES_OUT_FULL]\n\t"		\
00db10
+		  "    j 99f\n\t"					\
00db10
+		  "97: \n\t" /* Return incomplete input.  */		\
00db10
+		  "    lghi %[R_RES],%[RES_IN_FULL]\n\t"		\
00db10
+		  "    j 99f\n\t"					\
00db10
+		  "98:\n\t" /* Return Illegal character.  */		\
00db10
+		  "    lghi %[R_RES],%[RES_IN_ILL]\n\t"			\
00db10
+		  "99:\n\t"						\
00db10
+		  ".machine pop"					\
00db10
+		  : /* outputs */ [R_IN] "+a" (inptr)			\
00db10
+		    , [R_INLEN] "+d" (inlen), [R_OUT] "+a" (outptr)	\
00db10
+		    , [R_OUTLEN] "+d" (outlen), [R_TMP] "=a" (tmp)	\
00db10
+		    , [R_TMP2] "=d" (tmp2), [R_TMP3] "=a" (tmp3)	\
00db10
+		    , [R_RES] "+d" (result)				\
00db10
+		  : /* inputs */					\
00db10
+		    [RES_OUT_FULL] "i" (__GCONV_FULL_OUTPUT)		\
00db10
+		    , [RES_IN_ILL] "i" (__GCONV_ILLEGAL_INPUT)		\
00db10
+		    , [RES_IN_FULL] "i" (__GCONV_INCOMPLETE_INPUT)	\
00db10
+		  : /* clobber list */ "memory", "cc"			\
00db10
+		    ASM_CLOBBER_VR ("v16") ASM_CLOBBER_VR ("v17")	\
00db10
+		    ASM_CLOBBER_VR ("v18") ASM_CLOBBER_VR ("v19")	\
00db10
+		    ASM_CLOBBER_VR ("v30") ASM_CLOBBER_VR ("v31")	\
00db10
+		  );							\
00db10
+    if (__glibc_likely (inptr == inend)					\
00db10
+	|| result != __GCONV_ILLEGAL_INPUT)				\
00db10
+      break;								\
00db10
+									\
00db10
+    STANDARD_FROM_LOOP_ERR_HANDLER (2);					\
00db10
+  }
00db10
+
00db10
+
00db10
+/* Generate loop-function with software routing.  */
00db10
+#define MIN_NEEDED_INPUT	MIN_NEEDED_FROM
00db10
+#define MAX_NEEDED_INPUT	MAX_NEEDED_FROM
00db10
+#define MIN_NEEDED_OUTPUT	MIN_NEEDED_TO
00db10
+#if defined HAVE_S390_VX_ASM_SUPPORT
00db10
+# define LOOPFCT		__from_utf16_loop_c
00db10
+# define LOOP_NEED_FLAGS
00db10
+# define BODY			BODY_FROM_C
00db10
+# include <iconv/loop.c>
00db10
+
00db10
+/* Generate loop-function with hardware vector instructions.  */
00db10
+# define MIN_NEEDED_INPUT	MIN_NEEDED_FROM
00db10
+# define MAX_NEEDED_INPUT	MAX_NEEDED_FROM
00db10
+# define MIN_NEEDED_OUTPUT	MIN_NEEDED_TO
00db10
+# define LOOPFCT		__from_utf16_loop_vx
00db10
+# define LOOP_NEED_FLAGS
00db10
+# define BODY			BODY_FROM_VX
00db10
+# include <iconv/loop.c>
00db10
+
00db10
+/* Generate ifunc'ed loop function.  */
00db10
+__typeof(__from_utf16_loop_c)
00db10
+__attribute__ ((ifunc ("__from_utf16_loop_resolver")))
00db10
+__from_utf16_loop;
00db10
+
00db10
+static void *
00db10
+__from_utf16_loop_resolver (unsigned long int dl_hwcap)
00db10
+{
00db10
+  if (dl_hwcap & HWCAP_S390_VX)
00db10
+    return __from_utf16_loop_vx;
00db10
+  else
00db10
+    return __from_utf16_loop_c;
00db10
+}
00db10
+
00db10
+strong_alias (__from_utf16_loop_c_single, __from_utf16_loop_single)
00db10
+#else
00db10
+# define LOOPFCT		FROM_LOOP
00db10
+# define LOOP_NEED_FLAGS
00db10
+# define BODY			BODY_FROM_C
00db10
+# include <iconv/loop.c>
00db10
+#endif
00db10
 
00db10
 /* Conversion from UTF-32 internal/BE to UTF-16.  */
00db10
 
00db10
-#define MIN_NEEDED_INPUT	MIN_NEEDED_TO
00db10
-#define MIN_NEEDED_OUTPUT	MIN_NEEDED_FROM
00db10
-#define MAX_NEEDED_OUTPUT	MAX_NEEDED_FROM
00db10
-#define LOOPFCT			TO_LOOP
00db10
 /* The software routine is copied from utf-16.c (minus bytes
00db10
    swapping).  */
00db10
-#define BODY								\
00db10
+#define BODY_TO_C							\
00db10
   {									\
00db10
-    if (GLRO (dl_hwcap) & HWCAP_S390_ETF3EH)				\
00db10
-      {									\
00db10
-	HARDWARE_CONVERT ("cu42 %0, %1");				\
00db10
-									\
00db10
-	if (inptr != inend)						\
00db10
-	  {								\
00db10
-	    result = __GCONV_INCOMPLETE_INPUT;				\
00db10
-	    break;							\
00db10
-	  }								\
00db10
-	continue;							\
00db10
-      }									\
00db10
-									\
00db10
     uint32_t c = get32 (inptr);						\
00db10
 									\
00db10
     if (__builtin_expect (c <= 0xd7ff, 1)				\
00db10
 	|| (c >=0xdc00 && c <= 0xffff))					\
00db10
       {									\
00db10
-        /* Two UTF-16 chars.  */					\
00db10
-        put16 (outptr, c);						\
00db10
+	/* Two UTF-16 chars.  */					\
00db10
+	put16 (outptr, c);						\
00db10
       }									\
00db10
     else if (__builtin_expect (c >= 0x10000, 1)				\
00db10
 	     && __builtin_expect (c <= 0x10ffff, 1))			\
00db10
       {									\
00db10
 	/* Four UTF-16 chars.  */					\
00db10
-        uint16_t zabcd = ((c & 0x1f0000) >> 16) - 1;			\
00db10
+	uint16_t zabcd = ((c & 0x1f0000) >> 16) - 1;			\
00db10
 	uint16_t out;							\
00db10
 									\
00db10
 	/* Generate a surrogate character.  */				\
00db10
-	if (__glibc_unlikely (outptr + 4 > outend))			      \
00db10
+	if (__glibc_unlikely (outptr + 4 > outend))			\
00db10
 	  {								\
00db10
 	    /* Overflow in the output buffer.  */			\
00db10
 	    result = __GCONV_FULL_OUTPUT;				\
00db10
@@ -326,12 +460,165 @@ gconv_end (struct __gconv_step *data)
00db10
       }									\
00db10
     else								\
00db10
       {									\
00db10
-        STANDARD_TO_LOOP_ERR_HANDLER (4);				\
00db10
+	STANDARD_TO_LOOP_ERR_HANDLER (4);				\
00db10
       }									\
00db10
     outptr += 2;							\
00db10
     inptr += 4;								\
00db10
   }
00db10
+
00db10
+#define BODY_TO_ETF3EH							\
00db10
+  {									\
00db10
+    HARDWARE_CONVERT ("cu42 %0, %1");					\
00db10
+									\
00db10
+    if (__glibc_likely (inptr == inend)					\
00db10
+	|| result == __GCONV_FULL_OUTPUT)				\
00db10
+      break;								\
00db10
+									\
00db10
+    if (inptr + 4 > inend)						\
00db10
+      {									\
00db10
+	result = __GCONV_INCOMPLETE_INPUT;				\
00db10
+	break;								\
00db10
+      }									\
00db10
+									\
00db10
+    STANDARD_TO_LOOP_ERR_HANDLER (4);					\
00db10
+  }
00db10
+
00db10
+#define BODY_TO_VX							\
00db10
+  {									\
00db10
+    register const unsigned char* pInput asm ("8") = inptr;		\
00db10
+    register size_t inlen asm ("9") = inend - inptr;			\
00db10
+    register unsigned char* pOutput asm ("10") = outptr;		\
00db10
+    register size_t outlen asm("11") = outend - outptr;			\
00db10
+    unsigned long tmp, tmp2, tmp3;					\
00db10
+    asm volatile (".machine push\n\t"					\
00db10
+		  ".machine \"z13\"\n\t"				\
00db10
+		  ".machinemode \"zarch_nohighgprs\"\n\t"		\
00db10
+		  /* Setup to check for surrogates.  */			\
00db10
+		  "    larl %[R_TMP],9f\n\t"				\
00db10
+		  "    vlm %%v30,%%v31,0(%[R_TMP])\n\t"			\
00db10
+		  /* Loop which handles UTF-16 chars			\
00db10
+		     ch < 0xd800 || (ch > 0xdfff && ch < 0x10000).  */	\
00db10
+		  "0:  clgijl %[R_INLEN],32,20f\n\t"			\
00db10
+		  "    clgijl %[R_OUTLEN],16,20f\n\t"			\
00db10
+		  "1:  vlm %%v16,%%v17,0(%[R_IN])\n\t"			\
00db10
+		  "    lghi %[R_TMP2],0\n\t"				\
00db10
+		  /* Shorten to UTF-16.  */				\
00db10
+		  "    vpkf %%v18,%%v16,%%v17\n\t"			\
00db10
+		  /* Check for surrogate chars.  */			\
00db10
+		  "    vstrcfs %%v19,%%v16,%%v30,%%v31\n\t"		\
00db10
+		  "    jno 10f\n\t"					\
00db10
+		  "    vstrcfs %%v19,%%v17,%%v30,%%v31\n\t"		\
00db10
+		  "    jno 11f\n\t"					\
00db10
+		  /* Store 16 bytes to buf_out.  */			\
00db10
+		  "    vst %%v18,0(%[R_OUT])\n\t"			\
00db10
+		  "    la %[R_IN],32(%[R_IN])\n\t"			\
00db10
+		  "    aghi %[R_INLEN],-32\n\t"				\
00db10
+		  "    aghi %[R_OUTLEN],-16\n\t"			\
00db10
+		  "    la %[R_OUT],16(%[R_OUT])\n\t"			\
00db10
+		  "    clgijl %[R_INLEN],32,20f\n\t"			\
00db10
+		  "    clgijl %[R_OUTLEN],16,20f\n\t"			\
00db10
+		  "    j 1b\n\t"					\
00db10
+		  /* Setup to check for ch >= 0xd800 && ch <= 0xdfff	\
00db10
+		     and check for ch >= 0x10000. (v30, v31)  */	\
00db10
+		  "9:  .long 0xd800,0xdfff,0x10000,0x10000\n\t"		\
00db10
+		  "    .long 0xa0000000,0xc0000000, 0xa0000000,0xa0000000\n\t" \
00db10
+		  /* At least on UTF32 char is in range of surrogates.	\
00db10
+		     Store the preceding characters.  */		\
00db10
+		  "11: ahi %[R_TMP2],16\n\t"				\
00db10
+		  "10: vlgvb %[R_TMP],%%v19,7\n\t"			\
00db10
+		  "    agr %[R_TMP],%[R_TMP2]\n\t"			\
00db10
+		  "    srlg %[R_TMP3],%[R_TMP],1\n\t" /* Number of out bytes.  */ \
00db10
+		  "    ahik %[R_TMP2],%[R_TMP3],-1\n\t" /* Highest index to store.  */ \
00db10
+		  "    jl 20f\n\t"					\
00db10
+		  "    vstl %%v18,%[R_TMP2],0(%[R_OUT])\n\t"		\
00db10
+		  /* Update pointers.  */				\
00db10
+		  "    la %[R_IN],0(%[R_TMP],%[R_IN])\n\t"		\
00db10
+		  "    slgr %[R_INLEN],%[R_TMP]\n\t"			\
00db10
+		  "    la %[R_OUT],0(%[R_TMP3],%[R_OUT])\n\t"		\
00db10
+		  "    slgr %[R_OUTLEN],%[R_TMP3]\n\t"			\
00db10
+		  /* Handles UTF16 surrogates with convert instruction.  */ \
00db10
+		  "20: cu42 %[R_OUT],%[R_IN]\n\t"			\
00db10
+		  "    jo 0b\n\t" /* Try vector implemenation again.  */ \
00db10
+		  "    lochil %[R_RES],%[RES_OUT_FULL]\n\t" /* cc == 1.  */ \
00db10
+		  "    lochih %[R_RES],%[RES_IN_ILL]\n\t" /* cc == 2.  */ \
00db10
+		  ".machine pop"					\
00db10
+		  : /* outputs */ [R_IN] "+a" (pInput)			\
00db10
+		    , [R_INLEN] "+d" (inlen), [R_OUT] "+a" (pOutput)	\
00db10
+		    , [R_OUTLEN] "+d" (outlen), [R_TMP] "=a" (tmp)	\
00db10
+		    , [R_TMP2] "=d" (tmp2), [R_TMP3] "=a" (tmp3)	\
00db10
+		    , [R_RES] "+d" (result)				\
00db10
+		  : /* inputs */					\
00db10
+		    [RES_OUT_FULL] "i" (__GCONV_FULL_OUTPUT)		\
00db10
+		    , [RES_IN_ILL] "i" (__GCONV_ILLEGAL_INPUT)		\
00db10
+		    , [RES_IN_FULL] "i" (__GCONV_INCOMPLETE_INPUT)	\
00db10
+		  : /* clobber list */ "memory", "cc"			\
00db10
+		    ASM_CLOBBER_VR ("v16") ASM_CLOBBER_VR ("v17")	\
00db10
+		    ASM_CLOBBER_VR ("v18") ASM_CLOBBER_VR ("v19")	\
00db10
+		    ASM_CLOBBER_VR ("v30") ASM_CLOBBER_VR ("v31")	\
00db10
+		  );							\
00db10
+    inptr = pInput;							\
00db10
+    outptr = pOutput;							\
00db10
+									\
00db10
+    if (__glibc_likely (inptr == inend)					\
00db10
+	|| result == __GCONV_FULL_OUTPUT)				\
00db10
+      break;								\
00db10
+    if (inptr + 4 > inend)						\
00db10
+      {									\
00db10
+	result = __GCONV_INCOMPLETE_INPUT;				\
00db10
+	break;								\
00db10
+      }									\
00db10
+    STANDARD_TO_LOOP_ERR_HANDLER (4);					\
00db10
+  }
00db10
+
00db10
+/* Generate loop-function with software routing.  */
00db10
+#define MIN_NEEDED_INPUT	MIN_NEEDED_TO
00db10
+#define MIN_NEEDED_OUTPUT	MIN_NEEDED_FROM
00db10
+#define MAX_NEEDED_OUTPUT	MAX_NEEDED_FROM
00db10
+#define LOOPFCT			__to_utf16_loop_c
00db10
+#define LOOP_NEED_FLAGS
00db10
+#define BODY			BODY_TO_C
00db10
+#include <iconv/loop.c>
00db10
+
00db10
+/* Generate loop-function with hardware utf-convert instruction.  */
00db10
+#define MIN_NEEDED_INPUT	MIN_NEEDED_TO
00db10
+#define MIN_NEEDED_OUTPUT	MIN_NEEDED_FROM
00db10
+#define MAX_NEEDED_OUTPUT	MAX_NEEDED_FROM
00db10
+#define LOOPFCT			__to_utf16_loop_etf3eh
00db10
 #define LOOP_NEED_FLAGS
00db10
+#define BODY			BODY_TO_ETF3EH
00db10
 #include <iconv/loop.c>
00db10
 
00db10
+#if defined HAVE_S390_VX_ASM_SUPPORT
00db10
+/* Generate loop-function with hardware vector instructions.  */
00db10
+# define MIN_NEEDED_INPUT	MIN_NEEDED_TO
00db10
+# define MIN_NEEDED_OUTPUT	MIN_NEEDED_FROM
00db10
+# define MAX_NEEDED_OUTPUT	MAX_NEEDED_FROM
00db10
+# define LOOPFCT		__to_utf16_loop_vx
00db10
+# define LOOP_NEED_FLAGS
00db10
+# define BODY			BODY_TO_VX
00db10
+# include <iconv/loop.c>
00db10
+#endif
00db10
+
00db10
+/* Generate ifunc'ed loop function.  */
00db10
+__typeof(__to_utf16_loop_c)
00db10
+__attribute__ ((ifunc ("__to_utf16_loop_resolver")))
00db10
+__to_utf16_loop;
00db10
+
00db10
+static void *
00db10
+__to_utf16_loop_resolver (unsigned long int dl_hwcap)
00db10
+{
00db10
+#if defined HAVE_S390_VX_ASM_SUPPORT
00db10
+  if (dl_hwcap & HWCAP_S390_VX)
00db10
+    return __to_utf16_loop_vx;
00db10
+  else
00db10
+#endif
00db10
+  if (dl_hwcap & HWCAP_S390_ETF3EH)
00db10
+    return __to_utf16_loop_etf3eh;
00db10
+  else
00db10
+    return __to_utf16_loop_c;
00db10
+}
00db10
+
00db10
+strong_alias (__to_utf16_loop_c_single, __to_utf16_loop_single)
00db10
+
00db10
+
00db10
 #include <iconv/skeleton.c>
00db10
-- 
00db10
1.8.3.1
00db10