/* * msvcrt.dll drive/directory functions * * Copyright 1996,1998 Marcus Meissner * Copyright 1996 Jukka Iivonen * Copyright 1997,2000 Uwe Bonnes * Copyright 2000 Jon Griffiths * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "config.h" #include "wine/port.h" #include #include "ntddk.h" #include "wine/unicode.h" #include "msvcrt.h" #include "ms_errno.h" #include "wine/unicode.h" #include "msvcrt/direct.h" #include "msvcrt/dos.h" #include "msvcrt/io.h" #include "msvcrt/stdlib.h" #include "msvcrt/string.h" #include "wine/debug.h" WINE_DEFAULT_DEBUG_CHANNEL(msvcrt); /* INTERNAL: Translate finddata_t to PWIN32_FIND_DATAA */ static void msvcrt_fttofd(LPWIN32_FIND_DATAA fd, struct _finddata_t* ft) { DWORD dw; if (fd->dwFileAttributes == FILE_ATTRIBUTE_NORMAL) ft->attrib = 0; else ft->attrib = fd->dwFileAttributes; RtlTimeToSecondsSince1970( &fd->ftCreationTime, &dw ); ft->time_create = dw; RtlTimeToSecondsSince1970( &fd->ftLastAccessTime, &dw ); ft->time_access = dw; RtlTimeToSecondsSince1970( &fd->ftLastWriteTime, &dw ); ft->time_write = dw; ft->size = fd->nFileSizeLow; strcpy(ft->name, fd->cFileName); } /* INTERNAL: Translate wfinddata_t to PWIN32_FIND_DATAA */ static void msvcrt_wfttofd(LPWIN32_FIND_DATAW fd, struct _wfinddata_t* ft) { DWORD dw; if (fd->dwFileAttributes == FILE_ATTRIBUTE_NORMAL) ft->attrib = 0; else ft->attrib = fd->dwFileAttributes; RtlTimeToSecondsSince1970( &fd->ftCreationTime, &dw ); ft->time_create = dw; RtlTimeToSecondsSince1970( &fd->ftLastAccessTime, &dw ); ft->time_access = dw; RtlTimeToSecondsSince1970( &fd->ftLastWriteTime, &dw ); ft->time_write = dw; ft->size = fd->nFileSizeLow; strcpyW(ft->name, fd->cFileName); } /********************************************************************* * _chdir (MSVCRT.@) */ int _chdir(const char * newdir) { if (!SetCurrentDirectoryA(newdir)) { MSVCRT__set_errno(newdir?GetLastError():0); return -1; } return 0; } /********************************************************************* * _wchdir (MSVCRT.@) */ int _wchdir(const WCHAR * newdir) { if (!SetCurrentDirectoryW(newdir)) { MSVCRT__set_errno(newdir?GetLastError():0); return -1; } return 0; } /********************************************************************* * _chdrive (MSVCRT.@) */ int _chdrive(int newdrive) { char buffer[3] = "A:"; buffer[0] += newdrive - 1; if (!SetCurrentDirectoryA( buffer )) { MSVCRT__set_errno(GetLastError()); if (newdrive <= 0) SET_THREAD_VAR(errno,MSVCRT_EACCES); return -1; } return 0; } /********************************************************************* * _findclose (MSVCRT.@) */ int _findclose(long hand) { TRACE(":handle %ld\n",hand); if (!FindClose((HANDLE)hand)) { MSVCRT__set_errno(GetLastError()); return -1; } return 0; } /********************************************************************* * _findfirst (MSVCRT.@) */ long _findfirst(const char * fspec, struct _finddata_t* ft) { WIN32_FIND_DATAA find_data; HANDLE hfind; hfind = FindFirstFileA(fspec, &find_data); if (hfind == INVALID_HANDLE_VALUE) { MSVCRT__set_errno(GetLastError()); return -1; } msvcrt_fttofd(&find_data,ft); TRACE(":got handle %d\n",hfind); return hfind; } /********************************************************************* * _wfindfirst (MSVCRT.@) */ long _wfindfirst(const WCHAR * fspec, struct _wfinddata_t* ft) { WIN32_FIND_DATAW find_data; HANDLE hfind; hfind = FindFirstFileW(fspec, &find_data); if (hfind == INVALID_HANDLE_VALUE) { MSVCRT__set_errno(GetLastError()); return -1; } msvcrt_wfttofd(&find_data,ft); TRACE(":got handle %d\n",hfind); return hfind; } /********************************************************************* * _findnext (MSVCRT.@) */ int _findnext(long hand, struct _finddata_t * ft) { WIN32_FIND_DATAA find_data; if (!FindNextFileA(hand, &find_data)) { SET_THREAD_VAR(errno,MSVCRT_ENOENT); return -1; } msvcrt_fttofd(&find_data,ft); return 0; } /********************************************************************* * _wfindnext (MSVCRT.@) */ int _wfindnext(long hand, struct _wfinddata_t * ft) { WIN32_FIND_DATAW find_data; if (!FindNextFileW(hand, &find_data)) { SET_THREAD_VAR(errno,MSVCRT_ENOENT); return -1; } msvcrt_wfttofd(&find_data,ft); return 0; } /********************************************************************* * _getcwd (MSVCRT.@) */ char* _getcwd(char * buf, int size) { char dir[MAX_PATH]; int dir_len = GetCurrentDirectoryA(MAX_PATH,dir); if (dir_len < 1) return NULL; /* FIXME: Real return value untested */ if (!buf) { if (size < 0) return _strdup(dir); return msvcrt_strndup(dir,size); } if (dir_len >= size) { SET_THREAD_VAR(errno,MSVCRT_ERANGE); return NULL; /* buf too small */ } strcpy(buf,dir); return buf; } /********************************************************************* * _wgetcwd (MSVCRT.@) */ WCHAR* _wgetcwd(WCHAR * buf, int size) { WCHAR dir[MAX_PATH]; int dir_len = GetCurrentDirectoryW(MAX_PATH,dir); if (dir_len < 1) return NULL; /* FIXME: Real return value untested */ if (!buf) { if (size < 0) return _wcsdup(dir); return msvcrt_wstrndup(dir,size); } if (dir_len >= size) { SET_THREAD_VAR(errno,MSVCRT_ERANGE); return NULL; /* buf too small */ } strcpyW(buf,dir); return buf; } /********************************************************************* * _getdrive (MSVCRT.@) */ int _getdrive(void) { char buffer[MAX_PATH]; if (!GetCurrentDirectoryA( sizeof(buffer), buffer )) return 0; if (buffer[1] != ':') return 0; return toupper(buffer[0]) - 'A' + 1; } /********************************************************************* * _getdcwd (MSVCRT.@) */ char* _getdcwd(int drive, char * buf, int size) { static char* dummy; TRACE(":drive %d(%c), size %d\n",drive, drive + 'A' - 1, size); if (!drive || drive == _getdrive()) return _getcwd(buf,size); /* current */ else { char dir[MAX_PATH]; char drivespec[4] = {'A', ':', '\\', 0}; int dir_len; drivespec[0] += drive - 1; if (GetDriveTypeA(drivespec) < DRIVE_REMOVABLE) { SET_THREAD_VAR(errno,MSVCRT_EACCES); return NULL; } dir_len = GetFullPathNameA(drivespec,MAX_PATH,dir,&dummy); if (dir_len >= size || dir_len < 1) { SET_THREAD_VAR(errno,MSVCRT_ERANGE); return NULL; /* buf too small */ } TRACE(":returning '%s'\n", dir); if (!buf) return _strdup(dir); /* allocate */ strcpy(buf,dir); } return buf; } /********************************************************************* * _wgetdcwd (MSVCRT.@) */ WCHAR* _wgetdcwd(int drive, WCHAR * buf, int size) { static WCHAR* dummy; TRACE(":drive %d(%c), size %d\n",drive, drive + 'A' - 1, size); if (!drive || drive == _getdrive()) return _wgetcwd(buf,size); /* current */ else { WCHAR dir[MAX_PATH]; WCHAR drivespec[4] = {'A', ':', '\\', 0}; int dir_len; drivespec[0] += drive - 1; if (GetDriveTypeW(drivespec) < DRIVE_REMOVABLE) { SET_THREAD_VAR(errno,MSVCRT_EACCES); return NULL; } dir_len = GetFullPathNameW(drivespec,MAX_PATH,dir,&dummy); if (dir_len >= size || dir_len < 1) { SET_THREAD_VAR(errno,MSVCRT_ERANGE); return NULL; /* buf too small */ } TRACE(":returning %s\n", debugstr_w(dir)); if (!buf) return _wcsdup(dir); /* allocate */ strcpyW(buf,dir); } return buf; } /********************************************************************* * _getdiskfree (MSVCRT.@) */ unsigned int _getdiskfree(unsigned int disk, struct _diskfree_t* d) { char drivespec[4] = {'@', ':', '\\', 0}; DWORD ret[4]; unsigned int err; if (disk > 26) return ERROR_INVALID_PARAMETER; /* MSVCRT doesn't set errno here */ drivespec[0] += disk; /* make a drive letter */ if (GetDiskFreeSpaceA(disk==0?NULL:drivespec,ret,ret+1,ret+2,ret+3)) { d->sectors_per_cluster = (unsigned)ret[0]; d->bytes_per_sector = (unsigned)ret[1]; d->avail_clusters = (unsigned)ret[2]; d->total_clusters = (unsigned)ret[3]; return 0; } err = GetLastError(); MSVCRT__set_errno(err); return err; } /********************************************************************* * _mkdir (MSVCRT.@) */ int _mkdir(const char * newdir) { if (CreateDirectoryA(newdir,NULL)) return 0; MSVCRT__set_errno(GetLastError()); return -1; } /********************************************************************* * _wmkdir (MSVCRT.@) */ int _wmkdir(const WCHAR* newdir) { if (CreateDirectoryW(newdir,NULL)) return 0; MSVCRT__set_errno(GetLastError()); return -1; } /********************************************************************* * _rmdir (MSVCRT.@) */ int _rmdir(const char * dir) { if (RemoveDirectoryA(dir)) return 0; MSVCRT__set_errno(GetLastError()); return -1; } /********************************************************************* * _wrmdir (MSVCRT.@) */ int _wrmdir(const WCHAR * dir) { if (RemoveDirectoryW(dir)) return 0; MSVCRT__set_errno(GetLastError()); return -1; } /********************************************************************* * _wsplitpath (MSVCRT.@) */ void _wsplitpath(const WCHAR *inpath, WCHAR *drv, WCHAR *dir, WCHAR *fname, WCHAR *ext ) { /* Modified PD code from 'snippets' collection. */ WCHAR ch, *ptr, *p; WCHAR pathbuff[MAX_PATH],*path=pathbuff; TRACE(":splitting path %s\n",debugstr_w(path)); /* FIXME: Should be an strncpyW or something */ strcpyW(pathbuff, inpath); /* convert slashes to backslashes for searching */ for (ptr = (WCHAR*)path; *ptr; ++ptr) if (*ptr == (WCHAR)L'/') *ptr = (WCHAR)L'\\'; /* look for drive spec */ if ((ptr = strchrW(path, (WCHAR)L':')) != (WCHAR)L'\0') { ++ptr; if (drv) { strncpyW(drv, path, ptr - path); drv[ptr - path] = (WCHAR)L'\0'; } path = ptr; } else if (drv) *drv = (WCHAR)L'\0'; /* find rightmost backslash or leftmost colon */ if ((ptr = strrchrW(path, (WCHAR)L'\\')) == NULL) ptr = (strchrW(path, (WCHAR)L':')); if (!ptr) { ptr = (WCHAR *)path; /* no path */ if (dir) *dir = (WCHAR)L'\0'; } else { ++ptr; /* skip the delimiter */ if (dir) { ch = *ptr; *ptr = (WCHAR)L'\0'; strcpyW(dir, path); *ptr = ch; } } if ((p = strrchrW(ptr, (WCHAR)L'.')) == NULL) { if (fname) strcpyW(fname, ptr); if (ext) *ext = (WCHAR)L'\0'; } else { *p = (WCHAR)L'\0'; if (fname) strcpyW(fname, ptr); *p = (WCHAR)L'.'; if (ext) strcpyW(ext, p); } /* Fix pathological case - Win returns ':' as part of the * directory when no drive letter is given. */ if (drv && drv[0] == (WCHAR)L':') { *drv = (WCHAR)L'\0'; if (dir) { pathbuff[0] = (WCHAR)L':'; pathbuff[1] = (WCHAR)L'\0'; strcatW(pathbuff,dir); strcpyW(dir, pathbuff); } } } /* INTERNAL: Helper for _fullpath. Modified PD code from 'snippets'. */ static void msvcrt_fln_fix(char *path) { int dir_flag = 0, root_flag = 0; char *r, *p, *q, *s; /* Skip drive */ if (NULL == (r = strrchr(path, ':'))) r = path; else ++r; /* Ignore leading slashes */ while ('\\' == *r) if ('\\' == r[1]) strcpy(r, &r[1]); else { root_flag = 1; ++r; } p = r; /* Change "\\" to "\" */ while (NULL != (p = strchr(p, '\\'))) if ('\\' == p[1]) strcpy(p, &p[1]); else ++p; while ('.' == *r) /* Scrunch leading ".\" */ { if ('.' == r[1]) { /* Ignore leading ".." */ for (p = (r += 2); *p && (*p != '\\'); ++p) ; } else { for (p = r + 1 ;*p && (*p != '\\'); ++p) ; } strcpy(r, p + ((*p) ? 1 : 0)); } while ('\\' == path[strlen(path)-1]) /* Strip last '\\' */ { dir_flag = 1; path[strlen(path)-1] = '\0'; } s = r; /* Look for "\." in path */ while (NULL != (p = strstr(s, "\\."))) { if ('.' == p[2]) { /* Execute this section if ".." found */ q = p - 1; while (q > r) /* Backup one level */ { if (*q == '\\') break; --q; } if (q > r) { strcpy(q, p + 3); s = q; } else if ('.' != *q) { strcpy(q + ((*q == '\\') ? 1 : 0), p + 3 + ((*(p + 3)) ? 1 : 0)); s = q; } else s = ++p; } else { /* Execute this section if "." found */ q = p + 2; for ( ;*q && (*q != '\\'); ++q) ; strcpy (p, q); } } if (root_flag) /* Embedded ".." could have bubbled up to root */ { for (p = r; *p && ('.' == *p || '\\' == *p); ++p) ; if (r != p) strcpy(r, p); } if (dir_flag) strcat(path, "\\"); } /********************************************************************* * _fullpath (MSVCRT.@) */ char *_fullpath(char * absPath, const char* relPath, unsigned int size) { char drive[5],dir[MAX_PATH],file[MAX_PATH],ext[MAX_PATH]; char res[MAX_PATH]; size_t len; res[0] = '\0'; if (!relPath || !*relPath) return _getcwd(absPath, size); if (size < 4) { SET_THREAD_VAR(errno,MSVCRT_ERANGE); return NULL; } TRACE(":resolving relative path '%s'\n",relPath); _splitpath(relPath, drive, dir, file, ext); /* Get Directory and drive into 'res' */ if (!dir[0] || (dir[0] != '/' && dir[0] != '\\')) { /* Relative or no directory given */ _getdcwd(drive[0] ? toupper(drive[0]) - 'A' + 1 : 0, res, MAX_PATH); strcat(res,"\\"); if (dir[0]) strcat(res,dir); if (drive[0]) res[0] = drive[0]; /* If given a drive, preserve the letter case */ } else { strcpy(res,drive); strcat(res,dir); } strcat(res,"\\"); strcat(res, file); strcat(res, ext); msvcrt_fln_fix(res); len = strlen(res); if (len >= MAX_PATH || len >= (size_t)size) return NULL; /* FIXME: errno? */ if (!absPath) return _strdup(res); strcpy(absPath,res); return absPath; } /********************************************************************* * _makepath (MSVCRT.@) */ VOID _makepath(char * path, const char * drive, const char *directory, const char * filename, const char * extension ) { char ch; TRACE("got %s %s %s %s\n", drive, directory, filename, extension); if ( !path ) return; path[0] = 0; if (drive && drive[0]) { path[0] = drive[0]; path[1] = ':'; path[2] = 0; } if (directory && directory[0]) { strcat(path, directory); ch = path[strlen(path)-1]; if (ch != '/' && ch != '\\') strcat(path,"\\"); } if (filename && filename[0]) { strcat(path, filename); if (extension && extension[0]) { if ( extension[0] != '.' ) strcat(path,"."); strcat(path,extension); } } TRACE("returning %s\n",path); } /********************************************************************* * _wmakepath (MSVCRT.@) */ VOID _wmakepath(WCHAR *path, const WCHAR *drive, const WCHAR *directory, const WCHAR *filename, const WCHAR *extension) { WCHAR ch; TRACE("%s %s %s %s\n", debugstr_w(drive), debugstr_w(directory), debugstr_w(filename), debugstr_w(extension)); if ( !path ) return; path[0] = 0; if (drive && drive[0]) { path[0] = drive[0]; path[1] = ':'; path[2] = 0; } if (directory && directory[0]) { strcatW(path, directory); ch = path[strlenW(path) - 1]; if (ch != '/' && ch != '\\') { static const WCHAR backslashW[] = {'\\',0}; strcatW(path, backslashW); } } if (filename && filename[0]) { strcatW(path, filename); if (extension && extension[0]) { if ( extension[0] != '.' ) { static const WCHAR dotW[] = {'.',0}; strcatW(path, dotW); } strcatW(path, extension); } } TRACE("returning %s\n", debugstr_w(path)); } /********************************************************************* * _searchenv (MSVCRT.@) */ void _searchenv(const char* file, const char* env, char *buf) { char*envVal, *penv; char curPath[MAX_PATH]; *buf = '\0'; /* Try CWD first */ if (GetFileAttributesA( file ) != 0xFFFFFFFF) { GetFullPathNameA( file, MAX_PATH, buf, NULL ); /* Sigh. This error is *always* set, regardless of success */ MSVCRT__set_errno(ERROR_FILE_NOT_FOUND); return; } /* Search given environment variable */ envVal = MSVCRT_getenv(env); if (!envVal) { MSVCRT__set_errno(ERROR_FILE_NOT_FOUND); return; } penv = envVal; TRACE(":searching for %s in paths %s\n", file, envVal); do { char *end = penv; while(*end && *end != ';') end++; /* Find end of next path */ if (penv == end || !*penv) { MSVCRT__set_errno(ERROR_FILE_NOT_FOUND); return; } strncpy(curPath, penv, end - penv); if (curPath[end - penv] != '/' || curPath[end - penv] != '\\') { curPath[end - penv] = '\\'; curPath[end - penv + 1] = '\0'; } else curPath[end - penv] = '\0'; strcat(curPath, file); TRACE("Checking for file %s\n", curPath); if (GetFileAttributesA( curPath ) != 0xFFFFFFFF) { strcpy(buf, curPath); MSVCRT__set_errno(ERROR_FILE_NOT_FOUND); return; /* Found */ } penv = *end ? end + 1 : end; } while(1); }