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