If using the default values, also set dwType to REG_SZ as our default
[wine] / dlls / winedos / int26.c
1 /*
2  * DOS interrupt 26h handler
3  *
4  * Copyright 1997 Andreas Mohr
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19  */
20
21 #include "config.h"
22
23 #include <stdlib.h>
24 #include <fcntl.h>
25 #ifdef HAVE_UNISTD_H
26 # include <unistd.h>
27 #endif
28 #include "dosexe.h"
29 #include "drive.h"
30 #include "wine/debug.h"
31
32 WINE_DEFAULT_DEBUG_CHANNEL(int);
33
34
35 /***********************************************************************
36  *           DOSVM_RawWrite
37  *
38  * Write raw sectors to a device.
39  */
40 BOOL DOSVM_RawWrite(BYTE drive, DWORD begin, DWORD nr_sect, BYTE *dataptr, BOOL fake_success)
41 {
42     int fd;
43
44     if ((fd = DRIVE_OpenDevice( drive, O_RDONLY )) != -1)
45     {
46         lseek( fd, begin * 512, SEEK_SET );
47         /* FIXME: check errors */
48         write( fd, dataptr, nr_sect * 512 );
49         close( fd );
50     }
51     else if (!fake_success)
52         return FALSE;
53
54     return TRUE;
55 }
56
57
58 /**********************************************************************
59  *          DOSVM_Int26Handler (WINEDOS16.138)
60  *
61  * Handler for int 26h (absolute disk read).
62  */
63 void WINAPI DOSVM_Int26Handler( CONTEXT86 *context )
64 {
65     WCHAR drivespec[4] = {'A', ':', '\\', 0};
66     BYTE *dataptr = CTX_SEG_OFF_TO_LIN( context, context->SegDs, context->Ebx );
67     DWORD begin;
68     DWORD length;
69
70     drivespec[0] += AL_reg( context );
71
72     if (GetDriveTypeW( drivespec ) == DRIVE_NO_ROOT_DIR || 
73         GetDriveTypeW( drivespec ) == DRIVE_UNKNOWN)
74     {
75         SET_CFLAG( context );
76         SET_AX( context, 0x0201 ); /* unknown unit */
77         return;
78     }
79
80     if (CX_reg( context ) == 0xffff)
81     {
82         begin   = *(DWORD *)dataptr;
83         length  = *(WORD *)(dataptr + 4);
84         dataptr = (BYTE *)CTX_SEG_OFF_TO_LIN( context,
85                                               *(WORD *)(dataptr + 8), 
86                                               *(DWORD *)(dataptr + 6) );
87     }
88     else
89     {
90         begin  = DX_reg( context );
91         length = CX_reg( context );
92     }
93
94     TRACE( "abs diskwrite, drive %d, sector %ld, "
95            "count %ld, buffer %p\n",
96            AL_reg( context ), begin, length, dataptr );
97
98     DOSVM_RawWrite( AL_reg( context ), begin, length, dataptr, TRUE );
99     RESET_CFLAG( context );
100 }