|  | /* idn-stub.c --- Stub to dlopen libcidn.so and invoke idna_to_ascii_lz. | 
|  | * Copyright (C) 2003, 2004  Simon Josefsson | 
|  | * | 
|  | * This file is part of GNU Libidn. | 
|  | * | 
|  | * GNU Libidn 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. | 
|  | * | 
|  | * GNU Libidn 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 GNU Libidn; if not, see <http://www.gnu.org/licenses/>. | 
|  | */ | 
|  |  | 
|  | #include <ctype.h> | 
|  | #include <stdio.h> | 
|  | #include <string.h> | 
|  | #include <dlfcn.h> | 
|  | #include <gnu/lib-names.h> | 
|  | #include <libc-lock.h> | 
|  |  | 
|  | /* Get specification for idna_to_ascii_lz. */ | 
|  | #include "idna.h" | 
|  |  | 
|  | /* Handle of the libidn  DSO.  */ | 
|  | static void *h; | 
|  |  | 
|  |  | 
|  | static int (*to_ascii_lz) (const char *input, char **output, int flags); | 
|  | static int (*to_unicode_lzlz) (const char *input, char **output, int flags); | 
|  |  | 
|  |  | 
|  | static void | 
|  | load_dso (void) | 
|  | { | 
|  | /* Lock protecting the DSO loading.  */ | 
|  | __libc_lock_define_initialized (static, lock); | 
|  |  | 
|  | __libc_lock_lock (lock); | 
|  |  | 
|  | /* Retest in case some other thread arrived here at the same time.  */ | 
|  | if (h == NULL) | 
|  | { | 
|  | h = __libc_dlopen (LIBCIDN_SO); | 
|  |  | 
|  | if (h == NULL) | 
|  | h = (void *) 1l; | 
|  | else | 
|  | { | 
|  | /* Get the function we are interested in.  */ | 
|  | to_ascii_lz = __libc_dlsym (h, "idna_to_ascii_lz"); | 
|  | to_unicode_lzlz = __libc_dlsym (h, "idna_to_unicode_lzlz"); | 
|  | if (to_ascii_lz == NULL || to_unicode_lzlz == NULL) | 
|  | { | 
|  | __libc_dlclose (h); | 
|  | h = (void *) 1l; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | __libc_lock_unlock (lock); | 
|  | } | 
|  |  | 
|  |  | 
|  | /* Stub to dlopen libcidn.so and invoke the real idna_to_ascii_lz, or | 
|  | return IDNA_DLOPEN_ERROR on failure.  */ | 
|  | int | 
|  | __idna_to_unicode_lzlz (const char *input, char **output, int flags) | 
|  | { | 
|  | /* If the input string contains no "xn--" prefix for a component of | 
|  | the name we can pass it up right away.  */ | 
|  | const char *cp = input; | 
|  | while (*cp != '\0') | 
|  | { | 
|  | if (strncmp (cp, IDNA_ACE_PREFIX, strlen (IDNA_ACE_PREFIX)) == 0) | 
|  | break; | 
|  |  | 
|  | /* On to the next part of the name.  */ | 
|  | cp = __strchrnul (cp, '.'); | 
|  | if (*cp == '.') | 
|  | ++cp; | 
|  | } | 
|  |  | 
|  | if (*cp == '\0') | 
|  | { | 
|  | *output = (char *) input; | 
|  | return IDNA_SUCCESS; | 
|  | } | 
|  |  | 
|  | if (h == NULL) | 
|  | load_dso (); | 
|  |  | 
|  | if (h == (void *) 1l) | 
|  | return IDNA_DLOPEN_ERROR; | 
|  |  | 
|  | return to_unicode_lzlz (input, output, flags); | 
|  | } | 
|  |  | 
|  |  | 
|  | /* Stub to dlopen libcidn.so and invoke the real idna_to_ascii_lz, or | 
|  | return IDNA_DLOPEN_ERROR on failure.  */ | 
|  | int | 
|  | __idna_to_ascii_lz (const char *input, char **output, int flags) | 
|  | { | 
|  | /* If the input string contains no non-ASCII character the output | 
|  | string will be the same.  No valid locale encoding does not have | 
|  | this property.  */ | 
|  | const char *cp = input; | 
|  | while (*cp != '\0' && isascii (*cp)) | 
|  | ++cp; | 
|  |  | 
|  | if (*cp == '\0') | 
|  | { | 
|  | *output = (char *) input; | 
|  | return IDNA_SUCCESS; | 
|  | } | 
|  |  | 
|  | if (h == NULL) | 
|  | load_dso (); | 
|  |  | 
|  | if (h == (void *) 1l) | 
|  | return IDNA_DLOPEN_ERROR; | 
|  |  | 
|  | return to_ascii_lz (input, output, flags); | 
|  | } | 
|  |  | 
|  |  | 
|  | #if IS_IN (libc) | 
|  | libc_freeres_fn (unload_libidn) | 
|  | { | 
|  | if (h != NULL && h != (void *) 1l) | 
|  | { | 
|  | __libc_dlclose (h); | 
|  | h = (void *) 1l; | 
|  | } | 
|  | } | 
|  | #endif |