--- a/src/locale/dcngettext.c 2017-02-06 14:39:17.860482624 +0000 +++ b/src/locale/dcngettext.c 2017-02-06 14:39:17.860482624 +0000 @@ -100,7 +100,9 @@ size_t map_size; void *volatile plural_rule; volatile int nplurals; - char name[]; + struct binding *binding; + struct __locale_map *lm; + struct msgcat cat; }; static char *dummy_gettextdomain() @@ -120,8 +122,8 @@ struct msgcat *p; struct __locale_struct *loc = CURRENT_LOCALE; const struct __locale_map *lm; - const char *dirname, *locname, *catname; - size_t dirlen, loclen, catlen, domlen; + size_t domlen; + struct binding *q; if ((unsigned)category >= LC_ALL) goto notrans; @@ -130,47 +132,62 @@ domlen = strnlen(domainname, NAME_MAX+1); if (domlen > NAME_MAX) goto notrans; - dirname = gettextdir(domainname, &dirlen); - if (!dirname) goto notrans; + for (q=bindings; q; q=q->next) + if (!strcmp(q->domainname, domainname) && q->active) + break; + if (!q) goto notrans; lm = loc->cat[category]; if (!lm) { notrans: return (char *) ((n == 1) ? msgid1 : msgid2); } - locname = lm->name; - - catname = catnames[category]; - catlen = catlens[category]; - loclen = strlen(locname); - - size_t namelen = dirlen+1 + loclen+1 + catlen+1 + domlen+3; - char name[namelen+1], *s = name; - - memcpy(s, dirname, dirlen); - s[dirlen] = '/'; - s += dirlen + 1; - memcpy(s, locname, loclen); - s[loclen] = '/'; - s += loclen + 1; - memcpy(s, catname, catlen); - s[catlen] = '/'; - s += catlen + 1; - memcpy(s, domainname, domlen); - s[domlen] = '.'; - s[domlen+1] = 'm'; - s[domlen+2] = 'o'; - s[domlen+3] = 0; for (p=cats; p; p=p->next) - if (!strcmp(p->name, name)) + if (p->binding == q && p->lm == lm && p->cat == category) break; if (!p) { + const char *dirname, *locname, *catname; + size_t dirlen, loclen, catlen; void *old_cats; size_t map_size; - const void *map = __map_file(name, &map_size); + + dirname = q->dirname; + locname = lm->name; + catname = catnames[category]; + + dirlen = q->dirlen; + loclen = strlen(locname); + catlen = catlens[category]; + + size_t namelen = dirlen+1 + loclen+1 + catlen+1 + domlen+3; + char name[namelen+1]; + char locbuf[loclen+1], *locp = locbuf; + const void *map; + + memcpy(locbuf, locname, loclen); + locbuf[loclen] = 0; + + for (;;) { + snprintf(name, namelen+1, "%s/%s/%s/%s.mo\0", dirname, locbuf, catname, domainname); + if (map = __map_file(name, &map_size)) break; + + if (locp = strchr(locbuf, '@')) { + *locp = 0; + locbuf[loclen] = '@'; + } else if (locp = strchr(locbuf, '_')) { + if (locbuf[loclen] == '@') { + locbuf[loclen] = 0; + *locp = '@'; + strcat(locp+1, locbuf + strlen(locbuf) + 1); + } else *locp = 0; + } else { + break; + } + } if (!map) goto notrans; + p = calloc(sizeof *p + namelen + 1, 1); if (!p) { __munmap((void *)map, map_size); @@ -178,7 +195,6 @@ } p->map = map; p->map_size = map_size; - memcpy(p->name, name, namelen+1); do { old_cats = cats; p->next = old_cats; @@ -193,6 +193,9 @@ __munmap((void *)map, map_size); goto notrans; } + p->cat = category; + p->binding = q; + p->lm = lm; p->map = map; p->map_size = map_size; do {