+2017-11-01 Jonathan Wakely <jwakely@redhat.com>
+
+ PR libstdc++/82777
+ * src/filesystem/std-path.cc (path::lexically_normal): Remove dot-dot
+ elements correctly.
+ * testsuite/27_io/filesystem/path/generation/normal.cc: Add testcase.
+ * testsuite/util/testsuite_fs.h (compare_paths): Improve exception
+ text.
+
2017-10-30 Jonathan Wakely <jwakely@redhat.com>
* include/Makefile.am (stamp-bits-sup): Do not create broken symlink
#endif
if (is_dotdot(p))
{
- if (ret.has_filename() && !is_dotdot(ret.filename()))
- ret.remove_filename();
- else if (ret.has_filename() || !ret.has_root_directory())
- ret /= p;
+ if (ret.has_filename())
+ {
+ // remove a non-dot-dot filename immediately followed by /..
+ if (!is_dotdot(ret.filename()))
+ ret.remove_filename();
+ else
+ ret /= p;
+ }
+ else if (!ret.has_relative_path())
+ {
+ if (!ret.is_absolute())
+ ret /= p;
+ }
+ else
+ {
+ // Got a path with a relative path (i.e. at least one non-root
+ // element) and no filename at the end (i.e. empty last element),
+ // so must have a trailing slash. See what is before it.
+ auto elem = std::prev(ret.end(), 2);
+ if (elem->has_filename() && !is_dotdot(*elem))
+ {
+ // Remove the filename before the trailing slash
+ // (equiv. to ret = ret.parent_path().remove_filename())
+ ret._M_pathname.erase(elem._M_cur->_M_pos);
+ ret._M_cmpts.erase(elem._M_cur, ret._M_cmpts.end());
+ }
+ else // ???
+ ret /= p;
+ }
}
else if (is_dot(p))
ret /= path();
compare_paths( path().lexically_normal(), "" );
compare_paths( path("/..").lexically_normal(), "/" );
+
+ // PR libstdc++/82777
+ compare_paths( path("./a/b/c/../.././b/c").lexically_normal(), "a/b/c" );
+ compare_paths( path("/a/b/c/../.././b/c").lexically_normal(), "/a/b/c" );
}
void
{
#define PATH_CHK(p1, p2, fn) \
if ( p1.fn() != p2.fn() ) \
- throw test_fs::filesystem_error( #fn, p1, p2, \
+ throw test_fs::filesystem_error("comparing '" #fn "' failed", p1, p2, \
std::make_error_code(std::errc::invalid_argument) )
void