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