XlcDL.c revision 5a473929
11ab64890Smrg/*
21ab64890SmrgCopyright 1985, 1986, 1987, 1991, 1998  The Open Group
31ab64890Smrg
41ab64890SmrgPermission is hereby granted, free of charge, to any person obtaining a
51ab64890Smrgcopy of this software and associated documentation files (the
61ab64890Smrg"Software"), to deal in the Software without restriction, including
71ab64890Smrgwithout limitation the rights to use, copy, modify, merge, publish,
81ab64890Smrgdistribute, sublicense, and/or sell copies of the Software, and to
91ab64890Smrgpermit persons to whom the Software is furnished to do so, subject to
101ab64890Smrgthe following conditions: The above copyright notice and this
111ab64890Smrgpermission notice shall be included in all copies or substantial
121ab64890Smrgportions of the Software.
131ab64890Smrg
141ab64890Smrg
155a473929SmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
165a473929SmrgIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
175a473929SmrgFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
185a473929SmrgOPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
195a473929SmrgAN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
205a473929SmrgCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE
215a473929SmrgEVEN IF ADVISED IN ADVANCE OF THE POSSIBILITY OF SUCH DAMAGES.
221ab64890Smrg
231ab64890Smrg
245a473929SmrgExcept as contained in this notice, the name of The Open Group shall not be
255a473929Smrgused in advertising or otherwise to promote the sale, use or other dealings
265a473929Smrgin this Software without prior written authorization from The Open Group.
271ab64890Smrg
281ab64890Smrg
291ab64890SmrgX Window System is a trademark of The Open Group
301ab64890Smrg
311ab64890SmrgOSF/1, OSF/Motif and Motif are registered trademarks, and OSF, the OSF
321ab64890Smrglogo, LBX, X Window System, and Xinerama are trademarks of the Open
331ab64890SmrgGroup. All other trademarks and registered trademarks mentioned herein
341ab64890Smrgare the property of their respective owners. No right, title or
351ab64890Smrginterest in or to any trademark, service mark, logo or trade name of
361ab64890SmrgSun Microsystems, Inc. or its licensors is granted.
371ab64890Smrg
381ab64890Smrg*/
395a473929Smrg/*
405a473929Smrg * Copyright 2000 Sun Microsystems, Inc.  All rights reserved.
415a473929Smrg *
425a473929Smrg * Permission is hereby granted, free of charge, to any person obtaining a
435a473929Smrg * copy of this software and associated documentation files (the "Software"),
445a473929Smrg * to deal in the Software without restriction, including without limitation
455a473929Smrg * the rights to use, copy, modify, merge, publish, distribute, sublicense,
465a473929Smrg * and/or sell copies of the Software, and to permit persons to whom the
475a473929Smrg * Software is furnished to do so, subject to the following conditions:
485a473929Smrg *
495a473929Smrg * The above copyright notice and this permission notice (including the next
505a473929Smrg * paragraph) shall be included in all copies or substantial portions of the
515a473929Smrg * Software.
525a473929Smrg *
535a473929Smrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
545a473929Smrg * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
555a473929Smrg * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
565a473929Smrg * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
575a473929Smrg * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
585a473929Smrg * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
595a473929Smrg * DEALINGS IN THE SOFTWARE.
605a473929Smrg */
615a473929Smrg
621ab64890Smrg
631ab64890Smrg#ifdef HAVE_CONFIG_H
641ab64890Smrg# include <config.h>
651ab64890Smrg#else
661ab64890Smrg# if defined(hpux)
671ab64890Smrg#  define HAVE_DL_H
681ab64890Smrg# else
691ab64890Smrg#  define HAVE_DLFCN_H
701ab64890Smrg# endif
711ab64890Smrg#endif
721ab64890Smrg
731ab64890Smrg#include <stdio.h>
741ab64890Smrg
751ab64890Smrg#ifdef HAVE_DL_H
761ab64890Smrg#include <dl.h>
771ab64890Smrg#endif
781ab64890Smrg
791ab64890Smrg#ifdef HAVE_DLFCN_H
801ab64890Smrg#include <dlfcn.h>
811ab64890Smrg#endif
821ab64890Smrg
831ab64890Smrg#include <ctype.h>
841ab64890Smrg
851ab64890Smrg#include "Xlibint.h"
861ab64890Smrg#include "XlcPublic.h"
871ab64890Smrg#include "XlcPubI.h"
881ab64890Smrg
899013130fSmartin#if !defined(__NetBSD__) && defined(_LP64)  && defined(__sparcv9)
901ab64890Smrg# define	_MACH64_NAME		"sparcv9"
911ab64890Smrg#else
921ab64890Smrg# undef _MACH64_NAME
931ab64890Smrg#endif /* defined(_LP64)  && defined(__sparcv9) */
941ab64890Smrg
951ab64890Smrg#ifdef _MACH64_NAME
961ab64890Smrg#  define	_MACH64_NAME_LEN	(sizeof (_MACH64_NAME) - 1)
971ab64890Smrg#endif
981ab64890Smrg
991ab64890Smrg#define XI18N_DLREL		2
1001ab64890Smrg
1011ab64890Smrg#define	iscomment(ch)	((ch) == '\0' || (ch) == '#')
1021ab64890Smrg
1031ab64890Smrgtypedef enum {
1041ab64890Smrg  XLC_OBJECT,
1051ab64890Smrg  XIM_OBJECT,
1061ab64890Smrg  XOM_OBJECT
1071ab64890Smrg} XI18NDLType;
1081ab64890Smrg
1091ab64890Smrgtypedef struct {
1101ab64890Smrg  XI18NDLType type;
1111ab64890Smrg  int	locale_name_len;
1121ab64890Smrg  char *locale_name;
1131ab64890Smrg  char *dl_name;
1141ab64890Smrg  char *open;
1151ab64890Smrg  char *im_register;
1161ab64890Smrg  char *im_unregister;
1171ab64890Smrg  int dl_release;
1181ab64890Smrg  unsigned int refcount;
1191ab64890Smrg#if defined(hpux)
1201ab64890Smrg  shl_t dl_module;
1211ab64890Smrg#else
1221ab64890Smrg  void *dl_module;
1231ab64890Smrg#endif
1241ab64890Smrg} XI18NObjectsListRec, *XI18NObjectsList;
1251ab64890Smrg
1261ab64890Smrg#define OBJECT_INIT_LEN 8
1271ab64890Smrg#define OBJECT_INC_LEN 4
1281ab64890Smrgstatic int lc_len = 0;
1291ab64890Smrgstatic XI18NObjectsListRec *xi18n_objects_list = NULL;
1301ab64890Smrgstatic int lc_count = 0;
1311ab64890Smrg
1321ab64890Smrgstatic int
1331ab64890Smrgparse_line(char *line, char **argv, int argsize)
1341ab64890Smrg{
1351ab64890Smrg    int argc = 0;
1361ab64890Smrg    char *p = line;
1371ab64890Smrg
1381ab64890Smrg    while (argc < argsize) {
1391ab64890Smrg	while (isspace(*p)) {
1401ab64890Smrg	    ++p;
1411ab64890Smrg	}
1421ab64890Smrg	if (iscomment(*p)){
1431ab64890Smrg	    break;
1441ab64890Smrg	}
1451ab64890Smrg	argv[argc++] = p;
1461ab64890Smrg	while (!isspace(*p)) {
1471ab64890Smrg	    ++p;
1481ab64890Smrg	}
1491ab64890Smrg	if (iscomment(*p)) {
1501ab64890Smrg	    break;
1511ab64890Smrg	}
1521ab64890Smrg	*p++ = '\0';
1531ab64890Smrg    }
1541ab64890Smrg    return argc;
1551ab64890Smrg}
1561ab64890Smrg
1571ab64890Smrgstatic char *
1581ab64890Smrgstrdup_with_underscore(const char *symbol)
1591ab64890Smrg{
1601ab64890Smrg	char *result;
1611ab64890Smrg
1622f44462dSmrg	if ((result = malloc(strlen(symbol) + 2)) == NULL)
1631ab64890Smrg		return NULL;
1641ab64890Smrg	result[0] = '_';
1651ab64890Smrg	strcpy(result + 1, symbol);
1661ab64890Smrg	return result;
1671ab64890Smrg}
1681ab64890Smrg
1691ab64890Smrg#ifndef hpux
1701ab64890Smrgstatic void *
1711ab64890Smrgtry_both_dlsym (void *handle, char *name)
1721ab64890Smrg{
1731ab64890Smrg    void    *ret;
1741ab64890Smrg
1751ab64890Smrg    ret = dlsym (handle, name);
1761ab64890Smrg    if (!ret)
1771ab64890Smrg    {
1781ab64890Smrg	 name = strdup_with_underscore (name);
1791ab64890Smrg	 if (name)
1801ab64890Smrg	 {
1811ab64890Smrg	     ret = dlsym (handle, name);
1821ab64890Smrg	     free (name);
1831ab64890Smrg	 }
1841ab64890Smrg    }
1851ab64890Smrg    return ret;
1861ab64890Smrg}
1871ab64890Smrg#endif
1881ab64890Smrg
1891ab64890Smrgstatic void
1901ab64890Smrgresolve_object(char *path, const char *lc_name)
1911ab64890Smrg{
1921ab64890Smrg    char filename[BUFSIZ];
1931ab64890Smrg    FILE *fp;
1941ab64890Smrg    char buf[BUFSIZ];
1951ab64890Smrg
1961ab64890Smrg    if (lc_len == 0) { /* True only for the 1st time */
1971ab64890Smrg      lc_len = OBJECT_INIT_LEN;
1981ab64890Smrg      xi18n_objects_list = (XI18NObjectsList)
1991ab64890Smrg	  Xmalloc(sizeof(XI18NObjectsListRec) * lc_len);
2001ab64890Smrg      if (!xi18n_objects_list) return;
2011ab64890Smrg    }
2021ab64890Smrg/*
2031ab64890Smrg1266793
2041ab64890SmrgLimit the length of path to prevent stack buffer corruption.
2051ab64890Smrg    sprintf(filename, "%s/%s", path, "XI18N_OBJS");
2061ab64890Smrg*/
2071ab64890Smrg    sprintf(filename, "%.*s/%s", BUFSIZ - 12, path, "XI18N_OBJS");
2081ab64890Smrg    fp = fopen(filename, "r");
2091ab64890Smrg    if (fp == (FILE *)NULL){
2101ab64890Smrg	return;
2111ab64890Smrg    }
2121ab64890Smrg
2131ab64890Smrg    while (fgets(buf, BUFSIZ, fp) != NULL){
2141ab64890Smrg	char *p = buf;
2151ab64890Smrg	int n;
2161ab64890Smrg	char *args[6];
2171ab64890Smrg	while (isspace(*p)){
2181ab64890Smrg	    ++p;
2191ab64890Smrg	}
2201ab64890Smrg	if (iscomment(*p)){
2211ab64890Smrg	    continue;
2221ab64890Smrg	}
2231ab64890Smrg
2241ab64890Smrg	if (lc_count == lc_len) {
2251ab64890Smrg	  lc_len += OBJECT_INC_LEN;
2261ab64890Smrg	  xi18n_objects_list = (XI18NObjectsList)
2271ab64890Smrg	    Xrealloc(xi18n_objects_list,
2281ab64890Smrg		     sizeof(XI18NObjectsListRec) * lc_len);
2291ab64890Smrg	  if (!xi18n_objects_list) return;
2301ab64890Smrg	}
2311ab64890Smrg	n = parse_line(p, args, 6);
2322f44462dSmrg
2331ab64890Smrg	if (n == 3 || n == 5) {
2341ab64890Smrg	  if (!strcmp(args[0], "XLC")){
2351ab64890Smrg	    xi18n_objects_list[lc_count].type = XLC_OBJECT;
2361ab64890Smrg	  } else if (!strcmp(args[0], "XOM")){
2371ab64890Smrg	    xi18n_objects_list[lc_count].type = XOM_OBJECT;
2381ab64890Smrg	  } else if (!strcmp(args[0], "XIM")){
2391ab64890Smrg	    xi18n_objects_list[lc_count].type = XIM_OBJECT;
2401ab64890Smrg	  }
2411ab64890Smrg	  xi18n_objects_list[lc_count].dl_name = strdup(args[1]);
2421ab64890Smrg	  xi18n_objects_list[lc_count].open = strdup(args[2]);
2431ab64890Smrg	  xi18n_objects_list[lc_count].dl_release = XI18N_DLREL;
2441ab64890Smrg	  xi18n_objects_list[lc_count].locale_name = strdup(lc_name);
2451ab64890Smrg	  xi18n_objects_list[lc_count].refcount = 0;
2461ab64890Smrg	  xi18n_objects_list[lc_count].dl_module = (void*)NULL;
2471ab64890Smrg	  if (n == 5) {
2481ab64890Smrg	    xi18n_objects_list[lc_count].im_register = strdup(args[3]);
2491ab64890Smrg	    xi18n_objects_list[lc_count].im_unregister = strdup(args[4]);
2501ab64890Smrg	  } else {
2511ab64890Smrg	    xi18n_objects_list[lc_count].im_register = NULL;
2521ab64890Smrg	    xi18n_objects_list[lc_count].im_unregister = NULL;
2531ab64890Smrg	  }
2541ab64890Smrg	  lc_count++;
2551ab64890Smrg	}
2561ab64890Smrg    }
2571ab64890Smrg    fclose(fp);
2581ab64890Smrg}
2591ab64890Smrg
2601ab64890Smrgstatic char*
2611ab64890Smrg__lc_path(const char *dl_name, const char *lc_dir)
2621ab64890Smrg{
2631ab64890Smrg    char *path;
2641ab64890Smrg    size_t len;
2651ab64890Smrg
2661ab64890Smrg    /*
2671ab64890Smrg     * reject this for possible security issue
2681ab64890Smrg     */
2691ab64890Smrg    if (strstr (dl_name, "../"))
2701ab64890Smrg	return NULL;
2711ab64890Smrg
2721ab64890Smrg#if defined (_LP64) && defined (_MACH64_NAME)
2731ab64890Smrg    len = (lc_dir ? strlen(lc_dir) : 0 ) +
2741ab64890Smrg	(dl_name ? strlen(dl_name) : 0) + _MACH64_NAME_LEN + 10;
2751ab64890Smrg    path = Xmalloc(len + 1);
2761ab64890Smrg
2771ab64890Smrg    if (strchr(dl_name, '/') != NULL) {
2781ab64890Smrg	char *tmp = strdup(dl_name);
2791ab64890Smrg	char *dl_dir, *dl_file;
2801ab64890Smrg	char *slash_p;
2811ab64890Smrg	slash_p = strchr(tmp, '/');
2821ab64890Smrg	*slash_p = '\0';
2831ab64890Smrg	dl_dir = tmp;
2841ab64890Smrg	dl_file = ++slash_p;
2851ab64890Smrg
2861ab64890Smrg	slash_p = strrchr(lc_dir, '/');
2871ab64890Smrg	*slash_p = '\0';
2881ab64890Smrg	strcpy(path, lc_dir); strcat(path, "/");
2891ab64890Smrg	strcat(path, dl_dir); strcat(path, "/");
2901ab64890Smrg	strcat(path, _MACH64_NAME); strcat(path, "/");
2911ab64890Smrg	strcat(path, dl_file); strcat(path, ".so.2");
2921ab64890Smrg
2931ab64890Smrg	*slash_p = '/';
2941ab64890Smrg	Xfree(tmp);
2951ab64890Smrg    } else {
2961ab64890Smrg	strcpy(path, lc_dir); strcat(path, "/");
2971ab64890Smrg	strcat(path, _MACH64_NAME); strcat(path, "/");
2981ab64890Smrg	strcat(path, dl_name); strcat(path, ".so.2");
2991ab64890Smrg    }
3001ab64890Smrg#else
3011ab64890Smrg    len = (lc_dir ? strlen(lc_dir) : 0 ) +
3021ab64890Smrg	(dl_name ? strlen(dl_name) : 0) + 10;
3031ab64890Smrg#if defined POSTLOCALELIBDIR
3041ab64890Smrg    len += (strlen(POSTLOCALELIBDIR) + 1);
3051ab64890Smrg#endif
3061ab64890Smrg    path = Xmalloc(len + 1);
3071ab64890Smrg
3081ab64890Smrg    if (strchr(dl_name, '/') != NULL) {
3091ab64890Smrg	char *slash_p;
3101ab64890Smrg	slash_p = strrchr(lc_dir, '/');
3111ab64890Smrg	*slash_p = '\0';
3121ab64890Smrg	strcpy(path, lc_dir); strcat(path, "/");
3131ab64890Smrg#if defined POSTLOCALELIBDIR
3141ab64890Smrg	strcat(path, POSTLOCALELIBDIR); strcat(path, "/");
3151ab64890Smrg#endif
3161ab64890Smrg	strcat(path, dl_name); strcat(path, ".so.2");
3171ab64890Smrg	*slash_p = '/';
3181ab64890Smrg    } else {
3191ab64890Smrg	strcpy(path, lc_dir); strcat(path, "/");
3201ab64890Smrg#if defined POSTLOCALELIBDIR
3211ab64890Smrg	strcat(path, POSTLOCALELIBDIR); strcat(path, "/");
3221ab64890Smrg#endif
3231ab64890Smrg	strcat(path, dl_name); strcat(path, ".so.2");
3241ab64890Smrg    }
3251ab64890Smrg#endif
3261ab64890Smrg    return path;
3271ab64890Smrg}
3281ab64890Smrg
3291ab64890Smrg/* We reference count dlopen() and dlclose() of modules; unfortunately,
3301ab64890Smrg * since XCloseIM, XCloseOM, XlcClose aren't wrapped, but directly
3311ab64890Smrg * call the close method of the object, we leak a reference count every
3321ab64890Smrg * time we open then close a module. Fixing this would require
3331ab64890Smrg * either creating proxy objects or hooks for close_im/close_om
3341ab64890Smrg * in XLCd
3351ab64890Smrg */
3361ab64890Smrgstatic Bool
3371ab64890Smrgopen_object(
3381ab64890Smrg     XI18NObjectsList object,
3391ab64890Smrg     char *lc_dir)
3401ab64890Smrg{
3411ab64890Smrg  char *path;
3422f44462dSmrg
3431ab64890Smrg  if (object->refcount == 0) {
3441ab64890Smrg      path = __lc_path(object->dl_name, lc_dir);
3451ab64890Smrg      if (!path)
3461ab64890Smrg	  return False;
3471ab64890Smrg#if defined(hpux)
3481ab64890Smrg      object->dl_module = shl_load(path, BIND_DEFERRED, 0L);
3491ab64890Smrg#else
3501ab64890Smrg      object->dl_module = dlopen(path, RTLD_LAZY);
3511ab64890Smrg#endif
3521ab64890Smrg      Xfree(path);
3531ab64890Smrg
3541ab64890Smrg      if (!object->dl_module)
3551ab64890Smrg	  return False;
3561ab64890Smrg    }
3571ab64890Smrg
3581ab64890Smrg  object->refcount++;
3591ab64890Smrg  return True;
3601ab64890Smrg}
3611ab64890Smrg
3621ab64890Smrgstatic void *
3631ab64890Smrgfetch_symbol(
3641ab64890Smrg     XI18NObjectsList object,
3651ab64890Smrg     char *symbol)
3661ab64890Smrg{
3671ab64890Smrg    void *result = NULL;
3681ab64890Smrg#if defined(hpux)
3691ab64890Smrg    int getsyms_cnt, i;
3701ab64890Smrg    struct shl_symbol *symbols;
3711ab64890Smrg#endif
3721ab64890Smrg
3731ab64890Smrg    if (symbol == NULL)
3741ab64890Smrg    	return NULL;
3751ab64890Smrg
3761ab64890Smrg#if defined(hpux)
3771ab64890Smrg    getsyms_cnt = shl_getsymbols(object->dl_module, TYPE_PROCEDURE,
3781ab64890Smrg				 EXPORT_SYMBOLS, malloc, &symbols);
3791ab64890Smrg
3801ab64890Smrg    for(i=0; i<getsyms_cnt; i++) {
3811ab64890Smrg        if(!strcmp(symbols[i].name, symbol)) {
3821ab64890Smrg	    result = symbols[i].value;
3831ab64890Smrg	    break;
3841ab64890Smrg         }
3851ab64890Smrg    }
3861ab64890Smrg
3871ab64890Smrg    if(getsyms_cnt > 0) {
3881ab64890Smrg        free(symbols);
3891ab64890Smrg    }
3901ab64890Smrg#else
3911ab64890Smrg    result = try_both_dlsym(object->dl_module, symbol);
3921ab64890Smrg#endif
3931ab64890Smrg
3941ab64890Smrg    return result;
3951ab64890Smrg}
3961ab64890Smrg
3971ab64890Smrgstatic void
3981ab64890Smrgclose_object(XI18NObjectsList object)
3991ab64890Smrg{
4001ab64890Smrg  object->refcount--;
4011ab64890Smrg  if (object->refcount == 0)
4021ab64890Smrg    {
4031ab64890Smrg#if defined(hpux)
4041ab64890Smrg        shl_unload(object->dl_module);
4051ab64890Smrg#else
4061ab64890Smrg        dlclose(object->dl_module);
4071ab64890Smrg#endif
4081ab64890Smrg        object->dl_module = NULL;
4091ab64890Smrg    }
4101ab64890Smrg}
4111ab64890Smrg
4121ab64890Smrg
4131ab64890Smrgtypedef XLCd (*dynamicLoadProc)(const char *);
4141ab64890Smrg
4151ab64890SmrgXLCd
4161ab64890Smrg_XlcDynamicLoad(const char *lc_name)
4171ab64890Smrg{
4181ab64890Smrg    XLCd lcd = (XLCd)NULL;
4191ab64890Smrg    dynamicLoadProc lc_loader = (dynamicLoadProc)NULL;
4201ab64890Smrg    int count;
4211ab64890Smrg    XI18NObjectsList objects_list;
4221ab64890Smrg    char lc_dir[BUFSIZE], lc_lib_dir[BUFSIZE];
4231ab64890Smrg
4241ab64890Smrg    if (lc_name == NULL) return (XLCd)NULL;
4251ab64890Smrg
4261ab64890Smrg    if (_XlcLocaleDirName(lc_dir, BUFSIZE, (char *)lc_name) == (char *)NULL)
4271ab64890Smrg        return (XLCd)NULL;
4281ab64890Smrg    if (_XlcLocaleLibDirName(lc_lib_dir, BUFSIZE, (char *)lc_name) == (char*)NULL)
4291ab64890Smrg	return (XLCd)NULL;
4301ab64890Smrg
4311ab64890Smrg    resolve_object(lc_dir, lc_name);
4321ab64890Smrg    resolve_object(lc_lib_dir, lc_name);
4331ab64890Smrg
4341ab64890Smrg    objects_list = xi18n_objects_list;
4351ab64890Smrg    count = lc_count;
4361ab64890Smrg    for (; count-- > 0; objects_list++) {
4371ab64890Smrg        if (objects_list->type != XLC_OBJECT ||
4381ab64890Smrg	    strcmp(objects_list->locale_name, lc_name)) continue;
4391ab64890Smrg	if (!open_object (objects_list, lc_dir) && \
4401ab64890Smrg            !open_object (objects_list, lc_lib_dir))
4411ab64890Smrg	    continue;
4421ab64890Smrg
4431ab64890Smrg	lc_loader = (dynamicLoadProc)fetch_symbol (objects_list, objects_list->open);
4441ab64890Smrg	if (!lc_loader) continue;
4451ab64890Smrg	lcd = (*lc_loader)(lc_name);
4461ab64890Smrg	if (lcd != (XLCd)NULL) {
4471ab64890Smrg	    break;
4481ab64890Smrg	}
4492f44462dSmrg
4501ab64890Smrg	close_object (objects_list);
4511ab64890Smrg    }
4521ab64890Smrg    return (XLCd)lcd;
4531ab64890Smrg}
4541ab64890Smrg
4551ab64890Smrg
4561ab64890Smrgtypedef XIM (*dynamicOpenProcp)(XLCd, Display *, XrmDatabase, char *, char *);
4571ab64890Smrg
4581ab64890Smrgstatic XIM
4591ab64890Smrg_XDynamicOpenIM(XLCd lcd, Display *display, XrmDatabase rdb,
4601ab64890Smrg		char *res_name, char *res_class)
4611ab64890Smrg{
4621ab64890Smrg  XIM im = (XIM)NULL;
4631ab64890Smrg  char lc_dir[BUFSIZE];
4641ab64890Smrg  char *lc_name;
4651ab64890Smrg  dynamicOpenProcp im_openIM = (dynamicOpenProcp)NULL;
4661ab64890Smrg  int count;
4671ab64890Smrg  XI18NObjectsList objects_list = xi18n_objects_list;
4681ab64890Smrg
4691ab64890Smrg  lc_name = lcd->core->name;
4701ab64890Smrg
4711ab64890Smrg  if (_XlcLocaleLibDirName(lc_dir, BUFSIZE, lc_name) == NULL) return (XIM)0;
4721ab64890Smrg
4731ab64890Smrg  count = lc_count;
4741ab64890Smrg  for (; count-- > 0; objects_list++) {
4751ab64890Smrg    if (objects_list->type != XIM_OBJECT ||
4761ab64890Smrg	strcmp(objects_list->locale_name, lc_name)) continue;
4771ab64890Smrg
4781ab64890Smrg    if (!open_object (objects_list, lc_dir))
4791ab64890Smrg        continue;
4801ab64890Smrg
4811ab64890Smrg    im_openIM = (dynamicOpenProcp)fetch_symbol(objects_list, objects_list->open);
4821ab64890Smrg    if (!im_openIM) continue;
4831ab64890Smrg    im = (*im_openIM)(lcd, display, rdb, res_name, res_class);
4841ab64890Smrg    if (im != (XIM)NULL) {
4851ab64890Smrg        break;
4861ab64890Smrg    }
4872f44462dSmrg
4881ab64890Smrg    close_object (objects_list);
4891ab64890Smrg  }
4901ab64890Smrg  return (XIM)im;
4911ab64890Smrg}
4921ab64890Smrg
4931ab64890Smrgtypedef Bool (*dynamicRegisterCBProcp)(
4941ab64890Smrg    XLCd, Display *, XrmDatabase, char *, char *, XIDProc, XPointer);
4951ab64890Smrg
4961ab64890Smrgstatic Bool
4971ab64890Smrg_XDynamicRegisterIMInstantiateCallback(
4981ab64890Smrg    XLCd	 lcd,
4991ab64890Smrg    Display	*display,
5001ab64890Smrg    XrmDatabase	 rdb,
5011ab64890Smrg    char	*res_name,
5021ab64890Smrg    char        *res_class,
5031ab64890Smrg    XIDProc	 callback,
5041ab64890Smrg    XPointer	 client_data)
5051ab64890Smrg{
5061ab64890Smrg  char lc_dir[BUFSIZE];
5071ab64890Smrg  char *lc_name;
5081ab64890Smrg  dynamicRegisterCBProcp im_registerIM = (dynamicRegisterCBProcp)NULL;
5091ab64890Smrg  Bool ret_flag = False;
5101ab64890Smrg  int count;
5111ab64890Smrg  XI18NObjectsList objects_list = xi18n_objects_list;
5121ab64890Smrg#if defined(hpux)
5131ab64890Smrg  int getsyms_cnt, i;
5141ab64890Smrg  struct shl_symbol *symbols;
5151ab64890Smrg#endif
5161ab64890Smrg
5171ab64890Smrg  lc_name = lcd->core->name;
5181ab64890Smrg
5191ab64890Smrg  if (_XlcLocaleLibDirName(lc_dir, BUFSIZE, lc_name) == NULL) return False;
5201ab64890Smrg
5211ab64890Smrg  count = lc_count;
5221ab64890Smrg  for (; count-- > 0; objects_list++) {
5231ab64890Smrg    if (objects_list->type != XIM_OBJECT ||
5241ab64890Smrg	strcmp(objects_list->locale_name, lc_name)) continue;
5251ab64890Smrg
5261ab64890Smrg    if (!open_object (objects_list, lc_dir))
5271ab64890Smrg        continue;
5281ab64890Smrg    im_registerIM = (dynamicRegisterCBProcp)fetch_symbol(objects_list,
5291ab64890Smrg					    objects_list->im_register);
5301ab64890Smrg    if (!im_registerIM) continue;
5311ab64890Smrg    ret_flag = (*im_registerIM)(lcd, display, rdb,
5321ab64890Smrg				res_name, res_class,
5331ab64890Smrg				callback, client_data);
5341ab64890Smrg    if (ret_flag) break;
5351ab64890Smrg
5361ab64890Smrg    close_object (objects_list);
5371ab64890Smrg  }
5381ab64890Smrg  return (Bool)ret_flag;
5391ab64890Smrg}
5401ab64890Smrg
5411ab64890Smrgtypedef Bool (*dynamicUnregisterProcp)(
5421ab64890Smrg    XLCd, Display *, XrmDatabase, char *, char *, XIDProc, XPointer);
5431ab64890Smrg
5441ab64890Smrgstatic Bool
5451ab64890Smrg_XDynamicUnRegisterIMInstantiateCallback(
5461ab64890Smrg    XLCd	 lcd,
5471ab64890Smrg    Display	*display,
5481ab64890Smrg    XrmDatabase	 rdb,
5491ab64890Smrg    char	*res_name,
5501ab64890Smrg    char        *res_class,
5511ab64890Smrg    XIDProc	 callback,
5521ab64890Smrg    XPointer	 client_data)
5531ab64890Smrg{
5541ab64890Smrg  char lc_dir[BUFSIZE];
5551ab64890Smrg  char *lc_name;
5561ab64890Smrg  dynamicUnregisterProcp im_unregisterIM = (dynamicUnregisterProcp)NULL;
5571ab64890Smrg  Bool ret_flag = False;
5581ab64890Smrg  int count;
5591ab64890Smrg  XI18NObjectsList objects_list = xi18n_objects_list;
5601ab64890Smrg#if defined(hpux)
5611ab64890Smrg  int getsyms_cnt, i;
5621ab64890Smrg  struct shl_symbol *symbols;
5631ab64890Smrg#endif
5641ab64890Smrg
5651ab64890Smrg  lc_name = lcd->core->name;
5661ab64890Smrg  if (_XlcLocaleDirName(lc_dir, BUFSIZE, lc_name) == NULL) return False;
5671ab64890Smrg
5681ab64890Smrg  count = lc_count;
5691ab64890Smrg  for (; count-- > 0; objects_list++) {
5701ab64890Smrg    if (objects_list->type != XIM_OBJECT ||
5711ab64890Smrg	strcmp(objects_list->locale_name, lc_name)) continue;
5721ab64890Smrg
5731ab64890Smrg    if (!objects_list->refcount) /* Must already be opened */
5741ab64890Smrg        continue;
5751ab64890Smrg
5761ab64890Smrg    im_unregisterIM = (dynamicUnregisterProcp)fetch_symbol(objects_list,
5771ab64890Smrg					      objects_list->im_unregister);
5781ab64890Smrg
5791ab64890Smrg    if (!im_unregisterIM) continue;
5801ab64890Smrg    ret_flag = (*im_unregisterIM)(lcd, display, rdb,
5811ab64890Smrg				  res_name, res_class,
5821ab64890Smrg				  callback, client_data);
5831ab64890Smrg    if (ret_flag) {
5841ab64890Smrg        close_object (objects_list); /* opened in RegisterIMInstantiateCallback */
5851ab64890Smrg	break;
5861ab64890Smrg    }
5871ab64890Smrg  }
5881ab64890Smrg  return (Bool)ret_flag;
5891ab64890Smrg}
5901ab64890Smrg
5911ab64890SmrgBool
5921ab64890Smrg_XInitDynamicIM(XLCd lcd)
5931ab64890Smrg{
5941ab64890Smrg    if(lcd == (XLCd)NULL)
5951ab64890Smrg	return False;
5961ab64890Smrg    lcd->methods->open_im = _XDynamicOpenIM;
5971ab64890Smrg    lcd->methods->register_callback = _XDynamicRegisterIMInstantiateCallback;
5981ab64890Smrg    lcd->methods->unregister_callback = _XDynamicUnRegisterIMInstantiateCallback;
5991ab64890Smrg    return True;
6001ab64890Smrg}
6011ab64890Smrg
6021ab64890Smrg
6031ab64890Smrgtypedef XOM (*dynamicIOpenProcp)(
6041ab64890Smrg        XLCd, Display *, XrmDatabase, _Xconst char *, _Xconst char *);
6051ab64890Smrg
6061ab64890Smrgstatic XOM
6071ab64890Smrg_XDynamicOpenOM(XLCd lcd, Display *display, XrmDatabase rdb,
6081ab64890Smrg		_Xconst char *res_name, _Xconst char *res_class)
6091ab64890Smrg{
6101ab64890Smrg  XOM om = (XOM)NULL;
6111ab64890Smrg  int count;
6121ab64890Smrg  char lc_dir[BUFSIZE];
6131ab64890Smrg  char *lc_name;
6141ab64890Smrg  dynamicIOpenProcp om_openOM = (dynamicIOpenProcp)NULL;
6151ab64890Smrg  XI18NObjectsList objects_list = xi18n_objects_list;
6161ab64890Smrg#if defined(hpux)
6171ab64890Smrg  int getsyms_cnt, i;
6181ab64890Smrg  struct shl_symbol *symbols;
6191ab64890Smrg#endif
6201ab64890Smrg
6211ab64890Smrg  lc_name = lcd->core->name;
6221ab64890Smrg
6231ab64890Smrg  if (_XlcLocaleLibDirName(lc_dir, BUFSIZE, lc_name) == NULL) return (XOM)0;
6241ab64890Smrg
6251ab64890Smrg  count = lc_count;
6261ab64890Smrg  for (; count-- > 0; objects_list++) {
6271ab64890Smrg    if (objects_list->type != XOM_OBJECT ||
6281ab64890Smrg	strcmp(objects_list->locale_name, lc_name)) continue;
6291ab64890Smrg    if (!open_object (objects_list, lc_dir))
6301ab64890Smrg        continue;
6312f44462dSmrg
6321ab64890Smrg    om_openOM = (dynamicIOpenProcp)fetch_symbol(objects_list, objects_list->open);
6331ab64890Smrg    if (!om_openOM) continue;
6341ab64890Smrg    om = (*om_openOM)(lcd, display, rdb, res_name, res_class);
6351ab64890Smrg    if (om != (XOM)NULL) {
6361ab64890Smrg        break;
6371ab64890Smrg    }
6381ab64890Smrg    close_object(objects_list);
6391ab64890Smrg  }
6401ab64890Smrg  return (XOM)om;
6411ab64890Smrg}
6421ab64890Smrg
6431ab64890SmrgBool
6441ab64890Smrg_XInitDynamicOM(XLCd lcd)
6451ab64890Smrg{
6461ab64890Smrg    if(lcd == (XLCd)NULL)
6471ab64890Smrg	return False;
6481ab64890Smrg
6491ab64890Smrg    lcd->methods->open_om = _XDynamicOpenOM;
6501ab64890Smrg
6511ab64890Smrg    return True;
6521ab64890Smrg}
653