XlcDL.c revision ff3f3c6f
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/* 40ff3f3c6fSmrg * Copyright (c) 2000, Oracle and/or its affiliates. 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" 88f2d49d05Smrg#include "reallocarray.h" 891ab64890Smrg 901ab64890Smrg#define XI18N_DLREL 2 911ab64890Smrg 921ab64890Smrg#define iscomment(ch) ((ch) == '\0' || (ch) == '#') 931ab64890Smrg 941ab64890Smrgtypedef enum { 951ab64890Smrg XLC_OBJECT, 961ab64890Smrg XIM_OBJECT, 971ab64890Smrg XOM_OBJECT 981ab64890Smrg} XI18NDLType; 991ab64890Smrg 1001ab64890Smrgtypedef struct { 1011ab64890Smrg XI18NDLType type; 1021ab64890Smrg int locale_name_len; 1031ab64890Smrg char *locale_name; 1041ab64890Smrg char *dl_name; 1051ab64890Smrg char *open; 1061ab64890Smrg char *im_register; 1071ab64890Smrg char *im_unregister; 1081ab64890Smrg int dl_release; 1091ab64890Smrg unsigned int refcount; 1101ab64890Smrg#if defined(hpux) 1111ab64890Smrg shl_t dl_module; 1121ab64890Smrg#else 1131ab64890Smrg void *dl_module; 1141ab64890Smrg#endif 1151ab64890Smrg} XI18NObjectsListRec, *XI18NObjectsList; 1161ab64890Smrg 1171ab64890Smrg#define OBJECT_INIT_LEN 8 1181ab64890Smrg#define OBJECT_INC_LEN 4 1191ab64890Smrgstatic int lc_len = 0; 1201ab64890Smrgstatic XI18NObjectsListRec *xi18n_objects_list = NULL; 1211ab64890Smrgstatic int lc_count = 0; 1221ab64890Smrg 1231ab64890Smrgstatic int 1241ab64890Smrgparse_line(char *line, char **argv, int argsize) 1251ab64890Smrg{ 1261ab64890Smrg int argc = 0; 1271ab64890Smrg char *p = line; 1281ab64890Smrg 1291ab64890Smrg while (argc < argsize) { 1301ab64890Smrg while (isspace(*p)) { 1311ab64890Smrg ++p; 1321ab64890Smrg } 1331ab64890Smrg if (iscomment(*p)){ 1341ab64890Smrg break; 1351ab64890Smrg } 1361ab64890Smrg argv[argc++] = p; 1371ab64890Smrg while (!isspace(*p)) { 1381ab64890Smrg ++p; 1391ab64890Smrg } 1401ab64890Smrg if (iscomment(*p)) { 1411ab64890Smrg break; 1421ab64890Smrg } 1431ab64890Smrg *p++ = '\0'; 1441ab64890Smrg } 1451ab64890Smrg return argc; 1461ab64890Smrg} 1471ab64890Smrg 1481ab64890Smrgstatic char * 1491ab64890Smrgstrdup_with_underscore(const char *symbol) 1501ab64890Smrg{ 1511ab64890Smrg char *result; 1521ab64890Smrg 1532f44462dSmrg if ((result = malloc(strlen(symbol) + 2)) == NULL) 1541ab64890Smrg return NULL; 1551ab64890Smrg result[0] = '_'; 1561ab64890Smrg strcpy(result + 1, symbol); 1571ab64890Smrg return result; 1581ab64890Smrg} 1591ab64890Smrg 1601ab64890Smrg#ifndef hpux 1611ab64890Smrgstatic void * 1621ab64890Smrgtry_both_dlsym (void *handle, char *name) 1631ab64890Smrg{ 1641ab64890Smrg void *ret; 1651ab64890Smrg 1661ab64890Smrg ret = dlsym (handle, name); 1671ab64890Smrg if (!ret) 1681ab64890Smrg { 1691ab64890Smrg name = strdup_with_underscore (name); 1701ab64890Smrg if (name) 1711ab64890Smrg { 1721ab64890Smrg ret = dlsym (handle, name); 1731ab64890Smrg free (name); 1741ab64890Smrg } 1751ab64890Smrg } 1761ab64890Smrg return ret; 1771ab64890Smrg} 1781ab64890Smrg#endif 1791ab64890Smrg 1801ab64890Smrgstatic void 1811ab64890Smrgresolve_object(char *path, const char *lc_name) 1821ab64890Smrg{ 1831ab64890Smrg char filename[BUFSIZ]; 1841ab64890Smrg FILE *fp; 1851ab64890Smrg char buf[BUFSIZ]; 1861ab64890Smrg 1871ab64890Smrg if (lc_len == 0) { /* True only for the 1st time */ 1881ab64890Smrg lc_len = OBJECT_INIT_LEN; 189f2d49d05Smrg xi18n_objects_list = Xmallocarray(lc_len, sizeof(XI18NObjectsListRec)); 1901ab64890Smrg if (!xi18n_objects_list) return; 1911ab64890Smrg } 192c555af55Smrg snprintf(filename, sizeof(filename), "%s/%s", path, "XI18N_OBJS"); 1931ab64890Smrg fp = fopen(filename, "r"); 1941ab64890Smrg if (fp == (FILE *)NULL){ 1951ab64890Smrg return; 1961ab64890Smrg } 1971ab64890Smrg 1981ab64890Smrg while (fgets(buf, BUFSIZ, fp) != NULL){ 1991ab64890Smrg char *p = buf; 2001ab64890Smrg int n; 2011ab64890Smrg char *args[6]; 2021ab64890Smrg while (isspace(*p)){ 2031ab64890Smrg ++p; 2041ab64890Smrg } 2051ab64890Smrg if (iscomment(*p)){ 2061ab64890Smrg continue; 2071ab64890Smrg } 2081ab64890Smrg 2091ab64890Smrg if (lc_count == lc_len) { 210c555af55Smrg int new_len = lc_len + OBJECT_INC_LEN; 211f2d49d05Smrg XI18NObjectsListRec *tmp = 212f2d49d05Smrg Xreallocarray(xi18n_objects_list, new_len, 213f2d49d05Smrg sizeof(XI18NObjectsListRec)); 214c555af55Smrg if (tmp == NULL) 215c555af55Smrg goto done; 216c555af55Smrg xi18n_objects_list = tmp; 217c555af55Smrg lc_len = new_len; 2181ab64890Smrg } 2191ab64890Smrg n = parse_line(p, args, 6); 2202f44462dSmrg 2211ab64890Smrg if (n == 3 || n == 5) { 2221ab64890Smrg if (!strcmp(args[0], "XLC")){ 2231ab64890Smrg xi18n_objects_list[lc_count].type = XLC_OBJECT; 2241ab64890Smrg } else if (!strcmp(args[0], "XOM")){ 2251ab64890Smrg xi18n_objects_list[lc_count].type = XOM_OBJECT; 2261ab64890Smrg } else if (!strcmp(args[0], "XIM")){ 2271ab64890Smrg xi18n_objects_list[lc_count].type = XIM_OBJECT; 2281ab64890Smrg } 2291ab64890Smrg xi18n_objects_list[lc_count].dl_name = strdup(args[1]); 2301ab64890Smrg xi18n_objects_list[lc_count].open = strdup(args[2]); 2311ab64890Smrg xi18n_objects_list[lc_count].dl_release = XI18N_DLREL; 2321ab64890Smrg xi18n_objects_list[lc_count].locale_name = strdup(lc_name); 2331ab64890Smrg xi18n_objects_list[lc_count].refcount = 0; 2341ab64890Smrg xi18n_objects_list[lc_count].dl_module = (void*)NULL; 2351ab64890Smrg if (n == 5) { 2361ab64890Smrg xi18n_objects_list[lc_count].im_register = strdup(args[3]); 2371ab64890Smrg xi18n_objects_list[lc_count].im_unregister = strdup(args[4]); 2381ab64890Smrg } else { 2391ab64890Smrg xi18n_objects_list[lc_count].im_register = NULL; 2401ab64890Smrg xi18n_objects_list[lc_count].im_unregister = NULL; 2411ab64890Smrg } 2421ab64890Smrg lc_count++; 2431ab64890Smrg } 2441ab64890Smrg } 245c555af55Smrg done: 2461ab64890Smrg fclose(fp); 2471ab64890Smrg} 2481ab64890Smrg 2491ab64890Smrgstatic char* 2501ab64890Smrg__lc_path(const char *dl_name, const char *lc_dir) 2511ab64890Smrg{ 2521ab64890Smrg char *path; 2531ab64890Smrg size_t len; 254dac667f7Smrg char *slash_p; 2551ab64890Smrg 2561ab64890Smrg /* 2571ab64890Smrg * reject this for possible security issue 2581ab64890Smrg */ 2591ab64890Smrg if (strstr (dl_name, "../")) 2601ab64890Smrg return NULL; 2611ab64890Smrg 2621ab64890Smrg len = (lc_dir ? strlen(lc_dir) : 0 ) + 2631ab64890Smrg (dl_name ? strlen(dl_name) : 0) + 10; 2641ab64890Smrg#if defined POSTLOCALELIBDIR 2651ab64890Smrg len += (strlen(POSTLOCALELIBDIR) + 1); 2661ab64890Smrg#endif 2671ab64890Smrg path = Xmalloc(len + 1); 2681ab64890Smrg 2691ab64890Smrg if (strchr(dl_name, '/') != NULL) { 2701ab64890Smrg slash_p = strrchr(lc_dir, '/'); 2711ab64890Smrg *slash_p = '\0'; 272dac667f7Smrg } else 273dac667f7Smrg slash_p = NULL; 274dac667f7Smrg 2751ab64890Smrg#if defined POSTLOCALELIBDIR 276dac667f7Smrg snprintf(path, len + 1, "%s/%s/%s.so.2", 277dac667f7Smrg lc_dir, POSTLOCALELIBDIR, dl_name); 278dac667f7Smrg#else 279dac667f7Smrg snprintf(path, len + 1, "%s/%s.so.2", lc_dir, dl_name); 2801ab64890Smrg#endif 281dac667f7Smrg 282dac667f7Smrg if (slash_p != NULL) 2831ab64890Smrg *slash_p = '/'; 284dac667f7Smrg 2851ab64890Smrg return path; 2861ab64890Smrg} 2871ab64890Smrg 2881ab64890Smrg/* We reference count dlopen() and dlclose() of modules; unfortunately, 2891ab64890Smrg * since XCloseIM, XCloseOM, XlcClose aren't wrapped, but directly 2901ab64890Smrg * call the close method of the object, we leak a reference count every 2911ab64890Smrg * time we open then close a module. Fixing this would require 2921ab64890Smrg * either creating proxy objects or hooks for close_im/close_om 2931ab64890Smrg * in XLCd 2941ab64890Smrg */ 2951ab64890Smrgstatic Bool 2961ab64890Smrgopen_object( 2971ab64890Smrg XI18NObjectsList object, 2981ab64890Smrg char *lc_dir) 2991ab64890Smrg{ 3001ab64890Smrg char *path; 3012f44462dSmrg 3021ab64890Smrg if (object->refcount == 0) { 3031ab64890Smrg path = __lc_path(object->dl_name, lc_dir); 3041ab64890Smrg if (!path) 3051ab64890Smrg return False; 3061ab64890Smrg#if defined(hpux) 3071ab64890Smrg object->dl_module = shl_load(path, BIND_DEFERRED, 0L); 3081ab64890Smrg#else 3091ab64890Smrg object->dl_module = dlopen(path, RTLD_LAZY); 3101ab64890Smrg#endif 3111ab64890Smrg Xfree(path); 3121ab64890Smrg 3131ab64890Smrg if (!object->dl_module) 3141ab64890Smrg return False; 3151ab64890Smrg } 3161ab64890Smrg 3171ab64890Smrg object->refcount++; 3181ab64890Smrg return True; 3191ab64890Smrg} 3201ab64890Smrg 3211ab64890Smrgstatic void * 3221ab64890Smrgfetch_symbol( 3231ab64890Smrg XI18NObjectsList object, 3241ab64890Smrg char *symbol) 3251ab64890Smrg{ 3261ab64890Smrg void *result = NULL; 3271ab64890Smrg#if defined(hpux) 3281ab64890Smrg int getsyms_cnt, i; 3291ab64890Smrg struct shl_symbol *symbols; 3301ab64890Smrg#endif 3311ab64890Smrg 3321ab64890Smrg if (symbol == NULL) 3331ab64890Smrg return NULL; 3341ab64890Smrg 3351ab64890Smrg#if defined(hpux) 3361ab64890Smrg getsyms_cnt = shl_getsymbols(object->dl_module, TYPE_PROCEDURE, 3371ab64890Smrg EXPORT_SYMBOLS, malloc, &symbols); 3381ab64890Smrg 3391ab64890Smrg for(i=0; i<getsyms_cnt; i++) { 3401ab64890Smrg if(!strcmp(symbols[i].name, symbol)) { 3411ab64890Smrg result = symbols[i].value; 3421ab64890Smrg break; 3431ab64890Smrg } 3441ab64890Smrg } 3451ab64890Smrg 3461ab64890Smrg if(getsyms_cnt > 0) { 3471ab64890Smrg free(symbols); 3481ab64890Smrg } 3491ab64890Smrg#else 3501ab64890Smrg result = try_both_dlsym(object->dl_module, symbol); 3511ab64890Smrg#endif 3521ab64890Smrg 3531ab64890Smrg return result; 3541ab64890Smrg} 3551ab64890Smrg 3561ab64890Smrgstatic void 3571ab64890Smrgclose_object(XI18NObjectsList object) 3581ab64890Smrg{ 3591ab64890Smrg object->refcount--; 3601ab64890Smrg if (object->refcount == 0) 3611ab64890Smrg { 3621ab64890Smrg#if defined(hpux) 3631ab64890Smrg shl_unload(object->dl_module); 3641ab64890Smrg#else 3651ab64890Smrg dlclose(object->dl_module); 3661ab64890Smrg#endif 3671ab64890Smrg object->dl_module = NULL; 3681ab64890Smrg } 3691ab64890Smrg} 3701ab64890Smrg 3711ab64890Smrg 3721ab64890Smrgtypedef XLCd (*dynamicLoadProc)(const char *); 3731ab64890Smrg 3741ab64890SmrgXLCd 3751ab64890Smrg_XlcDynamicLoad(const char *lc_name) 3761ab64890Smrg{ 3771ab64890Smrg XLCd lcd = (XLCd)NULL; 3781ab64890Smrg dynamicLoadProc lc_loader = (dynamicLoadProc)NULL; 3791ab64890Smrg int count; 3801ab64890Smrg XI18NObjectsList objects_list; 3811ab64890Smrg char lc_dir[BUFSIZE], lc_lib_dir[BUFSIZE]; 3821ab64890Smrg 3831ab64890Smrg if (lc_name == NULL) return (XLCd)NULL; 3841ab64890Smrg 385c555af55Smrg if (_XlcLocaleDirName(lc_dir, BUFSIZE, lc_name) == NULL) 3861ab64890Smrg return (XLCd)NULL; 387c555af55Smrg if (_XlcLocaleLibDirName(lc_lib_dir, BUFSIZE, lc_name) == NULL) 3881ab64890Smrg return (XLCd)NULL; 3891ab64890Smrg 3901ab64890Smrg resolve_object(lc_dir, lc_name); 3911ab64890Smrg resolve_object(lc_lib_dir, lc_name); 3921ab64890Smrg 3931ab64890Smrg objects_list = xi18n_objects_list; 3941ab64890Smrg count = lc_count; 3951ab64890Smrg for (; count-- > 0; objects_list++) { 3961ab64890Smrg if (objects_list->type != XLC_OBJECT || 3971ab64890Smrg strcmp(objects_list->locale_name, lc_name)) continue; 3981ab64890Smrg if (!open_object (objects_list, lc_dir) && \ 3991ab64890Smrg !open_object (objects_list, lc_lib_dir)) 4001ab64890Smrg continue; 4011ab64890Smrg 4021ab64890Smrg lc_loader = (dynamicLoadProc)fetch_symbol (objects_list, objects_list->open); 4031ab64890Smrg if (!lc_loader) continue; 4041ab64890Smrg lcd = (*lc_loader)(lc_name); 4051ab64890Smrg if (lcd != (XLCd)NULL) { 4061ab64890Smrg break; 4071ab64890Smrg } 4082f44462dSmrg 4091ab64890Smrg close_object (objects_list); 4101ab64890Smrg } 4111ab64890Smrg return (XLCd)lcd; 4121ab64890Smrg} 4131ab64890Smrg 4141ab64890Smrg 4151ab64890Smrgtypedef XIM (*dynamicOpenProcp)(XLCd, Display *, XrmDatabase, char *, char *); 4161ab64890Smrg 4171ab64890Smrgstatic XIM 4181ab64890Smrg_XDynamicOpenIM(XLCd lcd, Display *display, XrmDatabase rdb, 4191ab64890Smrg char *res_name, char *res_class) 4201ab64890Smrg{ 4211ab64890Smrg XIM im = (XIM)NULL; 4221ab64890Smrg char lc_dir[BUFSIZE]; 4231ab64890Smrg char *lc_name; 4241ab64890Smrg dynamicOpenProcp im_openIM = (dynamicOpenProcp)NULL; 4251ab64890Smrg int count; 4261ab64890Smrg XI18NObjectsList objects_list = xi18n_objects_list; 4271ab64890Smrg 4281ab64890Smrg lc_name = lcd->core->name; 4291ab64890Smrg 4301ab64890Smrg if (_XlcLocaleLibDirName(lc_dir, BUFSIZE, lc_name) == NULL) return (XIM)0; 4311ab64890Smrg 4321ab64890Smrg count = lc_count; 4331ab64890Smrg for (; count-- > 0; objects_list++) { 4341ab64890Smrg if (objects_list->type != XIM_OBJECT || 4351ab64890Smrg strcmp(objects_list->locale_name, lc_name)) continue; 4361ab64890Smrg 4371ab64890Smrg if (!open_object (objects_list, lc_dir)) 4381ab64890Smrg continue; 4391ab64890Smrg 4401ab64890Smrg im_openIM = (dynamicOpenProcp)fetch_symbol(objects_list, objects_list->open); 4411ab64890Smrg if (!im_openIM) continue; 4421ab64890Smrg im = (*im_openIM)(lcd, display, rdb, res_name, res_class); 4431ab64890Smrg if (im != (XIM)NULL) { 4441ab64890Smrg break; 4451ab64890Smrg } 4462f44462dSmrg 4471ab64890Smrg close_object (objects_list); 4481ab64890Smrg } 4491ab64890Smrg return (XIM)im; 4501ab64890Smrg} 4511ab64890Smrg 4521ab64890Smrgtypedef Bool (*dynamicRegisterCBProcp)( 4531ab64890Smrg XLCd, Display *, XrmDatabase, char *, char *, XIDProc, XPointer); 4541ab64890Smrg 4551ab64890Smrgstatic Bool 4561ab64890Smrg_XDynamicRegisterIMInstantiateCallback( 4571ab64890Smrg XLCd lcd, 4581ab64890Smrg Display *display, 4591ab64890Smrg XrmDatabase rdb, 4601ab64890Smrg char *res_name, 4611ab64890Smrg char *res_class, 4621ab64890Smrg XIDProc callback, 4631ab64890Smrg XPointer client_data) 4641ab64890Smrg{ 4651ab64890Smrg char lc_dir[BUFSIZE]; 4661ab64890Smrg char *lc_name; 4671ab64890Smrg dynamicRegisterCBProcp im_registerIM = (dynamicRegisterCBProcp)NULL; 4681ab64890Smrg Bool ret_flag = False; 4691ab64890Smrg int count; 4701ab64890Smrg XI18NObjectsList objects_list = xi18n_objects_list; 4711ab64890Smrg#if defined(hpux) 4721ab64890Smrg int getsyms_cnt, i; 4731ab64890Smrg struct shl_symbol *symbols; 4741ab64890Smrg#endif 4751ab64890Smrg 4761ab64890Smrg lc_name = lcd->core->name; 4771ab64890Smrg 4781ab64890Smrg if (_XlcLocaleLibDirName(lc_dir, BUFSIZE, lc_name) == NULL) return False; 4791ab64890Smrg 4801ab64890Smrg count = lc_count; 4811ab64890Smrg for (; count-- > 0; objects_list++) { 4821ab64890Smrg if (objects_list->type != XIM_OBJECT || 4831ab64890Smrg strcmp(objects_list->locale_name, lc_name)) continue; 4841ab64890Smrg 4851ab64890Smrg if (!open_object (objects_list, lc_dir)) 4861ab64890Smrg continue; 4871ab64890Smrg im_registerIM = (dynamicRegisterCBProcp)fetch_symbol(objects_list, 4881ab64890Smrg objects_list->im_register); 4891ab64890Smrg if (!im_registerIM) continue; 4901ab64890Smrg ret_flag = (*im_registerIM)(lcd, display, rdb, 4911ab64890Smrg res_name, res_class, 4921ab64890Smrg callback, client_data); 4931ab64890Smrg if (ret_flag) break; 4941ab64890Smrg 4951ab64890Smrg close_object (objects_list); 4961ab64890Smrg } 4971ab64890Smrg return (Bool)ret_flag; 4981ab64890Smrg} 4991ab64890Smrg 5001ab64890Smrgtypedef Bool (*dynamicUnregisterProcp)( 5011ab64890Smrg XLCd, Display *, XrmDatabase, char *, char *, XIDProc, XPointer); 5021ab64890Smrg 5031ab64890Smrgstatic Bool 5041ab64890Smrg_XDynamicUnRegisterIMInstantiateCallback( 5051ab64890Smrg XLCd lcd, 5061ab64890Smrg Display *display, 5071ab64890Smrg XrmDatabase rdb, 5081ab64890Smrg char *res_name, 5091ab64890Smrg char *res_class, 5101ab64890Smrg XIDProc callback, 5111ab64890Smrg XPointer client_data) 5121ab64890Smrg{ 5131ab64890Smrg char lc_dir[BUFSIZE]; 514c555af55Smrg const char *lc_name; 5151ab64890Smrg dynamicUnregisterProcp im_unregisterIM = (dynamicUnregisterProcp)NULL; 5161ab64890Smrg Bool ret_flag = False; 5171ab64890Smrg int count; 5181ab64890Smrg XI18NObjectsList objects_list = xi18n_objects_list; 5191ab64890Smrg#if defined(hpux) 5201ab64890Smrg int getsyms_cnt, i; 5211ab64890Smrg struct shl_symbol *symbols; 5221ab64890Smrg#endif 5231ab64890Smrg 5241ab64890Smrg lc_name = lcd->core->name; 5251ab64890Smrg if (_XlcLocaleDirName(lc_dir, BUFSIZE, lc_name) == NULL) return False; 5261ab64890Smrg 5271ab64890Smrg count = lc_count; 5281ab64890Smrg for (; count-- > 0; objects_list++) { 5291ab64890Smrg if (objects_list->type != XIM_OBJECT || 5301ab64890Smrg strcmp(objects_list->locale_name, lc_name)) continue; 5311ab64890Smrg 5321ab64890Smrg if (!objects_list->refcount) /* Must already be opened */ 5331ab64890Smrg continue; 5341ab64890Smrg 5351ab64890Smrg im_unregisterIM = (dynamicUnregisterProcp)fetch_symbol(objects_list, 5361ab64890Smrg objects_list->im_unregister); 5371ab64890Smrg 5381ab64890Smrg if (!im_unregisterIM) continue; 5391ab64890Smrg ret_flag = (*im_unregisterIM)(lcd, display, rdb, 5401ab64890Smrg res_name, res_class, 5411ab64890Smrg callback, client_data); 5421ab64890Smrg if (ret_flag) { 5431ab64890Smrg close_object (objects_list); /* opened in RegisterIMInstantiateCallback */ 5441ab64890Smrg break; 5451ab64890Smrg } 5461ab64890Smrg } 5471ab64890Smrg return (Bool)ret_flag; 5481ab64890Smrg} 5491ab64890Smrg 5501ab64890SmrgBool 5511ab64890Smrg_XInitDynamicIM(XLCd lcd) 5521ab64890Smrg{ 5531ab64890Smrg if(lcd == (XLCd)NULL) 5541ab64890Smrg return False; 5551ab64890Smrg lcd->methods->open_im = _XDynamicOpenIM; 5561ab64890Smrg lcd->methods->register_callback = _XDynamicRegisterIMInstantiateCallback; 5571ab64890Smrg lcd->methods->unregister_callback = _XDynamicUnRegisterIMInstantiateCallback; 5581ab64890Smrg return True; 5591ab64890Smrg} 5601ab64890Smrg 5611ab64890Smrg 5621ab64890Smrgtypedef XOM (*dynamicIOpenProcp)( 5631ab64890Smrg XLCd, Display *, XrmDatabase, _Xconst char *, _Xconst char *); 5641ab64890Smrg 5651ab64890Smrgstatic XOM 5661ab64890Smrg_XDynamicOpenOM(XLCd lcd, Display *display, XrmDatabase rdb, 5671ab64890Smrg _Xconst char *res_name, _Xconst char *res_class) 5681ab64890Smrg{ 5691ab64890Smrg XOM om = (XOM)NULL; 5701ab64890Smrg int count; 5711ab64890Smrg char lc_dir[BUFSIZE]; 5721ab64890Smrg char *lc_name; 5731ab64890Smrg dynamicIOpenProcp om_openOM = (dynamicIOpenProcp)NULL; 5741ab64890Smrg XI18NObjectsList objects_list = xi18n_objects_list; 5751ab64890Smrg#if defined(hpux) 5761ab64890Smrg int getsyms_cnt, i; 5771ab64890Smrg struct shl_symbol *symbols; 5781ab64890Smrg#endif 5791ab64890Smrg 5801ab64890Smrg lc_name = lcd->core->name; 5811ab64890Smrg 5821ab64890Smrg if (_XlcLocaleLibDirName(lc_dir, BUFSIZE, lc_name) == NULL) return (XOM)0; 5831ab64890Smrg 5841ab64890Smrg count = lc_count; 5851ab64890Smrg for (; count-- > 0; objects_list++) { 5861ab64890Smrg if (objects_list->type != XOM_OBJECT || 5871ab64890Smrg strcmp(objects_list->locale_name, lc_name)) continue; 5881ab64890Smrg if (!open_object (objects_list, lc_dir)) 5891ab64890Smrg continue; 5902f44462dSmrg 5911ab64890Smrg om_openOM = (dynamicIOpenProcp)fetch_symbol(objects_list, objects_list->open); 5921ab64890Smrg if (!om_openOM) continue; 5931ab64890Smrg om = (*om_openOM)(lcd, display, rdb, res_name, res_class); 5941ab64890Smrg if (om != (XOM)NULL) { 5951ab64890Smrg break; 5961ab64890Smrg } 5971ab64890Smrg close_object(objects_list); 5981ab64890Smrg } 5991ab64890Smrg return (XOM)om; 6001ab64890Smrg} 6011ab64890Smrg 6021ab64890SmrgBool 6031ab64890Smrg_XInitDynamicOM(XLCd lcd) 6041ab64890Smrg{ 6051ab64890Smrg if(lcd == (XLCd)NULL) 6061ab64890Smrg return False; 6071ab64890Smrg 6081ab64890Smrg lcd->methods->open_om = _XDynamicOpenOM; 6091ab64890Smrg 6101ab64890Smrg return True; 6111ab64890Smrg} 612