da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin/***********************************************************************
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin* This software is part of the ast package *
3e14f97f673e8a630f076077de35afdd43dc1587Roger A. Faulkner* Copyright (c) 1985-2010 AT&T Intellectual Property *
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin* and is licensed under the *
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin* Common Public License, Version 1.0 *
7c2fbfb345896881c631598ee3852ce9ce33fb07April Chin* by AT&T Intellectual Property *
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin* A copy of the License is available at *
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin* (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9) *
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin* Information and Software Systems Research *
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin* AT&T Research *
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin* Florham Park NJ *
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin* Glenn Fowler <gsf@research.att.com> *
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin* David Korn <dgk@research.att.com> *
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin* Phong Vo <kpv@research.att.com> *
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin***********************************************************************/
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * locale state implementation
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin { &debug_lc, 0, 0 },
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin { &debug_lc, 0, 0 },
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin { &debug_lc, 0, 0 },
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin { &debug_lc, 0, 0 },
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin { &debug_lc, 0, 0 },
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin { &debug_lc, 0, 0 },
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin { &debug_lc, 0, 0 },
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin { &debug_lc, 0, 0 },
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin { &debug_lc, 0, 0 },
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin { &debug_lc, 0, 0 },
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin { &debug_lc, 0, 0 },
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin { &debug_lc, 0, 0 },
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * return the internal category index for category
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin return -1;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * return the first category table entry
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * return the current info for category
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * return 1 if s matches the alternation pattern p
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * if minimum!=0 then at least that many chars must match
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * if standard!=0 and s[0] is a digit leading non-digits are ignored in p
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chinmatch(const char* s, register const char* p, int minimum, int standard)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin register const char* t;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin const char* x;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin while (*p && !isdigit(*p))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin else if (isdigit(*p))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin while (*t && !isdigit(*t))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin while (*p && *p != '|')
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (!*t || *t == ',')
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin else if (*t == *p)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin else if (*t == '-')
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin while (*p && *p != '-')
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin else if (*p == '-')
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin while (*t && *t != '-')
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin return p - x;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin z = p - x;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin while (*p && *p != '|')
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin } while (*p++);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * return 1 if s matches the charset names in cp
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chinmatch_charset(register const char* s, register const Lc_charset_t* cp)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin return match(s, cp->code, 0, 1) || match(s, cp->alternates, 3, 1) || cp->ms && match(s, cp->ms, 0, 1);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * low level for lccanon
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chincanonical(const Lc_language_t* lp, const Lc_territory_t* tp, const Lc_charset_t* cp, const Lc_attribute_list_t* ap, unsigned long flags, char* buf, size_t siz)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin register int c;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin register int u;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin register char* s;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin register char* e;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin register const char* t;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (!(flags & (LC_abbreviated|LC_default|LC_local|LC_qualified|LC_verbose)))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin return s - buf;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin while (s < e && (c = *t++))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin else if (!isalnum(c))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (s < e)
7c2fbfb345896881c631598ee3852ce9ce33fb07April Chin if (tp && tp != &lc_territories[0] && (!(flags & (LC_abbreviated|LC_default)) || !lp || !streq(lp->code, tp->code)))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin *s++ = '_';
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin while (s < e && (c = *t++) && c != '|')
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin else if (!isalnum(c))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin for (t = tp->code; s < e && (*s = toupper(*t++)); s++);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (lp && (!(flags & (LC_abbreviated|LC_default)) || cp != lp->charset) && s < e)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin *s++ = '.';
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (!(flags & (LC_abbreviated|LC_default|LC_verbose)) || !(ap->attribute->flags & LC_default))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin for (t = ap->attribute->name; s < e && (*s = *t++); s++);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin return s - buf;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * generate a canonical locale name in buf
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chinlccanon(Lc_t* lc, unsigned long flags, char* buf, size_t siz)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if ((flags & LC_local) && (!lc->language || !(lc->language->flags & (LC_debug|LC_default))))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin GetLocaleInfo(lc->index, LOCALE_SENGLANGUAGE, lang, sizeof(lang)) &&
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin GetLocaleInfo(lc->index, LOCALE_SENGCOUNTRY, ctry, sizeof(ctry)))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (!GetLocaleInfo(lc->index, LOCALE_IDEFAULTANSICODEPAGE, code, sizeof(code)))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin return sfsprintf(buf, siz, "%s_%s.%s", lang, ctry, code);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin return sfsprintf(buf, siz, "%s_%s.%s,%s", lang, ctry, code, lc->charset->ms);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin return canonical(lc->language, lc->territory, lc->charset, lc->attributes, flags, buf, siz);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * make an Lc_t from a locale name
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin register int c;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin register char* s;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin register char* e;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin register const char* t;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin const char* a;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (!(t = name) || !*t)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (!strcasecmp(t, lc->code) || !strcasecmp(t, lc->name))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin while (s < e && (c = *t++))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (isspace(c) || (c == '(' || c == '-' && *t == '-') && ++n)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin while ((c = *t++) && (isspace(c) || (c == '-' || c == '(' || c == ')') && ++n))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (isalnum(c) && !n)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin *s++ = '-';
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin a = t - 1;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if ((t = a) && s < e)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin *s++ = ',';
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin while (s < e && (c = *t++))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (isspace(c) || (c == '(' || c == ')' || c == '-' && *t == '-') && ++n)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin while ((c = *t++) && (isspace(c) || (c == '-' || c == '(' || c == ')') && ++n))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (isalnum(c) && !n)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin *s++ = '-';
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * language
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (n == 2)
7c2fbfb345896881c631598ee3852ce9ce33fb07April Chin for (lp = lc_languages; lp->code && !streq(s, lp->code); lp++);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin else if (n == 3)
7c2fbfb345896881c631598ee3852ce9ce33fb07April Chin for (lp = lc_languages; lp->code && (!lp->alternates || !match(s, lp->alternates, n, 0)); lp++);
7c2fbfb345896881c631598ee3852ce9ce33fb07April Chin for (lp = lc_languages; lp->code && !streq(s, lp->code); lp++);
7c2fbfb345896881c631598ee3852ce9ce33fb07April Chin for (lp = lc_languages; lp->code && !match(s, lp->name, 0, 0); lp++);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (n == 2)
7c2fbfb345896881c631598ee3852ce9ce33fb07April Chin for (tp = lc_territories; tp->code && !streq(s, tp->code); tp++);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if ((z = i) == n)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * name not in the tables so let
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * _ast_setlocale() and/or setlocale()
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * handle the validity checks
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin s = (char*)name;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin name = ((Lc_language_t*)lp)->code = ((Lc_language_t*)lp)->name = (const char*)(lp + 1);
7c2fbfb345896881c631598ee3852ce9ce33fb07April Chin cp = ((Lc_language_t*)lp)->charset = &lc_charsets[0];
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * territory
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (n == 1)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (n == 2)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin for (i = 0; i < elementsof(tp->languages) && lp != tp->languages[i]; i++);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin for (i = 0; i < elementsof(tp->languages) && lp != tp->languages[i]; i++);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * attributes -- done here to catch misplaced charset references
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin for (w = s; *s && *s != ','; s++);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin for (i = 0; i < elementsof(lp->attributes) && (ap = lp->attributes[i]); i++)
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (i >= elementsof(lp->attributes) && (ap = newof(0, Lc_attribute_t, 1, sizeof(Lc_attribute_list_t) + s - w + 1)))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin strcpy((char*)(((Lc_attribute_t*)ap)->name = (const char*)(ai + 1)), w);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin } while (*s++);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin z = canonical(lp, tp, cp, al, 0, s = tmp, sizeof(tmp));
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * add to the list of possibly active locales
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin strcpy((char*)(lc->name = (const char*)(lc + 1)), name);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin lc->index = MAKELCID(MAKELANGID(lp->index, n), SORT_DEFAULT);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * return an Lc_t* for each locale in the tables
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * one Lc_t is allocated on the first call with lc==0
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * this is freed when 0 returned
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin * the return value is not part of the lcmake() cache
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin ls->language = elementsof(ls->lc.territory->languages);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin ls->attribute = elementsof(ls->lc.language->attributes);
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (++ls->attribute >= elementsof(ls->lc.language->attributes) || !(ls->list.attribute = ls->lc.language->attributes[ls->attribute]))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if (++ls->language >= elementsof(ls->lc.territory->languages) || !(ls->lc.language = ls->lc.territory->languages[ls->language]))
7c2fbfb345896881c631598ee3852ce9ce33fb07April Chin ls->lc.territory = &lc_territories[ls->territory];
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin ls->lc.language = ls->lc.territory->languages[ls->language = 0];
7c2fbfb345896881c631598ee3852ce9ce33fb07April Chin ls->lc.charset = ls->lc.language->charset ? ls->lc.language->charset : &lc_charsets[0];
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin ls->list.attribute = ls->lc.language->attributes[ls->attribute = 0];
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin ls->lc.attributes = ls->list.attribute ? &ls->list : (Lc_attribute_list_t*)0;
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin if ((!ls->list.attribute || !(ls->lc.index = ls->list.attribute->index)) &&
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin (!ls->lc.territory || !(ls->lc.index = ls->lc.territory->indices[ls->language])))
da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968chin ls->lc.index = MAKELCID(MAKELANGID(ls->lc.language->index, ls->lc.index), SORT_DEFAULT);