]> git.ipfire.org Git - thirdparty/postgresql.git/commitdiff
Fix race condition in gettext() initialization in libpq and ecpglib.
authorTom Lane <tgl@sss.pgh.pa.us>
Fri, 21 Jan 2022 20:36:12 +0000 (15:36 -0500)
committerTom Lane <tgl@sss.pgh.pa.us>
Fri, 21 Jan 2022 20:36:29 +0000 (15:36 -0500)
In libpq and ecpglib, multiple threads can concurrently enter the
initialization logic for message localization.  Since we set the
its-done flag before actually doing the work, it'd be possible
for some threads to reach gettext() before anyone has called
bindtextdomain().  Barring bugs in libintl itself, this would not
result in anything worse than failure to localize some early
messages.  Nonetheless, it's a bug, and an easy one to fix.

Noted while investigating bug #17299 from Clemens Zeidler
(much thanks to Liam Bowen for followup investigation on that).
It currently appears that that actually *is* a bug in libintl itself,
but that doesn't let us off the hook for this bit.

Back-patch to all supported versions.

Discussion: https://postgr.es/m/17299-7270741958c0b1ab@postgresql.org
Discussion: https://postgr.es/m/CAE7q7Eit4Eq2=bxce=Fm8HAStECjaXUE=WBQc-sDDcgJQ7s7eg@mail.gmail.com

src/interfaces/ecpg/ecpglib/misc.c
src/interfaces/libpq/fe-misc.c

index 85a308f537954e4322fae7de72b1bc0675a08c08..03d46cd2abed438bb9944d5e7704216261b49bd9 100644 (file)
@@ -489,7 +489,14 @@ win32_pthread_once(volatile pthread_once_t *once, void (*fn) (void))
 char *
 ecpg_gettext(const char *msgid)
 {
-       static bool already_bound = false;
+       /*
+        * If multiple threads come through here at about the same time, it's okay
+        * for more than one of them to call bindtextdomain().  But it's not okay
+        * for any of them to reach dgettext() before bindtextdomain() is
+        * complete, so don't set the flag till that's done.  Use "volatile" just
+        * to be sure the compiler doesn't try to get cute.
+        */
+       static volatile bool already_bound = false;
 
        if (!already_bound)
        {
@@ -501,12 +508,12 @@ ecpg_gettext(const char *msgid)
 #endif
                const char *ldir;
 
-               already_bound = true;
                /* No relocatable lookup here because the binary could be anywhere */
                ldir = getenv("PGLOCALEDIR");
                if (!ldir)
                        ldir = LOCALEDIR;
                bindtextdomain(PG_TEXTDOMAIN("ecpglib"), ldir);
+               already_bound = true;
 #ifdef WIN32
                SetLastError(save_errno);
 #else
index 14d042e220e63d1c8596cb99c69f7277ddcadc90..aa38eb187277f36e1916648b5758bf3961383f14 100644 (file)
@@ -1217,7 +1217,14 @@ PQenv2encoding(void)
 static void
 libpq_binddomain()
 {
-       static bool already_bound = false;
+       /*
+        * If multiple threads come through here at about the same time, it's okay
+        * for more than one of them to call bindtextdomain().  But it's not okay
+        * for any of them to return to caller before bindtextdomain() is
+        * complete, so don't set the flag till that's done.  Use "volatile" just
+        * to be sure the compiler doesn't try to get cute.
+        */
+       static volatile bool already_bound = false;
 
        if (!already_bound)
        {
@@ -1229,12 +1236,12 @@ libpq_binddomain()
 #endif
                const char *ldir;
 
-               already_bound = true;
                /* No relocatable lookup here because the binary could be anywhere */
                ldir = getenv("PGLOCALEDIR");
                if (!ldir)
                        ldir = LOCALEDIR;
                bindtextdomain(PG_TEXTDOMAIN("libpq"), ldir);
+               already_bound = true;
 #ifdef WIN32
                SetLastError(save_errno);
 #else