305 lines
6.2 KiB
C
305 lines
6.2 KiB
C
|
/* $Header: /src/pub/tcsh/tc.nls.c,v 3.6 2005/02/15 21:09:02 christos Exp $ */
|
||
|
/*
|
||
|
* tc.nls.c: NLS handling
|
||
|
*/
|
||
|
/*-
|
||
|
* Copyright (c) 1980, 1991 The Regents of the University of California.
|
||
|
* All rights reserved.
|
||
|
*
|
||
|
* Redistribution and use in source and binary forms, with or without
|
||
|
* modification, are permitted provided that the following conditions
|
||
|
* are met:
|
||
|
* 1. Redistributions of source code must retain the above copyright
|
||
|
* notice, this list of conditions and the following disclaimer.
|
||
|
* 2. Redistributions in binary form must reproduce the above copyright
|
||
|
* notice, this list of conditions and the following disclaimer in the
|
||
|
* documentation and/or other materials provided with the distribution.
|
||
|
* 3. Neither the name of the University nor the names of its contributors
|
||
|
* may be used to endorse or promote products derived from this software
|
||
|
* without specific prior written permission.
|
||
|
*
|
||
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
||
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
||
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
||
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
||
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||
|
* SUCH DAMAGE.
|
||
|
*/
|
||
|
#include "sh.h"
|
||
|
|
||
|
RCSID("$Id: tc.nls.c,v 3.6 2005/02/15 21:09:02 christos Exp $")
|
||
|
|
||
|
#ifdef SHORT_STRINGS
|
||
|
int
|
||
|
NLSWidth(c)
|
||
|
NLSChar c;
|
||
|
{
|
||
|
# ifdef HAVE_WCWIDTH
|
||
|
int l;
|
||
|
if (c & NLS_ILLEGAL)
|
||
|
return 1;
|
||
|
l = wcwidth(c);
|
||
|
return l >= 0 ? l : 0;
|
||
|
# else
|
||
|
return c != 0;
|
||
|
# endif
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#if defined (WIDE_STRINGS) || !defined (SHORT_STRINGS)
|
||
|
Char *
|
||
|
NLSChangeCase(Char *p, int mode)
|
||
|
{
|
||
|
Char c, *op = p, *n, c2 = 0;
|
||
|
for (; (c = *p) != 0; p++) {
|
||
|
if (mode == 0 && Islower(c)) {
|
||
|
c2 = Toupper(c);
|
||
|
break;
|
||
|
} else if (mode && Isupper(c)) {
|
||
|
c2 = Tolower(c);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (!*p)
|
||
|
return 0;
|
||
|
n = Strsave(op);
|
||
|
n[p - op] = c2;
|
||
|
return n;
|
||
|
}
|
||
|
|
||
|
int
|
||
|
NLSExtend(Char *from, int max, int num)
|
||
|
{
|
||
|
(void)from;
|
||
|
num = abs (num);
|
||
|
if (num > max)
|
||
|
num = max;
|
||
|
return num;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#ifdef WIDE_STRINGS
|
||
|
|
||
|
int
|
||
|
NLSStringWidth(Char *s)
|
||
|
{
|
||
|
int w = 0;
|
||
|
while (*s)
|
||
|
w += wcwidth(*s++);
|
||
|
return w;
|
||
|
}
|
||
|
|
||
|
#elif defined (SHORT_STRINGS)
|
||
|
|
||
|
int
|
||
|
NLSFrom(const Char *p, size_t l, NLSChar *cp)
|
||
|
{
|
||
|
size_t i;
|
||
|
int len;
|
||
|
wchar_t c;
|
||
|
char b[MB_LEN_MAX];
|
||
|
|
||
|
if (l == NLSZEROT) {
|
||
|
for (i = 0; i < MB_CUR_MAX && *p; i++)
|
||
|
b[i] = p[i] & CHAR;
|
||
|
} else {
|
||
|
for (i = 0; i < MB_CUR_MAX && i < l; i++)
|
||
|
b[i] = p[i] & CHAR;
|
||
|
}
|
||
|
mbtowc(0, 0, 0);
|
||
|
len = rt_mbtowc(&c, b, i);
|
||
|
if (len <= 0) {
|
||
|
if (cp)
|
||
|
*cp = *p ? *p | NLS_ILLEGAL : 0;
|
||
|
return 1;
|
||
|
}
|
||
|
if (cp)
|
||
|
*cp = (int)c;
|
||
|
return len;
|
||
|
}
|
||
|
|
||
|
int
|
||
|
NLSFinished(Char *p, size_t l, eChar extra)
|
||
|
{
|
||
|
size_t i, r;
|
||
|
wchar_t c;
|
||
|
char b[MB_LEN_MAX + 1], back[MB_LEN_MAX];
|
||
|
mbstate_t state;
|
||
|
for (i = 0; i < MB_CUR_MAX && i < l; i++)
|
||
|
b[i] = p[i];
|
||
|
if (extra != CHAR_ERR)
|
||
|
b[i++] = extra;
|
||
|
memset(&state, 0, sizeof(state));
|
||
|
r = mbrtowc((wchar_t *)&c, b, i, (mbstate_t *)&state);
|
||
|
if (r == (size_t)-2)
|
||
|
return 0;
|
||
|
if (r == (size_t)-1 || (size_t)wctomb(back, c) != r ||
|
||
|
memcmp(b, back, r) != 0)
|
||
|
return -1;
|
||
|
return r == i ? 1 : 2;
|
||
|
}
|
||
|
|
||
|
int
|
||
|
NLSChars(Char *s)
|
||
|
{
|
||
|
int l;
|
||
|
for (l = 0; *s; l++)
|
||
|
s += NLSSize(s, -1);
|
||
|
return l;
|
||
|
}
|
||
|
|
||
|
int
|
||
|
NLSStringWidth(Char *s)
|
||
|
{
|
||
|
int w = 0;
|
||
|
NLSChar c;
|
||
|
while (*s) {
|
||
|
s += NLSFrom(s, NLSZEROT, &c);
|
||
|
w += NLSWidth(c);
|
||
|
}
|
||
|
return w;
|
||
|
}
|
||
|
|
||
|
int
|
||
|
NLSTo(Char *p, NLSChar c)
|
||
|
{
|
||
|
char b[MB_LEN_MAX];
|
||
|
int i, j;
|
||
|
|
||
|
if (c & NLS_ILLEGAL) {
|
||
|
if (p)
|
||
|
*p = c;
|
||
|
return 1;
|
||
|
}
|
||
|
i = wctomb(b, (wchar_t)c);
|
||
|
if (i == -1)
|
||
|
return 0;
|
||
|
if (p)
|
||
|
for (j = 0; j < i; j++)
|
||
|
p[j] = b[j];
|
||
|
return i;
|
||
|
}
|
||
|
|
||
|
|
||
|
int
|
||
|
NLSExtend(Char *from, int max, int num)
|
||
|
{
|
||
|
int l, n, i;
|
||
|
Char *p;
|
||
|
|
||
|
if (num == 0)
|
||
|
return 0;
|
||
|
if (num > 0) {
|
||
|
n = 0;
|
||
|
while (num > 0 && max > 0) {
|
||
|
l = NLSSize(from, max);
|
||
|
n += l;
|
||
|
from += l;
|
||
|
max -= l;
|
||
|
num--;
|
||
|
}
|
||
|
return n;
|
||
|
}
|
||
|
from -= max;
|
||
|
p = from;
|
||
|
i = max;
|
||
|
n = 0;
|
||
|
while (i > 0) {
|
||
|
l = NLSSize(p, i);
|
||
|
p += l;
|
||
|
i -= l;
|
||
|
n++;
|
||
|
}
|
||
|
if (n >= -num)
|
||
|
n += num;
|
||
|
else
|
||
|
n = 0;
|
||
|
i = max;
|
||
|
while (n > 0) {
|
||
|
l = NLSSize(from, max);
|
||
|
from += l;
|
||
|
max -= l;
|
||
|
i -= l;
|
||
|
n--;
|
||
|
}
|
||
|
return i;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
NLSQuote(Char *cp)
|
||
|
{
|
||
|
int l;
|
||
|
while (*cp) {
|
||
|
l = NLSSize(cp, -1);
|
||
|
cp++;
|
||
|
while (l-- > 1)
|
||
|
*cp++ |= QUOTE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Char *
|
||
|
NLSChangeCase(Char *p, int mode)
|
||
|
{
|
||
|
Char *n, *op = p;
|
||
|
NLSChar c, c2 = 0;
|
||
|
int l, l2;
|
||
|
|
||
|
while (*p) {
|
||
|
l = NLSFrom(p, NLSZEROT, &c);
|
||
|
if (mode == 0 && iswlower((wint_t)c)) {
|
||
|
c2 = (int)towupper((wint_t)c);
|
||
|
break;
|
||
|
} else if (mode && iswupper((wint_t)c)) {
|
||
|
c2 = (int)towlower((wint_t)c);
|
||
|
break;
|
||
|
}
|
||
|
p += l;
|
||
|
}
|
||
|
if (!*p)
|
||
|
return 0;
|
||
|
l2 = NLSTo((Char *)0, c2);
|
||
|
n = (Char *)xmalloc((size_t)((op - p + l2 + Strlen(p + l) + 1) * sizeof(Char)));
|
||
|
if (p != op)
|
||
|
memcpy(n, op, (p - op) * sizeof(Char));
|
||
|
NLSTo(n + (p - op), c2);
|
||
|
memcpy(n + (p - op + l2), p + l, (Strlen(p + l) + 1) * sizeof(Char));
|
||
|
return n;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
int
|
||
|
NLSClassify(c, nocomb)
|
||
|
NLSChar c;
|
||
|
int nocomb;
|
||
|
{
|
||
|
int w;
|
||
|
if (c & NLS_ILLEGAL)
|
||
|
return NLSCLASS_ILLEGAL;
|
||
|
w = NLSWidth(c);
|
||
|
if (w > 0 || (Iswprint(c) && !nocomb))
|
||
|
return w;
|
||
|
if (Iswcntrl(c) && c < 0x100) {
|
||
|
if (c == '\n')
|
||
|
return NLSCLASS_NL;
|
||
|
if (c == '\t')
|
||
|
return NLSCLASS_TAB;
|
||
|
#ifndef ASCII
|
||
|
if (!Isupper(_toebcdic[_toascii[c]|0100]) && !strchr("@[\\]^_", _toebcdic[_toascii[c]|0100]))
|
||
|
return NLSCLASS_ILLEGAL;
|
||
|
#endif
|
||
|
return NLSCLASS_CTRL;
|
||
|
}
|
||
|
if (c >= 0x1000000)
|
||
|
return NLSCLASS_ILLEGAL4;
|
||
|
if (c >= 0x10000)
|
||
|
return NLSCLASS_ILLEGAL3;
|
||
|
if (c >= 0x100)
|
||
|
return NLSCLASS_ILLEGAL2;
|
||
|
return NLSCLASS_ILLEGAL;
|
||
|
}
|