/* demfcb.c - SVC handlers for misc. FCB operations * * demCloseFCB * demCreateFCB * demDate16 * demDeleteFCB * demFCBIO * demGetFileInfo * demOpenFCB * demRenameFCB * * Modification History: * * Sudeepb 09-Apr-1991 Created * Sudeepb 21-Nov-1991 Added FCB based IO functions * Jonle 30-Jun-1994 add wild card support for fcb rename */ #include "dem.h" #include "demmsg.h" #include #include #include #include "dpmtbls.h" #define DOT '.' #define QMARK '?' /* demDeleteFCB - FCB based File Delete * * * Entry - Client (ES:DI) - Full File Path * Client (AL) - 0 if not extended FCB * Client (DL) - File Attr. to be deleted (valid only if Al !=0 ) * * Exit * SUCCESS * Client (CF) = 0 * * FAILURE * Client (CF) = 1 * Client (AX) = system status code * HARD ERROR * Client (CF) = 1 * Client (AX) = 0ffffh * * Notes: Following are the rules for FCB based delete: * 1. If normal FCB than dont allow delete on hidden,system files * 2. if extended FCB than search attributes should include hidden, * system or read-only if that kind of file is to be deleted. */ VOID demDeleteFCB (VOID) { HANDLE hFind; LPSTR lpFileName; BYTE bClientAttr=0; BOOL fExtendedFCB=FALSE; WIN32_FIND_DATA wfBuffer; BOOL fSuccess = FALSE; DWORD dwAttr; USHORT uErr; CHAR szPath_buffer[_MAX_PATH]; CHAR szDrive[_MAX_DRIVE]; CHAR szDir[_MAX_DIR]; CHAR szFname[_MAX_FNAME]; CHAR szExt[_MAX_EXT]; DWORD dwErrCode = 0, dwErrCodeKeep = 0; // Get the file name lpFileName = (LPSTR) GetVDMAddr (getES(),getDI()); _splitpath( lpFileName, szDrive, szDir, szFname, szExt ); // Check if handling extended FCB if(getAL() != 0){ bClientAttr = getDL(); /* Special case for delete volume label (INT 21 Func 13H, Attr = 8H */ if((bClientAttr == ATTR_VOLUME_ID)) { if((uErr = demDeleteLabel(lpFileName[DRIVEBYTE]))) { setCF(1); setAX(uErr); return; } setAX(0); setCF(0); return; } bClientAttr &= (ATTR_READ_ONLY | ATTR_HIDDEN | ATTR_SYSTEM); fExtendedFCB = TRUE; } // Find the first instance of file if((hFind = FindFirstFileOem (lpFileName,&wfBuffer)) == (HANDLE)-1){ demClientError(INVALID_HANDLE_VALUE, *lpFileName); return; } // loop for all files which match the name and attributes do { // Check if read_only,hidden or system file if((dwAttr= wfBuffer.dwFileAttributes & (FILE_ATTRIBUTE_READONLY | FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM))){ // if so, try next file if normal FCB case. If extended fcb case // then check if right attributes are given by client. if(fExtendedFCB && ((dwAttr & (DWORD)bClientAttr) == dwAttr)){ // Yes, right attributes are given. So if the file is read // only then change the modes to normal. Note NT will // delete hidden and system files anyway. if (dwAttr & FILE_ATTRIBUTE_READONLY){ strcpy( szPath_buffer, szDrive); strcat( szPath_buffer, szDir); strncat( szPath_buffer, wfBuffer.cFileName,sizeof(szPath_buffer)-strlen(szPath_buffer)); szPath_buffer[sizeof(szPath_buffer)-1] = 0; // if set attributes fail try next file if(SetFileAttributesOemSys (szPath_buffer, FILE_ATTRIBUTE_NORMAL, FALSE) == -1) continue; } } else { dwErrCodeKeep = ERROR_ACCESS_DENIED; continue; } } strcpy( szPath_buffer, szDrive); strcat( szPath_buffer, szDir); strncat( szPath_buffer, wfBuffer.cFileName,sizeof(szPath_buffer)-strlen(szPath_buffer)); szPath_buffer[sizeof(szPath_buffer)-1] = 0; if(DeleteFileOem(szPath_buffer) == FALSE) { dwErrCode = GetLastError(); SetLastError(dwErrCode); if (((dwErrCode >= ERROR_WRITE_PROTECT) && (dwErrCode <= ERROR_GEN_FAILURE)) || dwErrCode == ERROR_WRONG_DISK ) { demClientError(INVALID_HANDLE_VALUE, szPath_buffer[0]); return; } continue; } // We have deleted at least one file, so report success fSuccess = TRUE; } while (FindNextFileOem(hFind,&wfBuffer) == TRUE); if(DPM_FindClose(hFind) == FALSE) demPrintMsg (MSG_INVALID_HFIND); if (fSuccess == TRUE){ setCF(0); return; } setCF(1); if(dwErrCodeKeep) setAX((SHORT) dwErrCodeKeep); else setAX(ERROR_FILE_NOT_FOUND); return; } /* demRenameFCB - FCB based Rename file * * Entry - Client (DS:SI) Sources file to be renamed * Client (ES:DI) Destination file to be renamed to * * Exit - SUCCESS * Client (CF) = 0 * * FAILURE * Client(CF) = 1 * Client(AX) = error code */ VOID demRenameFCB (VOID) { LPSTR lpSrc,lpDst; DWORD dw; HANDLE hFind; PCHAR pNewDstFilePart; PCHAR pDstFilePart; PCHAR pCurrSrcFilePart; WIN32_FIND_DATA W32FindData; CHAR NewDst[MAX_PATH]; CHAR CurrSrc[MAX_PATH]; lpSrc = (LPSTR) GetVDMAddr (getDS(),getSI()); lpDst = (LPSTR) GetVDMAddr (getES(),getDI()); // Find the first instance of the source file hFind = FindFirstFileOem (lpSrc,&W32FindData); if (hFind == INVALID_HANDLE_VALUE) { dw = GetLastError(); if (dw == ERROR_BAD_PATHNAME || dw == ERROR_DIRECTORY ) { SetLastError(ERROR_PATH_NOT_FOUND); } demClientError(INVALID_HANDLE_VALUE, *lpSrc); return; } // // Source string consists of the path taken from the original // source specified plus the filename part retrieved from the // FindFile call // strncpy(CurrSrc, lpSrc,sizeof(CurrSrc)); CurrSrc[sizeof(CurrSrc)-1] = 0; pCurrSrcFilePart = strrchr(CurrSrc, '\\'); pCurrSrcFilePart++; // // Destination string is template for meta character substitution // pDstFilePart = strrchr(lpDst, '\\'); pDstFilePart++; // // NewDst string is constructed from template and the source string // when doing meta file character substitution. // strncpy(NewDst, lpDst, sizeof(NewDst)); NewDst[sizeof(NewDst)-1] = 0; pNewDstFilePart = strrchr(NewDst, '\\'); pNewDstFilePart++; do { PCHAR pNew; PCHAR pSrc; PCHAR pDst; strncpy(pCurrSrcFilePart, W32FindData.cAlternateFileName[0] ? W32FindData.cAlternateFileName : W32FindData.cFileName, //// ??? hpfs lfns ???? sizeof(CurrSrc)+CurrSrc-pCurrSrcFilePart); CurrSrc[sizeof(CurrSrc)-1] = 0; pSrc = pCurrSrcFilePart; // source fname pNew = pNewDstFilePart; // dest fname to be constructed pDst = pDstFilePart; // raw dest fname template (with metas) while (*pDst) { // // If Found a '?' in Dest template, use character from src // if (*pDst == QMARK) { if (*pSrc != DOT && *pSrc) *pNew++ = *pSrc++; } // // if Found a DOT in Dest template, Align DOTS between Src\Dst // else if (*pDst == DOT) { while (*pSrc != DOT && *pSrc) { // mov src to one past DOT pSrc++; } if (*pSrc) pSrc++; *pNew++ = DOT; } // // Nothing special found, use character from Dest template // else { if (*pSrc != DOT && *pSrc) pSrc++; *pNew++ = *pDst; } pDst++; } *pNew = '\0'; // // MoveFile does not return error if dst and src are the same, // but DOS does, so check first.. // if (!_stricmp (CurrSrc, NewDst)) { setCF(1); setAX(0x5); DPM_FindClose(hFind); return; } if (!MoveFileOem(CurrSrc, NewDst)){ demClientError(INVALID_HANDLE_VALUE, *lpSrc); DPM_FindClose(hFind); return; } } while (FindNextFileOem(hFind,&W32FindData)); // // If the search on the source string for any reason besides // no more files, then its a genuine error. // dw = GetLastError(); if (dw != ERROR_NO_MORE_FILES) { if (dw == ERROR_BAD_PATHNAME || dw == ERROR_DIRECTORY ) { SetLastError(ERROR_PATH_NOT_FOUND); } demClientError(INVALID_HANDLE_VALUE, *lpSrc); } else { setCF(0); } DPM_FindClose(hFind); return; } /* demCloseFCB - Close the NT handle associated with the FCB being closed. * * Entry - Client (AX:SI) DWORD NT handle * * Exit - SUCCESS * Client (CF) = 0 * * FAILURE * Client(CF) = 1 * Client(AX) = error code */ VOID demCloseFCB (VOID) { HANDLE hFile; hFile = GETHANDLE (getAX(),getSI()); if(hFile == 0) { setCF(0); return; } if (DPM_CloseHandle (hFile) == FALSE){ demClientError(hFile, (CHAR)-1); return; } setCF(0); return; } /* demCreateFCB - An FCB is being created get the NT handle. * * Entry - Client (AL) Creation Mode * 00 - Normal File * 01 - Read-only file * 02 - Hidden File * 04 - System file * Client (DS:SI) Full path filename * Client (ES:DI) SFT address * * Exit - SUCCESS * Client (CF) = 0 * Client (AX:BP) = NT Handle * Client (BX) = Time * Client (CX) = Date * Client (DX:SI) = Size * * FAILURE * Client(CF) = 1 * Client(AX) = error code */ VOID demCreateFCB (VOID) { demFCBCommon (CREATE_ALWAYS); return; } /* demDate16 - Get the current date/time in DOS FCB format. * * Entry - None * * Exit - Always Success * Client (AX) has date * Client (DX) has time * NOTES: * * DemDate16 returns the current date in AX, current time in DX in this format * AX - YYYYYYYMMMMDDDDD years months days * DX - HHHHHMMMMMMSSSSS hours minutes seconds/2 */ VOID demDate16 (VOID) { SYSTEMTIME TimeDate; GetLocalTime(&TimeDate); // date is stored in a packed word: ((year-1980)*512) + (month*32) + day setAX ( (USHORT) (((TimeDate.wYear-1980) << 9 ) | ((TimeDate.wMonth & 0xf) << 5 ) | (TimeDate.wDay & 0x1f)) ); setDX ( (USHORT) ((TimeDate.wHour << 11) | ((TimeDate.wMinute & 0x3f) << 5) | ((TimeDate.wSecond / 2) & 0x1f)) ); return; } /* demFCBIO - Carry out the FCB based IO operation. * * Entry - Client (BX) = 1 if read operation, 0 if write * Client (AX:BP) NT Handle * Client (DI:DX) offset to start the operation with * Client (CX) Count of bytes * * Exit - SUCCESS * Client (CF) = 0 * Client (CX) = counts of bytes read/written * Client (AX:BX) = size * * FAILURE * Client(CF) = 1 * Client(AX) = error code */ VOID demFCBIO (VOID) { HANDLE hFile; ULONG CurOffset; PVOID pBuf; DWORD dwBytesIO=0; DWORD dwSize,dwSizeHigh; DWORD dwErrCode; hFile = GETHANDLE (getAX(),getBP()); CurOffset = (((ULONG)getDI()) << 16) + (ULONG)getDX(); if (DPM_SetFilePointer (hFile, (LONG)CurOffset, NULL, (DWORD)FILE_BEGIN) == -1L){ demClientError(hFile, (CHAR)-1); return ; } pBuf = (PVOID)GetVDMAddr(*((PUSHORT)pulDTALocation + 1), *((PUSHORT)pulDTALocation)); if(getBX()) { // Read Operation if (DPM_ReadFile (hFile, pBuf, (DWORD)getCX(), &dwBytesIO, NULL) == FALSE){ Sim32FlushVDMPointer(*pulDTALocation, getCX(), pBuf, FALSE); Sim32FreeVDMPointer(*pulDTALocation, getCX(), pBuf, FALSE); demClientError(hFile, (CHAR)-1); return ; } Sim32FlushVDMPointer (*pulDTALocation, getCX(),pBuf, FALSE); Sim32FreeVDMPointer (*pulDTALocation, getCX(), pBuf, FALSE); } else { if (getCX() == 0) { //0 byte write, adjust file size if(!DPM_SetEndOfFile(hFile)) { dwErrCode = GetLastError(); SetLastError(dwErrCode); demClientError(hFile,(CHAR)-1); return; } } else if (DPM_WriteFile (hFile, pBuf, (DWORD)getCX(), &dwBytesIO, NULL) == FALSE) { // If disk is full then we should return number of bytes written // AX = 1 and CF = 1 dwErrCode = GetLastError(); if(dwErrCode == ERROR_DISK_FULL) { setCX( (USHORT) dwBytesIO); setAX(1); setCF(1); return; } SetLastError(dwErrCode); demClientError(hFile, (CHAR)-1); return ; } } // Get File Size if((dwSize = DPM_GetFileSize(hFile,&dwSizeHigh)) == -1){ demPrintMsg(MSG_FILEINFO); ASSERT(FALSE); demClientError(hFile, (CHAR)-1); return; } if(dwSizeHigh) { demPrintMsg(MSG_FILESIZE_TOOBIG); ASSERT(FALSE); demClientError(hFile, (CHAR)-1); return; } // Setup the exit registers setCX((USHORT)dwBytesIO); setBX((USHORT)dwSize); setAX((USHORT)(dwSize >> 16 )); setCF(0); return; } /* demGetFileInfo - Get Misc. file info in FCB format. * * Entry - Client (DS:SI) full path file name * * Exit - SUCCESS * Client (CF) = 0 * Client (AX) = Attribute of file * Client (CX) = Time stamp of file * Client (DX = Date stamp of file * Client (BX:DI)= Size of file (32 bit) * * FAILURE * Client(CF) = 1 * Client(AX) = error code */ VOID demGetFileInfo (VOID) { HANDLE hFile; LPSTR lpFileName; WORD wDate,wTime; DWORD dwSize,dwAttr; lpFileName = (LPSTR) GetVDMAddr (getDS(),getSI()); if ((hFile = CreateFileOem(lpFileName, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL)) == (HANDLE)-1){ demClientError(INVALID_HANDLE_VALUE, *lpFileName); return; } // Get Misc. INfo if (demGetMiscInfo (hFile,&wTime, &wDate, &dwSize) == FALSE) { DPM_CloseHandle (hFile); return; } DPM_CloseHandle (hFile); if ((dwAttr = GetFileAttributesOemSys (lpFileName, FALSE)) == -1) { demClientError(INVALID_HANDLE_VALUE, *lpFileName); return; } if (dwAttr == FILE_ATTRIBUTE_NORMAL) dwAttr = 0; setAX((USHORT)dwAttr); setCX(wTime); setDX(wDate); setDI((USHORT)dwSize); setBX((USHORT)(dwSize >> 16)); return; } /* demOpenFCB - An FCB is being opened get the NT handle. * * Entry - Client (AL) Open Mode * Client (DS:SI) Full path filename * * Exit - SUCCESS * Client (CF) = 0 * Client (AX:BP) = NT Handle * Client (BX) = Time * Client (CX) = Date * Client (DX:SI) = Size * * FAILURE * Client(CF) = 1 * Client(AX) = error code */ VOID demOpenFCB (VOID) { demFCBCommon (OPEN_EXISTING); return; } /* demFCBCommon - FCB Open/Create. * * Entry - CreateDirective - Open/Create * Client (AL) Open Mode * Client (DS:SI) Full path filename * * Exit - SUCCESS * Client (CF) = 0 * Client (AX:BP) = NT Handle * Client (BX) = Time * Client (CX) = Date * Client (DX:SI) = Size * * FAILURE * Client(CF) = 1 * Client(AX) = error code */ VOID demFCBCommon (ULONG CreateDirective) { HANDLE hFile; LPSTR lpFileName; UCHAR uchMode,uchAccess; DWORD dwDesiredAccess = GENERIC_WRITE | GENERIC_READ; DWORD dwShareMode = FILE_SHARE_READ | FILE_SHARE_WRITE; WORD wDate,wTime; DWORD dwSize,dwAttr=0; USHORT uErr; SECURITY_ATTRIBUTES sa; lpFileName = (LPSTR) GetVDMAddr (getDS(),getSI()); uchMode = getAL(); /* Special case for delete volume label (INT 21 Func 13H, Attr = 8H */ if((uchMode == ATTR_VOLUME_ID) && (CreateDirective == CREATE_ALWAYS)) { if((uErr = demCreateLabel(lpFileName[DRIVEBYTE], lpFileName+LABELOFF))) { setCF(1); setAX(uErr); return; } setAX(0); setBP(0); setCF(0); return; } // In create case AL has creation attributes. By default // Access is for read/write and sharing for both. In open // case AL has appropriate access and sharing information. if((CreateDirective == CREATE_ALWAYS) && ((uchMode &0xff) == 0)) { dwAttr = FILE_ATTRIBUTE_NORMAL; dwShareMode = FILE_SHARE_WRITE | FILE_SHARE_READ; } else { uchAccess = uchMode & (UCHAR)ACCESS_MASK; if (uchAccess == OPEN_FOR_READ) dwDesiredAccess = GENERIC_READ; else if (uchAccess == OPEN_FOR_WRITE) dwDesiredAccess = GENERIC_WRITE; uchMode = uchMode & (UCHAR)SHARING_MASK; switch (uchMode) { case SHARING_DENY_BOTH: dwShareMode = 0; break; case SHARING_DENY_WRITE: dwShareMode = FILE_SHARE_READ; break; case SHARING_DENY_READ: dwShareMode = FILE_SHARE_WRITE; break; } } sa.nLength = sizeof (SECURITY_ATTRIBUTES); sa.lpSecurityDescriptor = NULL; sa.bInheritHandle = TRUE; if ((hFile = CreateFileOem(lpFileName, dwDesiredAccess, dwShareMode | FILE_SHARE_DELETE, &sa, CreateDirective, dwAttr, NULL)) == (HANDLE)-1){ demClientError(INVALID_HANDLE_VALUE, *lpFileName); return; } // Get Misc. INfo if (demGetMiscInfo (hFile,&wTime, &wDate, &dwSize) == FALSE) return; // Setup the exit registers setBX(wTime); setCX(wDate); setBP((USHORT)hFile); setAX((USHORT)((ULONG)hFile >> 16)); setSI((USHORT)dwSize); setDX((USHORT)(dwSize >> 16)); setCF(0); return; } BOOL demGetMiscInfo (hFile, lpTime, lpDate, lpSize) HANDLE hFile; LPWORD lpTime; LPWORD lpDate; LPDWORD lpSize; { FILETIME LastWriteTime,ftLocal; DWORD dwSizeHigh=0; if(GetFileTime (hFile,NULL,NULL,&LastWriteTime) == -1){ demPrintMsg(MSG_FILEINFO); ASSERT(FALSE); demClientError(hFile, (CHAR)-1); DPM_CloseHandle (hFile); return FALSE; } FileTimeToLocalFileTime (&LastWriteTime,&ftLocal); if(FileTimeToDosDateTime(&ftLocal, lpDate, lpTime) == FALSE){ demPrintMsg(MSG_FILEINFO); ASSERT(FALSE); demClientError(hFile, (CHAR)-1); return FALSE; } if((*lpSize = DPM_GetFileSize(hFile,&dwSizeHigh)) == -1){ demPrintMsg(MSG_FILEINFO); ASSERT(FALSE); demClientError(hFile, (CHAR)-1); return FALSE; } if(dwSizeHigh) { demPrintMsg(MSG_FILESIZE_TOOBIG); ASSERT(FALSE); demClientError(hFile, (CHAR)-1); return FALSE; } return TRUE; }