| #define _BSD_SOURCE |
| #include <nl_types.h> |
| #include <string.h> |
| #include <stdint.h> |
| #include <endian.h> |
| #include <errno.h> |
| #include <langinfo.h> |
| #include <locale.h> |
| #include <sys/mman.h> |
| #include "libc.h" |
| |
| #define V(p) be32toh(*(uint32_t *)(p)) |
| |
| static nl_catd do_catopen(const char *name) |
| { |
| size_t size; |
| const unsigned char *map = __map_file(name, &size); |
| /* Size recorded in the file must match file size; otherwise |
| * the information needed to unmap the file will be lost. */ |
| if (!map || V(map) != 0xff88ff89 || 20+V(map+8) != size) { |
| if(map) munmap((void *)map, size); |
| errno = ENOENT; |
| return (nl_catd)-1; |
| } |
| return (nl_catd)map; |
| } |
| |
| nl_catd catopen(const char *name, int oflag) |
| { |
| nl_catd catd; |
| |
| if (strchr(name, '/')) return do_catopen(name); |
| |
| char buf[PATH_MAX]; |
| size_t i; |
| const char *path, *lang, *p, *z; |
| if (libc.secure || !(path = getenv("NLSPATH"))) { |
| errno = ENOENT; |
| return (nl_catd)-1; |
| } |
| lang = oflag ? nl_langinfo(_NL_LOCALE_NAME(LC_MESSAGES)) : getenv("LANG"); |
| if (!lang) lang = ""; |
| for (p=path; *p; p=z) { |
| i = 0; |
| z = __strchrnul(p, ':'); |
| for (; p<z; p++) { |
| const char *v; |
| size_t l; |
| if (*p!='%') v=p, l=1; |
| else switch (*++p) { |
| case 'N': v=name; l=strlen(v); break; |
| case 'L': v=lang; l=strlen(v); break; |
| case 'l': v=lang; l=strcspn(v,"_.@"); break; |
| case 't': |
| v=__strchrnul(lang,'_'); |
| if (*v) v++; |
| l=strcspn(v,".@"); |
| break; |
| case 'c': v="UTF-8"; l=5; break; |
| case '%': v="%"; l=1; break; |
| default: v=0; |
| } |
| if (!v || l >= sizeof buf - i) { |
| break; |
| } |
| memcpy(buf+i, v, l); |
| i += l; |
| } |
| if (!*z && (p<z || !i)) break; |
| if (p<z) continue; |
| if (*z) z++; |
| buf[i] = 0; |
| /* Leading : or :: in NLSPATH is same as %N */ |
| catd = do_catopen(i ? buf : name); |
| if (catd != (nl_catd)-1) return catd; |
| } |
| errno = ENOENT; |
| return (nl_catd)-1; |
| } |