Skip to content

Commit 2c4a532

Browse files
macdiceadunstan
authored andcommitted
Fix readlink() for non-PostgreSQL junction points on Windows.
Since commit c5cb8f3 taught stat() to follow symlinks, and since initdb uses pg_mkdir_p(), and that examines parent directories, our humble readlink() implementation can now be exposed to junction points not of PostgreSQL origin. Those might be corrupted by our naive path mangling, which doesn't really understand NT paths in general. Simply decline to transform paths that don't look like a drive absolute path. That means that readlink() returns the NT path directly when checking a parent directory of PGDATA that happen to point to a drive using "rooted" format. That works for the purposes of our stat() emulation. Reported-by: Roman Zharkov <r.zharkov@postgrespro.ru> Reviewed-by: Roman Zharkov <r.zharkov@postgrespro.ru> Discussion: https://postgr.es/m/4590c37927d7b8ee84f9855d83229018%40postgrespro.ru Discussion: https://postgr.es/m/CA%2BhUKG%2BajSQ_8eu2AogTncOnZ5me2D-Cn66iN_-wZnRjLN%2Bicg%40mail.gmail.com Backpatched commit f71007f as above by Thomas Munro into releases 13 thru 15 Discussion: https://postgr.es/m/CA+hUKGLbnv+pe3q1fYOVkLD3pMra7GuihfMxUN-1831YH9RYQg@mail.gmail.com
1 parent fbf8efb commit 2c4a532

File tree

1 file changed

+14
-3
lines changed

1 file changed

+14
-3
lines changed

src/port/dirmod.c

+14-3
Original file line numberDiff line numberDiff line change
@@ -351,10 +351,21 @@ pgreadlink(const char *path, char *buf, size_t size)
351351
}
352352

353353
/*
354-
* If the path starts with "\??\", which it will do in most (all?) cases,
355-
* strip those out.
354+
* If the path starts with "\??\" followed by a "drive absolute" path
355+
* (known to Windows APIs as RtlPathTypeDriveAbsolute), then strip that
356+
* prefix. This undoes some of the transformation performed by
357+
* pqsymlink(), to get back to a format that users are used to seeing. We
358+
* don't know how to transform other path types that might be encountered
359+
* outside PGDATA, so we just return them directly.
356360
*/
357-
if (r > 4 && strncmp(buf, "\\??\\", 4) == 0)
361+
if (r >= 7 &&
362+
buf[0] == '\\' &&
363+
buf[1] == '?' &&
364+
buf[2] == '?' &&
365+
buf[3] == '\\' &&
366+
isalpha(buf[4]) &&
367+
buf[5] == ':' &&
368+
buf[6] == '\\')
358369
{
359370
memmove(buf, buf + 4, strlen(buf + 4) + 1);
360371
r -= 4;

0 commit comments

Comments
 (0)