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