2018-03-08 Richard Biener <rguenther@suse.de>
[official-gcc.git] / libstdc++-v3 / testsuite / util / testsuite_fs.h
blob5a9cdac2f837dd6ed560a1f37d7380299fe01f16
1 // -*- C++ -*-
2 // Filesystem utils for the C++ library testsuite.
3 //
4 // Copyright (C) 2014-2018 Free Software Foundation, Inc.
5 //
6 // This file is part of the GNU ISO C++ Library. This library is free
7 // software; you can redistribute it and/or modify it under the
8 // terms of the GNU General Public License as published by the
9 // Free Software Foundation; either version 3, or (at your option)
10 // any later version.
12 // This library is distributed in the hope that it will be useful,
13 // but WITHOUT ANY WARRANTY; without even the implied warranty of
14 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 // GNU General Public License for more details.
17 // You should have received a copy of the GNU General Public License along
18 // with this library; see the file COPYING3. If not see
19 // <http://www.gnu.org/licenses/>.
22 #ifndef _TESTSUITE_FS_H
23 #define _TESTSUITE_FS_H 1
25 // Assume we want std::filesystem in C++17, unless USE_FILESYSTEM_TS defined:
26 #if __cplusplus >= 201703L && ! defined USE_FILESYSTEM_TS
27 #include <filesystem>
28 namespace test_fs = std::filesystem;
29 #else
30 #include <experimental/filesystem>
31 namespace test_fs = std::experimental::filesystem;
32 #endif
33 #include <fstream>
34 #include <string>
35 #include <cstdio>
36 #include <stdlib.h>
37 #include <unistd.h>
39 namespace __gnu_test
41 #define PATH_CHK(p1, p2, fn) \
42 if ( p1.fn() != p2.fn() ) \
43 throw test_fs::filesystem_error("comparing '" #fn "' failed", p1, p2, \
44 std::make_error_code(std::errc::invalid_argument) )
46 void
47 compare_paths(const test_fs::path& p1,
48 const test_fs::path& p2)
50 PATH_CHK( p1, p2, native );
51 PATH_CHK( p1, p2, string );
52 PATH_CHK( p1, p2, empty );
53 PATH_CHK( p1, p2, has_root_path );
54 PATH_CHK( p1, p2, has_root_name );
55 PATH_CHK( p1, p2, has_root_directory );
56 PATH_CHK( p1, p2, has_relative_path );
57 PATH_CHK( p1, p2, has_parent_path );
58 PATH_CHK( p1, p2, has_filename );
59 PATH_CHK( p1, p2, has_stem );
60 PATH_CHK( p1, p2, has_extension );
61 PATH_CHK( p1, p2, is_absolute );
62 PATH_CHK( p1, p2, is_relative );
63 auto d1 = std::distance(p1.begin(), p1.end());
64 auto d2 = std::distance(p2.begin(), p2.end());
65 if( d1 != d2 )
66 throw test_fs::filesystem_error(
67 "distance(begin, end)", p1, p2,
68 std::make_error_code(std::errc::invalid_argument) );
71 const std::string test_paths[] = {
72 "", "/", "//", "/.", "/./", "/a", "/a/", "/a//", "/a/b/c/d", "/a//b",
73 "a", "a/b", "a/b/", "a/b/c", "a/b/c.d", "a/b/..", "a/b/c.", "a/b/.c"
76 // This is NOT supposed to be a secure way to get a unique name!
77 // We just need a path that doesn't exist for testing purposes.
78 test_fs::path
79 nonexistent_path()
81 test_fs::path p;
82 #if defined(_GNU_SOURCE) || _XOPEN_SOURCE >= 500 || _POSIX_C_SOURCE >= 200112L
83 char tmp[] = "filesystem-test.XXXXXX";
84 int fd = ::mkstemp(tmp);
85 if (fd == -1)
86 throw test_fs::filesystem_error("mkstemp failed",
87 std::error_code(errno, std::generic_category()));
88 ::unlink(tmp);
89 ::close(fd);
90 p = tmp;
91 #else
92 char buf[64];
93 static int counter;
94 #if _GLIBCXX_USE_C99_STDIO
95 std::snprintf(buf, 64,
96 #else
97 std::sprintf(buf,
98 #endif
99 "filesystem-test.%d.%lu", counter++, (unsigned long) ::getpid());
100 p = buf;
101 #endif
102 return p;
105 // RAII helper to remove a file on scope exit.
106 struct scoped_file
108 using path_type = test_fs::path;
110 enum adopt_file_t { adopt_file };
112 explicit
113 scoped_file(const path_type& p = nonexistent_path()) : path(p)
114 { std::ofstream{p.native()}; }
116 scoped_file(path_type p, adopt_file_t) : path(p) { }
118 ~scoped_file() { if (!path.empty()) remove(path); }
120 path_type path;
123 } // namespace __gnu_test
124 #endif