lcWrap.c revision 0f8248bf
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"
691ab64890Smrg
701ab64890Smrg#ifdef XTHREADS
711ab64890SmrgLockInfoPtr _Xi18n_lock;
721ab64890Smrg#endif
731ab64890Smrg
741ab64890Smrgchar *
751ab64890SmrgXSetLocaleModifiers(
761ab64890Smrg    const char *modifiers)
771ab64890Smrg{
781ab64890Smrg    XLCd lcd = _XlcCurrentLC();
791ab64890Smrg    char *user_mods;
80818534a1Smrg    char *mapped_mods;
811ab64890Smrg
821ab64890Smrg    if (!lcd)
831ab64890Smrg	return (char *) NULL;
841ab64890Smrg    if (!modifiers)
851ab64890Smrg	return lcd->core->modifiers;
861ab64890Smrg    user_mods = getenv("XMODIFIERS");
87818534a1Smrg    mapped_mods = (*lcd->methods->map_modifiers) (lcd, user_mods, modifiers);
88818534a1Smrg    if (mapped_mods) {
890f8248bfSmrg	Xfree(lcd->core->modifiers);
90818534a1Smrg	lcd->core->modifiers = mapped_mods;
911ab64890Smrg    }
92818534a1Smrg    return mapped_mods;
931ab64890Smrg}
941ab64890Smrg
951ab64890SmrgBool
9661b2299dSmrgXSupportsLocale(void)
971ab64890Smrg{
981ab64890Smrg    return _XlcCurrentLC() != (XLCd)NULL;
991ab64890Smrg}
1001ab64890Smrg
1011ab64890SmrgBool _XlcValidModSyntax(
1021ab64890Smrg    const char * mods,
1031ab64890Smrg    const char * const *valid_mods)
1041ab64890Smrg{
1051ab64890Smrg    int i;
1061ab64890Smrg    const char * const *ptr;
1071ab64890Smrg
1081ab64890Smrg    while (mods && (*mods == '@')) {
1091ab64890Smrg	mods++;
1101ab64890Smrg	if (*mods == '@')
1111ab64890Smrg	    break;
1121ab64890Smrg	for (ptr = valid_mods; *ptr; ptr++) {
1131ab64890Smrg	    i = strlen(*ptr);
1141ab64890Smrg	    if (strncmp(mods, *ptr, i) || ((mods[i] != '=')
1151ab64890Smrg#ifdef WIN32
1161ab64890Smrg					   && (mods[i] != '#')
1171ab64890Smrg#endif
1181ab64890Smrg					   ))
1191ab64890Smrg		continue;
1201ab64890Smrg	    mods = strchr(mods+i+1, '@');
1211ab64890Smrg	    break;
1221ab64890Smrg	}
1231ab64890Smrg    }
1241ab64890Smrg    return !mods || !*mods;
1251ab64890Smrg}
1261ab64890Smrg
1271ab64890Smrgstatic const char *im_valid[] = {"im", (const char *)NULL};
1281ab64890Smrg
1291ab64890Smrg/*ARGSUSED*/
1301ab64890Smrgchar *
1311ab64890Smrg_XlcDefaultMapModifiers(
1321ab64890Smrg    XLCd lcd,
1331ab64890Smrg    const char *user_mods,
1341ab64890Smrg    const char *prog_mods)
1351ab64890Smrg{
1361ab64890Smrg    int i;
1371ab64890Smrg    char *mods;
1381ab64890Smrg
1391ab64890Smrg    if (!_XlcValidModSyntax(prog_mods, im_valid))
1401ab64890Smrg	return (char *)NULL;
1411ab64890Smrg    if (!_XlcValidModSyntax(user_mods, im_valid))
1421ab64890Smrg	return (char *)NULL;
1431ab64890Smrg    i = strlen(prog_mods) + 1;
1441ab64890Smrg    if (user_mods)
1451ab64890Smrg	i += strlen(user_mods);
1461ab64890Smrg    mods = Xmalloc(i);
1471ab64890Smrg    if (mods) {
1481ab64890Smrg	strcpy(mods, prog_mods);
1491ab64890Smrg	if (user_mods)
1501ab64890Smrg	    strcat(mods, user_mods);
1511ab64890Smrg#ifdef WIN32
1521ab64890Smrg	{
1531ab64890Smrg	    char *s;
15461b2299dSmrg	    for (s = mods; (s = strchr(s, '@')); s++) {
1551ab64890Smrg		for (s++; *s && *s != '='; s++) {
1561ab64890Smrg		    if (*s == '#') {
1571ab64890Smrg			*s = '=';
1581ab64890Smrg			break;
1591ab64890Smrg		    }
1601ab64890Smrg		}
1611ab64890Smrg	    }
1621ab64890Smrg	}
1631ab64890Smrg#endif
1641ab64890Smrg    }
1651ab64890Smrg    return mods;
1661ab64890Smrg}
1671ab64890Smrg
1681ab64890Smrgtypedef struct _XLCdListRec {
1691ab64890Smrg    struct _XLCdListRec *next;
1701ab64890Smrg    XLCd lcd;
1711ab64890Smrg    int ref_count;
1721ab64890Smrg} XLCdListRec, *XLCdList;
1731ab64890Smrg
1741ab64890Smrgstatic XLCdList lcd_list = NULL;
1751ab64890Smrg
1761ab64890Smrgtypedef struct _XlcLoaderListRec {
1771ab64890Smrg    struct _XlcLoaderListRec *next;
1781ab64890Smrg    XLCdLoadProc proc;
1791ab64890Smrg} XlcLoaderListRec, *XlcLoaderList;
1801ab64890Smrg
1811ab64890Smrgstatic XlcLoaderList loader_list = NULL;
1821ab64890Smrg
1831ab64890Smrgvoid
1841ab64890Smrg_XlcRemoveLoader(
1851ab64890Smrg    XLCdLoadProc proc)
1861ab64890Smrg{
1871ab64890Smrg    XlcLoaderList loader, prev;
1881ab64890Smrg
1891ab64890Smrg    if (loader_list == NULL)
1901ab64890Smrg	return;
1911ab64890Smrg
1921ab64890Smrg    prev = loader = loader_list;
1931ab64890Smrg    if (loader->proc == proc) {
1941ab64890Smrg	loader_list = loader->next;
1951ab64890Smrg	Xfree(loader);
1961ab64890Smrg	return;
1971ab64890Smrg    }
1981ab64890Smrg
1991ab64890Smrg    while ((loader = loader->next)) {
2001ab64890Smrg	if (loader->proc == proc) {
2011ab64890Smrg	    prev->next = loader->next;
2021ab64890Smrg	    Xfree(loader);
2031ab64890Smrg	    return;
2041ab64890Smrg	}
2051ab64890Smrg	prev = loader;
2061ab64890Smrg    }
2071ab64890Smrg
2081ab64890Smrg    return;
2091ab64890Smrg}
2101ab64890Smrg
2111ab64890SmrgBool
2121ab64890Smrg_XlcAddLoader(
2131ab64890Smrg    XLCdLoadProc proc,
2141ab64890Smrg    XlcPosition position)
2151ab64890Smrg{
2161ab64890Smrg    XlcLoaderList loader, last;
2171ab64890Smrg
2181ab64890Smrg    _XlcRemoveLoader(proc);		/* remove old loader, if exist */
2191ab64890Smrg
220818534a1Smrg    loader = Xmalloc(sizeof(XlcLoaderListRec));
2211ab64890Smrg    if (loader == NULL)
2221ab64890Smrg	return False;
2231ab64890Smrg
2241ab64890Smrg    loader->proc = proc;
2251ab64890Smrg
2261ab64890Smrg    if (loader_list == NULL)
2271ab64890Smrg	position = XlcHead;
2281ab64890Smrg
2291ab64890Smrg    if (position == XlcHead) {
2301ab64890Smrg	loader->next = loader_list;
2311ab64890Smrg	loader_list = loader;
2321ab64890Smrg    } else {
2331ab64890Smrg	last = loader_list;
2341ab64890Smrg	while (last->next)
2351ab64890Smrg	    last = last->next;
23661b2299dSmrg
2371ab64890Smrg	loader->next = NULL;
2381ab64890Smrg	last->next = loader;
2391ab64890Smrg    }
2401ab64890Smrg
2411ab64890Smrg    return True;
2421ab64890Smrg}
2431ab64890Smrg
2441ab64890SmrgXLCd
2451ab64890Smrg_XOpenLC(
2461ab64890Smrg    char *name)
2471ab64890Smrg{
2481ab64890Smrg    XLCd lcd;
2491ab64890Smrg    XlcLoaderList loader;
2501ab64890Smrg    XLCdList cur;
2511ab64890Smrg#if !defined(X_LOCALE)
2521ab64890Smrg    int len;
2531ab64890Smrg    char sinamebuf[256];
2541ab64890Smrg    char* siname = sinamebuf;
2551ab64890Smrg#endif
2561ab64890Smrg
2571ab64890Smrg    if (name == NULL) {
2581ab64890Smrg	name = setlocale (LC_CTYPE, (char *)NULL);
2591ab64890Smrg#if !defined(X_LOCALE)
26061b2299dSmrg        /*
26161b2299dSmrg         * _XlMapOSLocaleName will return the same string or a substring
26261b2299dSmrg         * of name, so strlen(name) is okay
2631ab64890Smrg         */
2641ab64890Smrg        if ((len = strlen(name)) >= sizeof sinamebuf) {
2651ab64890Smrg            siname = Xmalloc (len + 1);
2661ab64890Smrg            if (siname == NULL)
2671ab64890Smrg                return NULL;
2681ab64890Smrg        }
2691ab64890Smrg        name = _XlcMapOSLocaleName(name, siname);
2701ab64890Smrg#endif
2711ab64890Smrg    }
2721ab64890Smrg
2731ab64890Smrg    _XLockMutex(_Xi18n_lock);
2741ab64890Smrg
2751ab64890Smrg    /*
2761ab64890Smrg     * search for needed lcd, if found return it
2771ab64890Smrg     */
2781ab64890Smrg    for (cur = lcd_list; cur; cur = cur->next) {
2791ab64890Smrg	if (!strcmp (cur->lcd->core->name, name)) {
2801ab64890Smrg	    lcd = cur->lcd;
2811ab64890Smrg	    cur->ref_count++;
2821ab64890Smrg	    goto found;
2831ab64890Smrg	}
2841ab64890Smrg    }
2851ab64890Smrg
2861ab64890Smrg    if (!loader_list)
2871ab64890Smrg	_XlcInitLoader();
2881ab64890Smrg
2891ab64890Smrg    /*
2901ab64890Smrg     * not there, so try to get and add to list
2911ab64890Smrg     */
2921ab64890Smrg    for (loader = loader_list; loader; loader = loader->next) {
2931ab64890Smrg	lcd = (*loader->proc)(name);
2941ab64890Smrg	if (lcd) {
295818534a1Smrg	    cur = Xmalloc (sizeof(XLCdListRec));
2961ab64890Smrg	    if (cur) {
2971ab64890Smrg		cur->lcd = lcd;
2981ab64890Smrg		cur->ref_count = 1;
2991ab64890Smrg		cur->next = lcd_list;
3001ab64890Smrg		lcd_list = cur;
3011ab64890Smrg	    } else {
3021ab64890Smrg		(*lcd->methods->close)(lcd);
3031ab64890Smrg		lcd = (XLCd) NULL;
3041ab64890Smrg	    }
3051ab64890Smrg	    goto found;
3061ab64890Smrg	}
3071ab64890Smrg    }
3081ab64890Smrg
3091ab64890Smrg    lcd = NULL;
3101ab64890Smrg
3111ab64890Smrgfound:
3121ab64890Smrg    _XUnlockMutex(_Xi18n_lock);
3131ab64890Smrg
3141ab64890Smrg#if !defined(X_LOCALE)
3151ab64890Smrg    if (siname != sinamebuf) Xfree(siname);
3161ab64890Smrg#endif
3171ab64890Smrg
3181ab64890Smrg    return lcd;
3191ab64890Smrg}
3201ab64890Smrg
3211ab64890Smrgvoid
3221ab64890Smrg_XCloseLC(
3231ab64890Smrg    XLCd lcd)
3241ab64890Smrg{
3251ab64890Smrg    XLCdList cur, *prev;
3261ab64890Smrg
3271ab64890Smrg    for (prev = &lcd_list; (cur = *prev); prev = &cur->next) {
3281ab64890Smrg	if (cur->lcd == lcd) {
3291ab64890Smrg	    if (--cur->ref_count < 1) {
3301ab64890Smrg		(*lcd->methods->close)(lcd);
3311ab64890Smrg		*prev = cur->next;
3321ab64890Smrg		Xfree(cur);
3331ab64890Smrg	    }
3341ab64890Smrg	    break;
3351ab64890Smrg	}
3361ab64890Smrg    }
3371ab64890Smrg
3381ab64890Smrg    if(loader_list) {
3391ab64890Smrg	_XlcDeInitLoader();
3401ab64890Smrg	loader_list = NULL;
3411ab64890Smrg    }
3421ab64890Smrg}
3431ab64890Smrg
3441ab64890Smrg/*
3451ab64890Smrg * Get the XLCd for the current locale
3461ab64890Smrg */
3471ab64890Smrg
3481ab64890SmrgXLCd
34961b2299dSmrg_XlcCurrentLC(void)
3501ab64890Smrg{
3511ab64890Smrg    XLCd lcd;
3521ab64890Smrg    static XLCd last_lcd = NULL;
3531ab64890Smrg
3541ab64890Smrg    lcd = _XOpenLC((char *) NULL);
3551ab64890Smrg
3561ab64890Smrg    if (last_lcd)
3571ab64890Smrg	_XCloseLC(last_lcd);
35861b2299dSmrg
3591ab64890Smrg    last_lcd = lcd;
3601ab64890Smrg
3611ab64890Smrg    return lcd;
3621ab64890Smrg}
3631ab64890Smrg
3641ab64890SmrgXrmMethods
3651ab64890Smrg_XrmInitParseInfo(
3661ab64890Smrg    XPointer *state)
3671ab64890Smrg{
3681ab64890Smrg    XLCd lcd = _XOpenLC((char *) NULL);
36961b2299dSmrg
3701ab64890Smrg    if (lcd == (XLCd) NULL)
3711ab64890Smrg	return (XrmMethods) NULL;
37261b2299dSmrg
3731ab64890Smrg    return (*lcd->methods->init_parse_info)(lcd, state);
3741ab64890Smrg}
3751ab64890Smrg
3761ab64890Smrgint
3771ab64890SmrgXmbTextPropertyToTextList(
3781ab64890Smrg    Display *dpy,
3791ab64890Smrg    const XTextProperty *text_prop,
3801ab64890Smrg    char ***list_ret,
3811ab64890Smrg    int *count_ret)
3821ab64890Smrg{
3831ab64890Smrg    XLCd lcd = _XlcCurrentLC();
38461b2299dSmrg
3851ab64890Smrg    if (lcd == NULL)
3861ab64890Smrg	return XLocaleNotSupported;
3871ab64890Smrg
3881ab64890Smrg    return (*lcd->methods->mb_text_prop_to_list)(lcd, dpy, text_prop, list_ret,
3891ab64890Smrg						 count_ret);
3901ab64890Smrg}
3911ab64890Smrg
3921ab64890Smrgint
3931ab64890SmrgXwcTextPropertyToTextList(
3941ab64890Smrg    Display *dpy,
3951ab64890Smrg    const XTextProperty *text_prop,
3961ab64890Smrg    wchar_t ***list_ret,
3971ab64890Smrg    int *count_ret)
3981ab64890Smrg{
3991ab64890Smrg    XLCd lcd = _XlcCurrentLC();
40061b2299dSmrg
4011ab64890Smrg    if (lcd == NULL)
4021ab64890Smrg	return XLocaleNotSupported;
4031ab64890Smrg
4041ab64890Smrg    return (*lcd->methods->wc_text_prop_to_list)(lcd, dpy, text_prop, list_ret,
4051ab64890Smrg						 count_ret);
4061ab64890Smrg}
4071ab64890Smrg
4081ab64890Smrgint
4091ab64890SmrgXutf8TextPropertyToTextList(
4101ab64890Smrg    Display *dpy,
4111ab64890Smrg    const XTextProperty *text_prop,
4121ab64890Smrg    char ***list_ret,
4131ab64890Smrg    int *count_ret)
4141ab64890Smrg{
4151ab64890Smrg    XLCd lcd = _XlcCurrentLC();
41661b2299dSmrg
4171ab64890Smrg    if (lcd == NULL)
4181ab64890Smrg	return XLocaleNotSupported;
4191ab64890Smrg
4201ab64890Smrg    return (*lcd->methods->utf8_text_prop_to_list)(lcd, dpy, text_prop,
4211ab64890Smrg						   list_ret, count_ret);
4221ab64890Smrg}
4231ab64890Smrg
4241ab64890Smrgint
4251ab64890SmrgXmbTextListToTextProperty(
4261ab64890Smrg    Display *dpy,
4271ab64890Smrg    char **list,
4281ab64890Smrg    int count,
4291ab64890Smrg    XICCEncodingStyle style,
4301ab64890Smrg    XTextProperty *text_prop)
4311ab64890Smrg{
4321ab64890Smrg    XLCd lcd = _XlcCurrentLC();
43361b2299dSmrg
4341ab64890Smrg    if (lcd == NULL)
4351ab64890Smrg	return XLocaleNotSupported;
4361ab64890Smrg
4371ab64890Smrg    return (*lcd->methods->mb_text_list_to_prop)(lcd, dpy, list, count, style,
4381ab64890Smrg						 text_prop);
4391ab64890Smrg}
4401ab64890Smrg
4411ab64890Smrgint
4421ab64890SmrgXwcTextListToTextProperty(
4431ab64890Smrg    Display *dpy,
4441ab64890Smrg    wchar_t **list,
4451ab64890Smrg    int count,
4461ab64890Smrg    XICCEncodingStyle style,
4471ab64890Smrg    XTextProperty *text_prop)
4481ab64890Smrg{
4491ab64890Smrg    XLCd lcd = _XlcCurrentLC();
45061b2299dSmrg
4511ab64890Smrg    if (lcd == NULL)
4521ab64890Smrg	return XLocaleNotSupported;
4531ab64890Smrg
4541ab64890Smrg    return (*lcd->methods->wc_text_list_to_prop)(lcd, dpy, list, count, style,
4551ab64890Smrg						 text_prop);
4561ab64890Smrg}
4571ab64890Smrg
4581ab64890Smrgint
4591ab64890SmrgXutf8TextListToTextProperty(
4601ab64890Smrg    Display *dpy,
4611ab64890Smrg    char **list,
4621ab64890Smrg    int count,
4631ab64890Smrg    XICCEncodingStyle style,
4641ab64890Smrg    XTextProperty *text_prop)
4651ab64890Smrg{
4661ab64890Smrg    XLCd lcd = _XlcCurrentLC();
46761b2299dSmrg
4681ab64890Smrg    if (lcd == NULL)
4691ab64890Smrg	return XLocaleNotSupported;
4701ab64890Smrg
4711ab64890Smrg    return (*lcd->methods->utf8_text_list_to_prop)(lcd, dpy, list, count,
4721ab64890Smrg						   style, text_prop);
4731ab64890Smrg}
4741ab64890Smrg
4751ab64890Smrgvoid
4761ab64890SmrgXwcFreeStringList(
4771ab64890Smrg    wchar_t **list)
4781ab64890Smrg{
4791ab64890Smrg    XLCd lcd = _XlcCurrentLC();
48061b2299dSmrg
4811ab64890Smrg    if (lcd == NULL)
4821ab64890Smrg	return;
4831ab64890Smrg
4841ab64890Smrg    (*lcd->methods->wc_free_string_list)(lcd, list);
4851ab64890Smrg}
4861ab64890Smrg
4871ab64890Smrgconst char *
48861b2299dSmrgXDefaultString(void)
4891ab64890Smrg{
4901ab64890Smrg    XLCd lcd = _XlcCurrentLC();
49161b2299dSmrg
4921ab64890Smrg    if (lcd == NULL)
4931ab64890Smrg	return (char *) NULL;
49461b2299dSmrg
4951ab64890Smrg    return (*lcd->methods->default_string)(lcd);
4961ab64890Smrg}
4971ab64890Smrg
4981ab64890Smrgvoid
4991ab64890Smrg_XlcCopyFromArg(
5001ab64890Smrg    char *src,
5011ab64890Smrg    char *dst,
5021ab64890Smrg    int size)
5031ab64890Smrg{
5041ab64890Smrg    if (size == sizeof(long))
5051ab64890Smrg	*((long *) dst) = (long) src;
5061ab64890Smrg#ifdef LONG64
5071ab64890Smrg    else if (size == sizeof(int))
5081ab64890Smrg	*((int *) dst) = (int)(long) src;
5091ab64890Smrg#endif
5101ab64890Smrg    else if (size == sizeof(short))
5111ab64890Smrg	*((short *) dst) = (short)(long) src;
5121ab64890Smrg    else if (size == sizeof(char))
5131ab64890Smrg	*((char *) dst) = (char)(long) src;
5141ab64890Smrg    else if (size == sizeof(XPointer))
5151ab64890Smrg	*((XPointer *) dst) = (XPointer) src;
5161ab64890Smrg    else if (size > sizeof(XPointer))
5171ab64890Smrg	memcpy(dst, (char *) src, size);
5181ab64890Smrg    else
5191ab64890Smrg	memcpy(dst, (char *) &src, size);
5201ab64890Smrg}
5211ab64890Smrg
5221ab64890Smrgvoid
5231ab64890Smrg_XlcCopyToArg(
5241ab64890Smrg    char *src,
5251ab64890Smrg    char **dst,
5261ab64890Smrg    int size)
5271ab64890Smrg{
5281ab64890Smrg    /* FIXME:
5291ab64890Smrg       On Big Endian machines, this behaves differently than _XCopyToArg. */
5301ab64890Smrg    if (size == sizeof(long))
5311ab64890Smrg	*((long *) *dst) = *((long *) src);
5321ab64890Smrg#ifdef LONG64
5331ab64890Smrg    else if (size == sizeof(int))
5341ab64890Smrg	*((int *) *dst) = *((int *) src);
5351ab64890Smrg#endif
5361ab64890Smrg    else if (size == sizeof(short))
5371ab64890Smrg	*((short *) *dst) = *((short *) src);
5381ab64890Smrg    else if (size == sizeof(char))
5391ab64890Smrg	*((char *) *dst) = *((char *) src);
5401ab64890Smrg    else if (size == sizeof(XPointer))
5411ab64890Smrg	*((XPointer *) *dst) = *((XPointer *) src);
5421ab64890Smrg    else
5431ab64890Smrg	memcpy(*dst, src, size);
5441ab64890Smrg}
5451ab64890Smrg
5461ab64890Smrgvoid
5471ab64890Smrg_XlcCountVaList(
5481ab64890Smrg    va_list var,
5491ab64890Smrg    int *count_ret)
5501ab64890Smrg{
5511ab64890Smrg    int count;
5521ab64890Smrg
5531ab64890Smrg    for (count = 0; va_arg(var, char *); count++)
5541ab64890Smrg	(void)va_arg(var, XPointer);
55561b2299dSmrg
5561ab64890Smrg    *count_ret = count;
5571ab64890Smrg}
5581ab64890Smrg
5591ab64890Smrgvoid
5601ab64890Smrg_XlcVaToArgList(
5611ab64890Smrg    va_list var,
5621ab64890Smrg    int count,
5631ab64890Smrg    XlcArgList *args_ret)
5641ab64890Smrg{
5651ab64890Smrg    XlcArgList args;
5661ab64890Smrg
567818534a1Smrg    *args_ret = args = Xmalloc(sizeof(XlcArg) * count);
5681ab64890Smrg    if (args == (XlcArgList) NULL)
5691ab64890Smrg	return;
57061b2299dSmrg
5711ab64890Smrg    for ( ; count-- > 0; args++) {
5721ab64890Smrg	args->name = va_arg(var, char *);
5731ab64890Smrg	args->value = va_arg(var, XPointer);
5741ab64890Smrg    }
5751ab64890Smrg}
5761ab64890Smrg
5771ab64890Smrgvoid
5781ab64890Smrg_XlcCompileResourceList(
5791ab64890Smrg    XlcResourceList resources,
5801ab64890Smrg    int num_resources)
5811ab64890Smrg{
5821ab64890Smrg    for ( ; num_resources-- > 0; resources++)
5831ab64890Smrg	resources->xrm_name = XrmPermStringToQuark(resources->name);
5841ab64890Smrg}
5851ab64890Smrg
5861ab64890Smrgchar *
5871ab64890Smrg_XlcGetValues(
5881ab64890Smrg    XPointer base,
5891ab64890Smrg    XlcResourceList resources,
5901ab64890Smrg    int num_resources,
5911ab64890Smrg    XlcArgList args,
5921ab64890Smrg    int num_args,
5931ab64890Smrg    unsigned long mask)
5941ab64890Smrg{
5951ab64890Smrg    XlcResourceList res;
5961ab64890Smrg    XrmQuark xrm_name;
5971ab64890Smrg    int count;
5981ab64890Smrg
5991ab64890Smrg    for ( ; num_args-- > 0; args++) {
6001ab64890Smrg	res = resources;
6011ab64890Smrg	count = num_resources;
6021ab64890Smrg	xrm_name = XrmPermStringToQuark(args->name);
6031ab64890Smrg
6041ab64890Smrg	for ( ; count-- > 0; res++) {
6051ab64890Smrg	    if (xrm_name == res->xrm_name && (mask & res->mask)) {
6061ab64890Smrg		    _XlcCopyToArg(base + res->offset, &args->value, res->size);
6071ab64890Smrg		break;
6081ab64890Smrg	    }
6091ab64890Smrg	}
6101ab64890Smrg
6111ab64890Smrg	if (count < 0)
6121ab64890Smrg	    return args->name;
6131ab64890Smrg    }
6141ab64890Smrg
6151ab64890Smrg    return NULL;
6161ab64890Smrg}
6171ab64890Smrg
6181ab64890Smrgchar *
6191ab64890Smrg_XlcSetValues(
6201ab64890Smrg    XPointer base,
6211ab64890Smrg    XlcResourceList resources,
6221ab64890Smrg    int num_resources,
6231ab64890Smrg    XlcArgList args,
6241ab64890Smrg    int num_args,
6251ab64890Smrg    unsigned long mask)
6261ab64890Smrg{
6271ab64890Smrg    XlcResourceList res;
6281ab64890Smrg    XrmQuark xrm_name;
6291ab64890Smrg    int count;
6301ab64890Smrg
6311ab64890Smrg    for ( ; num_args-- > 0; args++) {
6321ab64890Smrg	res = resources;
6331ab64890Smrg	count = num_resources;
6341ab64890Smrg	xrm_name = XrmPermStringToQuark(args->name);
6351ab64890Smrg
6361ab64890Smrg	for ( ; count-- > 0; res++) {
6371ab64890Smrg	    if (xrm_name == res->xrm_name && (mask & res->mask)) {
6381ab64890Smrg		_XlcCopyFromArg(args->value, base + res->offset, res->size);
6391ab64890Smrg		break;
6401ab64890Smrg	    }
6411ab64890Smrg	}
6421ab64890Smrg
6431ab64890Smrg	if (count < 0)
6441ab64890Smrg	    return args->name;
6451ab64890Smrg    }
6461ab64890Smrg
6471ab64890Smrg    return NULL;
6481ab64890Smrg}
649