2020-09-30 16:53:55 +02:00

554 lines
15 KiB
C

/* Installable drivers for windows. Less common code.
*/
#include "user.h"
/*--------------------------------------------------------------------------*\
**
** NewSignalProc() -
**
\*--------------------------------------------------------------------------*/
#define SG_EXIT 0x0020
#define SG_LOAD_DLL 0x0040
#define SG_EXIT_DLL 0x0080
#define SG_GP_FAULT 0x0666
BOOL
CALLBACK NewSignalProc(
HTASK hTask,
WORD message,
WPARAM wParam,
LPARAM lParam
)
{
BOOL fRet;
// Notify installable drivers this app is going away.
if ( message == SG_EXIT || message == SG_GP_FAULT ) {
InternalBroadcastDriverMessage( NULL, DRV_EXITAPPLICATION,
(message == SG_GP_FAULT
? DRVEA_ABNORMALEXIT
: DRVEA_NORMALEXIT),
0L, IBDM_FIRSTINSTANCEONLY );
}
//
// Pass notification on to WOW32 (which passes on to USER32)
//
fRet = SignalProc( hTask, message, wParam, lParam );
//
// After letting WOW32 and User32 cleanup, destroy the shadow
// message queue created by InitApp.
//
if ( message == SG_EXIT || message == SG_GP_FAULT ) {
DeleteQueue();
}
return fRet;
}
HINSTANCE LoadAliasedLibrary(LPCSTR szLibFileName,
LPCSTR szSection,
LPCSTR szIniFile,
LPSTR lpstrTail,
WORD cbTail)
{
char sz[128];
LPSTR pch;
HINSTANCE h;
WORD errMode;
if (!szLibFileName || !*szLibFileName)
return((HINSTANCE)2); /* File not found */
/* read the filename and additional info. into sz
*/
GetPrivateProfileString(szSection, // ini section
szLibFileName, // key name
szLibFileName, // default if no match
sz, // return buffer
sizeof(sz), // return buffer size
szIniFile); // ini. file
sz[sizeof(sz)-1] = 0;
/* strip off the additional info. Remember, DS!=SS so we need to get a lpstr
* to our stack allocated sz.
*/
pch = (LPSTR)sz;
while (*pch)
{
if (*pch == ' ')
{
*pch++ = '\0';
break;
}
pch++;
}
// pch pts to ch after first space or null ch
/* copy additional info. to lpstrTail
*/
if (lpstrTail && cbTail)
{
while (cbTail-- && (*lpstrTail++ = *pch++))
;
*(lpstrTail-1) = 0;
}
errMode = SetErrorMode(0x8001);
h = LoadLibrary(sz);
SetErrorMode(errMode);
return (h);
}
int GetDrvrUsage(HMODULE hModule)
/* effects: Runs through the driver list and figures out how many instances of
* this driver module handle we have. We use this instead of GetModuleUsage
* so that we can have drivers loaded as normal DLLs and as installable
* drivers.
*/
{
LPDRIVERTABLE lpdt;
int index;
int count;
if (!hInstalledDriverList || !cInstalledDrivers || !hModule)
return(0);
count = 0;
lpdt = (LPDRIVERTABLE)MAKELP(hInstalledDriverList,0);
for (index=0;index<cInstalledDrivers;index++)
{
if (lpdt->hModule==hModule)
count++;
lpdt++;
}
return(count);
}
BOOL PASCAL CheckValidDriverProc(LPDRIVERTABLE lpdt, HDRVR hdrv)
/* effects: Some vendors shipped multimedia style installable drivers with
* bogus entry procs. This test checks for these bogus drivers and refuses to
* install them.
*/
{
WORD currentSP;
WORD saveSP;
_asm mov saveSP, sp
(void)(lpdt->lpDriverEntryPoint)(0, hdrv, 0, 0L, 0L);
_asm mov currentSP, sp
_asm mov sp, saveSP
if (saveSP != currentSP)
DebugErr(DBF_ERROR, "Invalid driver entry proc address");
return (saveSP == currentSP);
}
LRESULT FAR InternalLoadDriver(LPCSTR szDriverName,
LPCSTR szSectionName,
LPCSTR lpstrTail,
WORD cbTail,
BOOL fSendEnable)
{
int index;
int i;
LPDRIVERTABLE lpdt;
LPDRIVERTABLE lpdtBegin;
LRESULT result;
HGLOBAL h;
HINSTANCE hInstance;
char szDrivers[20];
char szSystemIni[20];
/* Drivers receive the following messages: if the driver was loaded,
* DRV_LOAD. If DRV_LOAD returns non-zero and fSendEnable, DRV_ENABLE.
*/
if (!hInstalledDriverList)
h = GlobalAlloc(GHND|GMEM_SHARE, (DWORD)((WORD)sizeof(DRIVERTABLE)));
else
/* Alloc space for the next driver we will install. We may not really
* install the driver in the last slot but rather in an intermediate
* slot which was freed.
*/
h = GlobalReAlloc(hInstalledDriverList,
(DWORD)((WORD)sizeof(DRIVERTABLE)*(cInstalledDrivers+1)),
GHND|GMEM_SHARE);
if (!h)
return(0L);
cInstalledDrivers++;
hInstalledDriverList = h;
if (!szSectionName)
LoadString(hInstanceWin, STR_DRIVERS, szDrivers, sizeof(szDrivers));
LoadString(hInstanceWin, STR_SYSTEMINI, szSystemIni, sizeof(szSystemIni));
lpdtBegin = lpdt = (LPDRIVERTABLE)MAKELP(hInstalledDriverList, NULL);
/* Find an empty driver entry */
for (i = 0; i < cInstalledDrivers; i++)
{
if (lpdt->hModule == NULL)
{
index = i;
break;
}
lpdt++;
}
if (index + 1 < cInstalledDrivers)
/* The driver went into an unused slot in the middle somewhere so
* decrement cInstalledDrivers count.
*/
cInstalledDrivers--;
/* Temporarly use an hModule to 1 to reserve this entry in case the driver
* loads another driver in its LibMain.
*/
lpdt->hModule = (HMODULE)1;
hInstance = LoadAliasedLibrary((LPSTR)szDriverName,
(LPSTR)(szSectionName ? szSectionName : szDrivers),
szSystemIni,
(LPSTR)lpstrTail,
cbTail);
if (hInstance < HINSTANCE_ERROR)
{
lpdt->hModule = NULL;
/* Load failed with an error. Return error code in highword.
*/
return(MAKELRESULT(0, hInstance));
}
(FARPROC)lpdt->lpDriverEntryPoint = GetProcAddress(hInstance, "DriverProc");
if (!lpdt->lpDriverEntryPoint)
{
FreeLibrary(hInstance);
lpdt->hModule = 0;
result = 0L;
goto LoadCleanUp;
}
lpdt->hModule = hInstance;
/* Save either the alias or filename of this driver. (depends on what the
* app passed to us to load it)
*/
lstrcpy(lpdt->szAliasName, szDriverName);
if (GetDrvrUsage(hInstance) == 1)
{
/* If this is the first instance, send the drv_load message. Don't use
* SendDriverMessage because we haven't initialized the linked list yet
*/
if (!CheckValidDriverProc(lpdt, (HDRVR)(index+1)) ||
!(lpdt->lpDriverEntryPoint)(lpdt->dwDriverIdentifier,
(HDRVR)(index+1),
DRV_LOAD,
0L, 0L))
{
/* Driver failed load call.
*/
lpdt->lpDriverEntryPoint = NULL;
lpdt->hModule = NULL;
FreeLibrary(hInstance);
result = 0L;
goto LoadCleanUp;
}
lpdt->fFirstEntry = 1;
}
/* Put driver in the load order linked list
*/
if (idFirstDriver == -1)
{
/* Initialize everything when first driver is loaded.
*/
idFirstDriver = index;
idLastDriver = index;
lpdt->idNextDriver = -1;
lpdt->idPrevDriver = -1;
}
else
{
/* Insert this driver at the end of the load chain.
*/
lpdtBegin[idLastDriver].idNextDriver = index;
lpdt->idPrevDriver = idLastDriver;
lpdt->idNextDriver = -1;
idLastDriver = index;
}
if (fSendEnable && lpdt->fFirstEntry)
SendDriverMessage((HDRVR)(index+1), DRV_ENABLE, 0L, 0L);
result = MAKELRESULT(index+1, hInstance);
LoadCleanUp:
return(result);
}
WORD FAR InternalFreeDriver(HDRVR hDriver, BOOL fSendDisable)
{
LPDRIVERTABLE lpdt;
WORD w;
int id;
/* The driver will receive the following message sequence:
*
* if usage count of driver is 1
* DRV_DISABLE (normally)
* DRV_FREE
*/
if ((int)hDriver > cInstalledDrivers || !hDriver)
return(0);
lpdt = (LPDRIVERTABLE)MAKELP(hInstalledDriverList,0);
if (!lpdt[(int)hDriver-1].hModule)
return(0);
/* If the driver usage count is 1, then send free and disable messages.
*/
/* Clear dwDriverIdentifier so that the sendmessage for DRV_OPEN and
* DRV_ENABLE have dwDriverIdentifier = 0 if an entry gets reused and so
* that the DRV_DISABLE and DRV_FREE messages below also get
* dwDriverIdentifier = 0.
*/
lpdt[(int)hDriver-1].dwDriverIdentifier = 0;
w = GetDrvrUsage(lpdt[(int)hDriver-1].hModule);
if (w == 1)
{
if (fSendDisable)
SendDriverMessage(hDriver, DRV_DISABLE, 0L, 0L);
SendDriverMessage(hDriver, DRV_FREE, 0L, 0L);
}
FreeLibrary(lpdt[(int)hDriver-1].hModule);
// Clear the rest of the table entry
lpdt[(int)hDriver-1].hModule = 0; // this indicates free entry
lpdt[(int)hDriver-1].fFirstEntry = 0; // this is also just to be tidy
lpdt[(int)hDriver-1].lpDriverEntryPoint = 0; // this is also just to be tidy
/* Fix up the driver load linked list */
if (idFirstDriver == (int)hDriver-1)
{
idFirstDriver = lpdt[(int)hDriver-1].idNextDriver;
if (idFirstDriver == -1)
{
/* No more drivers in the chain */
idFirstDriver = -1;
idLastDriver = -1;
cInstalledDrivers= 0;
goto Done;
}
else
{
/* Make prev entry of new first driver -1 */
lpdt[idFirstDriver].idPrevDriver = -1;
}
}
else if (idLastDriver == (int)hDriver-1)
{
/* We are freeing the last driver. So find a new last driver. */
idLastDriver = lpdt[(int)hDriver-1].idPrevDriver;
lpdt[idLastDriver].idNextDriver = -1;
}
else
{
/* We are freeing a driver in the middle of the list somewhere. */
id = lpdt[(int)hDriver-1].idPrevDriver;
lpdt[id].idNextDriver = lpdt[(int)hDriver-1].idNextDriver;
id = lpdt[(int)hDriver-1].idNextDriver;
lpdt[id].idPrevDriver = lpdt[(int)hDriver-1].idPrevDriver;
}
Done:
return(w-1);
}
LRESULT InternalOpenDriver(LPCSTR szDriverName,
LPCSTR szSectionName,
LPARAM lParam2,
BOOL fSendEnable)
{
HDRVR hDriver;
LPDRIVERTABLE lpdt;
LRESULT result;
char sz[128];
if (hDriver = (HDRVR)LOWORD(InternalLoadDriver(szDriverName, szSectionName,
sz, sizeof(sz), fSendEnable)))
{
/* Set the driver identifier to the DRV_OPEN call to the driver
* handle. This will let people build helper functions that the driver
* can call with a unique identifier if they want to.
*/
lpdt = (LPDRIVERTABLE)MAKELP(hInstalledDriverList,0);
lpdt[(int)hDriver-1].dwDriverIdentifier = (DWORD)(WORD)hDriver;
result = SendDriverMessage(hDriver,
DRV_OPEN,
(LPARAM)(LPSTR)sz,
lParam2);
if (!result)
InternalFreeDriver(hDriver, fSendEnable);
else
{
lpdt = (LPDRIVERTABLE)MAKELONG(0,hInstalledDriverList);
lpdt[(int)hDriver-1].dwDriverIdentifier = (DWORD)result;
result = (LRESULT)(DWORD)(WORD)hDriver;
}
}
else
result = 0L;
return(result);
}
LRESULT InternalCloseDriver(HDRVR hDriver, LPARAM lParam1, LPARAM lParam2, BOOL fSendDisable)
{
LPDRIVERTABLE lpdt;
LRESULT result;
int index;
BOOL f;
HMODULE hm;
// check handle in valid range.
if ((int)hDriver > cInstalledDrivers)
return(FALSE);
lpdt = (LPDRIVERTABLE)MAKELP(hInstalledDriverList,0);
if (!lpdt[(int)hDriver-1].hModule)
return(FALSE);
result = SendDriverMessage(hDriver, DRV_CLOSE, lParam1, lParam2);
if (result)
{
// Driver didn't abort close
f = lpdt[(int)hDriver-1].fFirstEntry;
hm = lpdt[(int)hDriver-1].hModule;
if (InternalFreeDriver(hDriver, fSendDisable) && f)
{
lpdt = (LPDRIVERTABLE)MAKELP(hInstalledDriverList,0);
/* Only one entry for the driver in the driver list has the first
* instance flag set. This is to make it easier to handle system
* messages that only need to be sent to a driver once.
*
* To maintain the flag, we must set the flag in one of the other
* entries if we remove the driver entry with the flag set.
*
* Note that InternalFreeDriver returns the new usage count of
* the driver so if it is zero, we know that there are no other
* entries for the driver in the list and so we don't have to
* do this loop.
*/
for (index=0;index<cInstalledDrivers;index++)
if (lpdt[index].hModule == hm && !lpdt[index].fFirstEntry)
{
lpdt[index].fFirstEntry = 1;
break;
}
}
}
return(result);
}
HDRVR API IOpenDriver(LPCSTR szDriverName, LPCSTR szSectionName, LPARAM lParam)
{
LRESULT result;
/* The driver receives the following messages when it is opened. If it isn't
* loaded, the library is loaded and the DRV_LOAD message is sent. If
* DRV_LOAD returns nonzero, the DRV_ENABLE message is sent. Once the
* driver is loaded or if it was previously loaded, the DRV_OPEN message is
* sent.
*/
result = InternalOpenDriver(szDriverName, szSectionName, lParam, TRUE);
return((HDRVR)LOWORD(result));
}
LRESULT API ICloseDriver(HDRVR hDriver, LPARAM lParam1, LPARAM lParam2)
{
/* The driver will receive the following message sequence:
*
* DRV_CLOSE
* if DRV_CLOSE returns non-zero
* if driver usage count = 1
* DRV_DISABLE
* DRV_FREE
*/
return(InternalCloseDriver(hDriver, lParam1, lParam2, TRUE));
}
HINSTANCE API IGetDriverModuleHandle(HDRVR hDriver)
/* effects: Returns the module handle associated with the given driver ID.
*/
{
LPDRIVERTABLE lpdt;
HINSTANCE hModule = NULL;
if (hDriver && ((int)hDriver <= cInstalledDrivers))
{
lpdt = (LPDRIVERTABLE)MAKELP(hInstalledDriverList,0);
return lpdt[(int)hDriver-1].hModule;
}
else
return NULL;
}