| /* Get descriptor for character set conversion. |
| Copyright (C) 1997-2014 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; |
| } |