2000-05-06 21:34:20 +02:00
|
|
|
/****************************************************************/
|
|
|
|
/* */
|
|
|
|
/* ioctl.c */
|
|
|
|
/* */
|
|
|
|
/* DOS-C ioctl system call */
|
|
|
|
/* */
|
|
|
|
/* Copyright (c) 1995,1998 */
|
|
|
|
/* Pasquale J. Villani */
|
|
|
|
/* All Rights Reserved */
|
|
|
|
/* */
|
|
|
|
/* This file is part of DOS-C. */
|
|
|
|
/* */
|
|
|
|
/* DOS-C is free software; you can redistribute it and/or */
|
|
|
|
/* modify it under the terms of the GNU General Public License */
|
|
|
|
/* as published by the Free Software Foundation; either version */
|
|
|
|
/* 2, or (at your option) any later version. */
|
|
|
|
/* */
|
|
|
|
/* DOS-C 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 General Public License for more details. */
|
|
|
|
/* */
|
|
|
|
/* You should have received a copy of the GNU General Public */
|
|
|
|
/* License along with DOS-C; see the file COPYING. If not, */
|
|
|
|
/* write to the Free Software Foundation, 675 Mass Ave, */
|
|
|
|
/* Cambridge, MA 02139, USA. */
|
|
|
|
/****************************************************************/
|
|
|
|
|
|
|
|
#include "portab.h"
|
|
|
|
#include "globals.h"
|
|
|
|
|
|
|
|
#ifdef VERSION_STRINGS
|
|
|
|
static BYTE *RcsId = "$Id$";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* $Log$
|
2000-06-21 20:16:46 +02:00
|
|
|
* Revision 1.6 2000/06/21 18:16:46 jimtabor
|
|
|
|
* Add UMB code, patch, and code fixes
|
|
|
|
*
|
2000-05-25 22:56:23 +02:00
|
|
|
* Revision 1.5 2000/05/25 20:56:21 jimtabor
|
|
|
|
* Fixed project history
|
|
|
|
*
|
2000-05-17 21:15:12 +02:00
|
|
|
* Revision 1.4 2000/05/17 19:15:12 jimtabor
|
|
|
|
* Cleanup, add and fix source.
|
|
|
|
*
|
2000-05-11 06:26:26 +02:00
|
|
|
* Revision 1.3 2000/05/11 04:26:26 jimtabor
|
|
|
|
* Added code for DOS FN 69 & 6C
|
|
|
|
*
|
2000-05-08 06:30:00 +02:00
|
|
|
* Revision 1.2 2000/05/08 04:30:00 jimtabor
|
|
|
|
* Update CVS to 2020
|
|
|
|
*
|
2000-05-25 22:56:23 +02:00
|
|
|
* Revision 1.1.1.1 2000/05/06 19:34:53 jhall1
|
|
|
|
* The FreeDOS Kernel. A DOS kernel that aims to be 100% compatible with
|
|
|
|
* MS-DOS. Distributed under the GNU GPL.
|
2000-05-11 06:26:26 +02:00
|
|
|
*
|
2000-05-08 06:30:00 +02:00
|
|
|
* Revision 1.4 2000/04/29 05:13:16 jtabor
|
|
|
|
* Added new functions and clean up code
|
2000-05-06 21:34:20 +02:00
|
|
|
*
|
|
|
|
* Revision 1.3 2000/03/09 06:07:11 kernel
|
|
|
|
* 2017f updates by James Tabor
|
|
|
|
*
|
|
|
|
* Revision 1.2 1999/04/04 18:51:43 jprice
|
|
|
|
* no message
|
|
|
|
*
|
|
|
|
* Revision 1.1.1.1 1999/03/29 15:41:09 jprice
|
|
|
|
* New version without IPL.SYS
|
|
|
|
*
|
|
|
|
* Revision 1.4 1999/02/08 05:55:57 jprice
|
|
|
|
* Added Pat's 1937 kernel patches
|
|
|
|
*
|
|
|
|
* Revision 1.3 1999/02/01 01:48:41 jprice
|
|
|
|
* Clean up; Now you can use hex numbers in config.sys. added config.sys screen function to change screen mode (28 or 43/50 lines)
|
|
|
|
*
|
|
|
|
* Revision 1.2 1999/01/22 04:13:26 jprice
|
|
|
|
* Formating
|
|
|
|
*
|
|
|
|
* Revision 1.1.1.1 1999/01/20 05:51:01 jprice
|
|
|
|
* Imported sources
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* Rev 1.7 06 Dec 1998 8:48:22 patv
|
|
|
|
* Expanded due to new I/O subsystem.
|
|
|
|
*
|
|
|
|
* Rev 1.6 11 Jan 1998 2:06:22 patv
|
|
|
|
* Added functionality to ioctl.
|
|
|
|
*
|
|
|
|
* Rev 1.5 04 Jan 1998 23:15:18 patv
|
|
|
|
* Changed Log for strip utility
|
|
|
|
*
|
|
|
|
* Rev 1.4 16 Jan 1997 12:46:54 patv
|
|
|
|
* pre-Release 0.92 feature additions
|
|
|
|
*
|
|
|
|
* Rev 1.3 29 May 1996 21:03:30 patv
|
|
|
|
* bug fixes for v0.91a
|
|
|
|
*
|
|
|
|
* Rev 1.2 19 Feb 1996 3:21:34 patv
|
|
|
|
* Added NLS, int2f and config.sys processing
|
|
|
|
*
|
|
|
|
* Rev 1.1 01 Sep 1995 17:54:16 patv
|
|
|
|
* First GPL release.
|
|
|
|
*
|
|
|
|
* Rev 1.0 02 Jul 1995 8:32:04 patv
|
|
|
|
* Initial revision.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef PROTO
|
|
|
|
sft FAR *get_sft(COUNT);
|
|
|
|
#else
|
|
|
|
sft FAR *get_sft();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* WARNING: this code is non-portable (8086 specific).
|
|
|
|
*/
|
|
|
|
|
|
|
|
COUNT DosDevIOctl(iregs FAR * r, COUNT FAR * err)
|
|
|
|
{
|
|
|
|
sft FAR *s;
|
|
|
|
struct dpb FAR *dpbp;
|
2000-05-08 06:30:00 +02:00
|
|
|
struct cds FAR *cdsp;
|
2000-05-06 21:34:20 +02:00
|
|
|
BYTE FAR *pBuffer = MK_FP(r->DS, r->DX);
|
2000-06-21 20:16:46 +02:00
|
|
|
COUNT nMode , dev;
|
2000-05-06 21:34:20 +02:00
|
|
|
|
|
|
|
/* Test that the handle is valid */
|
|
|
|
switch (r->AL)
|
|
|
|
{
|
|
|
|
case 0x00:
|
|
|
|
case 0x01:
|
|
|
|
case 0x02:
|
|
|
|
case 0x03:
|
|
|
|
case 0x06:
|
|
|
|
case 0x07:
|
|
|
|
case 0x0a:
|
|
|
|
case 0x0c:
|
|
|
|
|
|
|
|
/* Get the SFT block that contains the SFT */
|
|
|
|
if ((s = get_sft(r->BX)) == (sft FAR *) - 1)
|
|
|
|
{
|
|
|
|
*err = DE_INVLDHNDL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x04:
|
|
|
|
case 0x05:
|
|
|
|
case 0x08:
|
|
|
|
case 0x09:
|
|
|
|
case 0x0d:
|
|
|
|
case 0x0e:
|
|
|
|
case 0x0f:
|
|
|
|
case 0x10:
|
|
|
|
case 0x11:
|
2000-05-08 06:30:00 +02:00
|
|
|
|
2000-05-06 21:34:20 +02:00
|
|
|
/*
|
|
|
|
This line previously returned the deviceheader at r->bl. But,
|
|
|
|
DOS numbers its drives starting at 1, not 0. A=1, B=2, and so
|
|
|
|
on. Changed this line so it is now zero-based.
|
|
|
|
|
|
|
|
-SRM
|
|
|
|
*/
|
|
|
|
/* JPP - changed to use default drive if drive=0 */
|
2000-05-08 06:30:00 +02:00
|
|
|
/* JT Fixed it */
|
|
|
|
|
2000-06-21 20:16:46 +02:00
|
|
|
dev = ( r->BL == 0 ? default_drive : r->BL - 1);
|
2000-05-11 06:26:26 +02:00
|
|
|
|
2000-06-21 20:16:46 +02:00
|
|
|
if (dev > (lastdrive -1))
|
2000-05-08 06:30:00 +02:00
|
|
|
{
|
|
|
|
*err = DE_INVLDDRV;
|
|
|
|
return 0;
|
2000-05-06 21:34:20 +02:00
|
|
|
}
|
2000-05-08 06:30:00 +02:00
|
|
|
else
|
2000-05-11 06:26:26 +02:00
|
|
|
{
|
2000-06-21 20:16:46 +02:00
|
|
|
cdsp = &CDSp->cds_table[dev];
|
2000-05-08 06:30:00 +02:00
|
|
|
dpbp = cdsp->cdsDpb;
|
2000-05-11 06:26:26 +02:00
|
|
|
}
|
2000-05-06 21:34:20 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x0b:
|
|
|
|
/* skip, it's a special case. */
|
|
|
|
|
|
|
|
NetDelay = r->CX;
|
|
|
|
if (!r->DX)
|
|
|
|
NetRetry = r->DX;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
*err = DE_INVLDFUNC;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (r->AL)
|
|
|
|
{
|
|
|
|
case 0x00:
|
|
|
|
/* Get the flags from the SFT */
|
2000-06-21 20:16:46 +02:00
|
|
|
r->AX = s->sft_dev->dh_attr;
|
|
|
|
r->DH = r->AH;
|
|
|
|
/* Undocumented result, Ax = Dx seen using Pcwatch */
|
|
|
|
r->DL = r->AL = s->sft_flags;
|
2000-05-06 21:34:20 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x01:
|
|
|
|
/* sft_flags is a file, return an error because you */
|
|
|
|
/* can't set the status of a file. */
|
2000-05-11 06:26:26 +02:00
|
|
|
if (!(s->sft_flags & SFT_FDEVICE))
|
2000-05-06 21:34:20 +02:00
|
|
|
{
|
|
|
|
*err = DE_INVLDFUNC;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set it to what we got in the DL register from the */
|
|
|
|
/* user. */
|
|
|
|
r->AL = s->sft_flags_lo = SFT_FDEVICE | r->DL;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x0c:
|
|
|
|
nMode = C_GENIOCTL;
|
|
|
|
goto IoCharCommon;
|
|
|
|
case 0x02:
|
|
|
|
nMode = C_IOCTLIN;
|
|
|
|
goto IoCharCommon;
|
|
|
|
case 0x10:
|
|
|
|
nMode = C_IOCTLQRY;
|
|
|
|
goto IoCharCommon;
|
|
|
|
case 0x03:
|
|
|
|
nMode = C_IOCTLOUT;
|
|
|
|
IoCharCommon:
|
|
|
|
if ((s->sft_flags & SFT_FDEVICE)
|
2000-05-11 06:26:26 +02:00
|
|
|
|| ((r->AL == 0x02 ) && (s->sft_dev->dh_attr & SFT_FIOCTL))
|
|
|
|
|| ((r->AL == 0x03 ) && (s->sft_dev->dh_attr & SFT_FIOCTL))
|
|
|
|
|| ((r->AL == 0x10) && (s->sft_dev->dh_attr & ATTR_QRYIOCTL))
|
|
|
|
|| ((r->AL == 0x0c) && (s->sft_dev->dh_attr & ATTR_GENIOCTL)))
|
2000-05-06 21:34:20 +02:00
|
|
|
{
|
|
|
|
CharReqHdr.r_unit = 0;
|
|
|
|
CharReqHdr.r_length = sizeof(request);
|
|
|
|
CharReqHdr.r_command = nMode;
|
|
|
|
CharReqHdr.r_count = r->CX;
|
|
|
|
CharReqHdr.r_trans = pBuffer;
|
|
|
|
CharReqHdr.r_status = 0;
|
|
|
|
execrh((request FAR *) & CharReqHdr, s->sft_dev);
|
|
|
|
|
|
|
|
if (CharReqHdr.r_status & S_ERROR)
|
2000-06-21 20:16:46 +02:00
|
|
|
{
|
|
|
|
*err = DE_DEVICE;
|
|
|
|
return 0;
|
|
|
|
}
|
2000-05-06 21:34:20 +02:00
|
|
|
if (r->AL == 0x07)
|
|
|
|
{
|
2000-06-21 20:16:46 +02:00
|
|
|
r->AL = CharReqHdr.r_status & S_BUSY ? 00 : 0xff;
|
|
|
|
|
|
|
|
}
|
|
|
|
if (r->AL == 0x02 || r->AL == 0x03)
|
|
|
|
{
|
|
|
|
r->AX = CharReqHdr.r_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (r->AL == 0x0c || r->AL == 0x10)
|
|
|
|
{
|
|
|
|
r->AX = CharReqHdr.r_status;
|
2000-05-06 21:34:20 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
*err = DE_INVLDFUNC;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case 0x0d:
|
|
|
|
nMode = C_GENIOCTL;
|
|
|
|
goto IoBlockCommon;
|
|
|
|
case 0x04:
|
|
|
|
nMode = C_IOCTLIN;
|
|
|
|
goto IoBlockCommon;
|
|
|
|
case 0x11:
|
|
|
|
nMode = C_IOCTLQRY;
|
|
|
|
goto IoBlockCommon;
|
|
|
|
case 0x05:
|
|
|
|
nMode = C_IOCTLOUT;
|
|
|
|
IoBlockCommon:
|
2000-06-21 20:16:46 +02:00
|
|
|
if(!dpbp)
|
|
|
|
{
|
|
|
|
*err = DE_INVLDDRV;
|
|
|
|
return 0;
|
|
|
|
}
|
2000-05-11 06:26:26 +02:00
|
|
|
if ( ((r->AL == 0x04 ) && !(dpbp->dpb_device->dh_attr & ATTR_IOCTL))
|
|
|
|
|| ((r->AL == 0x05 ) && !(dpbp->dpb_device->dh_attr & ATTR_IOCTL))
|
|
|
|
|| ((r->AL == 0x11) && !(dpbp->dpb_device->dh_attr & ATTR_QRYIOCTL))
|
|
|
|
|| ((r->AL == 0x0d) && !(dpbp->dpb_device->dh_attr & ATTR_GENIOCTL)))
|
2000-05-06 21:34:20 +02:00
|
|
|
{
|
|
|
|
*err = DE_INVLDFUNC;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2000-06-21 20:16:46 +02:00
|
|
|
|
|
|
|
CharReqHdr.r_unit = dev;
|
2000-05-06 21:34:20 +02:00
|
|
|
CharReqHdr.r_length = sizeof(request);
|
|
|
|
CharReqHdr.r_command = nMode;
|
|
|
|
CharReqHdr.r_count = r->CX;
|
|
|
|
CharReqHdr.r_trans = pBuffer;
|
|
|
|
CharReqHdr.r_status = 0;
|
|
|
|
execrh((request FAR *) & CharReqHdr,
|
|
|
|
dpbp->dpb_device);
|
2000-06-21 20:16:46 +02:00
|
|
|
|
2000-05-06 21:34:20 +02:00
|
|
|
if (CharReqHdr.r_status & S_ERROR)
|
|
|
|
{
|
2000-06-21 20:16:46 +02:00
|
|
|
*err = DE_DEVICE;
|
|
|
|
return 0;
|
2000-05-06 21:34:20 +02:00
|
|
|
}
|
2000-06-21 20:16:46 +02:00
|
|
|
if (r->AL == 0x08)
|
2000-05-06 21:34:20 +02:00
|
|
|
{
|
2000-06-21 20:16:46 +02:00
|
|
|
r->AX = (CharReqHdr.r_status & S_BUSY) ? 1 : 0;
|
|
|
|
|
2000-05-06 21:34:20 +02:00
|
|
|
}
|
2000-06-21 20:16:46 +02:00
|
|
|
|
|
|
|
if (r->AL == 0x04 || r->AL == 0x05)
|
|
|
|
{
|
|
|
|
r->AX = CharReqHdr.r_count;
|
|
|
|
|
|
|
|
}
|
|
|
|
if (r->AL == 0x0d || r->AL == 0x11)
|
|
|
|
{
|
|
|
|
r->AX = CharReqHdr.r_status;
|
|
|
|
}
|
|
|
|
break;
|
2000-05-06 21:34:20 +02:00
|
|
|
|
|
|
|
case 0x06:
|
|
|
|
if (s->sft_flags & SFT_FDEVICE)
|
|
|
|
{
|
|
|
|
r->AL = s->sft_flags & SFT_FEOF ? 0 : 0xFF;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
r->AL = s->sft_posit >= s->sft_size ? 0xFF : 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x07:
|
|
|
|
if (s->sft_flags & SFT_FDEVICE)
|
|
|
|
{
|
2000-06-21 20:16:46 +02:00
|
|
|
nMode = C_OSTAT;
|
2000-05-06 21:34:20 +02:00
|
|
|
goto IoCharCommon;
|
|
|
|
}
|
|
|
|
r->AL = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x08:
|
2000-06-21 20:16:46 +02:00
|
|
|
if(!dpbp)
|
|
|
|
{
|
|
|
|
*err = DE_INVLDDRV;
|
|
|
|
return 0;
|
|
|
|
}
|
2000-05-06 21:34:20 +02:00
|
|
|
if (dpbp->dpb_device->dh_attr & ATTR_EXCALLS)
|
|
|
|
{
|
|
|
|
nMode = C_REMMEDIA;
|
|
|
|
goto IoBlockCommon;
|
|
|
|
}
|
|
|
|
*err = DE_INVLDFUNC;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case 0x09:
|
2000-05-08 06:30:00 +02:00
|
|
|
if(cdsp->cdsFlags & CDSNETWDRV)
|
2000-06-21 20:16:46 +02:00
|
|
|
{
|
|
|
|
r->DX = ATTR_REMOTE ;
|
|
|
|
r->AX = S_DONE|S_BUSY;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if(!dpbp)
|
|
|
|
{
|
|
|
|
*err = DE_INVLDDRV;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/* Need to add subst bit 15 */
|
|
|
|
r->DX = dpbp->dpb_device->dh_attr;
|
|
|
|
r->AX = S_DONE|S_BUSY;
|
|
|
|
}
|
2000-05-06 21:34:20 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x0a:
|
2000-06-21 20:16:46 +02:00
|
|
|
r->DX = s->sft_flags;
|
|
|
|
r->AX = 0;
|
2000-05-06 21:34:20 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x0e:
|
|
|
|
nMode = C_GETLDEV;
|
|
|
|
goto IoLogCommon;
|
|
|
|
case 0x0f:
|
|
|
|
nMode = C_SETLDEV;
|
|
|
|
IoLogCommon:
|
2000-06-21 20:16:46 +02:00
|
|
|
if(!dpbp)
|
|
|
|
{
|
|
|
|
*err = DE_INVLDDRV;
|
|
|
|
return 0;
|
|
|
|
}
|
2000-05-06 21:34:20 +02:00
|
|
|
if ((dpbp->dpb_device->dh_attr & ATTR_GENIOCTL))
|
|
|
|
{
|
|
|
|
|
2000-06-21 20:16:46 +02:00
|
|
|
|
|
|
|
CharReqHdr.r_unit = dev;
|
2000-05-06 21:34:20 +02:00
|
|
|
CharReqHdr.r_length = sizeof(request);
|
|
|
|
CharReqHdr.r_command = nMode;
|
|
|
|
CharReqHdr.r_count = r->CX;
|
|
|
|
CharReqHdr.r_trans = pBuffer;
|
|
|
|
CharReqHdr.r_status = 0;
|
|
|
|
execrh((request FAR *) & CharReqHdr,
|
|
|
|
dpbp->dpb_device);
|
|
|
|
|
|
|
|
if (CharReqHdr.r_status & S_ERROR)
|
|
|
|
*err = DE_ACCESS;
|
|
|
|
else
|
2000-06-21 20:16:46 +02:00
|
|
|
{
|
|
|
|
r->AL = CharReqHdr.r_unit;
|
|
|
|
*err = SUCCESS;
|
|
|
|
}
|
2000-05-06 21:34:20 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
*err = DE_INVLDFUNC;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
default:
|
|
|
|
*err = DE_INVLDFUNC;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
*err = SUCCESS;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|