lcWrap.c revision 9c019ec5
11ab64890Smrg/*
21ab64890Smrg
31ab64890SmrgCopyright 1991, 1998  The Open Group
41ab64890Smrg
51ab64890SmrgPermission to use, copy, modify, distribute, and sell this software and its
61ab64890Smrgdocumentation for any purpose is hereby granted without fee, provided that
71ab64890Smrgthe above copyright notice appear in all copies and that both that
81ab64890Smrgcopyright notice and this permission notice appear in supporting
91ab64890Smrgdocumentation.
101ab64890Smrg
111ab64890SmrgThe above copyright notice and this permission notice shall be included
121ab64890Smrgin all copies or substantial portions of the Software.
131ab64890Smrg
141ab64890SmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
151ab64890SmrgOR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
161ab64890SmrgMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
171ab64890SmrgIN NO EVENT SHALL THE OPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR
181ab64890SmrgOTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
191ab64890SmrgARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
201ab64890SmrgOTHER DEALINGS IN THE SOFTWARE.
211ab64890Smrg
221ab64890SmrgExcept as contained in this notice, the name of The Open Group shall
231ab64890Smrgnot be used in advertising or otherwise to promote the sale, use or
241ab64890Smrgother dealings in this Software without prior written authorization
251ab64890Smrgfrom The Open Group.
261ab64890Smrg
271ab64890Smrg*/
281ab64890Smrg/*
291ab64890Smrg * Copyright 1991 by the Open Software Foundation
301ab64890Smrg * Copyright 1993 by the TOSHIBA Corp.
311ab64890Smrg *
321ab64890Smrg * Permission to use, copy, modify, distribute, and sell this software and its
331ab64890Smrg * documentation for any purpose is hereby granted without fee, provided that
341ab64890Smrg * the above copyright notice appear in all copies and that both that
351ab64890Smrg * copyright notice and this permission notice appear in supporting
361ab64890Smrg * documentation, and that the names of Open Software Foundation and TOSHIBA
371ab64890Smrg * not be used in advertising or publicity pertaining to distribution of the
381ab64890Smrg * software without specific, written prior permission.  Open Software
391ab64890Smrg * Foundation and TOSHIBA make no representations about the suitability of this
401ab64890Smrg * software for any purpose.  It is provided "as is" without express or
411ab64890Smrg * implied warranty.
421ab64890Smrg *
431ab64890Smrg * OPEN SOFTWARE FOUNDATION AND TOSHIBA DISCLAIM ALL WARRANTIES WITH REGARD TO
441ab64890Smrg * THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
451ab64890Smrg * FITNESS, IN NO EVENT SHALL OPEN SOFTWARE FOUNDATIONN OR TOSHIBA BE
4661b2299dSmrg * LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
471ab64890Smrg * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
481ab64890Smrg * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
491ab64890Smrg * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
5061b2299dSmrg *
5161b2299dSmrg *		 M. Collins		OSF
521ab64890Smrg *
531ab64890Smrg *		 Katsuhisa Yano		TOSHIBA Corp.
5461b2299dSmrg */
551ab64890Smrg
561ab64890Smrg#ifdef HAVE_CONFIG_H
571ab64890Smrg#include <config.h>
581ab64890Smrg#endif
591ab64890Smrg#include <stdlib.h>
601ab64890Smrg#include "Xlibint.h"
611ab64890Smrg#include "Xlcint.h"
621ab64890Smrg#include <X11/Xlocale.h>
631ab64890Smrg#include <X11/Xos.h>
641ab64890Smrg#ifdef WIN32
651ab64890Smrg#undef close
661ab64890Smrg#endif
671ab64890Smrg#include <X11/Xutil.h>
681ab64890Smrg#include "XlcPubI.h"
69258a0ebeSmrg#include "reallocarray.h"
701ab64890Smrg
711ab64890Smrg#ifdef XTHREADS
721ab64890SmrgLockInfoPtr _Xi18n_lock;
731ab64890Smrg#endif
741ab64890Smrg
751ab64890Smrgchar *
761ab64890SmrgXSetLocaleModifiers(
771ab64890Smrg    const char *modifiers)
781ab64890Smrg{
791ab64890Smrg    XLCd lcd = _XlcCurrentLC();
801ab64890Smrg    char *user_mods;
81818534a1Smrg    char *mapped_mods;
821ab64890Smrg
831ab64890Smrg    if (!lcd)
841ab64890Smrg	return (char *) NULL;
851ab64890Smrg    if (!modifiers)
861ab64890Smrg	return lcd->core->modifiers;
871ab64890Smrg    user_mods = getenv("XMODIFIERS");
88818534a1Smrg    mapped_mods = (*lcd->methods->map_modifiers) (lcd, user_mods, modifiers);
89818534a1Smrg    if (mapped_mods) {
900f8248bfSmrg	Xfree(lcd->core->modifiers);
91818534a1Smrg	lcd->core->modifiers = mapped_mods;
921ab64890Smrg    }
93818534a1Smrg    return mapped_mods;
941ab64890Smrg}
951ab64890Smrg
961ab64890SmrgBool
9761b2299dSmrgXSupportsLocale(void)
981ab64890Smrg{
991ab64890Smrg    return _XlcCurrentLC() != (XLCd)NULL;
1001ab64890Smrg}
1011ab64890Smrg
1021ab64890SmrgBool _XlcValidModSyntax(
1031ab64890Smrg    const char * mods,
1041ab64890Smrg    const char * const *valid_mods)
1051ab64890Smrg{
1061ab64890Smrg    int i;
1071ab64890Smrg    const char * const *ptr;
1081ab64890Smrg
1091ab64890Smrg    while (mods && (*mods == '@')) {
1101ab64890Smrg	mods++;
1111ab64890Smrg	if (*mods == '@')
1121ab64890Smrg	    break;
1131ab64890Smrg	for (ptr = valid_mods; *ptr; ptr++) {
1149c019ec5Smaya	    i = (int) strlen(*ptr);
1159c019ec5Smaya	    if (strncmp(mods, *ptr, (size_t) i) || ((mods[i] != '=')
1161ab64890Smrg#ifdef WIN32
1171ab64890Smrg					   && (mods[i] != '#')
1181ab64890Smrg#endif
1191ab64890Smrg					   ))
1201ab64890Smrg		continue;
1211ab64890Smrg	    mods = strchr(mods+i+1, '@');
1221ab64890Smrg	    break;
1231ab64890Smrg	}
1241ab64890Smrg    }
1251ab64890Smrg    return !mods || !*mods;
1261ab64890Smrg}
1271ab64890Smrg
1281ab64890Smrgstatic const char *im_valid[] = {"im", (const char *)NULL};
1291ab64890Smrg
1301ab64890Smrg/*ARGSUSED*/
1311ab64890Smrgchar *
1321ab64890Smrg_XlcDefaultMapModifiers(
1331ab64890Smrg    XLCd lcd,
1341ab64890Smrg    const char *user_mods,
1351ab64890Smrg    const char *prog_mods)
1361ab64890Smrg{
1371ab64890Smrg    int i;
1381ab64890Smrg    char *mods;
1391ab64890Smrg
1401ab64890Smrg    if (!_XlcValidModSyntax(prog_mods, im_valid))
1411ab64890Smrg	return (char *)NULL;
1421ab64890Smrg    if (!_XlcValidModSyntax(user_mods, im_valid))
1431ab64890Smrg	return (char *)NULL;
1449c019ec5Smaya    i = (int) strlen(prog_mods) + 1;
1451ab64890Smrg    if (user_mods)
1469c019ec5Smaya	i = (int) ((size_t) i + strlen(user_mods));
1471ab64890Smrg    mods = Xmalloc(i);
1481ab64890Smrg    if (mods) {
1491ab64890Smrg	strcpy(mods, prog_mods);
1501ab64890Smrg	if (user_mods)
1511ab64890Smrg	    strcat(mods, user_mods);
1521ab64890Smrg#ifdef WIN32
1531ab64890Smrg	{
1541ab64890Smrg	    char *s;
15561b2299dSmrg	    for (s = mods; (s = strchr(s, '@')); s++) {
1561ab64890Smrg		for (s++; *s && *s != '='; s++) {
1571ab64890Smrg		    if (*s == '#') {
1581ab64890Smrg			*s = '=';
1591ab64890Smrg			break;
1601ab64890Smrg		    }
1611ab64890Smrg		}
1621ab64890Smrg	    }
1631ab64890Smrg	}
1641ab64890Smrg#endif
1651ab64890Smrg    }
1661ab64890Smrg    return mods;
1671ab64890Smrg}
1681ab64890Smrg
1691ab64890Smrgtypedef struct _XLCdListRec {
1701ab64890Smrg    struct _XLCdListRec *next;
1711ab64890Smrg    XLCd lcd;
1721ab64890Smrg    int ref_count;
1731ab64890Smrg} XLCdListRec, *XLCdList;
1741ab64890Smrg
1751ab64890Smrgstatic XLCdList lcd_list = NULL;
1761ab64890Smrg
1771ab64890Smrgtypedef struct _XlcLoaderListRec {
1781ab64890Smrg    struct _XlcLoaderListRec *next;
1791ab64890Smrg    XLCdLoadProc proc;
1801ab64890Smrg} XlcLoaderListRec, *XlcLoaderList;
1811ab64890Smrg
1821ab64890Smrgstatic XlcLoaderList loader_list = NULL;
1831ab64890Smrg
1841ab64890Smrgvoid
1851ab64890Smrg_XlcRemoveLoader(
1861ab64890Smrg    XLCdLoadProc proc)
1871ab64890Smrg{
1881ab64890Smrg    XlcLoaderList loader, prev;
1891ab64890Smrg
1901ab64890Smrg    if (loader_list == NULL)
1911ab64890Smrg	return;
1921ab64890Smrg
1931ab64890Smrg    prev = loader = loader_list;
1941ab64890Smrg    if (loader->proc == proc) {
1951ab64890Smrg	loader_list = loader->next;
1961ab64890Smrg	Xfree(loader);
1971ab64890Smrg	return;
1981ab64890Smrg    }
1991ab64890Smrg
2001ab64890Smrg    while ((loader = loader->next)) {
2011ab64890Smrg	if (loader->proc == proc) {
2021ab64890Smrg	    prev->next = loader->next;
2031ab64890Smrg	    Xfree(loader);
2041ab64890Smrg	    return;
2051ab64890Smrg	}
2061ab64890Smrg	prev = loader;
2071ab64890Smrg    }
2081ab64890Smrg
2091ab64890Smrg    return;
2101ab64890Smrg}
2111ab64890Smrg
2121ab64890SmrgBool
2131ab64890Smrg_XlcAddLoader(
2141ab64890Smrg    XLCdLoadProc proc,
2151ab64890Smrg    XlcPosition position)
2161ab64890Smrg{
2171ab64890Smrg    XlcLoaderList loader, last;
2181ab64890Smrg
2191ab64890Smrg    _XlcRemoveLoader(proc);		/* remove old loader, if exist */
2201ab64890Smrg
221818534a1Smrg    loader = Xmalloc(sizeof(XlcLoaderListRec));
2221ab64890Smrg    if (loader == NULL)
2231ab64890Smrg	return False;
2241ab64890Smrg
2251ab64890Smrg    loader->proc = proc;
2261ab64890Smrg
2271ab64890Smrg    if (loader_list == NULL)
2281ab64890Smrg	position = XlcHead;
2291ab64890Smrg
2301ab64890Smrg    if (position == XlcHead) {
2311ab64890Smrg	loader->next = loader_list;
2321ab64890Smrg	loader_list = loader;
2331ab64890Smrg    } else {
2341ab64890Smrg	last = loader_list;
2351ab64890Smrg	while (last->next)
2361ab64890Smrg	    last = last->next;
23761b2299dSmrg
2381ab64890Smrg	loader->next = NULL;
2391ab64890Smrg	last->next = loader;
2401ab64890Smrg    }
2411ab64890Smrg
2421ab64890Smrg    return True;
2431ab64890Smrg}
2441ab64890Smrg
2451ab64890SmrgXLCd
2461ab64890Smrg_XOpenLC(
2471ab64890Smrg    char *name)
2481ab64890Smrg{
2491ab64890Smrg    XLCd lcd;
2501ab64890Smrg    XlcLoaderList loader;
2511ab64890Smrg    XLCdList cur;
2521ab64890Smrg#if !defined(X_LOCALE)
2531ab64890Smrg    int len;
2541ab64890Smrg    char sinamebuf[256];
2551ab64890Smrg    char* siname = sinamebuf;
2561ab64890Smrg#endif
2571ab64890Smrg
2581ab64890Smrg    if (name == NULL) {
2591ab64890Smrg	name = setlocale (LC_CTYPE, (char *)NULL);
2601ab64890Smrg#if !defined(X_LOCALE)
26161b2299dSmrg        /*
26261b2299dSmrg         * _XlMapOSLocaleName will return the same string or a substring
26361b2299dSmrg         * of name, so strlen(name) is okay
2641ab64890Smrg         */
2659c019ec5Smaya        if ((len = (int) strlen(name)) >= sizeof sinamebuf) {
2661ab64890Smrg            siname = Xmalloc (len + 1);
2671ab64890Smrg            if (siname == NULL)
2681ab64890Smrg                return NULL;
2691ab64890Smrg        }
2701ab64890Smrg        name = _XlcMapOSLocaleName(name, siname);
2711ab64890Smrg#endif
2721ab64890Smrg    }
2731ab64890Smrg
2741ab64890Smrg    _XLockMutex(_Xi18n_lock);
2751ab64890Smrg
2761ab64890Smrg    /*
2771ab64890Smrg     * search for needed lcd, if found return it
2781ab64890Smrg     */
2791ab64890Smrg    for (cur = lcd_list; cur; cur = cur->next) {
2801ab64890Smrg	if (!strcmp (cur->lcd->core->name, name)) {
2811ab64890Smrg	    lcd = cur->lcd;
2821ab64890Smrg	    cur->ref_count++;
2831ab64890Smrg	    goto found;
2841ab64890Smrg	}
2851ab64890Smrg    }
2861ab64890Smrg
2871ab64890Smrg    if (!loader_list)
2881ab64890Smrg	_XlcInitLoader();
2891ab64890Smrg
2901ab64890Smrg    /*
2911ab64890Smrg     * not there, so try to get and add to list
2921ab64890Smrg     */
2931ab64890Smrg    for (loader = loader_list; loader; loader = loader->next) {
2941ab64890Smrg	lcd = (*loader->proc)(name);
2951ab64890Smrg	if (lcd) {
296818534a1Smrg	    cur = Xmalloc (sizeof(XLCdListRec));
2971ab64890Smrg	    if (cur) {
2981ab64890Smrg		cur->lcd = lcd;
2991ab64890Smrg		cur->ref_count = 1;
3001ab64890Smrg		cur->next = lcd_list;
3011ab64890Smrg		lcd_list = cur;
3021ab64890Smrg	    } else {
3031ab64890Smrg		(*lcd->methods->close)(lcd);
3041ab64890Smrg		lcd = (XLCd) NULL;
3051ab64890Smrg	    }
3061ab64890Smrg	    goto found;
3071ab64890Smrg	}
3081ab64890Smrg    }
3091ab64890Smrg
3101ab64890Smrg    lcd = NULL;
3111ab64890Smrg
3121ab64890Smrgfound:
3131ab64890Smrg    _XUnlockMutex(_Xi18n_lock);
3141ab64890Smrg
3151ab64890Smrg#if !defined(X_LOCALE)
3161ab64890Smrg    if (siname != sinamebuf) Xfree(siname);
3171ab64890Smrg#endif
3181ab64890Smrg
3191ab64890Smrg    return lcd;
3201ab64890Smrg}
3211ab64890Smrg
3221ab64890Smrgvoid
3231ab64890Smrg_XCloseLC(
3241ab64890Smrg    XLCd lcd)
3251ab64890Smrg{
3261ab64890Smrg    XLCdList cur, *prev;
3271ab64890Smrg
3281ab64890Smrg    for (prev = &lcd_list; (cur = *prev); prev = &cur->next) {
3291ab64890Smrg	if (cur->lcd == lcd) {
3301ab64890Smrg	    if (--cur->ref_count < 1) {
331258a0ebeSmrg		_XlcDestroyLC(lcd);
3321ab64890Smrg		*prev = cur->next;
3331ab64890Smrg		Xfree(cur);
3341ab64890Smrg	    }
3351ab64890Smrg	    break;
3361ab64890Smrg	}
3371ab64890Smrg    }
3381ab64890Smrg
3391ab64890Smrg    if(loader_list) {
3401ab64890Smrg	_XlcDeInitLoader();
3411ab64890Smrg	loader_list = NULL;
3421ab64890Smrg    }
3431ab64890Smrg}
3441ab64890Smrg
3451ab64890Smrg/*
3461ab64890Smrg * Get the XLCd for the current locale
3471ab64890Smrg */
3481ab64890Smrg
3491ab64890SmrgXLCd
35061b2299dSmrg_XlcCurrentLC(void)
3511ab64890Smrg{
3521ab64890Smrg    XLCd lcd;
3531ab64890Smrg    static XLCd last_lcd = NULL;
3541ab64890Smrg
3551ab64890Smrg    lcd = _XOpenLC((char *) NULL);
3561ab64890Smrg
3571ab64890Smrg    if (last_lcd)
3581ab64890Smrg	_XCloseLC(last_lcd);
35961b2299dSmrg
3601ab64890Smrg    last_lcd = lcd;
3611ab64890Smrg
3621ab64890Smrg    return lcd;
3631ab64890Smrg}
3641ab64890Smrg
3651ab64890SmrgXrmMethods
3661ab64890Smrg_XrmInitParseInfo(
3671ab64890Smrg    XPointer *state)
3681ab64890Smrg{
3691ab64890Smrg    XLCd lcd = _XOpenLC((char *) NULL);
37061b2299dSmrg
3711ab64890Smrg    if (lcd == (XLCd) NULL)
3721ab64890Smrg	return (XrmMethods) NULL;
37361b2299dSmrg
3741ab64890Smrg    return (*lcd->methods->init_parse_info)(lcd, state);
3751ab64890Smrg}
3761ab64890Smrg
3771ab64890Smrgint
3781ab64890SmrgXmbTextPropertyToTextList(
3791ab64890Smrg    Display *dpy,
3801ab64890Smrg    const XTextProperty *text_prop,
3811ab64890Smrg    char ***list_ret,
3821ab64890Smrg    int *count_ret)
3831ab64890Smrg{
3841ab64890Smrg    XLCd lcd = _XlcCurrentLC();
38561b2299dSmrg
3861ab64890Smrg    if (lcd == NULL)
3871ab64890Smrg	return XLocaleNotSupported;
3881ab64890Smrg
3891ab64890Smrg    return (*lcd->methods->mb_text_prop_to_list)(lcd, dpy, text_prop, list_ret,
3901ab64890Smrg						 count_ret);
3911ab64890Smrg}
3921ab64890Smrg
3931ab64890Smrgint
3941ab64890SmrgXwcTextPropertyToTextList(
3951ab64890Smrg    Display *dpy,
3961ab64890Smrg    const XTextProperty *text_prop,
3971ab64890Smrg    wchar_t ***list_ret,
3981ab64890Smrg    int *count_ret)
3991ab64890Smrg{
4001ab64890Smrg    XLCd lcd = _XlcCurrentLC();
40161b2299dSmrg
4021ab64890Smrg    if (lcd == NULL)
4031ab64890Smrg	return XLocaleNotSupported;
4041ab64890Smrg
4051ab64890Smrg    return (*lcd->methods->wc_text_prop_to_list)(lcd, dpy, text_prop, list_ret,
4061ab64890Smrg						 count_ret);
4071ab64890Smrg}
4081ab64890Smrg
4091ab64890Smrgint
4101ab64890SmrgXutf8TextPropertyToTextList(
4111ab64890Smrg    Display *dpy,
4121ab64890Smrg    const XTextProperty *text_prop,
4131ab64890Smrg    char ***list_ret,
4141ab64890Smrg    int *count_ret)
4151ab64890Smrg{
4161ab64890Smrg    XLCd lcd = _XlcCurrentLC();
41761b2299dSmrg
4181ab64890Smrg    if (lcd == NULL)
4191ab64890Smrg	return XLocaleNotSupported;
4201ab64890Smrg
4211ab64890Smrg    return (*lcd->methods->utf8_text_prop_to_list)(lcd, dpy, text_prop,
4221ab64890Smrg						   list_ret, count_ret);
4231ab64890Smrg}
4241ab64890Smrg
4251ab64890Smrgint
4261ab64890SmrgXmbTextListToTextProperty(
4271ab64890Smrg    Display *dpy,
4281ab64890Smrg    char **list,
4291ab64890Smrg    int count,
4301ab64890Smrg    XICCEncodingStyle style,
4311ab64890Smrg    XTextProperty *text_prop)
4321ab64890Smrg{
4331ab64890Smrg    XLCd lcd = _XlcCurrentLC();
43461b2299dSmrg
4351ab64890Smrg    if (lcd == NULL)
4361ab64890Smrg	return XLocaleNotSupported;
4371ab64890Smrg
4381ab64890Smrg    return (*lcd->methods->mb_text_list_to_prop)(lcd, dpy, list, count, style,
4391ab64890Smrg						 text_prop);
4401ab64890Smrg}
4411ab64890Smrg
4421ab64890Smrgint
4431ab64890SmrgXwcTextListToTextProperty(
4441ab64890Smrg    Display *dpy,
4451ab64890Smrg    wchar_t **list,
4461ab64890Smrg    int count,
4471ab64890Smrg    XICCEncodingStyle style,
4481ab64890Smrg    XTextProperty *text_prop)
4491ab64890Smrg{
4501ab64890Smrg    XLCd lcd = _XlcCurrentLC();
45161b2299dSmrg
4521ab64890Smrg    if (lcd == NULL)
4531ab64890Smrg	return XLocaleNotSupported;
4541ab64890Smrg
4551ab64890Smrg    return (*lcd->methods->wc_text_list_to_prop)(lcd, dpy, list, count, style,
4561ab64890Smrg						 text_prop);
4571ab64890Smrg}
4581ab64890Smrg
4591ab64890Smrgint
4601ab64890SmrgXutf8TextListToTextProperty(
4611ab64890Smrg    Display *dpy,
4621ab64890Smrg    char **list,
4631ab64890Smrg    int count,
4641ab64890Smrg    XICCEncodingStyle style,
4651ab64890Smrg    XTextProperty *text_prop)
4661ab64890Smrg{
4671ab64890Smrg    XLCd lcd = _XlcCurrentLC();
46861b2299dSmrg
4691ab64890Smrg    if (lcd == NULL)
4701ab64890Smrg	return XLocaleNotSupported;
4711ab64890Smrg
4721ab64890Smrg    return (*lcd->methods->utf8_text_list_to_prop)(lcd, dpy, list, count,
4731ab64890Smrg						   style, text_prop);
4741ab64890Smrg}
4751ab64890Smrg
4761ab64890Smrgvoid
4771ab64890SmrgXwcFreeStringList(
4781ab64890Smrg    wchar_t **list)
4791ab64890Smrg{
4801ab64890Smrg    XLCd lcd = _XlcCurrentLC();
48161b2299dSmrg
4821ab64890Smrg    if (lcd == NULL)
4831ab64890Smrg	return;
4841ab64890Smrg
4851ab64890Smrg    (*lcd->methods->wc_free_string_list)(lcd, list);
4861ab64890Smrg}
4871ab64890Smrg
4881ab64890Smrgconst char *
48961b2299dSmrgXDefaultString(void)
4901ab64890Smrg{
4911ab64890Smrg    XLCd lcd = _XlcCurrentLC();
49261b2299dSmrg
4931ab64890Smrg    if (lcd == NULL)
4941ab64890Smrg	return (char *) NULL;
49561b2299dSmrg
4961ab64890Smrg    return (*lcd->methods->default_string)(lcd);
4971ab64890Smrg}
4981ab64890Smrg
4991ab64890Smrgvoid
5001ab64890Smrg_XlcCopyFromArg(
5011ab64890Smrg    char *src,
5021ab64890Smrg    char *dst,
5031ab64890Smrg    int size)
5041ab64890Smrg{
5051ab64890Smrg    if (size == sizeof(long))
5061ab64890Smrg	*((long *) dst) = (long) src;
5071ab64890Smrg#ifdef LONG64
5081ab64890Smrg    else if (size == sizeof(int))
5091ab64890Smrg	*((int *) dst) = (int)(long) src;
5101ab64890Smrg#endif
5111ab64890Smrg    else if (size == sizeof(short))
5121ab64890Smrg	*((short *) dst) = (short)(long) src;
5131ab64890Smrg    else if (size == sizeof(char))
5141ab64890Smrg	*((char *) dst) = (char)(long) src;
5151ab64890Smrg    else if (size == sizeof(XPointer))
5161ab64890Smrg	*((XPointer *) dst) = (XPointer) src;
5171ab64890Smrg    else if (size > sizeof(XPointer))
5189c019ec5Smaya	memcpy(dst, (char *) src, (size_t) size);
5191ab64890Smrg    else
5209c019ec5Smaya	memcpy(dst, (char *) &src, (size_t) size);
5211ab64890Smrg}
5221ab64890Smrg
5231ab64890Smrgvoid
5241ab64890Smrg_XlcCopyToArg(
5251ab64890Smrg    char *src,
5261ab64890Smrg    char **dst,
5271ab64890Smrg    int size)
5281ab64890Smrg{
5291ab64890Smrg    /* FIXME:
5301ab64890Smrg       On Big Endian machines, this behaves differently than _XCopyToArg. */
5311ab64890Smrg    if (size == sizeof(long))
5321ab64890Smrg	*((long *) *dst) = *((long *) src);
5331ab64890Smrg#ifdef LONG64
5341ab64890Smrg    else if (size == sizeof(int))
5351ab64890Smrg	*((int *) *dst) = *((int *) src);
5361ab64890Smrg#endif
5371ab64890Smrg    else if (size == sizeof(short))
5381ab64890Smrg	*((short *) *dst) = *((short *) src);
5391ab64890Smrg    else if (size == sizeof(char))
5401ab64890Smrg	*((char *) *dst) = *((char *) src);
5411ab64890Smrg    else if (size == sizeof(XPointer))
5421ab64890Smrg	*((XPointer *) *dst) = *((XPointer *) src);
5431ab64890Smrg    else
5449c019ec5Smaya	memcpy(*dst, src, (size_t) size);
5451ab64890Smrg}
5461ab64890Smrg
5471ab64890Smrgvoid
5481ab64890Smrg_XlcCountVaList(
5491ab64890Smrg    va_list var,
5501ab64890Smrg    int *count_ret)
5511ab64890Smrg{
5521ab64890Smrg    int count;
5531ab64890Smrg
5541ab64890Smrg    for (count = 0; va_arg(var, char *); count++)
5551ab64890Smrg	(void)va_arg(var, XPointer);
55661b2299dSmrg
5571ab64890Smrg    *count_ret = count;
5581ab64890Smrg}
5591ab64890Smrg
5601ab64890Smrgvoid
5611ab64890Smrg_XlcVaToArgList(
5621ab64890Smrg    va_list var,
5631ab64890Smrg    int count,
5641ab64890Smrg    XlcArgList *args_ret)
5651ab64890Smrg{
5661ab64890Smrg    XlcArgList args;
5671ab64890Smrg
568258a0ebeSmrg    *args_ret = args = Xmallocarray(count, sizeof(XlcArg));
5691ab64890Smrg    if (args == (XlcArgList) NULL)
5701ab64890Smrg	return;
57161b2299dSmrg
5721ab64890Smrg    for ( ; count-- > 0; args++) {
5731ab64890Smrg	args->name = va_arg(var, char *);
5741ab64890Smrg	args->value = va_arg(var, XPointer);
5751ab64890Smrg    }
5761ab64890Smrg}
5771ab64890Smrg
5781ab64890Smrgvoid
5791ab64890Smrg_XlcCompileResourceList(
5801ab64890Smrg    XlcResourceList resources,
5811ab64890Smrg    int num_resources)
5821ab64890Smrg{
5831ab64890Smrg    for ( ; num_resources-- > 0; resources++)
5841ab64890Smrg	resources->xrm_name = XrmPermStringToQuark(resources->name);
5851ab64890Smrg}
5861ab64890Smrg
5871ab64890Smrgchar *
5881ab64890Smrg_XlcGetValues(
5891ab64890Smrg    XPointer base,
5901ab64890Smrg    XlcResourceList resources,
5911ab64890Smrg    int num_resources,
5921ab64890Smrg    XlcArgList args,
5931ab64890Smrg    int num_args,
5941ab64890Smrg    unsigned long mask)
5951ab64890Smrg{
5961ab64890Smrg    XlcResourceList res;
5971ab64890Smrg    XrmQuark xrm_name;
5981ab64890Smrg    int count;
5991ab64890Smrg
6001ab64890Smrg    for ( ; num_args-- > 0; args++) {
6011ab64890Smrg	res = resources;
6021ab64890Smrg	count = num_resources;
6031ab64890Smrg	xrm_name = XrmPermStringToQuark(args->name);
6041ab64890Smrg
6051ab64890Smrg	for ( ; count-- > 0; res++) {
6061ab64890Smrg	    if (xrm_name == res->xrm_name && (mask & res->mask)) {
6071ab64890Smrg		    _XlcCopyToArg(base + res->offset, &args->value, res->size);
6081ab64890Smrg		break;
6091ab64890Smrg	    }
6101ab64890Smrg	}
6111ab64890Smrg
6121ab64890Smrg	if (count < 0)
6131ab64890Smrg	    return args->name;
6141ab64890Smrg    }
6151ab64890Smrg
6161ab64890Smrg    return NULL;
6171ab64890Smrg}
6181ab64890Smrg
6191ab64890Smrgchar *
6201ab64890Smrg_XlcSetValues(
6211ab64890Smrg    XPointer base,
6221ab64890Smrg    XlcResourceList resources,
6231ab64890Smrg    int num_resources,
6241ab64890Smrg    XlcArgList args,
6251ab64890Smrg    int num_args,
6261ab64890Smrg    unsigned long mask)
6271ab64890Smrg{
6281ab64890Smrg    XlcResourceList res;
6291ab64890Smrg    XrmQuark xrm_name;
6301ab64890Smrg    int count;
6311ab64890Smrg
6321ab64890Smrg    for ( ; num_args-- > 0; args++) {
6331ab64890Smrg	res = resources;
6341ab64890Smrg	count = num_resources;
6351ab64890Smrg	xrm_name = XrmPermStringToQuark(args->name);
6361ab64890Smrg
6371ab64890Smrg	for ( ; count-- > 0; res++) {
6381ab64890Smrg	    if (xrm_name == res->xrm_name && (mask & res->mask)) {
6391ab64890Smrg		_XlcCopyFromArg(args->value, base + res->offset, res->size);
6401ab64890Smrg		break;
6411ab64890Smrg	    }
6421ab64890Smrg	}
6431ab64890Smrg
6441ab64890Smrg	if (count < 0)
6451ab64890Smrg	    return args->name;
6461ab64890Smrg    }
6471ab64890Smrg
6481ab64890Smrg    return NULL;
6491ab64890Smrg}
650