xref: /llvm-project/llvm/unittests/Support/ReplaceFileTest.cpp (revision 0d802a4923e6a603aa1ee06d0969a793cc93f858)
17f68a716SGreg Bedwell //===- llvm/unittest/Support/ReplaceFileTest.cpp - unit tests -------------===//
27f68a716SGreg Bedwell //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
67f68a716SGreg Bedwell //
77f68a716SGreg Bedwell //===----------------------------------------------------------------------===//
87f68a716SGreg Bedwell 
97f68a716SGreg Bedwell #include "llvm/Support/Errc.h"
107f68a716SGreg Bedwell #include "llvm/Support/ErrorHandling.h"
117f68a716SGreg Bedwell #include "llvm/Support/FileSystem.h"
127f68a716SGreg Bedwell #include "llvm/Support/MemoryBuffer.h"
137f68a716SGreg Bedwell #include "llvm/Support/Path.h"
147f68a716SGreg Bedwell #include "llvm/Support/Process.h"
157f68a716SGreg Bedwell #include "gtest/gtest.h"
167f68a716SGreg Bedwell 
177f68a716SGreg Bedwell using namespace llvm;
187f68a716SGreg Bedwell using namespace llvm::sys;
197f68a716SGreg Bedwell 
20*0d802a49SPavel Labath #define ASSERT_NO_ERROR(x)                                                 \
21*0d802a49SPavel Labath   do {                                                                     \
22*0d802a49SPavel Labath     if (std::error_code ASSERT_NO_ERROR_ec = x) {                          \
23*0d802a49SPavel Labath       errs() << #x ": did not return errc::success.\n"                     \
24*0d802a49SPavel Labath              << "error number: " << ASSERT_NO_ERROR_ec.value() << "\n"     \
25*0d802a49SPavel Labath              << "error message: " << ASSERT_NO_ERROR_ec.message() << "\n"; \
26*0d802a49SPavel Labath     }                                                                      \
27*0d802a49SPavel Labath   } while (false)
287f68a716SGreg Bedwell 
297f68a716SGreg Bedwell namespace {
CreateFileWithContent(const SmallString<128> & FilePath,const StringRef & content)307f68a716SGreg Bedwell std::error_code CreateFileWithContent(const SmallString<128> &FilePath,
317f68a716SGreg Bedwell                                       const StringRef &content) {
327f68a716SGreg Bedwell   int FD = 0;
331f67a3cbSZachary Turner   if (std::error_code ec = fs::openFileForWrite(FilePath, FD))
347f68a716SGreg Bedwell     return ec;
357f68a716SGreg Bedwell 
367f68a716SGreg Bedwell   const bool ShouldClose = true;
377f68a716SGreg Bedwell   raw_fd_ostream OS(FD, ShouldClose);
387f68a716SGreg Bedwell   OS << content;
397f68a716SGreg Bedwell 
407f68a716SGreg Bedwell   return std::error_code();
417f68a716SGreg Bedwell }
427f68a716SGreg Bedwell 
437f68a716SGreg Bedwell class ScopedFD {
447f68a716SGreg Bedwell   int FD;
457f68a716SGreg Bedwell 
467f68a716SGreg Bedwell   ScopedFD(const ScopedFD &) = delete;
477f68a716SGreg Bedwell   ScopedFD &operator=(const ScopedFD &) = delete;
487f68a716SGreg Bedwell 
497f68a716SGreg Bedwell  public:
ScopedFD(int Descriptor)507f68a716SGreg Bedwell   explicit ScopedFD(int Descriptor) : FD(Descriptor) {}
~ScopedFD()517f68a716SGreg Bedwell   ~ScopedFD() { Process::SafelyCloseFileDescriptor(FD); }
527f68a716SGreg Bedwell };
537f68a716SGreg Bedwell 
FDHasContent(int FD,StringRef Content)5480e31f1fSPeter Collingbourne bool FDHasContent(int FD, StringRef Content) {
55cc418a3aSReid Kleckner   auto Buffer =
56cc418a3aSReid Kleckner       MemoryBuffer::getOpenFile(sys::fs::convertFDToNativeFile(FD), "", -1);
5780e31f1fSPeter Collingbourne   assert(Buffer);
5880e31f1fSPeter Collingbourne   return Buffer.get()->getBuffer() == Content;
5980e31f1fSPeter Collingbourne }
6080e31f1fSPeter Collingbourne 
FileHasContent(StringRef File,StringRef Content)6180e31f1fSPeter Collingbourne bool FileHasContent(StringRef File, StringRef Content) {
6280e31f1fSPeter Collingbourne   int FD = 0;
6380e31f1fSPeter Collingbourne   auto EC = fs::openFileForRead(File, FD);
6480e31f1fSPeter Collingbourne   (void)EC;
6580e31f1fSPeter Collingbourne   assert(!EC);
6680e31f1fSPeter Collingbourne   ScopedFD EventuallyCloseIt(FD);
6780e31f1fSPeter Collingbourne   return FDHasContent(FD, Content);
6880e31f1fSPeter Collingbourne }
6980e31f1fSPeter Collingbourne 
TEST(rename,FileOpenedForReadingCanBeReplaced)707f68a716SGreg Bedwell TEST(rename, FileOpenedForReadingCanBeReplaced) {
717f68a716SGreg Bedwell   // Create unique temporary directory for this test.
727f68a716SGreg Bedwell   SmallString<128> TestDirectory;
737f68a716SGreg Bedwell   ASSERT_NO_ERROR(fs::createUniqueDirectory(
747f68a716SGreg Bedwell       "FileOpenedForReadingCanBeReplaced-test", TestDirectory));
757f68a716SGreg Bedwell 
767f68a716SGreg Bedwell   // Add a couple of files to the test directory.
777f68a716SGreg Bedwell   SmallString<128> SourceFileName(TestDirectory);
787f68a716SGreg Bedwell   path::append(SourceFileName, "source");
797f68a716SGreg Bedwell 
807f68a716SGreg Bedwell   SmallString<128> TargetFileName(TestDirectory);
817f68a716SGreg Bedwell   path::append(TargetFileName, "target");
827f68a716SGreg Bedwell 
837f68a716SGreg Bedwell   ASSERT_NO_ERROR(CreateFileWithContent(SourceFileName, "!!source!!"));
847f68a716SGreg Bedwell   ASSERT_NO_ERROR(CreateFileWithContent(TargetFileName, "!!target!!"));
857f68a716SGreg Bedwell 
867f68a716SGreg Bedwell   {
877f68a716SGreg Bedwell     // Open the target file for reading.
887f68a716SGreg Bedwell     int ReadFD = 0;
897f68a716SGreg Bedwell     ASSERT_NO_ERROR(fs::openFileForRead(TargetFileName, ReadFD));
907f68a716SGreg Bedwell     ScopedFD EventuallyCloseIt(ReadFD);
917f68a716SGreg Bedwell 
927f68a716SGreg Bedwell     // Confirm we can replace the file while it is open.
937f68a716SGreg Bedwell     EXPECT_TRUE(!fs::rename(SourceFileName, TargetFileName));
947f68a716SGreg Bedwell 
957f68a716SGreg Bedwell     // We should still be able to read the old data through the existing
967f68a716SGreg Bedwell     // descriptor.
9780e31f1fSPeter Collingbourne     EXPECT_TRUE(FDHasContent(ReadFD, "!!target!!"));
987f68a716SGreg Bedwell 
997f68a716SGreg Bedwell     // The source file should no longer exist
1007f68a716SGreg Bedwell     EXPECT_FALSE(fs::exists(SourceFileName));
1017f68a716SGreg Bedwell   }
1027f68a716SGreg Bedwell 
1037f68a716SGreg Bedwell   // If we obtain a new descriptor for the target file, we should find that it
1047f68a716SGreg Bedwell   // contains the content that was in the source file.
10580e31f1fSPeter Collingbourne   EXPECT_TRUE(FileHasContent(TargetFileName, "!!source!!"));
1067f68a716SGreg Bedwell 
1077f68a716SGreg Bedwell   // Rename the target file back to the source file name to confirm that rename
1087f68a716SGreg Bedwell   // still works if the destination does not already exist.
1097f68a716SGreg Bedwell   EXPECT_TRUE(!fs::rename(TargetFileName, SourceFileName));
1107f68a716SGreg Bedwell   EXPECT_FALSE(fs::exists(TargetFileName));
1117f68a716SGreg Bedwell   ASSERT_TRUE(fs::exists(SourceFileName));
1127f68a716SGreg Bedwell 
1137f68a716SGreg Bedwell   // Clean up.
1147f68a716SGreg Bedwell   ASSERT_NO_ERROR(fs::remove(SourceFileName));
1157f68a716SGreg Bedwell   ASSERT_NO_ERROR(fs::remove(TestDirectory.str()));
1167f68a716SGreg Bedwell }
1177f68a716SGreg Bedwell 
TEST(rename,ExistingTemp)11880e31f1fSPeter Collingbourne TEST(rename, ExistingTemp) {
11980e31f1fSPeter Collingbourne   // Test that existing .tmpN files don't get deleted by the Windows
12080e31f1fSPeter Collingbourne   // sys::fs::rename implementation.
12180e31f1fSPeter Collingbourne   SmallString<128> TestDirectory;
12280e31f1fSPeter Collingbourne   ASSERT_NO_ERROR(
12380e31f1fSPeter Collingbourne       fs::createUniqueDirectory("ExistingTemp-test", TestDirectory));
12480e31f1fSPeter Collingbourne 
12580e31f1fSPeter Collingbourne   SmallString<128> SourceFileName(TestDirectory);
12680e31f1fSPeter Collingbourne   path::append(SourceFileName, "source");
12780e31f1fSPeter Collingbourne 
12880e31f1fSPeter Collingbourne   SmallString<128> TargetFileName(TestDirectory);
12980e31f1fSPeter Collingbourne   path::append(TargetFileName, "target");
13080e31f1fSPeter Collingbourne 
13180e31f1fSPeter Collingbourne   SmallString<128> TargetTmp0FileName(TestDirectory);
13280e31f1fSPeter Collingbourne   path::append(TargetTmp0FileName, "target.tmp0");
13380e31f1fSPeter Collingbourne 
13480e31f1fSPeter Collingbourne   SmallString<128> TargetTmp1FileName(TestDirectory);
13580e31f1fSPeter Collingbourne   path::append(TargetTmp1FileName, "target.tmp1");
13680e31f1fSPeter Collingbourne 
13780e31f1fSPeter Collingbourne   ASSERT_NO_ERROR(CreateFileWithContent(SourceFileName, "!!source!!"));
13880e31f1fSPeter Collingbourne   ASSERT_NO_ERROR(CreateFileWithContent(TargetFileName, "!!target!!"));
13980e31f1fSPeter Collingbourne   ASSERT_NO_ERROR(CreateFileWithContent(TargetTmp0FileName, "!!target.tmp0!!"));
14080e31f1fSPeter Collingbourne 
14180e31f1fSPeter Collingbourne   {
14280e31f1fSPeter Collingbourne     // Use mapped_file_region to make sure that the destination file is mmap'ed.
14380e31f1fSPeter Collingbourne     // This will cause SetInformationByHandle to fail when renaming to the
14480e31f1fSPeter Collingbourne     // destination, and we will follow the code path that tries to give target
14580e31f1fSPeter Collingbourne     // a temporary name.
14680e31f1fSPeter Collingbourne     int TargetFD;
14780e31f1fSPeter Collingbourne     std::error_code EC;
14880e31f1fSPeter Collingbourne     ASSERT_NO_ERROR(fs::openFileForRead(TargetFileName, TargetFD));
14980e31f1fSPeter Collingbourne     ScopedFD X(TargetFD);
150cc418a3aSReid Kleckner     sys::fs::mapped_file_region MFR(sys::fs::convertFDToNativeFile(TargetFD),
151cc418a3aSReid Kleckner                                     sys::fs::mapped_file_region::readonly, 10,
152cc418a3aSReid Kleckner                                     0, EC);
15380e31f1fSPeter Collingbourne     ASSERT_FALSE(EC);
15480e31f1fSPeter Collingbourne 
15580e31f1fSPeter Collingbourne     ASSERT_NO_ERROR(fs::rename(SourceFileName, TargetFileName));
15680e31f1fSPeter Collingbourne 
15780e31f1fSPeter Collingbourne #ifdef _WIN32
15880e31f1fSPeter Collingbourne     // Make sure that target was temporarily renamed to target.tmp1 on Windows.
15980e31f1fSPeter Collingbourne     // This is signified by a permission denied error as opposed to no such file
16080e31f1fSPeter Collingbourne     // or directory when trying to open it.
16180e31f1fSPeter Collingbourne     int Tmp1FD;
16280e31f1fSPeter Collingbourne     EXPECT_EQ(errc::permission_denied,
16380e31f1fSPeter Collingbourne               fs::openFileForRead(TargetTmp1FileName, Tmp1FD));
16480e31f1fSPeter Collingbourne #endif
16580e31f1fSPeter Collingbourne   }
16680e31f1fSPeter Collingbourne 
16780e31f1fSPeter Collingbourne   EXPECT_TRUE(FileHasContent(TargetTmp0FileName, "!!target.tmp0!!"));
16880e31f1fSPeter Collingbourne 
16980e31f1fSPeter Collingbourne   ASSERT_NO_ERROR(fs::remove(TargetFileName));
17080e31f1fSPeter Collingbourne   ASSERT_NO_ERROR(fs::remove(TargetTmp0FileName));
17180e31f1fSPeter Collingbourne   ASSERT_NO_ERROR(fs::remove(TestDirectory.str()));
17280e31f1fSPeter Collingbourne }
17380e31f1fSPeter Collingbourne 
1747f68a716SGreg Bedwell }  // anonymous namespace
175