/* WDL - dirscan.h Copyright (C) 2005 and later Cockos Incorporated This software is provided 'as-is', without any express or implied warranty. In no event will the authors be held liable for any damages arising from the use of this software. Permission is granted to anyone to use this software for any purpose, including commercial applications, and to alter it and redistribute it freely, subject to the following restrictions: 1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required. 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software. 3. This notice may not be removed or altered from any source distribution. */ /* This file provides the interface and implementation for WDL_DirScan, a simple (and somewhat portable) directory reading class. On non-Win32 systems it wraps opendir()/readdir()/etc. On Win32, it uses FindFirst*, and supports wildcards as well. */ #ifndef _WDL_DIRSCAN_H_ #define _WDL_DIRSCAN_H_ #include "wdlstring.h" #ifndef _WIN32 #include #include #include extern struct stat wdl_stat_chk; // if this fails on linux, use CFLAGS += -D_FILE_OFFSET_BITS=64 typedef char wdl_dirscan_assert_failed_stat_not_64[sizeof(wdl_stat_chk.st_size)!=8 ? -1 : 1]; #endif class WDL_DirScan { public: WDL_DirScan() : #ifdef _WIN32 m_h(INVALID_HANDLE_VALUE) #ifndef WDL_NO_SUPPORT_UTF8 , m_wcmode(false) #endif #else m_h(NULL), m_ent(NULL) #endif { } ~WDL_DirScan() { Close(); } int First(const char *dirname #ifdef _WIN32 , int isExactSpec=0 #endif ) // returns 0 if success { WDL_FastString scanstr(dirname); const int l = scanstr.GetLength(); if (l < 1) return -1; #ifdef _WIN32 if (!isExactSpec) { if (dirname[l-1] == '\\' || dirname[l-1] == '/') scanstr.SetLen(l-1); m_leading_path = scanstr; scanstr.Append("\\*"); } else { m_leading_path = scanstr; // remove trailing wildcards and directory separator from m_leading_path const char *sp = m_leading_path.Get(); int idx = m_leading_path.GetLength() - 1; while (idx > 0 && sp[idx] != '/' && sp[idx] != '\\') idx--; if (idx > 0) m_leading_path.SetLen(idx); } #else if (dirname[l-1] == '\\' || dirname[l-1] == '/') scanstr.SetLen(l-1); m_leading_path = scanstr; if (!scanstr.GetLength()) scanstr.Set("/"); // fix for scanning / #endif Close(); #ifdef _WIN32 #ifndef WDL_NO_SUPPORT_UTF8 m_h=INVALID_HANDLE_VALUE; #ifdef WDL_SUPPORT_WIN9X m_wcmode = GetVersion()< 0x80000000; #else m_wcmode = true; #endif if (m_wcmode) { int reqbuf = MultiByteToWideChar(CP_UTF8,MB_ERR_INVALID_CHARS,scanstr.Get(),-1,NULL,0); if (reqbuf > 1000) { WDL_TypedBuf tmp; tmp.Resize(reqbuf+20); if (MultiByteToWideChar(CP_UTF8,MB_ERR_INVALID_CHARS,scanstr.Get(),-1,tmp.Get(),tmp.GetSize()-10)) { correctlongpath(tmp.Get()); m_h=FindFirstFileW(tmp.Get(),&m_fd); } } else { WCHAR wfilename[1024]; if (MultiByteToWideChar(CP_UTF8,MB_ERR_INVALID_CHARS,scanstr.Get(),-1,wfilename,1024-10)) { correctlongpath(wfilename); m_h=FindFirstFileW(wfilename,&m_fd); } } } if (m_h==INVALID_HANDLE_VALUE) m_wcmode=false; if (m_h==INVALID_HANDLE_VALUE) #endif m_h=FindFirstFileA(scanstr.Get(),(WIN32_FIND_DATAA*)&m_fd); return (m_h == INVALID_HANDLE_VALUE); #else m_ent=0; m_h=opendir(scanstr.Get()); return !m_h || Next(); #endif } int Next() // returns 0 on success { #ifdef _WIN32 if (m_h == INVALID_HANDLE_VALUE) return -1; #ifndef WDL_NO_SUPPORT_UTF8 if (m_wcmode) return !FindNextFileW(m_h,&m_fd); #endif return !FindNextFileA(m_h,(WIN32_FIND_DATAA*)&m_fd); #else if (!m_h) return -1; return !(m_ent=readdir(m_h)); #endif } void Close() { #ifdef _WIN32 if (m_h != INVALID_HANDLE_VALUE) FindClose(m_h); m_h=INVALID_HANDLE_VALUE; #else if (m_h) closedir(m_h); m_h=0; m_ent=0; #endif } #ifdef _WIN32 const char *GetCurrentFN() { #ifndef WDL_NO_SUPPORT_UTF8 if (m_wcmode) { if (!WideCharToMultiByte(CP_UTF8,0,m_fd.cFileName,-1,m_tmpbuf,sizeof(m_tmpbuf),NULL,NULL)) m_tmpbuf[0]=0; return m_tmpbuf; } #endif return ((WIN32_FIND_DATAA *)&m_fd)->cFileName; } #else const char *GetCurrentFN() const { return m_ent?m_ent->d_name : ""; } #endif template void GetCurrentFullFN(T *str) { str->Set(m_leading_path.Get()); #ifdef _WIN32 str->Append("\\"); #else str->Append("/"); #endif str->Append(GetCurrentFN()); } int GetCurrentIsDirectory() const // returns 1 if dir, 2 if symlink to dir, 4 if possibly-recursive symlink to dir { #ifdef _WIN32 return !!(m_fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY); #else char tmp[2048]; if (m_ent) switch (m_ent->d_type) { case DT_DIR: return 1; case DT_LNK: { snprintf(tmp,sizeof(tmp),"%s/%s",m_leading_path.Get(),m_ent->d_name); char *rp = realpath(tmp,NULL); if (!rp) return 0; struct stat sb; int ret = (!stat(rp,&sb) && (sb.st_mode & S_IFMT) == S_IFDIR) ? 2 : 0; if (ret) { // treat symlinks of /path/to/foo -> /path from being resolved (avoiding obvious feedback loops) const int rpl = (int) strlen(rp); if ( #ifdef __APPLE__ !strnicmp(rp,m_leading_path.Get(),rpl) #else !strncmp(rp,m_leading_path.Get(),rpl) #endif && (m_leading_path.Get()[rpl] == '/' || m_leading_path.Get()[rpl] == 0) ) ret = 4; } free(rp); return ret; } case DT_UNKNOWN: { snprintf(tmp,sizeof(tmp),"%s/%s",m_leading_path.Get(),m_ent->d_name); DIR *d = opendir(tmp); if (d) { closedir(d); return 1; } return 0; } } return 0; #endif } // these are somewhat windows specific calls, eh #ifdef _WIN32 DWORD GetCurrentFileSize(DWORD *HighWord=NULL) const { if (HighWord) *HighWord = m_fd.nFileSizeHigh; return m_fd.nFileSizeLow; } void GetCurrentLastWriteTime(FILETIME *ft) const { *ft = m_fd.ftLastWriteTime; } void GetCurrentLastAccessTime(FILETIME *ft) const { *ft = m_fd.ftLastAccessTime; } void GetCurrentCreationTime(FILETIME *ft) const { *ft = m_fd.ftCreationTime; } DWORD GetFileAttributes() const { return m_fd.dwFileAttributes; } #elif defined(_WDL_SWELL_H_) void GetCurrentCreationTime(FILETIME *ft) { char tmp[2048]; snprintf(tmp,sizeof(tmp),"%s/%s",m_leading_path.Get(),GetCurrentFN()); struct stat st={0,}; stat(tmp,&st); unsigned long long a=(unsigned long long)st.st_ctime; // seconds since january 1st, 1970 a+=11644473600ull; // 1601->1970 a*=10000000; // seconds to 1/10th microseconds (100 nanoseconds) ft->dwLowDateTime=a & 0xffffffff; ft->dwHighDateTime=a>>32; } void GetCurrentLastWriteTime(FILETIME *ft) { char tmp[2048]; snprintf(tmp,sizeof(tmp),"%s/%s",m_leading_path.Get(),GetCurrentFN()); struct stat st={0,}; stat(tmp,&st); unsigned long long a=(unsigned long long)st.st_mtime; // seconds since january 1st, 1970 a+=11644473600ull; // 1601->1970 a*=10000000; // seconds to 1/10th microseconds (100 nanoseconds) ft->dwLowDateTime=a & 0xffffffff; ft->dwHighDateTime=a>>32; } DWORD GetCurrentFileSize(DWORD *HighWord=NULL) { char tmp[2048]; snprintf(tmp,sizeof(tmp),"%s/%s",m_leading_path.Get(),GetCurrentFN()); struct stat st={0,}; stat(tmp,&st); if (HighWord) *HighWord = (DWORD)(st.st_size>>32); return (DWORD)(st.st_size&0xffffffff); } #endif private: #ifdef _WIN32 #ifndef WDL_NO_SUPPORT_UTF8 bool m_wcmode; WIN32_FIND_DATAW m_fd; char m_tmpbuf[MAX_PATH*5]; // even if each byte gets encoded as 4 utf-8 bytes this should be plenty ;) #else WIN32_FIND_DATAA m_fd; #endif HANDLE m_h; #else DIR *m_h; struct dirent *m_ent; #endif WDL_FastString m_leading_path; #ifdef _WIN32 static void correctlongpath(WCHAR *buf) // this also exists as wdl_utf8_correctlongpath { const WCHAR *insert; WCHAR *wr; int skip = 0; if (!buf || !buf[0] || wcslen(buf) < 256) return; if (buf[1] == ':') insert=L"\\\\?\\"; else if (buf[0] == '\\' && buf[1] == '\\') { insert = L"\\\\?\\UNC\\"; skip=2; } else return; wr = buf + wcslen(insert); memmove(wr, buf + skip, (wcslen(buf+skip)+1)*2); memmove(buf,insert,wcslen(insert)*2); while (*wr) { if (*wr == '/') *wr = '\\'; wr++; } } #endif } WDL_FIXALIGN; #endif