/* 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) // 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 namespace KWSYS_NAMESPACE { class DirectoryInternals { public: // 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 { if (dindex >= this->Internal->Files.size()) { return nullptr; } return this->Internal->Files[dindex].c_str(); } 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__) # include # include # include # include # include # include # include # include # include namespace KWSYS_NAMESPACE { Status Directory::Load(std::string const& name, std::string* errorMessage) { this->Clear(); intptr_t 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()); } } struct _wfinddata_t data; // data of current file // Now put them into the file array srchHandle = _wfindfirst((wchar_t*)Encoding::ToWindowsExtendedPath(buf).c_str(), &data); delete[] buf; if (srchHandle == -1) { Status status = Status::POSIX_errno(); if (errorMessage) { *errorMessage = status.GetString(); } return status; } // Loop through names do { this->Internal->Files.push_back(Encoding::ToNarrow(data.name)); } while (_wfindnext(srchHandle, &data) != -1); this->Internal->Path = name; if (_findclose(srchHandle) == -1) { 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) { intptr_t 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()); } struct _wfinddata_t data; // data of current file // Now put them into the file array srchHandle = _wfindfirst((wchar_t*)Encoding::ToWide(buf).c_str(), &data); delete[] buf; if (srchHandle == -1) { 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 (_wfindnext(srchHandle, &data) != -1); _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: // http://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 != nullptr) { *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 != nullptr) { *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 != nullptr) { *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 != nullptr) { *errorMessage = std::string(strerror(errno)); } return false; } closedir(dir); return count; } } // namespace KWSYS_NAMESPACE #endif