| /* Get descriptor for character set conversion. | 
 |    Copyright (C) 1997-2016 Free Software Foundation, Inc. | 
 |    This file is part of the GNU C Library. | 
 |    Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997. | 
 |  | 
 |    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 <alloca.h> | 
 | #include <errno.h> | 
 | #include <iconv.h> | 
 | #include <stdbool.h> | 
 | #include <stdlib.h> | 
 | #include <string.h> | 
 |  | 
 | #include <gconv_int.h> | 
 | #include "gconv_charset.h" | 
 |  | 
 |  | 
 | iconv_t | 
 | iconv_open (const char *tocode, const char *fromcode) | 
 | { | 
 |   /* Normalize the name.  We remove all characters beside alpha-numeric, | 
 |      '_', '-', '/', '.', and ':'.  */ | 
 |   size_t tocode_len = strlen (tocode) + 3; | 
 |   char *tocode_conv; | 
 |   bool tocode_usealloca = __libc_use_alloca (tocode_len); | 
 |   if (tocode_usealloca) | 
 |     tocode_conv = (char *) alloca (tocode_len); | 
 |   else | 
 |     { | 
 |       tocode_conv = (char *) malloc (tocode_len); | 
 |       if (tocode_conv == NULL) | 
 | 	return (iconv_t) -1; | 
 |     } | 
 |   strip (tocode_conv, tocode); | 
 |   tocode = (tocode_conv[2] == '\0' && tocode[0] != '\0' | 
 | 	    ? upstr (tocode_conv, tocode) : tocode_conv); | 
 |  | 
 |   size_t fromcode_len = strlen (fromcode) + 3; | 
 |   char *fromcode_conv; | 
 |   bool fromcode_usealloca = __libc_use_alloca (fromcode_len); | 
 |   if (fromcode_usealloca) | 
 |     fromcode_conv = (char *) alloca (fromcode_len); | 
 |   else | 
 |     { | 
 |       fromcode_conv = (char *) malloc (fromcode_len); | 
 |       if (fromcode_conv == NULL) | 
 | 	{ | 
 | 	  if (! tocode_usealloca) | 
 | 	    free (tocode_conv); | 
 | 	  return (iconv_t) -1; | 
 | 	} | 
 |     } | 
 |   strip (fromcode_conv, fromcode); | 
 |   fromcode = (fromcode_conv[2] == '\0' && fromcode[0] != '\0' | 
 | 	      ? upstr (fromcode_conv, fromcode) : fromcode_conv); | 
 |  | 
 |   __gconv_t cd; | 
 |   int res = __gconv_open (tocode, fromcode, &cd, 0); | 
 |  | 
 |   if (! fromcode_usealloca) | 
 |     free (fromcode_conv); | 
 |   if (! tocode_usealloca) | 
 |     free (tocode_conv); | 
 |  | 
 |   if (__builtin_expect (res, __GCONV_OK) != __GCONV_OK) | 
 |     { | 
 |       /* We must set the error number according to the specs.  */ | 
 |       if (res == __GCONV_NOCONV || res == __GCONV_NODB) | 
 | 	__set_errno (EINVAL); | 
 |  | 
 |       cd = (iconv_t) -1; | 
 |     } | 
 |  | 
 |   return (iconv_t) cd; | 
 | } |