diff options
author | Corinna Vinschen <corinna@vinschen.de> | 2013-06-17 12:37:09 +0000 |
---|---|---|
committer | Corinna Vinschen <corinna@vinschen.de> | 2013-06-17 12:37:09 +0000 |
commit | 2566c2e60060d7734554d6036e71ad198d945e35 (patch) | |
tree | c3d9d012ea8900ff0df74e512e83d0f096d1ed9b /winsup | |
parent | 42c8e85109d9c8edaf1e46eb26d5484ebdf16881 (diff) | |
download | newlib-2566c2e60060d7734554d6036e71ad198d945e35.zip newlib-2566c2e60060d7734554d6036e71ad198d945e35.tar.gz newlib-2566c2e60060d7734554d6036e71ad198d945e35.tar.bz2 |
* path.cc (cnt_bs): New inline function.
(symlink_native): Fix creating relative native symlink.
Diffstat (limited to 'winsup')
-rw-r--r-- | winsup/cygwin/ChangeLog | 5 | ||||
-rw-r--r-- | winsup/cygwin/path.cc | 59 | ||||
-rw-r--r-- | winsup/cygwin/release/1.7.21 | 3 |
3 files changed, 63 insertions, 4 deletions
diff --git a/winsup/cygwin/ChangeLog b/winsup/cygwin/ChangeLog index 7c2f442..00ba880 100644 --- a/winsup/cygwin/ChangeLog +++ b/winsup/cygwin/ChangeLog @@ -1,5 +1,10 @@ 2013-06-17 Corinna Vinschen <corinna@vinschen.de> + * path.cc (cnt_bs): New inline function. + (symlink_native): Fix creating relative native symlink. + +2013-06-17 Corinna Vinschen <corinna@vinschen.de> + * fhandler_clipboard.cc (fhandler_dev_clipboard::read): Fix buffer read access overrun when pos > 0. diff --git a/winsup/cygwin/path.cc b/winsup/cygwin/path.cc index e0fa376..9492590 100644 --- a/winsup/cygwin/path.cc +++ b/winsup/cygwin/path.cc @@ -1530,12 +1530,25 @@ symlink_nfs (const char *oldpath, path_conv &win32_newpath) return 0; } +/* Count backslashes between s and e. */ +static inline int +cnt_bs (PWCHAR s, PWCHAR e) +{ + int num = 0; + + while (s < e) + if (*s++ == L'\\') + ++num; + return num; +} + static int symlink_native (const char *oldpath, path_conv &win32_newpath) { tmp_pathbuf tp; path_conv win32_oldpath; PUNICODE_STRING final_oldpath, final_newpath; + UNICODE_STRING final_oldpath_buf; if (isabspath (oldpath)) { @@ -1554,10 +1567,48 @@ symlink_native (const char *oldpath, path_conv &win32_newpath) stpcpy (stpncpy (absoldpath, win32_newpath.normalized_path, len), oldpath); win32_oldpath.check (absoldpath, PC_SYM_NOFOLLOW, stat_suffixes); - UNICODE_STRING dirpath; - RtlSplitUnicodePath (win32_newpath.get_nt_native_path (), &dirpath, NULL); - final_oldpath = win32_oldpath.get_nt_native_path (); - final_oldpath->Buffer += dirpath.Length / sizeof (WCHAR); + + /* Try hard to keep Windows symlink path relative. */ + + /* 1. Find common path prefix. */ + PWCHAR c_old = win32_oldpath.get_nt_native_path ()->Buffer; + PWCHAR c_new = win32_newpath.get_nt_native_path ()->Buffer; + /* Windows compatible == always check case insensitive. */ + while (towupper (*c_old++) == towupper (*c_new++)) + ; + /* The last component could share a common prefix, so make sure we end + up on the first char after the last common backslash. */ + while (c_old[-1] != L'\\') + --c_old, --c_new; + + /* 2. Check if prefix is long enough. The prefix must at least points to + a complete device: \\?\X:\ or \\?\UNC\server\share\ are the minimum + prefix strings. We start counting behind the \\?\ for speed. */ + int num = cnt_bs (win32_oldpath.get_nt_native_path ()->Buffer + 4, c_old); + if (num < 1 /* locale drive. */ + || (win32_oldpath.get_nt_native_path ()->Buffer[6] != L':' + && num < 3)) /* UNC path. */ + { + /* 3a. No valid common path prefix: Create absolute symlink. */ + final_oldpath = win32_oldpath.get_nt_native_path (); + final_oldpath->Buffer[1] = L'\\'; + } + else + { + /* 3b. Common path prefx. Count number of additional directories + in symlink's path, and prepend as much ".." path components + to the target path. */ + PWCHAR e_new = win32_newpath.get_nt_native_path ()->Buffer + + win32_newpath.get_nt_native_path ()->Length + / sizeof (WCHAR); + num = cnt_bs (c_new, e_new); + final_oldpath = &final_oldpath_buf; + final_oldpath->Buffer = tp.w_get (); + PWCHAR e_old = final_oldpath->Buffer; + while (num-- > 0) + e_old = wcpcpy (e_old, L"..\\"); + wcpcpy (e_old, c_old); + } } /* If the symlink target doesn't exist, don't create native symlink. Otherwise the directory flag in the symlink is potentially wrong diff --git a/winsup/cygwin/release/1.7.21 b/winsup/cygwin/release/1.7.21 index 2ea2677..b1a045e 100644 --- a/winsup/cygwin/release/1.7.21 +++ b/winsup/cygwin/release/1.7.21 @@ -11,3 +11,6 @@ Bug fixes: - Fix EFAULT when reading large clipboard. Fixes: http://cygwin.com/ml/cygwin/2013-06/msg00311.html + +- Fix creation of relative native symlinks. + Fixes: http://cygwin.com/ml/cygwin/2013-06/msg00340.html |