| /* | 
 |  * DOS interrupt 25h handler | 
 |  * | 
 |  * Copyright 1997 Andreas Mohr | 
 |  * | 
 |  * 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 <stdlib.h> | 
 | #include <stdio.h> | 
 | #include <string.h> | 
 | #include <fcntl.h> | 
 | #ifdef HAVE_UNISTD_H | 
 | # include <unistd.h> | 
 | #endif | 
 | #include "dosexe.h" | 
 | #include "wine/debug.h" | 
 |  | 
 | WINE_DEFAULT_DEBUG_CHANNEL(int); | 
 |  | 
 |  | 
 | /*********************************************************************** | 
 |  *           DOSVM_RawRead | 
 |  * | 
 |  * Read raw sectors from a device. | 
 |  */ | 
 | BOOL DOSVM_RawRead(BYTE drive, DWORD begin, DWORD nr_sect, BYTE *dataptr, BOOL fake_success) | 
 | {     | 
 |     WCHAR root[] = {'\\','\\','.','\\','A',':',0}; | 
 |     HANDLE h; | 
 |  | 
 |     TRACE( "abs diskread, drive %d, sector %ld, " | 
 |            "count %ld, buffer %p\n", | 
 |            drive, begin, nr_sect, dataptr ); | 
 |  | 
 |     root[4] += drive; | 
 |     h = CreateFileW(root, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, | 
 |                     FILE_FLAG_BACKUP_SEMANTICS, NULL); | 
 |     if (h != INVALID_HANDLE_VALUE) | 
 |     { | 
 |         SetFilePointer(h, begin * 512, NULL, SEEK_SET ); | 
 |         /* FIXME: check errors */ | 
 |         ReadFile(h, dataptr, nr_sect * 512, NULL, NULL ); | 
 |         CloseHandle(h); | 
 |     } | 
 |     else | 
 |     { | 
 |         memset( dataptr, 0, nr_sect * 512 ); | 
 |         if (fake_success) | 
 |         { | 
 |             /* FIXME: explain what happens here */ | 
 |             if (begin == 0 && nr_sect > 1) *(dataptr + 512) = 0xf8; | 
 |             if (begin == 1) *dataptr = 0xf8; | 
 |         } | 
 |         else | 
 |             return FALSE; | 
 |     } | 
 |  | 
 |     return TRUE; | 
 | } | 
 |  | 
 |  | 
 | /********************************************************************** | 
 |  *	    DOSVM_Int25Handler (WINEDOS16.137) | 
 |  * | 
 |  * Handler for int 25h (absolute disk read). | 
 |  */ | 
 | void WINAPI DOSVM_Int25Handler( CONTEXT86 *context ) | 
 | { | 
 |     WCHAR drivespec[4] = {'A', ':', '\\', 0}; | 
 |     BYTE *dataptr = CTX_SEG_OFF_TO_LIN( context, context->SegDs, context->Ebx ); | 
 |     DWORD begin; | 
 |     DWORD length; | 
 |  | 
 |     drivespec[0] += AL_reg( context ); | 
 |  | 
 |     if (GetDriveTypeW( drivespec ) == DRIVE_NO_ROOT_DIR ||  | 
 |         GetDriveTypeW( drivespec ) == DRIVE_UNKNOWN) | 
 |     { | 
 |         SET_CFLAG( context ); | 
 |         SET_AX( context, 0x0201 ); /* unknown unit */ | 
 |         return; | 
 |     } | 
 |  | 
 |     if (CX_reg( context ) == 0xffff) | 
 |     { | 
 |         begin   = *(DWORD *)dataptr; | 
 |         length  = *(WORD *)(dataptr + 4); | 
 |         dataptr = (BYTE *)CTX_SEG_OFF_TO_LIN( context, | 
 |                                               *(WORD *)(dataptr + 8),  | 
 |                                               *(DWORD *)(dataptr + 6) ); | 
 |     } | 
 |     else | 
 |     { | 
 |         begin  = DX_reg( context ); | 
 |         length = CX_reg( context ); | 
 |     } | 
 |  | 
 |     DOSVM_RawRead( AL_reg( context ), begin, length, dataptr, TRUE ); | 
 |     RESET_CFLAG( context ); | 
 | } |