/* * Unit test suite for various Path and Directory Functions * * Copyright 2002 Geoffrey Hausheer * Copyright 2006 Detlef Riekenberg * * 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include #include #include #include "wine/test.h" #include "windef.h" #include "winbase.h" #include "winuser.h" #include "winerror.h" #include "winnls.h" #define HAS_TRAIL_SLASH_A(string) (string[lstrlenA(string)-1]=='\\') #define LONGFILE "Long File test.path" #define SHORTFILE "pathtest.pth" #define SHORTDIR "shortdir" #define LONGDIR "Long Directory" #define NONFILE_SHORT "noexist.pth" #define NONFILE_LONG "NonExistent File" #define NONDIR_SHORT "notadir" #define NONDIR_LONG "NonExistent Directory" #define NOT_A_VALID_DRIVE '@' #ifdef __i386__ #define ARCH "x86" #elif defined __x86_64__ #define ARCH "amd64" #elif defined __arm__ #define ARCH "arm" #elif defined __aarch64__ #define ARCH "arm64" #else #define ARCH "none" #endif /* the following characters don't work well with GetFullPathNameA in Win98. I don't know if this is a FAT thing, or if it is an OS thing but I don't test these characters now. NOTE: Win2k allows GetFullPathNameA to work with them though |<>" */ static const CHAR funny_chars[]="!@#$%^&*()=+{}[],?'`"; static const CHAR is_char_ok[] ="11111110111111111011"; static DWORD (WINAPI *pGetLongPathNameA)(LPCSTR,LPSTR,DWORD); static DWORD (WINAPI *pGetLongPathNameW)(LPWSTR,LPWSTR,DWORD); /* Present in Win2003+ */ static BOOL (WINAPI *pNeedCurrentDirectoryForExePathA)(LPCSTR); static BOOL (WINAPI *pNeedCurrentDirectoryForExePathW)(LPCWSTR); static DWORD (WINAPI *pSearchPathA)(LPCSTR,LPCSTR,LPCSTR,DWORD,LPSTR,LPSTR*); static DWORD (WINAPI *pSearchPathW)(LPCWSTR,LPCWSTR,LPCWSTR,DWORD,LPWSTR,LPWSTR*); static BOOL (WINAPI *pActivateActCtx)(HANDLE,ULONG_PTR*); static HANDLE (WINAPI *pCreateActCtxW)(PCACTCTXW); static BOOL (WINAPI *pDeactivateActCtx)(DWORD,ULONG_PTR); static BOOL (WINAPI *pGetCurrentActCtx)(HANDLE *); static void (WINAPI *pReleaseActCtx)(HANDLE); static BOOL (WINAPI *pCheckNameLegalDOS8Dot3W)(const WCHAR *, char *, DWORD, BOOL *, BOOL *); static BOOL (WINAPI *pCheckNameLegalDOS8Dot3A)(const char *, char *, DWORD, BOOL *, BOOL *); /* a structure to deal with wine todos somewhat cleanly */ typedef struct { DWORD shortlen; DWORD shorterror; DWORD s2llen; DWORD s2lerror; DWORD longlen; DWORD longerror; } SLpassfail; /* function that tests GetFullPathNameA, GetShortPathNameA,GetLongPathNameA */ /* NOTE: the passfail structure is used to allow customizable todo checking for wine. It is not very pretty, but it sure beats duplicating this function lots of times */ static void test_ValidPathA(const CHAR *curdir, const CHAR *subdir, const CHAR *filename, CHAR *shortstr, SLpassfail *passfail, const CHAR *errstr) { CHAR tmpstr[MAX_PATH], fullpath[MAX_PATH], /*full path to the file (not short/long) */ subpath[MAX_PATH], /*relative path to the file */ fullpathshort[MAX_PATH], /*absolute path to the file (short format) */ fullpathlong[MAX_PATH], /*absolute path to the file (long format) */ curdirshort[MAX_PATH], /*absolute path to the current dir (short) */ curdirlong[MAX_PATH]; /*absolute path to the current dir (long) */ LPSTR strptr; /*ptr to the filename portion of the path */ DWORD len; /* if passfail is NULL, we can perform all checks within this function, otherwise, we will return the relevant data in the passfail struct, so we must initialize it first */ if(passfail!=NULL) { passfail->shortlen=-1;passfail->s2llen=-1;passfail->longlen=-1; passfail->shorterror=0;passfail->s2lerror=0;passfail->longerror=0; } /* GetLongPathNameA is only supported on Win2k+ and Win98+ */ if(pGetLongPathNameA) { ok((len=pGetLongPathNameA(curdir,curdirlong,MAX_PATH)), "%s: GetLongPathNameA failed\n",errstr); /*GetLongPathNameA can return a trailing '\\' but shouldn't do so here */ ok(! HAS_TRAIL_SLASH_A(curdirlong), "%s: GetLongPathNameA should not have a trailing \\\n",errstr); } ok((len=GetShortPathNameA(curdir,curdirshort,MAX_PATH)), "%s: GetShortPathNameA failed\n",errstr); /*GetShortPathNameA can return a trailing '\\' but shouldn't do so here */ ok(! HAS_TRAIL_SLASH_A(curdirshort), "%s: GetShortPathNameA should not have a trailing \\\n",errstr); /* build relative and absolute paths from inputs */ if(lstrlenA(subdir)) { sprintf(subpath,"%s\\%s",subdir,filename); } else { lstrcpyA(subpath,filename); } sprintf(fullpath,"%s\\%s",curdir,subpath); sprintf(fullpathshort,"%s\\%s",curdirshort,subpath); sprintf(fullpathlong,"%s\\%s",curdirlong,subpath); /* Test GetFullPathNameA functionality */ len=GetFullPathNameA(subpath,MAX_PATH,tmpstr,&strptr); ok(len, "GetFullPathNameA failed for: '%s'\n",subpath); if(HAS_TRAIL_SLASH_A(subpath)) { ok(strptr==NULL, "%s: GetFullPathNameA should not return a filename ptr\n",errstr); ok(lstrcmpiA(fullpath,tmpstr)==0, "%s: GetFullPathNameA returned '%s' instead of '%s'\n", errstr,tmpstr,fullpath); } else { ok(lstrcmpiA(strptr,filename)==0, "%s: GetFullPathNameA returned '%s' instead of '%s'\n", errstr,strptr,filename); ok(lstrcmpiA(fullpath,tmpstr)==0, "%s: GetFullPathNameA returned '%s' instead of '%s'\n", errstr,tmpstr,fullpath); } /* Test GetShortPathNameA functionality */ SetLastError(0); len=GetShortPathNameA(fullpathshort,shortstr,MAX_PATH); if(passfail==NULL) { ok(len, "%s: GetShortPathNameA failed\n",errstr); } else { passfail->shortlen=len; passfail->shorterror=GetLastError(); } /* Test GetLongPathNameA functionality We test both conversion from GetFullPathNameA and from GetShortPathNameA */ if(pGetLongPathNameA) { if(len!=0) { SetLastError(0); len=pGetLongPathNameA(shortstr,tmpstr,MAX_PATH); if(passfail==NULL) { ok(len, "%s: GetLongPathNameA failed during Short->Long conversion\n", errstr); ok(lstrcmpiA(fullpathlong,tmpstr)==0, "%s: GetLongPathNameA returned '%s' instead of '%s'\n", errstr,tmpstr,fullpathlong); } else { passfail->s2llen=len; passfail->s2lerror=GetLastError(); } } SetLastError(0); len=pGetLongPathNameA(fullpath,tmpstr,MAX_PATH); if(passfail==NULL) { ok(len, "%s: GetLongPathNameA failed\n",errstr); if(HAS_TRAIL_SLASH_A(fullpath)) { ok(lstrcmpiA(fullpathlong,tmpstr)==0, "%s: GetLongPathNameA returned '%s' instead of '%s'\n", errstr,tmpstr,fullpathlong); } else { ok(lstrcmpiA(fullpathlong,tmpstr)==0, "%s: GetLongPathNameA returned '%s' instead of '%s'\n", errstr,tmpstr,fullpathlong); } } else { passfail->longlen=len; passfail->longerror=GetLastError(); } } } /* split path into leading directory, and 8.3 filename */ static void test_SplitShortPathA(CHAR *path,CHAR *dir,CHAR *eight,CHAR *three) { BOOL done = FALSE, error = FALSE; int ext,fil; int len,i; len=lstrlenA(path); ext=len; fil=len; /* walk backwards over path looking for '.' or '\\' separators */ for(i=len-1;(i>=0) && (!done);i--) { if(path[i]=='.') if(ext!=len) error=TRUE; else ext=i; else if(path[i]=='\\') { if(i==len-1) { error=TRUE; } else { fil=i; done=TRUE; } } } /* Check that we didn't find a trailing '\\' or multiple '.' */ ok(!error,"Illegal file found in 8.3 path '%s'\n",path); /* Separate dir, root, and extension */ if(ext!=len) lstrcpyA(three,path+ext+1); else lstrcpyA(three,""); if(fil!=len) { lstrcpynA(eight,path+fil+1,ext-fil); lstrcpynA(dir,path,fil+1); } else { lstrcpynA(eight,path,ext+1); lstrcpyA(dir,""); } /* Validate that root and extension really are 8.3 */ ok(lstrlenA(eight)<=8 && lstrlenA(three)<=3, "GetShortPathNAmeA did not return an 8.3 path\n"); } /* Check that GetShortPathNameA returns a valid 8.3 path */ static void test_LongtoShortA(CHAR *teststr,const CHAR *goodstr, const CHAR *ext,const CHAR *errstr) { CHAR dir[MAX_PATH],eight[MAX_PATH],three[MAX_PATH]; test_SplitShortPathA(teststr,dir,eight,three); ok(lstrcmpiA(dir,goodstr)==0, "GetShortPathNameA returned '%s' instead of '%s'\n",dir,goodstr); ok(lstrcmpiA(three,ext)==0, "GetShortPathNameA returned '%s' with incorrect extension\n",three); } /* Test that Get(Short|Long|Full)PathNameA work correctly with interesting characters in the filename. 'valid' indicates whether this would be an allowed filename 'todo' indicates that wine doesn't get this right yet. NOTE: We always call this routine with a nonexistent filename, so Get(Short|Long)PathNameA should never pass, but GetFullPathNameA should. */ static void test_FunnyChars(CHAR *curdir,CHAR *curdir_short,CHAR *filename, INT valid,CHAR *errstr) { CHAR tmpstr[MAX_PATH],tmpstr1[MAX_PATH]; SLpassfail passfail; test_ValidPathA(curdir,"",filename,tmpstr,&passfail,errstr); if(valid) { sprintf(tmpstr1,"%s\\%s",curdir_short,filename); ok((passfail.shortlen==0 && (passfail.shorterror==ERROR_FILE_NOT_FOUND || passfail.shorterror==ERROR_PATH_NOT_FOUND || !passfail.shorterror)) || (passfail.shortlen==strlen(tmpstr1) && lstrcmpiA(tmpstr,tmpstr1)==0), "%s: GetShortPathNameA error: len=%d error=%d tmpstr=[%s]\n", errstr,passfail.shortlen,passfail.shorterror,tmpstr); } else { ok(passfail.shortlen==0 && (passfail.shorterror==ERROR_INVALID_NAME || passfail.shorterror==ERROR_FILE_NOT_FOUND || !passfail.shorterror), "%s: GetShortPathA should have failed len=%d, error=%d\n", errstr,passfail.shortlen,passfail.shorterror); } if(pGetLongPathNameA) { ok(passfail.longlen==0,"GetLongPathNameA passed when it shouldn't have\n"); if(valid) { ok(passfail.longerror==ERROR_FILE_NOT_FOUND, "%s: GetLongPathA returned %d and not %d\n", errstr,passfail.longerror,ERROR_FILE_NOT_FOUND); } else { ok(passfail.longerror==ERROR_INVALID_NAME || passfail.longerror==ERROR_FILE_NOT_FOUND, "%s: GetLongPathA returned %d and not %d or %d'\n", errstr, passfail.longerror,ERROR_INVALID_NAME,ERROR_FILE_NOT_FOUND); } } } /* Routine to test that SetCurrentDirectory behaves as expected. */ static void test_setdir(CHAR *olddir,CHAR *newdir, CHAR *cmprstr, INT pass, const CHAR *errstr) { CHAR tmppath[MAX_PATH], *dirptr; DWORD val,len,chklen; val=SetCurrentDirectoryA(newdir); len=GetCurrentDirectoryA(MAX_PATH,tmppath); /* if 'pass' then the SetDirectoryA was supposed to pass */ if(pass) { dirptr=(cmprstr==NULL) ? newdir : cmprstr; chklen=lstrlenA(dirptr); ok(val,"%s: SetCurrentDirectoryA failed\n",errstr); ok(len==chklen, "%s: SetCurrentDirectory did not change the directory, though it passed\n", errstr); ok(lstrcmpiA(dirptr,tmppath)==0, "%s: SetCurrentDirectory did not change the directory, though it passed\n", errstr); ok(SetCurrentDirectoryA(olddir), "%s: Couldn't set directory to its original value\n",errstr); } else { /* else thest that it fails correctly */ chklen=lstrlenA(olddir); ok(val==0, "%s: SetCurrentDirectoryA passed when it should have failed\n",errstr); ok(len==chklen, "%s: SetCurrentDirectory changed the directory, though it failed\n", errstr); ok(lstrcmpiA(olddir,tmppath)==0, "%s: SetCurrentDirectory changed the directory, though it failed\n", errstr); } } static void test_InitPathA(CHAR *newdir, CHAR *curDrive, CHAR *otherDrive) { CHAR tmppath[MAX_PATH], /*path to TEMP */ tmpstr[MAX_PATH], tmpstr1[MAX_PATH], invalid_dir[MAX_PATH]; DWORD len,len1,drives; INT id; HANDLE hndl; BOOL bRes; UINT unique; *curDrive = *otherDrive = NOT_A_VALID_DRIVE; /* Get the current drive letter */ if( GetCurrentDirectoryA( MAX_PATH, tmpstr)) *curDrive = tmpstr[0]; else trace( "Unable to discover current drive, some tests will not be conducted.\n"); /* Test GetTempPathA */ len=GetTempPathA(MAX_PATH,tmppath); ok(len!=0 && len < MAX_PATH,"GetTempPathA failed\n"); ok(HAS_TRAIL_SLASH_A(tmppath), "GetTempPathA returned a path that did not end in '\\'\n"); lstrcpyA(tmpstr,"aaaaaaaa"); len1=GetTempPathA(len,tmpstr); ok(len1==len+1 || broken(len1 == len), /* WinME */ "GetTempPathA should return string length %d instead of %d\n",len+1,len1); /* Test GetTmpFileNameA */ ok((id=GetTempFileNameA(tmppath,"path",0,newdir)),"GetTempFileNameA failed\n"); sprintf(tmpstr,"pat%.4x.tmp",id & 0xffff); sprintf(tmpstr1,"pat%x.tmp",id & 0xffff); ok(lstrcmpiA(newdir+lstrlenA(tmppath),tmpstr)==0 || lstrcmpiA(newdir+lstrlenA(tmppath),tmpstr1)==0, "GetTempFileNameA returned '%s' which doesn't match '%s' or '%s'. id=%x\n", newdir,tmpstr,tmpstr1,id); ok(DeleteFileA(newdir),"Couldn't delete the temporary file we just created\n"); id=GetTempFileNameA(tmppath,NULL,0,newdir); /* Windows 95, 98 return 0==id, while Windows 2000, XP return 0!=id */ if (id) { sprintf(tmpstr,"%.4x.tmp",id & 0xffff); sprintf(tmpstr1,"%x.tmp",id & 0xffff); ok(lstrcmpiA(newdir+lstrlenA(tmppath),tmpstr)==0 || lstrcmpiA(newdir+lstrlenA(tmppath),tmpstr1)==0, "GetTempFileNameA returned '%s' which doesn't match '%s' or '%s'. id=%x\n", newdir,tmpstr,tmpstr1,id); ok(DeleteFileA(newdir),"Couldn't delete the temporary file we just created\n"); } for(unique=0;unique<3;unique++) { /* Nonexistent path */ sprintf(invalid_dir, "%s\\%s",tmppath,"non_existent_dir_1jwj3y32nb3"); SetLastError(0xdeadbeef); ok(!GetTempFileNameA(invalid_dir,"tfn",unique,newdir),"GetTempFileNameA should have failed\n"); ok(GetLastError()==ERROR_DIRECTORY || broken(GetLastError()==ERROR_PATH_NOT_FOUND)/*win98*/, "got %d, expected ERROR_DIRECTORY\n", GetLastError()); /* Check return value for unique !=0 */ if(unique) { ok((GetTempFileNameA(tmppath,"tfn",unique,newdir) == unique),"GetTempFileNameA unexpectedly failed\n"); /* if unique != 0, the actual temp files are not created: */ ok(!DeleteFileA(newdir) && GetLastError() == ERROR_FILE_NOT_FOUND,"Deleted a file that shouldn't exist!\n"); } } /* Find first valid drive letter that is neither newdir[0] nor curDrive */ drives = GetLogicalDrives() & ~(1<<(newdir[0]-'A')); if( *curDrive != NOT_A_VALID_DRIVE) drives &= ~(1<<(*curDrive-'A')); if( drives) for( *otherDrive='A'; (drives & 1) == 0; drives>>=1, (*otherDrive)++); else trace( "Could not find alternative drive, some tests will not be conducted.\n"); /* Do some CreateDirectoryA tests */ /* It would be nice to do test the SECURITY_ATTRIBUTES, but I don't really understand how they work. More formal tests should be done along with CreateFile tests */ ok((id=GetTempFileNameA(tmppath,"path",0,newdir)),"GetTempFileNameA failed\n"); ok(CreateDirectoryA(newdir,NULL)==0, "CreateDirectoryA succeeded even though a file of the same name exists\n"); ok(DeleteFileA(newdir),"Couldn't delete the temporary file we just created\n"); ok(CreateDirectoryA(newdir,NULL),"CreateDirectoryA failed\n"); /* Create some files to test other functions. Note, we will test CreateFileA at some later point */ sprintf(tmpstr,"%s\\%s",newdir,SHORTDIR); ok(CreateDirectoryA(tmpstr,NULL),"CreateDirectoryA failed\n"); sprintf(tmpstr,"%s\\%s",newdir,LONGDIR); ok(CreateDirectoryA(tmpstr,NULL),"CreateDirectoryA failed\n"); sprintf(tmpstr,"%c:", *curDrive); bRes = CreateDirectoryA(tmpstr,NULL); ok(!bRes && (GetLastError() == ERROR_ACCESS_DENIED || GetLastError() == ERROR_ALREADY_EXISTS), "CreateDirectoryA(\"%s\" should have failed (%d)\n", tmpstr, GetLastError()); sprintf(tmpstr,"%c:\\", *curDrive); bRes = CreateDirectoryA(tmpstr,NULL); ok(!bRes && (GetLastError() == ERROR_ACCESS_DENIED || GetLastError() == ERROR_ALREADY_EXISTS), "CreateDirectoryA(\"%s\" should have failed (%d)\n", tmpstr, GetLastError()); sprintf(tmpstr,"%s\\%s\\%s",newdir,SHORTDIR,SHORTFILE); hndl=CreateFileA(tmpstr,GENERIC_WRITE,0,NULL, CREATE_NEW,FILE_ATTRIBUTE_NORMAL,NULL); ok(hndl!=INVALID_HANDLE_VALUE,"CreateFileA failed\n"); ok(CloseHandle(hndl),"CloseHandle failed\n"); sprintf(tmpstr,"%s\\%s\\%s",newdir,SHORTDIR,LONGFILE); hndl=CreateFileA(tmpstr,GENERIC_WRITE,0,NULL, CREATE_NEW,FILE_ATTRIBUTE_NORMAL,NULL); ok(hndl!=INVALID_HANDLE_VALUE,"CreateFileA failed\n"); ok(CloseHandle(hndl),"CloseHandle failed\n"); sprintf(tmpstr,"%s\\%s\\%s",newdir,LONGDIR,SHORTFILE); hndl=CreateFileA(tmpstr,GENERIC_WRITE,0,NULL, CREATE_NEW,FILE_ATTRIBUTE_NORMAL,NULL); ok(hndl!=INVALID_HANDLE_VALUE,"CreateFileA failed\n"); ok(CloseHandle(hndl),"CloseHandle failed\n"); sprintf(tmpstr,"%s\\%s\\%s",newdir,LONGDIR,LONGFILE); hndl=CreateFileA(tmpstr,GENERIC_WRITE,0,NULL, CREATE_NEW,FILE_ATTRIBUTE_NORMAL,NULL); ok(hndl!=INVALID_HANDLE_VALUE,"CreateFileA failed\n"); ok(CloseHandle(hndl),"CloseHandle failed\n"); } /* Test GetCurrentDirectory & SetCurrentDirectory */ static void test_CurrentDirectoryA(CHAR *origdir, CHAR *newdir) { CHAR tmpstr[MAX_PATH],tmpstr1[MAX_PATH]; char *buffer; DWORD len,len1; /* Save the original directory, so that we can return to it at the end of the test */ len=GetCurrentDirectoryA(MAX_PATH,origdir); ok(len!=0 && len < MAX_PATH,"GetCurrentDirectoryA failed\n"); /* Make sure that CetCurrentDirectoryA doesn't overwrite the buffer when the buffer size is too small to hold the current directory */ lstrcpyA(tmpstr,"aaaaaaa"); len1=GetCurrentDirectoryA(len,tmpstr); ok(len1==len+1, "GetCurrentDirectoryA returned %d instead of %d\n",len1,len+1); ok(lstrcmpiA(tmpstr,"aaaaaaa")==0, "GetCurrentDirectoryA should not have modified the buffer\n"); buffer = HeapAlloc( GetProcessHeap(), 0, 2 * 65536 ); SetLastError( 0xdeadbeef ); strcpy( buffer, "foo" ); len = GetCurrentDirectoryA( 32767, buffer ); ok( len != 0 && len < MAX_PATH, "GetCurrentDirectoryA failed %u err %u\n", len, GetLastError() ); if (len) ok( !strcmp( buffer, origdir ), "wrong result %s\n", buffer ); SetLastError( 0xdeadbeef ); strcpy( buffer, "foo" ); len = GetCurrentDirectoryA( 32768, buffer ); ok( len != 0 && len < MAX_PATH, "GetCurrentDirectoryA failed %u err %u\n", len, GetLastError() ); if (len) ok( !strcmp( buffer, origdir ), "wrong result %s\n", buffer ); SetLastError( 0xdeadbeef ); strcpy( buffer, "foo" ); len = GetCurrentDirectoryA( 65535, buffer ); ok( (len != 0 && len < MAX_PATH) || broken(!len), /* nt4, win2k, xp */ "GetCurrentDirectoryA failed %u err %u\n", len, GetLastError() ); if (len) ok( !strcmp( buffer, origdir ), "wrong result %s\n", buffer ); SetLastError( 0xdeadbeef ); strcpy( buffer, "foo" ); len = GetCurrentDirectoryA( 65536, buffer ); ok( (len != 0 && len < MAX_PATH) || broken(!len), /* nt4 */ "GetCurrentDirectoryA failed %u err %u\n", len, GetLastError() ); if (len) ok( !strcmp( buffer, origdir ), "wrong result %s\n", buffer ); SetLastError( 0xdeadbeef ); strcpy( buffer, "foo" ); len = GetCurrentDirectoryA( 2 * 65536, buffer ); ok( (len != 0 && len < MAX_PATH) || broken(!len), /* nt4 */ "GetCurrentDirectoryA failed %u err %u\n", len, GetLastError() ); if (len) ok( !strcmp( buffer, origdir ), "wrong result %s\n", buffer ); HeapFree( GetProcessHeap(), 0, buffer ); /* Check for crash prevention on swapped args. Crashes all but Win9x. */ if (0) { GetCurrentDirectoryA( 42, (LPSTR)(MAX_PATH + 42) ); } /* SetCurrentDirectoryA shouldn't care whether the string has a trailing '\\' or not */ sprintf(tmpstr,"%s\\",newdir); test_setdir(origdir,tmpstr,newdir,1,"check 1"); test_setdir(origdir,newdir,NULL,1,"check 2"); /* Set the directory to the working area. We just tested that this works, so why check it again. */ SetCurrentDirectoryA(newdir); /* Check that SetCurrentDirectory fails when a nonexistent dir is specified */ sprintf(tmpstr,"%s\\%s\\%s",newdir,SHORTDIR,NONDIR_SHORT); test_setdir(newdir,tmpstr,NULL,0,"check 3"); /* Check that SetCurrentDirectory fails for a nonexistent lond directory */ sprintf(tmpstr,"%s\\%s\\%s",newdir,SHORTDIR,NONDIR_LONG); test_setdir(newdir,tmpstr,NULL,0,"check 4"); /* Check that SetCurrentDirectory passes with a long directory */ sprintf(tmpstr,"%s\\%s",newdir,LONGDIR); test_setdir(newdir,tmpstr,NULL,1,"check 5"); /* Check that SetCurrentDirectory passes with a short relative directory */ sprintf(tmpstr,"%s",SHORTDIR); sprintf(tmpstr1,"%s\\%s",newdir,SHORTDIR); test_setdir(newdir,tmpstr,tmpstr1,1,"check 6"); /* starting with a '.' */ sprintf(tmpstr,".\\%s",SHORTDIR); test_setdir(newdir,tmpstr,tmpstr1,1,"check 7"); /* Check that SetCurrentDirectory passes with a short relative directory */ sprintf(tmpstr,"%s",LONGDIR); sprintf(tmpstr1,"%s\\%s",newdir,LONGDIR); test_setdir(newdir,tmpstr,tmpstr1,1,"check 8"); /* starting with a '.' */ sprintf(tmpstr,".\\%s",LONGDIR); test_setdir(newdir,tmpstr,tmpstr1,1,"check 9"); /* change to root without a trailing backslash. The function call succeeds but the directory is not changed. */ sprintf(tmpstr, "%c:", newdir[0]); test_setdir(newdir,tmpstr,newdir,1,"check 10"); /* works however with a trailing backslash */ sprintf(tmpstr, "%c:\\", newdir[0]); test_setdir(newdir,tmpstr,NULL,1,"check 11"); } /* Cleanup the mess we made while executing these tests */ static void test_CleanupPathA(CHAR *origdir, CHAR *curdir) { CHAR tmpstr[MAX_PATH]; sprintf(tmpstr,"%s\\%s\\%s",curdir,SHORTDIR,SHORTFILE); ok(DeleteFileA(tmpstr),"DeleteFileA failed\n"); sprintf(tmpstr,"%s\\%s\\%s",curdir,SHORTDIR,LONGFILE); ok(DeleteFileA(tmpstr),"DeleteFileA failed\n"); sprintf(tmpstr,"%s\\%s\\%s",curdir,LONGDIR,SHORTFILE); ok(DeleteFileA(tmpstr),"DeleteFileA failed\n"); sprintf(tmpstr,"%s\\%s\\%s",curdir,LONGDIR,LONGFILE); ok(DeleteFileA(tmpstr),"DeleteFileA failed\n"); sprintf(tmpstr,"%s\\%s",curdir,SHORTDIR); ok(RemoveDirectoryA(tmpstr),"RemoveDirectoryA failed\n"); sprintf(tmpstr,"%s\\%s",curdir,LONGDIR); ok(RemoveDirectoryA(tmpstr),"RemoveDirectoryA failed\n"); ok(SetCurrentDirectoryA(origdir),"SetCurrentDirectoryA failed\n"); ok(RemoveDirectoryA(curdir),"RemoveDirectoryA failed\n"); } /* test that short path name functions work regardless of case */ static void test_ShortPathCase(const char *tmpdir, const char *dirname, const char *filename) { char buf[MAX_PATH], shortbuf[MAX_PATH]; HANDLE hndl; size_t i; assert(strlen(tmpdir) + strlen(dirname) + strlen(filename) + 2 < sizeof(buf)); sprintf(buf,"%s\\%s\\%s",tmpdir,dirname,filename); GetShortPathNameA(buf,shortbuf,sizeof(shortbuf)); hndl = CreateFileA(shortbuf,GENERIC_READ|GENERIC_WRITE,0,NULL,OPEN_EXISTING,0,NULL); ok(hndl!=INVALID_HANDLE_VALUE,"CreateFileA failed (%d)\n",GetLastError()); CloseHandle(hndl); /* Now for the real test */ for(i=0;i= len_with_null, "Expected >= %u, got %u\n", len_with_null, len); len = GetTempPathA(0, NULL); ok(len >= len_with_null, "Expected >= %u, got %u\n", len_with_null, len); /* The call above gave us the buffer size that Windows thinks is needed * so the next call should work */ lstrcpyA(buf, "foo"); len = GetTempPathA(len, buf); ok(lstrcmpiA(buf, tmp_dir) == 0, "expected [%s], got [%s]\n",tmp_dir,buf); ok(len == strlen(buf), "returned length should be equal to the length of string\n"); memset(buf, 'a', sizeof(buf)); len = GetTempPathA(sizeof(buf), buf); ok(lstrcmpiA(buf, tmp_dir) == 0, "expected [%s], got [%s]\n",tmp_dir,buf); ok(len == strlen(buf), "returned length should be equal to the length of string\n"); /* The rest of the buffer remains untouched */ slen = len + 1; for(len++; len < sizeof(buf); len++) ok(buf[len] == 'a', "expected 'a' at [%d], got 0x%x\n", len, buf[len]); /* When the buffer is not long enough it remains untouched */ memset(buf, 'a', sizeof(buf)); len = GetTempPathA(slen / 2, buf); ok(len == slen || broken(len == slen + 1) /* read the big comment above */ , "expected %d, got %d\n", slen, len); for(len = 0; len < sizeof(buf) / sizeof(buf[0]); len++) ok(buf[len] == 'a', "expected 'a' at [%d], got 0x%x\n", len, buf[len]); } static void test_GetTempPathW(char* tmp_dir) { DWORD len, slen, len_with_null; WCHAR buf[MAX_PATH], *long_buf; WCHAR tmp_dirW[MAX_PATH]; static const WCHAR fooW[] = {'f','o','o',0}; MultiByteToWideChar(CP_ACP,0,tmp_dir,-1,tmp_dirW,sizeof(tmp_dirW)/sizeof(*tmp_dirW)); len_with_null = lstrlenW(tmp_dirW) + 1; /* This one is different from ANSI version: ANSI version doesn't * touch the buffer, unicode version usually truncates the buffer * to zero size. NT still exaggerates the required buffer size * sometimes so we cannot test for an exact match. Finally, the * 'len_with_null - 1' case is so buggy on Windows it's not testable. * For instance on NT4 it will sometimes return a path without the * trailing '\\' and sometimes return an error. */ lstrcpyW(buf, fooW); len = GetTempPathW(MAX_PATH, buf); if (len == 0 && GetLastError() == ERROR_CALL_NOT_IMPLEMENTED) { win_skip("GetTempPathW is not available\n"); return; } ok(lstrcmpiW(buf, tmp_dirW) == 0, "GetTempPathW returned an incorrect temporary path\n"); ok(len == lstrlenW(buf), "returned length should be equal to the length of string\n"); lstrcpyW(buf, fooW); len = GetTempPathW(1, buf); ok(buf[0] == 0, "unicode version should truncate the buffer to zero size\n"); ok(len >= len_with_null, "Expected >= %u, got %u\n", len_with_null, len); len = GetTempPathW(0, NULL); ok(len >= len_with_null, "Expected >= %u, got %u\n", len_with_null, len); lstrcpyW(buf, fooW); len = GetTempPathW(len, buf); ok(lstrcmpiW(buf, tmp_dirW) == 0, "GetTempPathW returned an incorrect temporary path\n"); ok(len == lstrlenW(buf), "returned length should be equal to the length of string\n"); for(len = 0; len < sizeof(buf) / sizeof(buf[0]); len++) buf[len] = 'a'; len = GetTempPathW(len, buf); ok(lstrcmpiW(buf, tmp_dirW) == 0, "GetTempPathW returned an incorrect temporary path\n"); ok(len == lstrlenW(buf), "returned length should be equal to the length of string\n"); /* The rest of the buffer must be zeroed */ slen = len + 1; for(len++; len < sizeof(buf) / sizeof(buf[0]); len++) ok(buf[len] == '\0', "expected NULL at [%d], got 0x%x\n", len, buf[len]); /* When the buffer is not long enough the length passed is zeroed */ for(len = 0; len < sizeof(buf) / sizeof(buf[0]); len++) buf[len] = 'a'; len = GetTempPathW(slen / 2, buf); ok(len == slen || broken(len == slen + 1) /* read the big comment above */ , "expected %d, got %d\n", slen, len); { /* In Windows 8 when TMP var points to a drive only (like C:) instead of a * full directory the behavior changes. It will start filling the path but * will later truncate the buffer before returning. So the generic test * below will fail for this Windows 8 corner case. */ char tmp_var[64]; DWORD version = GetVersion(); GetEnvironmentVariableA("TMP", tmp_var, sizeof(tmp_var)); if (strlen(tmp_var) == 2 && version >= 0x00060002) return; } for(len = 0; len < slen / 2; len++) ok(buf[len] == '\0', "expected NULL at [%d], got 0x%x\n", len, buf[len]); for(; len < sizeof(buf) / sizeof(buf[0]); len++) ok(buf[len] == 'a', "expected 'a' at [%d], got 0x%x\n", len, buf[len]); /* bogus application from bug 38220 passes the count value in sizeof(buffer) * instead the correct count of WCHAR, this test catches this case. */ slen = 65534; long_buf = HeapAlloc(GetProcessHeap(), 0, slen * sizeof(WCHAR)); if (!long_buf) { skip("Could not allocate memory for the test\n"); return; } for(len = 0; len < slen; len++) long_buf[len] = 0xCC; len = GetTempPathW(slen, long_buf); ok(lstrcmpiW(long_buf, tmp_dirW) == 0, "GetTempPathW returned an incorrect temporary path\n"); ok(len == lstrlenW(long_buf), "returned length should be equal to the length of string\n"); /* the remaining buffer must be zeroed up to different values in different OS versions. * <= XP - 32766 * > XP - 32767 * to simplify testing we will test only until XP. */ for(; len < 32767; len++) ok(long_buf[len] == '\0', "expected NULL at [%d], got 0x%x\n", len, long_buf[len]); /* we will know skip the test that is in the middle of the OS difference by * incrementing len and then resume the test for the untouched part. */ for(len++; len < slen; len++) ok(long_buf[len] == 0xcc, "expected 0xcc at [%d], got 0x%x\n", len, long_buf[len]); HeapFree(GetProcessHeap(), 0, long_buf); } static void test_GetTempPath(void) { char save_TMP[MAX_PATH]; char windir[MAX_PATH]; char buf[MAX_PATH]; if (!GetEnvironmentVariableA("TMP", save_TMP, sizeof(save_TMP))) save_TMP[0] = 0; /* test default configuration */ trace("TMP=%s\n", save_TMP); if (save_TMP[0]) { strcpy(buf,save_TMP); if (buf[strlen(buf)-1]!='\\') strcat(buf,"\\"); test_GetTempPathA(buf); test_GetTempPathW(buf); } /* TMP=C:\WINDOWS */ GetWindowsDirectoryA(windir, sizeof(windir)); SetEnvironmentVariableA("TMP", windir); GetEnvironmentVariableA("TMP", buf, sizeof(buf)); trace("TMP=%s\n", buf); strcat(windir,"\\"); test_GetTempPathA(windir); test_GetTempPathW(windir); /* TMP=C:\ */ GetWindowsDirectoryA(windir, sizeof(windir)); windir[3] = 0; SetEnvironmentVariableA("TMP", windir); GetEnvironmentVariableA("TMP", buf, sizeof(buf)); trace("TMP=%s\n", buf); test_GetTempPathA(windir); test_GetTempPathW(windir); /* TMP=C: i.e. use current working directory of the specified drive */ GetWindowsDirectoryA(windir, sizeof(windir)); SetCurrentDirectoryA(windir); windir[2] = 0; SetEnvironmentVariableA("TMP", windir); GetEnvironmentVariableA("TMP", buf, sizeof(buf)); trace("TMP=%s\n", buf); GetWindowsDirectoryA(windir, sizeof(windir)); strcat(windir,"\\"); test_GetTempPathA(windir); test_GetTempPathW(windir); SetEnvironmentVariableA("TMP", save_TMP); } static void test_GetLongPathNameA(void) { DWORD length, explength, hostsize; char tempfile[MAX_PATH]; char longpath[MAX_PATH]; char unc_prefix[MAX_PATH]; char unc_short[MAX_PATH], unc_long[MAX_PATH]; char temppath[MAX_PATH], temppath2[MAX_PATH]; HANDLE file; if (!pGetLongPathNameA) return; GetTempPathA(MAX_PATH, tempfile); lstrcatA(tempfile, "longfilename.longext"); file = CreateFileA(tempfile, GENERIC_READ|GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); CloseHandle(file); /* Test a normal path with a small buffer size */ memset(temppath, 0, MAX_PATH); length = pGetLongPathNameA(tempfile, temppath, 4); /* We have a failure so length should be the minimum plus the terminating '0' */ ok(length >= strlen(tempfile) + 1, "Wrong length\n"); ok(temppath[0] == 0, "Buffer should not have been touched\n"); /* Some UNC syntax tests */ memset(temppath, 0, MAX_PATH); memset(temppath2, 0, MAX_PATH); lstrcpyA(temppath2, "\\\\?\\"); lstrcatA(temppath2, tempfile); explength = length + 4; SetLastError(0xdeadbeef); length = pGetLongPathNameA(temppath2, NULL, 0); if (length == 0 && GetLastError() == ERROR_BAD_NET_NAME) { win_skip("UNC syntax tests don't work on Win98/WinMe\n"); DeleteFileA(tempfile); return; } ok(length == explength, "Wrong length %d, expected %d\n", length, explength); length = pGetLongPathNameA(temppath2, NULL, MAX_PATH); ok(length == explength, "Wrong length %d, expected %d\n", length, explength); length = pGetLongPathNameA(temppath2, temppath, 4); ok(length == explength, "Wrong length %d, expected %d\n", length, explength); ok(temppath[0] == 0, "Buffer should not have been touched\n"); /* Now an UNC path with the computername */ lstrcpyA(unc_prefix, "\\\\"); hostsize = sizeof(unc_prefix) - 2; GetComputerNameA(unc_prefix + 2, &hostsize); lstrcatA(unc_prefix, "\\"); /* Create a short syntax for the whole unc path */ memset(unc_short, 0, MAX_PATH); GetShortPathNameA(tempfile, temppath, MAX_PATH); lstrcpyA(unc_short, unc_prefix); unc_short[lstrlenA(unc_short)] = temppath[0]; lstrcatA(unc_short, "$\\"); lstrcatA(unc_short, strchr(temppath, '\\') + 1); /* Create a long syntax for reference */ memset(longpath, 0, MAX_PATH); pGetLongPathNameA(tempfile, temppath, MAX_PATH); lstrcpyA(longpath, unc_prefix); longpath[lstrlenA(longpath)] = temppath[0]; lstrcatA(longpath, "$\\"); lstrcatA(longpath, strchr(temppath, '\\') + 1); /* NULL test */ SetLastError(0xdeadbeef); length = pGetLongPathNameA(unc_short, NULL, 0); if (length == 0 && GetLastError() == ERROR_BAD_NETPATH) { /* Seen on Window XP Home */ win_skip("UNC with computername is not supported\n"); DeleteFileA(tempfile); return; } explength = lstrlenA(longpath) + 1; todo_wine ok(length == explength, "Wrong length %d, expected %d\n", length, explength); length = pGetLongPathNameA(unc_short, NULL, MAX_PATH); todo_wine ok(length == explength, "Wrong length %d, expected %d\n", length, explength); memset(unc_long, 0, MAX_PATH); length = pGetLongPathNameA(unc_short, unc_long, lstrlenA(unc_short)); /* length will include terminating '0' on failure */ todo_wine ok(length == explength, "Wrong length %d, expected %d\n", length, explength); ok(unc_long[0] == 0, "Buffer should not have been touched\n"); memset(unc_long, 0, MAX_PATH); length = pGetLongPathNameA(unc_short, unc_long, length); /* length doesn't include terminating '0' on success */ explength--; todo_wine { ok(length == explength, "Wrong length %d, expected %d\n", length, explength); ok(!lstrcmpiA(unc_long, longpath), "Expected (%s), got (%s)\n", longpath, unc_long); } DeleteFileA(tempfile); } static void test_GetLongPathNameW(void) { DWORD length, expanded; BOOL ret; HANDLE file; WCHAR empty[MAX_PATH]; WCHAR tempdir[MAX_PATH], name[200]; WCHAR dirpath[4 + MAX_PATH + 200]; /* To ease removal */ WCHAR shortpath[4 + MAX_PATH + 200 + 1 + 200]; static const WCHAR prefix[] = { '\\','\\','?','\\', 0}; static const WCHAR backslash[] = { '\\', 0}; static const WCHAR letterX[] = { 'X', 0}; if (!pGetLongPathNameW) return; SetLastError(0xdeadbeef); length = pGetLongPathNameW(NULL,NULL,0); if (GetLastError() == ERROR_CALL_NOT_IMPLEMENTED) { win_skip("GetLongPathNameW is not implemented\n"); return; } ok(0==length,"GetLongPathNameW returned %d but expected 0\n",length); ok(GetLastError()==ERROR_INVALID_PARAMETER,"GetLastError returned %d but expected ERROR_INVALID_PARAMETER\n",GetLastError()); SetLastError(0xdeadbeef); empty[0]=0; length = pGetLongPathNameW(empty,NULL,0); ok(0==length,"GetLongPathNameW returned %d but expected 0\n",length); ok(GetLastError()==ERROR_PATH_NOT_FOUND,"GetLastError returned %d but expected ERROR_PATH_NOT_FOUND\n",GetLastError()); /* Create a long path name. The path needs to exist for these tests to * succeed so we need the "\\?\" prefix when creating directories and * files. */ name[0] = 0; while (lstrlenW(name) < (sizeof(name)/sizeof(WCHAR) - 1)) lstrcatW(name, letterX); GetTempPathW(MAX_PATH, tempdir); lstrcpyW(shortpath, prefix); lstrcatW(shortpath, tempdir); lstrcatW(shortpath, name); lstrcpyW(dirpath, shortpath); ret = CreateDirectoryW(shortpath, NULL); ok(ret, "Could not create the temporary directory : %d\n", GetLastError()); lstrcatW(shortpath, backslash); lstrcatW(shortpath, name); /* Path does not exist yet and we know it overruns MAX_PATH */ /* No prefix */ SetLastError(0xdeadbeef); length = pGetLongPathNameW(shortpath + 4, NULL, 0); ok(length == 0, "Expected 0, got %d\n", length); todo_wine ok(GetLastError() == ERROR_PATH_NOT_FOUND, "Expected ERROR_PATH_NOT_FOUND, got %d\n", GetLastError()); /* With prefix */ SetLastError(0xdeadbeef); length = pGetLongPathNameW(shortpath, NULL, 0); todo_wine { ok(length == 0, "Expected 0, got %d\n", length); ok(GetLastError() == ERROR_FILE_NOT_FOUND, "Expected ERROR_PATH_NOT_FOUND, got %d\n", GetLastError()); } file = CreateFileW(shortpath, GENERIC_READ|GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); ok(file != INVALID_HANDLE_VALUE, "Could not create the temporary file : %d.\n", GetLastError()); CloseHandle(file); /* Path exists */ /* No prefix */ SetLastError(0xdeadbeef); length = pGetLongPathNameW(shortpath + 4, NULL, 0); todo_wine { ok(length == 0, "Expected 0, got %d\n", length); ok(GetLastError() == ERROR_PATH_NOT_FOUND, "Expected ERROR_PATH_NOT_FOUND, got %d\n", GetLastError()); } /* With prefix */ expanded = 4 + (pGetLongPathNameW(tempdir, NULL, 0) - 1) + lstrlenW(name) + 1 + lstrlenW(name) + 1; SetLastError(0xdeadbeef); length = pGetLongPathNameW(shortpath, NULL, 0); ok(length == expanded, "Expected %d, got %d\n", expanded, length); /* NULL buffer with length crashes on Windows */ if (0) pGetLongPathNameW(shortpath, NULL, 20); ok(DeleteFileW(shortpath), "Could not delete temporary file\n"); ok(RemoveDirectoryW(dirpath), "Could not delete temporary directory\n"); } static void test_GetShortPathNameW(void) { static const WCHAR extended_prefix[] = {'\\','\\','?','\\',0}; static const WCHAR test_path[] = { 'L', 'o', 'n', 'g', 'D', 'i', 'r', 'e', 'c', 't', 'o', 'r', 'y', 'N', 'a', 'm', 'e', 0 }; static const WCHAR name[] = { 't', 'e', 's', 't', 0 }; static const WCHAR backSlash[] = { '\\', 0 }; static const WCHAR a_bcdeW[] = {'a','.','b','c','d','e',0}; WCHAR path[MAX_PATH], tmppath[MAX_PATH], *ptr; WCHAR short_path[MAX_PATH]; DWORD length; HANDLE file; int ret; SetLastError(0xdeadbeef); GetTempPathW( MAX_PATH, tmppath ); if (GetLastError() == ERROR_CALL_NOT_IMPLEMENTED) { win_skip("GetTempPathW is not implemented\n"); return; } lstrcpyW( path, tmppath ); lstrcatW( path, test_path ); lstrcatW( path, backSlash ); ret = CreateDirectoryW( path, NULL ); ok( ret, "Directory was not created. LastError = %d\n", GetLastError() ); /* Starting a main part of test */ /* extended path \\?\C:\path\ */ lstrcpyW( path, extended_prefix ); lstrcatW( path, tmppath ); lstrcatW( path, test_path ); lstrcatW( path, backSlash ); short_path[0] = 0; length = GetShortPathNameW( path, short_path, sizeof(short_path) / sizeof(*short_path) ); ok( length, "GetShortPathNameW returned 0.\n" ); lstrcpyW( path, tmppath ); lstrcatW( path, test_path ); lstrcatW( path, backSlash ); length = GetShortPathNameW( path, short_path, 0 ); ok( length, "GetShortPathNameW returned 0.\n" ); ret = GetShortPathNameW( path, short_path, length ); ok( ret && ret == length-1, "GetShortPathNameW returned 0.\n" ); lstrcatW( short_path, name ); /* GetShortPathName for a non-existent short file name should fail */ SetLastError(0xdeadbeef); length = GetShortPathNameW( short_path, path, 0 ); ok(!length, "GetShortPathNameW should fail\n"); ok(GetLastError() == ERROR_FILE_NOT_FOUND, "expected ERROR_FILE_NOT_FOUND, got %d\n", GetLastError()); file = CreateFileW( short_path, GENERIC_READ|GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL ); ok( file != INVALID_HANDLE_VALUE, "File was not created.\n" ); CloseHandle( file ); ret = DeleteFileW( short_path ); ok( ret, "Cannot delete file.\n" ); ptr = path + lstrlenW(path); lstrcpyW( ptr, a_bcdeW); file = CreateFileW( path, GENERIC_READ|GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL ); ok( file != INVALID_HANDLE_VALUE, "File was not created.\n" ); CloseHandle( file ); length = GetShortPathNameW( path, short_path, sizeof(short_path)/sizeof(*short_path) ); ok( length, "GetShortPathNameW failed: %u.\n", GetLastError() ); ret = DeleteFileW( path ); ok( ret, "Cannot delete file.\n" ); *ptr = 0; /* End test */ ret = RemoveDirectoryW( path ); ok( ret, "Cannot delete directory.\n" ); } static void test_GetSystemDirectory(void) { CHAR buffer[MAX_PATH + 4]; DWORD res; DWORD total; SetLastError(0xdeadbeef); res = GetSystemDirectoryA(NULL, 0); /* res includes the terminating Zero */ ok(res > 0, "returned %d with %d (expected '>0')\n", res, GetLastError()); total = res; /* this crashes on XP */ if (0) GetSystemDirectoryA(NULL, total); SetLastError(0xdeadbeef); res = GetSystemDirectoryA(NULL, total-1); /* 95+NT: total (includes the terminating Zero) 98+ME: 0 with ERROR_INVALID_PARAMETER */ ok( (res == total) || (!res && (GetLastError() == ERROR_INVALID_PARAMETER)), "returned %d with %d (expected '%d' or: '0' with " "ERROR_INVALID_PARAMETER)\n", res, GetLastError(), total); if (total > MAX_PATH) return; buffer[0] = '\0'; SetLastError(0xdeadbeef); res = GetSystemDirectoryA(buffer, total); /* res does not include the terminating Zero */ ok( (res == (total-1)) && (buffer[0]), "returned %d with %d and '%s' (expected '%d' and a string)\n", res, GetLastError(), buffer, total-1); buffer[0] = '\0'; SetLastError(0xdeadbeef); res = GetSystemDirectoryA(buffer, total + 1); /* res does not include the terminating Zero */ ok( (res == (total-1)) && (buffer[0]), "returned %d with %d and '%s' (expected '%d' and a string)\n", res, GetLastError(), buffer, total-1); memset(buffer, '#', total + 1); buffer[total + 2] = '\0'; SetLastError(0xdeadbeef); res = GetSystemDirectoryA(buffer, total-1); /* res includes the terminating Zero) */ ok( res == total, "returned %d with %d and '%s' (expected '%d')\n", res, GetLastError(), buffer, total); memset(buffer, '#', total + 1); buffer[total + 2] = '\0'; SetLastError(0xdeadbeef); res = GetSystemDirectoryA(buffer, total-2); /* res includes the terminating Zero) */ ok( res == total, "returned %d with %d and '%s' (expected '%d')\n", res, GetLastError(), buffer, total); } static void test_GetWindowsDirectory(void) { CHAR buffer[MAX_PATH + 4]; DWORD res; DWORD total; SetLastError(0xdeadbeef); res = GetWindowsDirectoryA(NULL, 0); /* res includes the terminating Zero */ ok(res > 0, "returned %d with %d (expected '>0')\n", res, GetLastError()); total = res; /* this crashes on XP */ if (0) GetWindowsDirectoryA(NULL, total); SetLastError(0xdeadbeef); res = GetWindowsDirectoryA(NULL, total-1); /* 95+NT: total (includes the terminating Zero) 98+ME: 0 with ERROR_INVALID_PARAMETER */ ok( (res == total) || (!res && (GetLastError() == ERROR_INVALID_PARAMETER)), "returned %d with %d (expected '%d' or: '0' with " "ERROR_INVALID_PARAMETER)\n", res, GetLastError(), total); if (total > MAX_PATH) return; buffer[0] = '\0'; SetLastError(0xdeadbeef); res = GetWindowsDirectoryA(buffer, total); /* res does not include the terminating Zero */ ok( (res == (total-1)) && (buffer[0]), "returned %d with %d and '%s' (expected '%d' and a string)\n", res, GetLastError(), buffer, total-1); buffer[0] = '\0'; SetLastError(0xdeadbeef); res = GetWindowsDirectoryA(buffer, total + 1); /* res does not include the terminating Zero */ ok( (res == (total-1)) && (buffer[0]), "returned %d with %d and '%s' (expected '%d' and a string)\n", res, GetLastError(), buffer, total-1); memset(buffer, '#', total + 1); buffer[total + 2] = '\0'; SetLastError(0xdeadbeef); res = GetWindowsDirectoryA(buffer, total-1); /* res includes the terminating Zero) */ ok( res == total, "returned %d with %d and '%s' (expected '%d')\n", res, GetLastError(), buffer, total); memset(buffer, '#', total + 1); buffer[total + 2] = '\0'; SetLastError(0xdeadbeef); res = GetWindowsDirectoryA(buffer, total-2); /* res includes the terminating Zero) */ ok( res == total, "returned %d with %d and '%s' (expected '%d')\n", res, GetLastError(), buffer, total); } static void test_NeedCurrentDirectoryForExePathA(void) { if (!pNeedCurrentDirectoryForExePathA) { win_skip("NeedCurrentDirectoryForExePathA is not available\n"); return; } /* Crashes in Windows */ if (0) pNeedCurrentDirectoryForExePathA(NULL); SetEnvironmentVariableA("NoDefaultCurrentDirectoryInExePath", NULL); ok(pNeedCurrentDirectoryForExePathA("."), "returned FALSE for \".\"\n"); ok(pNeedCurrentDirectoryForExePathA("c:\\"), "returned FALSE for \"c:\\\"\n"); ok(pNeedCurrentDirectoryForExePathA("cmd.exe"), "returned FALSE for \"cmd.exe\"\n"); SetEnvironmentVariableA("NoDefaultCurrentDirectoryInExePath", "nya"); ok(!pNeedCurrentDirectoryForExePathA("."), "returned TRUE for \".\"\n"); ok(pNeedCurrentDirectoryForExePathA("c:\\"), "returned FALSE for \"c:\\\"\n"); ok(!pNeedCurrentDirectoryForExePathA("cmd.exe"), "returned TRUE for \"cmd.exe\"\n"); } static void test_NeedCurrentDirectoryForExePathW(void) { const WCHAR thispath[] = {'.', 0}; const WCHAR fullpath[] = {'c', ':', '\\', 0}; const WCHAR cmdname[] = {'c', 'm', 'd', '.', 'e', 'x', 'e', 0}; if (!pNeedCurrentDirectoryForExePathW) { win_skip("NeedCurrentDirectoryForExePathW is not available\n"); return; } /* Crashes in Windows */ if (0) pNeedCurrentDirectoryForExePathW(NULL); SetEnvironmentVariableA("NoDefaultCurrentDirectoryInExePath", NULL); ok(pNeedCurrentDirectoryForExePathW(thispath), "returned FALSE for \".\"\n"); ok(pNeedCurrentDirectoryForExePathW(fullpath), "returned FALSE for \"c:\\\"\n"); ok(pNeedCurrentDirectoryForExePathW(cmdname), "returned FALSE for \"cmd.exe\"\n"); SetEnvironmentVariableA("NoDefaultCurrentDirectoryInExePath", "nya"); ok(!pNeedCurrentDirectoryForExePathW(thispath), "returned TRUE for \".\"\n"); ok(pNeedCurrentDirectoryForExePathW(fullpath), "returned FALSE for \"c:\\\"\n"); ok(!pNeedCurrentDirectoryForExePathW(cmdname), "returned TRUE for \"cmd.exe\"\n"); } /* Call various path/file name retrieving APIs and check the case of * the returned drive letter. Some apps (for instance Adobe Photoshop CS3 * installer) depend on the drive letter being in upper case. */ static void test_drive_letter_case(void) { UINT ret; char buf[MAX_PATH]; #define is_upper_case_letter(a) ((a) >= 'A' && (a) <= 'Z') memset(buf, 0, sizeof(buf)); SetLastError(0xdeadbeef); ret = GetWindowsDirectoryA(buf, sizeof(buf)); ok(ret, "GetWindowsDirectory error %u\n", GetLastError()); ok(ret < sizeof(buf), "buffer should be %u bytes\n", ret); ok(buf[1] == ':', "expected buf[1] == ':' got %c\n", buf[1]); ok(is_upper_case_letter(buf[0]), "expected buf[0] upper case letter got %c\n", buf[0]); /* re-use the buffer returned by GetFullPathName */ buf[2] = '/'; SetLastError(0xdeadbeef); ret = GetFullPathNameA(buf + 2, sizeof(buf), buf, NULL); ok(ret, "GetFullPathName error %u\n", GetLastError()); ok(ret < sizeof(buf), "buffer should be %u bytes\n", ret); ok(buf[1] == ':', "expected buf[1] == ':' got %c\n", buf[1]); ok(is_upper_case_letter(buf[0]), "expected buf[0] upper case letter got %c\n", buf[0]); memset(buf, 0, sizeof(buf)); SetLastError(0xdeadbeef); ret = GetSystemDirectoryA(buf, sizeof(buf)); ok(ret, "GetSystemDirectory error %u\n", GetLastError()); ok(ret < sizeof(buf), "buffer should be %u bytes\n", ret); ok(buf[1] == ':', "expected buf[1] == ':' got %c\n", buf[1]); ok(is_upper_case_letter(buf[0]), "expected buf[0] upper case letter got %c\n", buf[0]); memset(buf, 0, sizeof(buf)); SetLastError(0xdeadbeef); ret = GetCurrentDirectoryA(sizeof(buf), buf); ok(ret, "GetCurrentDirectory error %u\n", GetLastError()); ok(ret < sizeof(buf), "buffer should be %u bytes\n", ret); ok(buf[1] == ':', "expected buf[1] == ':' got %c\n", buf[1]); ok(is_upper_case_letter(buf[0]), "expected buf[0] upper case letter got %c\n", buf[0]); /* TEMP is an environment variable, so it can't be tested for case-sensitivity */ memset(buf, 0, sizeof(buf)); SetLastError(0xdeadbeef); ret = GetTempPathA(sizeof(buf), buf); ok(ret, "GetTempPath error %u\n", GetLastError()); ok(ret < sizeof(buf), "buffer should be %u bytes\n", ret); if (buf[0]) { ok(buf[1] == ':', "expected buf[1] == ':' got %c\n", buf[1]); ok(buf[strlen(buf)-1] == '\\', "Temporary path (%s) doesn't end in a slash\n", buf); } memset(buf, 0, sizeof(buf)); SetLastError(0xdeadbeef); ret = GetFullPathNameA(".", sizeof(buf), buf, NULL); ok(ret, "GetFullPathName error %u\n", GetLastError()); ok(ret < sizeof(buf), "buffer should be %u bytes\n", ret); ok(buf[1] == ':', "expected buf[1] == ':' got %c\n", buf[1]); ok(is_upper_case_letter(buf[0]), "expected buf[0] upper case letter got %c\n", buf[0]); /* re-use the buffer returned by GetFullPathName */ SetLastError(0xdeadbeef); ret = GetShortPathNameA(buf, buf, sizeof(buf)); ok(ret, "GetShortPathName error %u\n", GetLastError()); ok(ret < sizeof(buf), "buffer should be %u bytes\n", ret); ok(buf[1] == ':', "expected buf[1] == ':' got %c\n", buf[1]); ok(is_upper_case_letter(buf[0]), "expected buf[0] upper case letter got %c\n", buf[0]); if (pGetLongPathNameA) { /* re-use the buffer returned by GetShortPathName */ SetLastError(0xdeadbeef); ret = pGetLongPathNameA(buf, buf, sizeof(buf)); ok(ret, "GetLongPathNameA error %u\n", GetLastError()); ok(ret < sizeof(buf), "buffer should be %u bytes\n", ret); ok(buf[1] == ':', "expected buf[1] == ':' got %c\n", buf[1]); ok(is_upper_case_letter(buf[0]), "expected buf[0] upper case letter got %c\n", buf[0]); } #undef is_upper_case_letter } static const char manifest_dep[] = "" "" " " " " " " ""; static const char manifest_main[] = "" "" "" " " " " " " "" ""; static void create_manifest_file(const char *filename, const char *manifest) { WCHAR path[MAX_PATH], manifest_path[MAX_PATH]; HANDLE file; DWORD size; MultiByteToWideChar( CP_ACP, 0, filename, -1, path, MAX_PATH ); GetTempPathW(sizeof(manifest_path)/sizeof(WCHAR), manifest_path); lstrcatW(manifest_path, path); file = CreateFileW(manifest_path, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); ok(file != INVALID_HANDLE_VALUE, "CreateFile failed: %u\n", GetLastError()); WriteFile(file, manifest, strlen(manifest), &size, NULL); CloseHandle(file); } static void delete_manifest_file(const char *filename) { CHAR path[MAX_PATH]; GetTempPathA(sizeof(path), path); strcat(path, filename); DeleteFileA(path); } static HANDLE test_create(const char *file) { WCHAR path[MAX_PATH], manifest_path[MAX_PATH]; ACTCTXW actctx; HANDLE handle; MultiByteToWideChar(CP_ACP, 0, file, -1, path, MAX_PATH); GetTempPathW(sizeof(manifest_path)/sizeof(WCHAR), manifest_path); lstrcatW(manifest_path, path); memset(&actctx, 0, sizeof(ACTCTXW)); actctx.cbSize = sizeof(ACTCTXW); actctx.lpSource = manifest_path; handle = pCreateActCtxW(&actctx); ok(handle != INVALID_HANDLE_VALUE, "failed to create context, error %u\n", GetLastError()); ok(actctx.cbSize == sizeof(actctx), "cbSize=%d\n", actctx.cbSize); ok(actctx.dwFlags == 0, "dwFlags=%d\n", actctx.dwFlags); ok(actctx.lpSource == manifest_path, "lpSource=%p\n", actctx.lpSource); ok(actctx.wProcessorArchitecture == 0, "wProcessorArchitecture=%d\n", actctx.wProcessorArchitecture); ok(actctx.wLangId == 0, "wLangId=%d\n", actctx.wLangId); ok(actctx.lpAssemblyDirectory == NULL, "lpAssemblyDirectory=%p\n", actctx.lpAssemblyDirectory); ok(actctx.lpResourceName == NULL, "lpResourceName=%p\n", actctx.lpResourceName); ok(actctx.lpApplicationName == NULL, "lpApplicationName=%p\n", actctx.lpApplicationName); ok(actctx.hModule == NULL, "hModule=%p\n", actctx.hModule); return handle; } static void test_SearchPathA(void) { static const CHAR testdepA[] = "testdep.dll"; static const CHAR testdeprelA[] = "./testdep.dll"; static const CHAR kernel32A[] = "kernel32.dll"; static const CHAR fileA[] = ""; CHAR pathA[MAX_PATH], buffA[MAX_PATH], path2A[MAX_PATH]; CHAR *ptrA = NULL; ULONG_PTR cookie; HANDLE handle; DWORD ret; if (!pSearchPathA) { win_skip("SearchPathA isn't available\n"); return; } GetWindowsDirectoryA(pathA, sizeof(pathA)/sizeof(CHAR)); /* NULL filename */ SetLastError(0xdeadbeef); ret = pSearchPathA(pathA, NULL, NULL, sizeof(buffA)/sizeof(CHAR), buffA, &ptrA); ok(ret == 0, "Expected failure, got %d\n", ret); ok(GetLastError() == ERROR_INVALID_PARAMETER, "Expected ERROR_INVALID_PARAMETER, got %x\n", GetLastError()); /* empty filename */ SetLastError(0xdeadbeef); ret = pSearchPathA(pathA, fileA, NULL, sizeof(buffA)/sizeof(CHAR), buffA, &ptrA); ok(ret == 0, "Expected failure, got %d\n", ret); ok(GetLastError() == ERROR_INVALID_PARAMETER, "Expected ERROR_INVALID_PARAMETER, got %x\n", GetLastError()); if (!pActivateActCtx) return; create_manifest_file("testdep1.manifest", manifest_dep); create_manifest_file("main.manifest", manifest_main); handle = test_create("main.manifest"); delete_manifest_file("testdep1.manifest"); delete_manifest_file("main.manifest"); /* search fails without active context */ ret = pSearchPathA(NULL, testdepA, NULL, sizeof(buffA)/sizeof(CHAR), buffA, NULL); ok(ret == 0, "got %d\n", ret); ret = pSearchPathA(NULL, kernel32A, NULL, sizeof(path2A)/sizeof(CHAR), path2A, NULL); ok(ret && ret == strlen(path2A), "got %d\n", ret); ret = pActivateActCtx(handle, &cookie); ok(ret, "failed to activate context, %u\n", GetLastError()); /* works when activated */ ret = pSearchPathA(NULL, testdepA, NULL, sizeof(buffA)/sizeof(CHAR), buffA, NULL); ok(ret && ret == strlen(buffA), "got %d\n", ret); ret = pSearchPathA(NULL, "testdep.dll", ".ext", sizeof(buffA)/sizeof(CHAR), buffA, NULL); ok(ret && ret == strlen(buffA), "got %d\n", ret); ret = pSearchPathA(NULL, "testdep", ".dll", sizeof(buffA)/sizeof(CHAR), buffA, NULL); ok(ret && ret == strlen(buffA), "got %d\n", ret); ret = pSearchPathA(NULL, "testdep", ".ext", sizeof(buffA)/sizeof(CHAR), buffA, NULL); ok(!ret, "got %d\n", ret); /* name contains path */ ret = pSearchPathA(NULL, testdeprelA, NULL, sizeof(buffA)/sizeof(CHAR), buffA, NULL); ok(!ret, "got %d\n", ret); /* fails with specified path that doesn't contain this file */ ret = pSearchPathA(pathA, testdepA, NULL, sizeof(buffA)/sizeof(CHAR), buffA, NULL); ok(!ret, "got %d\n", ret); /* path is redirected for wellknown names too */ ret = pSearchPathA(NULL, kernel32A, NULL, sizeof(buffA)/sizeof(CHAR), buffA, NULL); ok(ret && ret == strlen(buffA), "got %d\n", ret); ok(strcmp(buffA, path2A), "got wrong path %s, %s\n", buffA, path2A); ret = pDeactivateActCtx(0, cookie); ok(ret, "failed to deactivate context, %u\n", GetLastError()); pReleaseActCtx(handle); } static void test_SearchPathW(void) { static const WCHAR testdeprelW[] = {'.','/','t','e','s','t','d','e','p','.','d','l','l',0}; static const WCHAR testdepW[] = {'t','e','s','t','d','e','p','.','d','l','l',0}; static const WCHAR testdep1W[] = {'t','e','s','t','d','e','p',0}; static const WCHAR kernel32dllW[] = {'k','e','r','n','e','l','3','2','.','d','l','l',0}; static const WCHAR kernel32W[] = {'k','e','r','n','e','l','3','2',0}; static const WCHAR ole32W[] = {'o','l','e','3','2',0}; static const WCHAR extW[] = {'.','e','x','t',0}; static const WCHAR dllW[] = {'.','d','l','l',0}; static const WCHAR fileW[] = { 0 }; WCHAR pathW[MAX_PATH], buffW[MAX_PATH], path2W[MAX_PATH]; WCHAR *ptrW = NULL; ULONG_PTR cookie; HANDLE handle; DWORD ret; if (!pSearchPathW) { win_skip("SearchPathW isn't available\n"); return; } if (0) { /* NULL filename, crashes on nt4 */ pSearchPathW(pathW, NULL, NULL, sizeof(buffW)/sizeof(WCHAR), buffW, &ptrW); } GetWindowsDirectoryW(pathW, sizeof(pathW)/sizeof(WCHAR)); /* empty filename */ SetLastError(0xdeadbeef); ret = pSearchPathW(pathW, fileW, NULL, sizeof(buffW)/sizeof(WCHAR), buffW, &ptrW); ok(ret == 0, "Expected failure, got %d\n", ret); ok(GetLastError() == ERROR_INVALID_PARAMETER, "Expected ERROR_INVALID_PARAMETER, got %x\n", GetLastError()); if (!pActivateActCtx) return; create_manifest_file("testdep1.manifest", manifest_dep); create_manifest_file("main.manifest", manifest_main); handle = test_create("main.manifest"); delete_manifest_file("testdep1.manifest"); delete_manifest_file("main.manifest"); /* search fails without active context */ ret = pSearchPathW(NULL, testdepW, NULL, sizeof(buffW)/sizeof(WCHAR), buffW, NULL); ok(ret == 0, "got %d\n", ret); ret = pSearchPathW(NULL, kernel32dllW, NULL, sizeof(path2W)/sizeof(WCHAR), path2W, NULL); ok(ret && ret == lstrlenW(path2W), "got %d\n", ret); /* full path, name without 'dll' extension */ GetSystemDirectoryW(pathW, sizeof(pathW)/sizeof(WCHAR)); ret = pSearchPathW(pathW, kernel32W, NULL, sizeof(path2W)/sizeof(WCHAR), path2W, NULL); ok(ret == 0, "got %d\n", ret); GetWindowsDirectoryW(pathW, sizeof(pathW)/sizeof(WCHAR)); ret = pActivateActCtx(handle, &cookie); ok(ret, "failed to activate context, %u\n", GetLastError()); /* works when activated */ ret = pSearchPathW(NULL, testdepW, NULL, sizeof(buffW)/sizeof(WCHAR), buffW, NULL); ok(ret && ret == lstrlenW(buffW), "got %d\n", ret); ret = pSearchPathW(NULL, testdepW, extW, sizeof(buffW)/sizeof(WCHAR), buffW, NULL); ok(ret && ret == lstrlenW(buffW), "got %d\n", ret); ret = pSearchPathW(NULL, testdep1W, dllW, sizeof(buffW)/sizeof(WCHAR), buffW, NULL); ok(ret && ret == lstrlenW(buffW), "got %d\n", ret); ret = pSearchPathW(NULL, testdep1W, extW, sizeof(buffW)/sizeof(WCHAR), buffW, NULL); ok(!ret, "got %d\n", ret); /* name contains path */ ret = pSearchPathW(NULL, testdeprelW, NULL, sizeof(buffW)/sizeof(WCHAR), buffW, NULL); ok(!ret, "got %d\n", ret); /* fails with specified path that doesn't contain this file */ ret = pSearchPathW(pathW, testdepW, NULL, sizeof(buffW)/sizeof(WCHAR), buffW, NULL); ok(!ret, "got %d\n", ret); /* path is redirected for wellknown names too, meaning it takes precedence over normal search order */ ret = pSearchPathW(NULL, kernel32dllW, NULL, sizeof(buffW)/sizeof(WCHAR), buffW, NULL); ok(ret && ret == lstrlenW(buffW), "got %d\n", ret); ok(lstrcmpW(buffW, path2W), "got wrong path %s, %s\n", wine_dbgstr_w(buffW), wine_dbgstr_w(path2W)); /* path is built using on manifest file name */ ret = pSearchPathW(NULL, ole32W, NULL, sizeof(buffW)/sizeof(WCHAR), buffW, NULL); ok(ret && ret == lstrlenW(buffW), "got %d\n", ret); ret = pDeactivateActCtx(0, cookie); ok(ret, "failed to deactivate context, %u\n", GetLastError()); pReleaseActCtx(handle); } static void test_GetFullPathNameA(void) { char output[MAX_PATH], *filepart; DWORD ret; int i; UINT acp; const struct { LPCSTR name; DWORD len; LPSTR buffer; LPSTR *lastpart; } invalid_parameters[] = { {NULL, 0, NULL, NULL}, {NULL, MAX_PATH, NULL, NULL}, {NULL, MAX_PATH, output, NULL}, {NULL, MAX_PATH, output, &filepart}, {"", 0, NULL, NULL}, {"", MAX_PATH, NULL, NULL}, {"", MAX_PATH, output, NULL}, {"", MAX_PATH, output, &filepart}, }; for (i = 0; i < sizeof(invalid_parameters)/sizeof(invalid_parameters[0]); i++) { SetLastError(0xdeadbeef); strcpy(output, "deadbeef"); filepart = (char *)0xdeadbeef; ret = GetFullPathNameA(invalid_parameters[i].name, invalid_parameters[i].len, invalid_parameters[i].buffer, invalid_parameters[i].lastpart); ok(!ret, "[%d] Expected GetFullPathNameA to return 0, got %u\n", i, ret); ok(!strcmp(output, "deadbeef"), "[%d] Expected the output buffer to be unchanged, got \"%s\"\n", i, output); ok(filepart == (char *)0xdeadbeef, "[%d] Expected output file part pointer to be untouched, got %p\n", i, filepart); ok(GetLastError() == 0xdeadbeef || GetLastError() == ERROR_INVALID_NAME, /* Win7 */ "[%d] Expected GetLastError() to return 0xdeadbeef, got %u\n", i, GetLastError()); } acp = GetACP(); if (acp != 932) skip("Skipping DBCS(Japanese) GetFullPathNameA test in this codepage (%d)\n", acp); else { const struct dbcs_case { const char *input; const char *expected; } testset[] = { { "c:\\a\\\x95\x5c\x97\xa0.txt", "\x95\x5c\x97\xa0.txt" }, { "c:\\\x83\x8f\x83\x43\x83\x93\\wine.c", "wine.c" }, { "c:\\demo\\\x97\xa0\x95\x5c", "\x97\xa0\x95\x5c" } }; for (i = 0; i < sizeof(testset)/sizeof(testset[0]); i++) { ret = GetFullPathNameA(testset[i].input, sizeof(output), output, &filepart); ok(ret, "[%d] GetFullPathName error %u\n", i, GetLastError()); ok(!lstrcmpA(filepart, testset[i].expected), "[%d] expected %s got %s\n", i, testset[i].expected, filepart); } } } static void test_GetFullPathNameW(void) { static const WCHAR emptyW[] = {0}; static const WCHAR deadbeefW[] = {'d','e','a','d','b','e','e','f',0}; WCHAR output[MAX_PATH], *filepart; DWORD ret; int i; const struct { LPCWSTR name; DWORD len; LPWSTR buffer; LPWSTR *lastpart; int win7_expect; } invalid_parameters[] = { {NULL, 0, NULL, NULL}, {NULL, 0, NULL, &filepart, 1}, {NULL, MAX_PATH, NULL, NULL}, {NULL, MAX_PATH, output, NULL}, {NULL, MAX_PATH, output, &filepart, 1}, {emptyW, 0, NULL, NULL}, {emptyW, 0, NULL, &filepart, 1}, {emptyW, MAX_PATH, NULL, NULL}, {emptyW, MAX_PATH, output, NULL}, {emptyW, MAX_PATH, output, &filepart, 1}, }; SetLastError(0xdeadbeef); ret = GetFullPathNameW(NULL, 0, NULL, NULL); if (!ret && GetLastError() == ERROR_CALL_NOT_IMPLEMENTED) { win_skip("GetFullPathNameW is not available\n"); return; } for (i = 0; i < sizeof(invalid_parameters)/sizeof(invalid_parameters[0]); i++) { SetLastError(0xdeadbeef); lstrcpyW(output, deadbeefW); filepart = (WCHAR *)0xdeadbeef; ret = GetFullPathNameW(invalid_parameters[i].name, invalid_parameters[i].len, invalid_parameters[i].buffer, invalid_parameters[i].lastpart); ok(!ret, "[%d] Expected GetFullPathNameW to return 0, got %u\n", i, ret); ok(!lstrcmpW(output, deadbeefW), "[%d] Expected the output buffer to be unchanged, got %s\n", i, wine_dbgstr_w(output)); ok(filepart == (WCHAR *)0xdeadbeef || (invalid_parameters[i].win7_expect && filepart == NULL), "[%d] Expected output file part pointer to be untouched, got %p\n", i, filepart); ok(GetLastError() == 0xdeadbeef || GetLastError() == ERROR_INVALID_NAME, /* Win7 */ "[%d] Expected GetLastError() to return 0xdeadbeef, got %u\n", i, GetLastError()); } } static void init_pointers(void) { HMODULE hKernel32 = GetModuleHandleA("kernel32.dll"); #define MAKEFUNC(f) (p##f = (void*)GetProcAddress(hKernel32, #f)) MAKEFUNC(GetLongPathNameA); MAKEFUNC(GetLongPathNameW); MAKEFUNC(NeedCurrentDirectoryForExePathA); MAKEFUNC(NeedCurrentDirectoryForExePathW); MAKEFUNC(SearchPathA); MAKEFUNC(SearchPathW); MAKEFUNC(ActivateActCtx); MAKEFUNC(CreateActCtxW); MAKEFUNC(DeactivateActCtx); MAKEFUNC(GetCurrentActCtx); MAKEFUNC(ReleaseActCtx); MAKEFUNC(CheckNameLegalDOS8Dot3W); MAKEFUNC(CheckNameLegalDOS8Dot3A); #undef MAKEFUNC } static void test_relative_path(void) { char path[MAX_PATH], buf[MAX_PATH]; HANDLE file; int ret; if (!pGetLongPathNameA) return; GetTempPathA(MAX_PATH, path); ret = SetCurrentDirectoryA(path); ok(ret, "SetCurrentDirectory error %d\n", GetLastError()); ret = CreateDirectoryA("foo", NULL); ok(ret, "CreateDirectory error %d\n", GetLastError()); file = CreateFileA("foo\\file", GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, 0, 0); ok(file != INVALID_HANDLE_VALUE, "failed to create temp file\n"); CloseHandle(file); ret = CreateDirectoryA("bar", NULL); ok(ret, "CreateDirectory error %d\n", GetLastError()); ret = SetCurrentDirectoryA("bar"); ok(ret, "SetCurrentDirectory error %d\n", GetLastError()); ret = GetFileAttributesA("..\\foo\\file"); ok(ret != INVALID_FILE_ATTRIBUTES, "GetFileAttributes error %d\n", GetLastError()); strcpy(buf, "deadbeef"); ret = pGetLongPathNameA(".", buf, MAX_PATH); ok(ret, "GetLongPathName error %d\n", GetLastError()); ok(!strcmp(buf, "."), "expected ., got %s\n", buf); strcpy(buf, "deadbeef"); ret = GetShortPathNameA(".", buf, MAX_PATH); ok(ret, "GetShortPathName error %d\n", GetLastError()); ok(!strcmp(buf, "."), "expected ., got %s\n", buf); strcpy(buf, "deadbeef"); ret = pGetLongPathNameA("..", buf, MAX_PATH); ok(ret, "GetLongPathName error %d\n", GetLastError()); ok(!strcmp(buf, ".."), "expected .., got %s\n", buf); strcpy(buf, "deadbeef"); ret = GetShortPathNameA("..", buf, MAX_PATH); ok(ret, "GetShortPathName error %d\n", GetLastError()); ok(!strcmp(buf, ".."), "expected .., got %s\n", buf); strcpy(buf, "deadbeef"); ret = pGetLongPathNameA("..\\foo\\file", buf, MAX_PATH); ok(ret, "GetLongPathName error %d\n", GetLastError()); ok(!strcmp(buf, "..\\foo\\file"), "expected ..\\foo\\file, got %s\n", buf); strcpy(buf, "deadbeef"); ret = GetShortPathNameA("..\\foo\\file", buf, MAX_PATH); ok(ret, "GetShortPathName error %d\n", GetLastError()); ok(!strcmp(buf, "..\\foo\\file"), "expected ..\\foo\\file, got %s\n", buf); SetCurrentDirectoryA(".."); DeleteFileA("foo\\file"); RemoveDirectoryA("foo"); RemoveDirectoryA("bar"); } static void test_CheckNameLegalDOS8Dot3(void) { static const WCHAR has_driveW[] = {'C',':','\\','a','.','t','x','t',0}; static const WCHAR has_pathW[] = {'b','\\','a','.','t','x','t',0}; static const WCHAR too_longW[] = {'a','l','o','n','g','f','i','l','e','n','a','m','e','.','t','x','t',0}; static const WCHAR twodotsW[] = {'t','e','s','t','.','e','s','t','.','t','x','t',0}; static const WCHAR longextW[] = {'t','e','s','t','.','t','x','t','t','x','t',0}; static const WCHAR emptyW[] = {0}; static const WCHAR funnycharsW[] = {'!','#','$','%','&','\'','(',')','.','-','@','^',0}; static const WCHAR length8W[] = {'t','e','s','t','t','e','s','t','.','t','x','t',0}; static const WCHAR length1W[] = {'t',0}; static const WCHAR withspaceW[] = {'t','e','s','t',' ','e','s','t','.','t','x','t',0}; static const struct { const WCHAR *name; BOOL should_be_legal, has_space; } cases[] = { {has_driveW, FALSE, FALSE}, {has_pathW, FALSE, FALSE}, {too_longW, FALSE, FALSE}, {twodotsW, FALSE, FALSE}, {longextW, FALSE, FALSE}, {emptyW, TRUE /* ! */, FALSE}, {funnycharsW, TRUE, FALSE}, {length8W, TRUE, FALSE}, {length1W, TRUE, FALSE}, {withspaceW, TRUE, TRUE}, }; BOOL br, is_legal, has_space; char astr[64]; DWORD i; if(!pCheckNameLegalDOS8Dot3W){ win_skip("Missing CheckNameLegalDOS8Dot3, skipping tests\n"); return; } br = pCheckNameLegalDOS8Dot3W(NULL, NULL, 0, NULL, &is_legal); ok(br == FALSE, "CheckNameLegalDOS8Dot3W should have failed\n"); br = pCheckNameLegalDOS8Dot3A(NULL, NULL, 0, NULL, &is_legal); ok(br == FALSE, "CheckNameLegalDOS8Dot3A should have failed\n"); br = pCheckNameLegalDOS8Dot3W(length8W, NULL, 0, NULL, NULL); ok(br == FALSE, "CheckNameLegalDOS8Dot3W should have failed\n"); br = pCheckNameLegalDOS8Dot3A("testtest.txt", NULL, 0, NULL, NULL); ok(br == FALSE, "CheckNameLegalDOS8Dot3A should have failed\n"); for(i = 0; i < sizeof(cases)/sizeof(*cases); ++i){ br = pCheckNameLegalDOS8Dot3W(cases[i].name, NULL, 0, &has_space, &is_legal); ok(br == TRUE, "CheckNameLegalDOS8Dot3W failed for %s\n", wine_dbgstr_w(cases[i].name)); ok(is_legal == cases[i].should_be_legal, "Got wrong legality for %s\n", wine_dbgstr_w(cases[i].name)); if(is_legal) ok(has_space == cases[i].has_space, "Got wrong space for %s\n", wine_dbgstr_w(cases[i].name)); WideCharToMultiByte(CP_ACP, 0, cases[i].name, -1, astr, sizeof(astr), NULL, NULL); br = pCheckNameLegalDOS8Dot3A(astr, NULL, 0, &has_space, &is_legal); ok(br == TRUE, "CheckNameLegalDOS8Dot3W failed for %s\n", astr); ok(is_legal == cases[i].should_be_legal, "Got wrong legality for %s\n", astr); if(is_legal) ok(has_space == cases[i].has_space, "Got wrong space for %s\n", wine_dbgstr_w(cases[i].name)); } } START_TEST(path) { CHAR origdir[MAX_PATH],curdir[MAX_PATH], curDrive, otherDrive; init_pointers(); /* Report only once */ if (!pGetLongPathNameA) win_skip("GetLongPathNameA is not available\n"); if (!pGetLongPathNameW) win_skip("GetLongPathNameW is not available\n"); if (!pActivateActCtx) win_skip("Activation contexts not supported, some tests will be skipped\n"); test_relative_path(); test_InitPathA(curdir, &curDrive, &otherDrive); test_CurrentDirectoryA(origdir,curdir); test_PathNameA(curdir, curDrive, otherDrive); test_CleanupPathA(origdir,curdir); test_GetTempPath(); test_GetLongPathNameA(); test_GetLongPathNameW(); test_GetShortPathNameW(); test_GetSystemDirectory(); test_GetWindowsDirectory(); test_NeedCurrentDirectoryForExePathA(); test_NeedCurrentDirectoryForExePathW(); test_drive_letter_case(); test_SearchPathA(); test_SearchPathW(); test_GetFullPathNameA(); test_GetFullPathNameW(); test_CheckNameLegalDOS8Dot3(); }