/* * Copyright 1992, 1993 by TOSHIBA Corp. * * Permission to use, copy, modify, and distribute this software and its * documentation for any purpose and without fee is hereby granted, provided * that the above copyright notice appear in all copies and that both that * copyright notice and this permission notice appear in supporting * documentation, and that the name of TOSHIBA not be used in advertising * or publicity pertaining to distribution of the software without specific, * written prior permission. TOSHIBA make no representations about the * suitability of this software for any purpose. It is provided "as is" * without express or implied warranty. * * TOSHIBA DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING * ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL * TOSHIBA BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR * ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, * WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, * ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS * SOFTWARE. * * Author: Katsuhisa Yano TOSHIBA Corp. * mopi@osa.ilab.toshiba.co.jp * Bug fixes: Bruno Haible XFree86 Inc. */ #ifdef HAVE_CONFIG_H #include <config.h> #endif #include "Xlibint.h" #include "XlcPubI.h" #include <stdio.h> /* * Default implementation of methods for Xrm parsing. */ /* ======================= Unibyte implementation ======================= */ /* Only for efficiency, to speed up things. */ /* This implementation must keep the locale, for lcname. */ typedef struct _UbStateRec { XLCd lcd; } UbStateRec, *UbState; /* Sets the state to the initial state. Initiates a sequence of calls to mbchar. */ static void ub_mbinit( XPointer state) { } /* Transforms one multibyte character, and return a 'char' in the same parsing class. Returns the number of consumed bytes in *lenp. */ static char ub_mbchar( XPointer state, const char *str, int *lenp) { *lenp = 1; return *str; } /* Terminates a sequence of calls to mbchar. */ static void ub_mbfinish( XPointer state) { } /* Returns the name of the state's locale, as a static string. */ static const char * ub_lcname( XPointer state) { return ((UbState) state)->lcd->core->name; } /* Frees the state, which was allocated by _XrmDefaultInitParseInfo. */ static void ub_destroy( XPointer state) { _XCloseLC(((UbState) state)->lcd); Xfree((char *) state); } static const XrmMethodsRec ub_methods = { ub_mbinit, ub_mbchar, ub_mbfinish, ub_lcname, ub_destroy }; /* ======================= Multibyte implementation ======================= */ /* This implementation uses an XlcConv from XlcNMultiByte to XlcNWideChar. */ typedef struct _MbStateRec { XLCd lcd; XlcConv conv; } MbStateRec, *MbState; /* Sets the state to the initial state. Initiates a sequence of calls to mbchar. */ static void mb_mbinit( XPointer state) { _XlcResetConverter(((MbState) state)->conv); } /* Transforms one multibyte character, and return a 'char' in the same parsing class. Returns the number of consumed bytes in *lenp. */ static char mb_mbchar( XPointer state, const char *str, int *lenp) { XlcConv conv = ((MbState) state)->conv; const char *from; wchar_t *to, wc; int cur_max, i, from_left, to_left, ret; cur_max = XLC_PUBLIC(((MbState) state)->lcd, mb_cur_max); from = str; /* Determine from_left. Avoid overrun error which could occur if from_left > strlen(str). */ from_left = cur_max; for (i = 0; i < cur_max; i++) if (str[i] == '\0') { from_left = i; break; } *lenp = from_left; to = &wc; to_left = 1; ret = _XlcConvert(conv, (XPointer *) &from, &from_left, (XPointer *) &to, &to_left, NULL, 0); *lenp -= from_left; if (ret < 0 || to_left > 0) { /* Invalid or incomplete multibyte character seen. */ *lenp = 1; return 0x7f; } /* Return a 'char' equivalent to wc. */ return (wc >= 0 && wc <= 0x7f ? wc : 0x7f); } /* Terminates a sequence of calls to mbchar. */ static void mb_mbfinish( XPointer state) { } /* Returns the name of the state's locale, as a static string. */ static const char * mb_lcname( XPointer state) { return ((MbState) state)->lcd->core->name; } /* Frees the state, which was allocated by _XrmDefaultInitParseInfo. */ static void mb_destroy( XPointer state) { _XlcCloseConverter(((MbState) state)->conv); _XCloseLC(((MbState) state)->lcd); Xfree((char *) state); } static const XrmMethodsRec mb_methods = { mb_mbinit, mb_mbchar, mb_mbfinish, mb_lcname, mb_destroy }; /* ======================= Exported function ======================= */ XrmMethods _XrmDefaultInitParseInfo( XLCd lcd, XPointer *rm_state) { if (XLC_PUBLIC(lcd, mb_cur_max) == 1) { /* Unibyte case. */ UbState state = (UbState) Xmalloc(sizeof(UbStateRec)); if (state == NULL) return (XrmMethods) NULL; state->lcd = lcd; *rm_state = (XPointer) state; return &ub_methods; } else { /* Multibyte case. */ MbState state = (MbState) Xmalloc(sizeof(MbStateRec)); if (state == NULL) return (XrmMethods) NULL; state->lcd = lcd; state->conv = _XlcOpenConverter(lcd, XlcNMultiByte, lcd, XlcNWideChar); if (state->conv == NULL) { Xfree((char *) state); return (XrmMethods) NULL; } *rm_state = (XPointer) state; return &mb_methods; } }