29#include <gnu/libc-version.h>
81#define TEXTBUFLEN 1024
86 const char *arg2, ssize_t len2,
89 const char *src, ssize_t srclen,
96static int strncoll_libc_win32_utf8(
const char *arg1, ssize_t len1,
97 const char *arg2, ssize_t len2,
101static size_t char2wchar(
wchar_t *to,
size_t tolen,
const char *from,
105 const char *src, ssize_t srclen,
108 const char *src, ssize_t srclen,
111 const char *src, ssize_t srclen,
114 const char *src, ssize_t srclen,
117 const char *src, ssize_t srclen,
120 const char *src, ssize_t srclen,
181 return isxdigit_l((
unsigned char) wc,
locale->lt);
183 return _isxdigit_l((
unsigned char) wc,
locale->lt);
252 return iswxdigit_l((wint_t) wc,
locale->lt);
254 return _iswxdigit_l((wint_t) wc,
locale->lt);
287 if (
sizeof(
wchar_t) >= 4 || wc <= (
pg_wchar) 0xFFFF)
315 if (
sizeof(
wchar_t) >= 4 || wc <= (
pg_wchar) 0xFFFF)
332 for (
i = 0;
i < srclen &&
i < dstsize;
i++)
334 unsigned char ch = (
unsigned char) src[
i];
336 if (ch >=
'A' && ch <=
'Z')
382 .downcase_ident = NULL,
405 .downcase_ident = NULL,
424 .strnxfrm_prefix = NULL,
436 .strxfrm_is_safe =
true,
438 .strxfrm_is_safe =
false,
444 .
strncoll = strncoll_libc_win32_utf8,
446 .strnxfrm_prefix = NULL,
448 .strxfrm_is_safe =
true,
450 .strxfrm_is_safe =
false,
460 srclen = strlen(src);
462 if (srclen + 1 <= destsize)
467 memcpy(
dest, src, srclen);
477 for (p =
dest; *p; p++)
481 if (*p >=
'A' && *p <=
'Z')
506 srclen = strlen(src);
509 if ((srclen + 1) > (INT_MAX /
sizeof(
wchar_t)))
511 (
errcode(ERRCODE_OUT_OF_MEMORY),
512 errmsg(
"out of memory")));
517 char2wchar(workspace, srclen + 1, src, srclen, loc);
519 for (curr_char = 0; workspace[curr_char] != 0; curr_char++)
520 workspace[curr_char] =
towlower_l(workspace[curr_char], loc);
526 result =
palloc(max_size + 1);
528 result_size =
wchar2char(result, workspace, max_size + 1, loc);
530 if (result_size + 1 > destsize)
533 memcpy(
dest, result, result_size);
534 dest[result_size] =
'\0';
547 srclen = strlen(src);
549 if (srclen + 1 <= destsize)
552 int wasalnum =
false;
555 memcpy(
dest, src, srclen);
565 for (p =
dest; *p; p++)
571 if (*p >=
'A' && *p <=
'Z')
578 if (*p >=
'a' && *p <=
'z')
591 wasalnum =
isalnum_l((
unsigned char) *p, loc);
603 int wasalnum =
false;
611 srclen = strlen(src);
614 if ((srclen + 1) > (INT_MAX /
sizeof(
wchar_t)))
616 (
errcode(ERRCODE_OUT_OF_MEMORY),
617 errmsg(
"out of memory")));
622 char2wchar(workspace, srclen + 1, src, srclen, loc);
624 for (curr_char = 0; workspace[curr_char] != 0; curr_char++)
627 workspace[curr_char] =
towlower_l(workspace[curr_char], loc);
629 workspace[curr_char] =
towupper_l(workspace[curr_char], loc);
630 wasalnum =
iswalnum_l(workspace[curr_char], loc);
637 result =
palloc(max_size + 1);
639 result_size =
wchar2char(result, workspace, max_size + 1, loc);
641 if (result_size + 1 > destsize)
644 memcpy(
dest, result, result_size);
645 dest[result_size] =
'\0';
658 srclen = strlen(src);
660 if (srclen + 1 <= destsize)
665 memcpy(
dest, src, srclen);
675 for (p =
dest; *p; p++)
679 if (*p >=
'a' && *p <=
'z')
704 srclen = strlen(src);
707 if ((srclen + 1) > (INT_MAX /
sizeof(
wchar_t)))
709 (
errcode(ERRCODE_OUT_OF_MEMORY),
710 errmsg(
"out of memory")));
715 char2wchar(workspace, srclen + 1, src, srclen, loc);
717 for (curr_char = 0; workspace[curr_char] != 0; curr_char++)
718 workspace[curr_char] =
towupper_l(workspace[curr_char], loc);
724 result =
palloc(max_size + 1);
726 result_size =
wchar2char(result, workspace, max_size + 1, loc);
728 if (result_size + 1 > destsize)
731 memcpy(
dest, result, result_size);
732 dest[result_size] =
'\0';
748 if (
collid == DEFAULT_COLLATION_OID)
757 Anum_pg_database_datcollate);
760 Anum_pg_database_datctype);
775 Anum_pg_collation_collcollate);
778 Anum_pg_collation_collctype);
790 (strcmp(collate,
"POSIX") == 0);
791 result->
ctype_is_c = (strcmp(ctype,
"C") == 0) ||
792 (strcmp(ctype,
"POSIX") == 0);
798 result->
collate = &collate_methods_libc_win32_utf8;
829 if (strcmp(collate, ctype) == 0)
831 if (strcmp(ctype,
"C") != 0 && strcmp(ctype,
"POSIX") != 0)
836 loc = newlocale(LC_COLLATE_MASK | LC_CTYPE_MASK, collate,
839 loc = _create_locale(LC_ALL, collate);
851 if (strcmp(collate,
"C") != 0 && strcmp(collate,
"POSIX") != 0)
854 loc1 = newlocale(LC_COLLATE_MASK, collate, NULL);
859 if (strcmp(ctype,
"C") != 0 && strcmp(ctype,
"POSIX") != 0)
862 loc = newlocale(LC_CTYPE_MASK, ctype, loc1);
880 (
errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
881 errmsg(
"collations with different collate and ctype values are not supported on this platform")));
896strncoll_libc(
const char *arg1, ssize_t len1,
const char *arg2, ssize_t len2,
901 size_t bufsize1 = (len1 == -1) ? 0 : len1 + 1;
902 size_t bufsize2 = (len2 == -1) ? 0 : len2 + 1;
919 memcpy(buf1, arg1, len1);
930 char *buf2 =
buf + bufsize1;
932 memcpy(buf2, arg2, len2);
968 memcpy(
buf, src, srclen);
977 Assert(result >= destsize ||
dest[result] ==
'\0');
985 char *collversion = NULL;
991#if defined(__GLIBC__)
993 collversion =
pstrdup(gnu_get_libc_version());
994#elif defined(LC_VERSION_MASK)
998 loc = newlocale(LC_COLLATE_MASK, collcollate, NULL);
1002 pstrdup(querylocale(LC_COLLATE_MASK | LC_VERSION_MASK, loc));
1007 (
errmsg(
"could not load locale \"%s\"", collcollate)));
1014 NLSVERSIONINFOEX version = {
sizeof(NLSVERSIONINFOEX)};
1015 WCHAR wide_collcollate[LOCALE_NAME_MAX_LENGTH];
1017 MultiByteToWideChar(CP_ACP, 0, collcollate, -1, wide_collcollate,
1018 LOCALE_NAME_MAX_LENGTH);
1019 if (!GetNLSVersionEx(COMPARE_STRING, wide_collcollate, &version))
1028 if (GetLastError() == ERROR_INVALID_PARAMETER)
1032 (
errmsg(
"could not get collation version for locale \"%s\": error code %lu",
1036 collversion =
psprintf(
"%lu.%lu,%lu.%lu",
1037 (version.dwNLSVersion >> 8) & 0xFFFF,
1038 version.dwNLSVersion & 0xFF,
1039 (version.dwDefinedVersion >> 8) & 0xFFFF,
1040 version.dwDefinedVersion & 0xFF);
1057strncoll_libc_win32_utf8(
const char *arg1, ssize_t len1,
const char *arg2,
1072 len1 = strlen(arg1);
1074 len2 = strlen(arg2);
1076 a1len = len1 * 2 + 2;
1077 a2len = len2 * 2 + 2;
1090 r = MultiByteToWideChar(CP_UTF8, 0, arg1, len1,
1091 (LPWSTR) a1p, a1len / 2);
1094 (
errmsg(
"could not convert string to UTF-16: error code %lu",
1097 ((LPWSTR) a1p)[r] = 0;
1103 r = MultiByteToWideChar(CP_UTF8, 0, arg2, len2,
1104 (LPWSTR) a2p, a2len / 2);
1107 (
errmsg(
"could not convert string to UTF-16: error code %lu",
1110 ((LPWSTR) a2p)[r] = 0;
1114 if (result == 2147483647)
1116 (
errmsg(
"could not compare Unicode strings: %m")));
1147 (
errcode(ERRCODE_INVALID_PARAMETER_VALUE),
1148 errmsg(
"could not create locale \"%s\": %m",
1150 (save_errno == ENOENT ?
1151 errdetail(
"The operating system could not find any locale data for the locale name \"%s\".",
1159#ifndef HAVE_MBSTOWCS_L
1164 return _mbstowcs_l(
dest, src, n, loc);
1167 locale_t save_locale = uselocale(loc);
1169 result = mbstowcs(
dest, src, n);
1170 uselocale(save_locale);
1175#ifndef HAVE_WCSTOMBS_L
1180 return _wcstombs_l(
dest, src, n, loc);
1183 locale_t save_locale = uselocale(loc);
1185 result = wcstombs(
dest, src, n);
1186 uselocale(save_locale);
1221 result = WideCharToMultiByte(CP_UTF8, 0, from, -1, to, tolen,
1238 result = wcstombs(to, from, tolen);
1259char2wchar(
wchar_t *to,
size_t tolen,
const char *from,
size_t fromlen,
1276 result = MultiByteToWideChar(CP_UTF8, 0, from, fromlen, to, tolen - 1);
1298 result = mbstowcs(to,
str, tolen);
1322 (
errcode(ERRCODE_CHARACTER_NOT_IN_REPERTOIRE),
1323 errmsg(
"invalid multibyte character for locale"),
1324 errhint(
"The server's LC_CTYPE locale is probably incompatible with the database encoding.")));
#define TextDatumGetCString(d)
#define IS_HIGHBIT_SET(ch)
int errdetail(const char *fmt,...)
int errhint(const char *fmt,...)
int errcode(int sqlerrcode)
int errmsg(const char *fmt,...)
#define ereport(elevel,...)
#define palloc_array(type, count)
Assert(PointerIsAligned(start, uint64))
#define HeapTupleIsValid(tuple)
if(TABLE==NULL||TABLE_index==NULL)
int GetDatabaseEncoding(void)
bool pg_verifymbstr(const char *mbstr, int len, bool noError)
int pg_database_encoding_max_length(void)
void * MemoryContextAllocZero(MemoryContext context, Size size)
char * pstrdup(const char *in)
void pfree(void *pointer)
char * pnstrdup(const char *in, Size len)
static bool wc_isalpha_libc_mb(pg_wchar wc, pg_locale_t locale)
static size_t strtitle_libc_mb(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
static bool wc_ispunct_libc_sb(pg_wchar wc, pg_locale_t locale)
static const struct ctype_methods ctype_methods_libc_other_mb
static const struct ctype_methods ctype_methods_libc_utf8
static pg_wchar toupper_libc_mb(pg_wchar wc, pg_locale_t locale)
static size_t downcase_ident_libc_sb(char *dst, size_t dstsize, const char *src, ssize_t srclen, pg_locale_t locale)
static bool wc_iscased_libc_sb(pg_wchar wc, pg_locale_t locale)
static size_t strupper_libc_sb(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
static bool wc_isprint_libc_mb(pg_wchar wc, pg_locale_t locale)
static bool wc_isdigit_libc_sb(pg_wchar wc, pg_locale_t locale)
pg_locale_t create_pg_locale_libc(Oid collid, MemoryContext context)
size_t wchar2char(char *to, const wchar_t *from, size_t tolen, locale_t loc)
static bool wc_isspace_libc_sb(pg_wchar wc, pg_locale_t locale)
static bool wc_islower_libc_sb(pg_wchar wc, pg_locale_t locale)
static bool wc_isupper_libc_mb(pg_wchar wc, pg_locale_t locale)
static pg_wchar toupper_libc_sb(pg_wchar wc, pg_locale_t locale)
static bool wc_isgraph_libc_mb(pg_wchar wc, pg_locale_t locale)
static size_t char2wchar(wchar_t *to, size_t tolen, const char *from, size_t fromlen, locale_t loc)
static bool wc_isalnum_libc_mb(pg_wchar wc, pg_locale_t locale)
static bool wc_isalnum_libc_sb(pg_wchar wc, pg_locale_t locale)
static bool wc_isalpha_libc_sb(pg_wchar wc, pg_locale_t locale)
static bool wc_isprint_libc_sb(pg_wchar wc, pg_locale_t locale)
char * get_collation_actual_version_libc(const char *collcollate)
static bool wc_isupper_libc_sb(pg_wchar wc, pg_locale_t locale)
static locale_t make_libc_collator(const char *collate, const char *ctype)
static bool wc_isgraph_libc_sb(pg_wchar wc, pg_locale_t locale)
static bool wc_iscased_libc_mb(pg_wchar wc, pg_locale_t locale)
static pg_wchar tolower_libc_sb(pg_wchar wc, pg_locale_t locale)
static size_t wcstombs_l(char *dest, const wchar_t *src, size_t n, locale_t loc)
static const struct collate_methods collate_methods_libc
static bool wc_ispunct_libc_mb(pg_wchar wc, pg_locale_t locale)
static size_t strupper_libc_mb(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
static int strncoll_libc(const char *arg1, ssize_t len1, const char *arg2, ssize_t len2, pg_locale_t locale)
static size_t strlower_libc_sb(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
static bool wc_islower_libc_mb(pg_wchar wc, pg_locale_t locale)
static const struct ctype_methods ctype_methods_libc_sb
static size_t strnxfrm_libc(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
static bool wc_isxdigit_libc_sb(pg_wchar wc, pg_locale_t locale)
void report_newlocale_failure(const char *localename)
static pg_wchar tolower_libc_mb(pg_wchar wc, pg_locale_t locale)
static bool wc_isdigit_libc_mb(pg_wchar wc, pg_locale_t locale)
static bool wc_isspace_libc_mb(pg_wchar wc, pg_locale_t locale)
static size_t strtitle_libc_sb(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
static bool wc_isxdigit_libc_mb(pg_wchar wc, pg_locale_t locale)
static size_t mbstowcs_l(wchar_t *dest, const char *src, size_t n, locale_t loc)
static size_t strlower_libc_mb(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
static char buf[DEFAULT_XLOG_SEG_SIZE]
int pg_strcasecmp(const char *s1, const char *s2)
static unsigned char pg_ascii_tolower(unsigned char ch)
static unsigned char pg_ascii_toupper(unsigned char ch)
int pg_strncasecmp(const char *s1, const char *s2, size_t n)
static Datum ObjectIdGetDatum(Oid X)
char * psprintf(const char *fmt,...)
int(* strncoll)(const char *arg1, ssize_t len1, const char *arg2, ssize_t len2, pg_locale_t locale)
size_t(* strlower)(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
const struct ctype_methods * ctype
const struct collate_methods * collate
void ReleaseSysCache(HeapTuple tuple)
HeapTuple SearchSysCache1(int cacheId, Datum key1)
Datum SysCacheGetAttrNotNull(int cacheId, HeapTuple tup, AttrNumber attributeNumber)