/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying file Copyright.txt or https://cmake.org/licensing#kwsys for details. */ #include "kwsysPrivate.h" #include KWSYS_HEADER(Directory.hxx) #include KWSYS_HEADER(Configure.hxx) #include KWSYS_HEADER(Encoding.hxx) #include KWSYS_HEADER(SystemTools.hxx) // Work-around CMake dependency scanning limitation. This must // duplicate the above list of headers. #if 0 # include "Configure.hxx.in" # include "Directory.hxx.in" # include "Encoding.hxx.in" #endif #include #include #include #if defined(_WIN32) && !defined(__CYGWIN__) # include # include # include # include # include # include # include # include # include #endif namespace KWSYS_NAMESPACE { class DirectoryInternals { public: struct FileData { std::string Name; #if defined(_WIN32) && !defined(__CYGWIN__) WIN32_FIND_DATAW FindData; #endif FileData(std::string name #if defined(_WIN32) && !defined(__CYGWIN__) , WIN32_FIND_DATAW data #endif ) : Name(std::move(name)) #if defined(_WIN32) && !defined(__CYGWIN__) , FindData(std::move(data)) #endif { } }; // Array of Files std::vector Files; // Path to Open'ed directory std::string Path; }; Directory::Directory() { this->Internal = new DirectoryInternals; } Directory::Directory(Directory&& other) { this->Internal = other.Internal; other.Internal = nullptr; } Directory& Directory::operator=(Directory&& other) { std::swap(this->Internal, other.Internal); return *this; } Directory::~Directory() { delete this->Internal; } unsigned long Directory::GetNumberOfFiles() const { return static_cast(this->Internal->Files.size()); } const char* Directory::GetFile(unsigned long dindex) const { return this->Internal->Files[dindex].Name.c_str(); } std::string const& Directory::GetFileName(std::size_t i) const { return this->Internal->Files[i].Name; } std::string Directory::GetFilePath(std::size_t i) const { std::string abs = this->Internal->Path; if (!abs.empty() && abs.back() != '/') { abs += '/'; } abs += this->Internal->Files[i].Name; return abs; } bool Directory::FileIsDirectory(std::size_t i) const { #if defined(_WIN32) && !defined(__CYGWIN__) auto const& data = this->Internal->Files[i].FindData; return (data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) != 0; #else std::string const& path = this->GetFilePath(i); return kwsys::SystemTools::FileIsDirectory(path); #endif } bool Directory::FileIsSymlink(std::size_t i) const { std::string const& path = this->GetFilePath(i); #if defined(_WIN32) && !defined(__CYGWIN__) auto const& data = this->Internal->Files[i].FindData; return kwsys::SystemTools::FileIsSymlinkWithAttr( Encoding::ToWindowsExtendedPath(path), data.dwFileAttributes); #else return kwsys::SystemTools::FileIsSymlink(path); #endif } const char* Directory::GetPath() const { return this->Internal->Path.c_str(); } void Directory::Clear() { this->Internal->Path.resize(0); this->Internal->Files.clear(); } } // namespace KWSYS_NAMESPACE // First Windows platforms #if defined(_WIN32) && !defined(__CYGWIN__) namespace KWSYS_NAMESPACE { Status Directory::Load(std::string const& name, std::string* errorMessage) { this->Clear(); HANDLE srchHandle; char* buf; size_t bufLength; size_t n = name.size(); if (name.back() == '/' || name.back() == '\\') { bufLength = n + 1 + 1; buf = new char[bufLength]; snprintf(buf, bufLength, "%s*", name.c_str()); } else { // Make sure the slashes in the wildcard suffix are consistent with the // rest of the path bufLength = n + 2 + 1; buf = new char[bufLength]; if (name.find('\\') != std::string::npos) { snprintf(buf, bufLength, "%s\\*", name.c_str()); } else { snprintf(buf, bufLength, "%s/*", name.c_str()); } } WIN32_FIND_DATAW data; // data of current file // Now put them into the file array srchHandle = FindFirstFileW(Encoding::ToWindowsExtendedPath(buf).c_str(), &data); delete[] buf; if (srchHandle == INVALID_HANDLE_VALUE) { Status status = Status::POSIX_errno(); if (errorMessage) { *errorMessage = status.GetString(); } return status; } // Loop through names do { this->Internal->Files.emplace_back(Encoding::ToNarrow(data.cFileName), data); } while (FindNextFileW(srchHandle, &data)); this->Internal->Path = name; if (!FindClose(srchHandle)) { Status status = Status::POSIX_errno(); if (errorMessage) { *errorMessage = status.GetString(); } return status; } return Status::Success(); } unsigned long Directory::GetNumberOfFilesInDirectory(const std::string& name, std::string* errorMessage) { HANDLE srchHandle; char* buf; size_t bufLength; size_t n = name.size(); if (name.back() == '/') { bufLength = n + 1 + 1; buf = new char[n + 1 + 1]; snprintf(buf, bufLength, "%s*", name.c_str()); } else { bufLength = n + 2 + 1; buf = new char[n + 2 + 1]; snprintf(buf, bufLength, "%s/*", name.c_str()); } WIN32_FIND_DATAW data; // data of current file // Now put them into the file array srchHandle = FindFirstFileW(Encoding::ToWide(buf).c_str(), &data); delete[] buf; if (srchHandle == INVALID_HANDLE_VALUE) { if (errorMessage) { if (unsigned int errorId = GetLastError()) { LPSTR message = nullptr; DWORD size = FormatMessageA( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, nullptr, errorId, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPSTR)&message, 0, nullptr); *errorMessage = std::string(message, size); LocalFree(message); } else { *errorMessage = "Unknown error."; } } return 0; } // Loop through names unsigned long count = 0; do { count++; } while (FindNextFileW(srchHandle, &data)); FindClose(srchHandle); return count; } } // namespace KWSYS_NAMESPACE #else // Now the POSIX style directory access # include # include # include # include // PGI with glibc has trouble with dirent and large file support: // https://www.pgroup.com/userforum/viewtopic.php? // p=1992&sid=f16167f51964f1a68fe5041b8eb213b6 // Work around the problem by mapping dirent the same way as readdir. # if defined(__PGI) && defined(__GLIBC__) # define kwsys_dirent_readdir dirent # define kwsys_dirent_readdir64 dirent64 # define kwsys_dirent kwsys_dirent_lookup(readdir) # define kwsys_dirent_lookup(x) kwsys_dirent_lookup_delay(x) # define kwsys_dirent_lookup_delay(x) kwsys_dirent_##x # else # define kwsys_dirent dirent # endif namespace KWSYS_NAMESPACE { Status Directory::Load(std::string const& name, std::string* errorMessage) { this->Clear(); errno = 0; DIR* dir = opendir(name.c_str()); if (!dir) { if (errorMessage) { *errorMessage = std::string(strerror(errno)); } return Status::POSIX_errno(); } errno = 0; for (kwsys_dirent* d = readdir(dir); d; d = readdir(dir)) { this->Internal->Files.emplace_back(d->d_name); } if (errno != 0) { if (errorMessage) { *errorMessage = std::string(strerror(errno)); } return Status::POSIX_errno(); } this->Internal->Path = name; closedir(dir); return Status::Success(); } unsigned long Directory::GetNumberOfFilesInDirectory(const std::string& name, std::string* errorMessage) { errno = 0; DIR* dir = opendir(name.c_str()); if (!dir) { if (errorMessage) { *errorMessage = std::string(strerror(errno)); } return 0; } unsigned long count = 0; for (kwsys_dirent* d = readdir(dir); d; d = readdir(dir)) { count++; } if (errno != 0) { if (errorMessage) { *errorMessage = std::string(strerror(errno)); } return false; } closedir(dir); return count; } } // namespace KWSYS_NAMESPACE #endif