From 38dae407317b11aa6bbc433698a5b1fb775adb91 Mon Sep 17 00:00:00 2001 From: Owen Reynolds Date: Mon, 12 Aug 2019 14:00:28 +0000 Subject: [PATCH] [llvm-ar] Accept file paths with windows format slashes The internal representation of llvm-ar archives uses linux style slashes for paths, no matter the OS. In the case of windows this meant file paths input intending to match existing members would only match if linux style slashes where used. This change allows either slash direction to be input by the user. This change includes removing an unnecessary call to normalisePath and moving the call of another. Differential Revision: https://reviews.llvm.org/D65743 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@368573 91177308-0d34-0410-b5e6-96231b3b80d8 --- test/tools/llvm-ar/windows-path.test | 44 ++++++++++++++++++++++++++++ tools/llvm-ar/llvm-ar.cpp | 16 +++++----- 2 files changed, 52 insertions(+), 8 deletions(-) create mode 100644 test/tools/llvm-ar/windows-path.test diff --git a/test/tools/llvm-ar/windows-path.test b/test/tools/llvm-ar/windows-path.test new file mode 100644 index 00000000000..e33fbf532c3 --- /dev/null +++ b/test/tools/llvm-ar/windows-path.test @@ -0,0 +1,44 @@ +# Test that windows path seperators are handled correctly. +REQUIRES: system-windows + +# Note: many of these tests depend on relative paths, so we have to cd to a +# test directory first. +RUN: mkdir -p %t && cd %t +RUN: rm -rf a b && mkdir a b +RUN: echo hello-a > a/foo.txt +RUN: echo hello-b > b/foo.txt +RUN: echo hello-parent > foo.txt + +# P is implied when using thin archives. +# Create an archive. +RUN: rm -f archive.a +RUN: llvm-ar rcST archive.a a\..\a\foo.txt +RUN: llvm-ar rcST archive.a foo.txt +RUN: llvm-ar rcST archive.a b\foo.txt b/foo.txt +RUN: llvm-ar dT archive.a foo.txt +RUN: llvm-ar t archive.a | FileCheck %s --check-prefix=ARCHIVE --implicit-check-not {{.}} + +ARCHIVE: a/foo.txt +ARCHIVE-NEXT: b/foo.txt +ARCHIVE-NEXT: b/foo.txt + +RUN: llvm-ar t archive.a | FileCheck %s --check-prefix=LIST-BOTH --implicit-check-not {{.}} +RUN: llvm-ar t archive.a a\foo.txt | FileCheck %s --check-prefix=LIST-A --implicit-check-not {{.}} +RUN: llvm-ar t archive.a b\foo.txt | FileCheck %s --check-prefix=LIST-B --implicit-check-not {{.}} + +LIST-BOTH: a/foo.txt +LIST-BOTH-NEXT: b/foo.txt +LIST-BOTH-NEXT: b/foo.txt +LIST-A: a/foo.txt +LIST-B: b/foo.txt + +# Nesting a thin archive with a name conflict. +RUN: rm -f a\nested.a b\nested.a nested.a +RUN: llvm-ar rcST a\nested.a a\foo.txt +RUN: llvm-ar rcST b\nested.a b\foo.txt +RUN: llvm-ar rcST nested.a a\nested.a foo.txt b\nested.a +RUN: llvm-ar t nested.a | FileCheck %s --check-prefix=NESTED --implicit-check-not {{.}} + +NESTED: a/foo.txt +NESTED-NEXT: foo.txt +NESTED-NEXT: b/foo.txt diff --git a/tools/llvm-ar/llvm-ar.cpp b/tools/llvm-ar/llvm-ar.cpp index 7c4cc26ad6d..0817e2db968 100644 --- a/tools/llvm-ar/llvm-ar.cpp +++ b/tools/llvm-ar/llvm-ar.cpp @@ -474,8 +474,9 @@ static void doDisplayTable(StringRef Name, const object::Archive::Child &C) { outs() << Name << "\n"; } -static StringRef normalizePath(StringRef Path) { - return CompareFullPath ? Path : sys::path::filename(Path); +static std::string normalizePath(StringRef Path) { + return CompareFullPath ? sys::path::convert_to_slash(Path) + : sys::path::filename(Path); } // Implement the 'x' operation. This function extracts files back to the file @@ -698,9 +699,8 @@ static InsertAction computeInsertAction(ArchiveOperation Operation, return IA_MoveOldMember; if (Operation == ReplaceOrInsert) { - StringRef PosName = normalizePath(RelPos); if (!OnlyUpdate) { - if (PosName.empty()) + if (RelPos.empty()) return IA_AddNewMember; return IA_MoveNewMember; } @@ -712,12 +712,12 @@ static InsertAction computeInsertAction(ArchiveOperation Operation, auto ModTimeOrErr = Member.getLastModified(); failIfError(ModTimeOrErr.takeError()); if (Status.getLastModificationTime() < ModTimeOrErr.get()) { - if (PosName.empty()) + if (RelPos.empty()) return IA_AddOldMember; return IA_MoveOldMember; } - if (PosName.empty()) + if (RelPos.empty()) return IA_AddNewMember; return IA_MoveNewMember; } @@ -732,8 +732,8 @@ computeNewArchiveMembers(ArchiveOperation Operation, std::vector Ret; std::vector Moved; int InsertPos = -1; - StringRef PosName = normalizePath(RelPos); if (OldArchive) { + std::string PosName = normalizePath(RelPos); Error Err = Error::success(); StringMap MemberCount; for (auto &Child : OldArchive->children(Err)) { @@ -1009,7 +1009,7 @@ static void runMRIScript() { ArchiveName = Rest; break; case MRICommand::Delete: { - StringRef Name = normalizePath(Rest); + std::string Name = normalizePath(Rest); llvm::erase_if(NewMembers, [=](NewArchiveMember &M) { return M.MemberName == Name; }); break; -- 2.50.1