From: Rafael Espindola Date: Fri, 19 Jul 2013 15:02:03 +0000 (+0000) Subject: Split openFileForWrite into windows and unix versions. X-Git-Url: http://review.tizen.org/git/?a=commitdiff_plain;h=67080cec2558a32ce3a50bc505e2197bc947b6a6;p=platform%2Fupstream%2Fllvm.git Split openFileForWrite into windows and unix versions. It is similar to 186511, but for creating files for writing. llvm-svn: 186679 --- diff --git a/llvm/lib/Support/Path.cpp b/llvm/lib/Support/Path.cpp index 171ccc3..88137ea 100644 --- a/llvm/lib/Support/Path.cpp +++ b/llvm/lib/Support/Path.cpp @@ -692,36 +692,6 @@ error_code createUniqueDirectory(const Twine &Prefix, true, 0, FS_Dir); } -error_code openFileForWrite(const Twine &Name, int &ResultFD, - sys::fs::OpenFlags Flags, unsigned Mode) { - // Verify that we don't have both "append" and "excl". - assert((!(Flags & sys::fs::F_Excl) || !(Flags & sys::fs::F_Append)) && - "Cannot specify both 'excl' and 'append' file creation flags!"); - - int OpenFlags = O_WRONLY | O_CREAT; - -#ifdef O_BINARY - if (Flags & F_Binary) - OpenFlags |= O_BINARY; -#endif - - if (Flags & F_Append) - OpenFlags |= O_APPEND; - else - OpenFlags |= O_TRUNC; - - if (Flags & F_Excl) - OpenFlags |= O_EXCL; - - SmallString<128> Storage; - StringRef P = Name.toNullTerminatedStringRef(Storage); - while ((ResultFD = open(P.begin(), OpenFlags, Mode)) < 0) { - if (errno != EINTR) - return error_code(errno, system_category()); - } - return error_code::success(); -} - error_code make_absolute(SmallVectorImpl &path) { StringRef p(path.data(), path.size()); diff --git a/llvm/lib/Support/Unix/Path.inc b/llvm/lib/Support/Unix/Path.inc index 8944008..ccd60e5 100644 --- a/llvm/lib/Support/Unix/Path.inc +++ b/llvm/lib/Support/Unix/Path.inc @@ -766,6 +766,31 @@ error_code openFileForRead(const Twine &Name, int &ResultFD) { return error_code::success(); } +error_code openFileForWrite(const Twine &Name, int &ResultFD, + sys::fs::OpenFlags Flags, unsigned Mode) { + // Verify that we don't have both "append" and "excl". + assert((!(Flags & sys::fs::F_Excl) || !(Flags & sys::fs::F_Append)) && + "Cannot specify both 'excl' and 'append' file creation flags!"); + + int OpenFlags = O_WRONLY | O_CREAT; + + if (Flags & F_Append) + OpenFlags |= O_APPEND; + else + OpenFlags |= O_TRUNC; + + if (Flags & F_Excl) + OpenFlags |= O_EXCL; + + SmallString<128> Storage; + StringRef P = Name.toNullTerminatedStringRef(Storage); + while ((ResultFD = open(P.begin(), OpenFlags, Mode)) < 0) { + if (errno != EINTR) + return error_code(errno, system_category()); + } + return error_code::success(); +} + } // end namespace fs } // end namespace sys } // end namespace llvm diff --git a/llvm/lib/Support/Windows/Path.inc b/llvm/lib/Support/Windows/Path.inc index dff89c7..1be7433 100644 --- a/llvm/lib/Support/Windows/Path.inc +++ b/llvm/lib/Support/Windows/Path.inc @@ -1071,6 +1071,60 @@ error_code openFileForRead(const Twine &Name, int &ResultFD) { return error_code::success(); } +error_code openFileForWrite(const Twine &Name, int &ResultFD, + sys::fs::OpenFlags Flags, unsigned Mode) { + // Verify that we don't have both "append" and "excl". + assert((!(Flags & sys::fs::F_Excl) || !(Flags & sys::fs::F_Append)) && + "Cannot specify both 'excl' and 'append' file creation flags!"); + + SmallString<128> PathStorage; + SmallVector PathUTF16; + + if (error_code EC = UTF8ToUTF16(Name.toStringRef(PathStorage), + PathUTF16)) + return EC; + + DWORD CreationDisposition; + if (Flags & F_Excl) + CreationDisposition = CREATE_NEW; + else if (Flags & F_Append) + CreationDisposition = OPEN_ALWAYS; + else + CreationDisposition = CREATE_ALWAYS; + + HANDLE H = ::CreateFileW(PathUTF16.begin(), GENERIC_WRITE, + FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, + CreationDisposition, FILE_ATTRIBUTE_NORMAL, NULL); + + if (H == INVALID_HANDLE_VALUE) { + error_code EC = windows_error(::GetLastError()); + // Provide a better error message when trying to open directories. + // This only runs if we failed to open the file, so there is probably + // no performances issues. + if (EC != windows_error::access_denied) + return EC; + if (is_directory(Name)) + return error_code(errc::is_a_directory, posix_category()); + return EC; + } + + int OpenFlags = 0; + if (Flags & F_Append) + OpenFlags |= _O_APPEND; + + if (!(Flags & F_Binary)) + OpenFlags |= _O_TEXT; + + int FD = ::_open_osfhandle(intptr_t(H), OpenFlags); + if (FD == -1) { + ::CloseHandle(H); + return windows_error::invalid_handle; + } + + ResultFD = FD; + return error_code::success(); +} + } // end namespace fs } // end namespace sys } // end namespace llvm diff --git a/llvm/test/Object/archive-extract-dir.test b/llvm/test/Object/archive-extract-dir.test new file mode 100644 index 0000000..c718f90 --- /dev/null +++ b/llvm/test/Object/archive-extract-dir.test @@ -0,0 +1,13 @@ +REQUIRES: shell + +RUN: mkdir -p %t +RUN: cd %t +RUN: rm -rf foo +RUN: echo foo > foo +RUN: rm -f test.a +RUN: llvm-ar rc test.a foo +RUN: rm foo +RUN: mkdir foo +RUN: not llvm-ar x test.a foo 2>&1 | FileCheck %s + +CHECK: foo: Is a directory