You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
537 lines
17 KiB
537 lines
17 KiB
//
|
|
// TheApp.cpp
|
|
//
|
|
// 16-bit code to install network components such as TCP/IP.
|
|
//
|
|
// History:
|
|
//
|
|
// 2/02/1999 KenSh Created for JetNet, largely from Internet Connection Wizard
|
|
// 9/29/1999 KenSh Adapterd for Home Networking Wizard
|
|
//
|
|
|
|
#include "stdafx.h"
|
|
#include <string.h>
|
|
#include <regstr.h>
|
|
#include "NConn16.h"
|
|
#include "strstri.h"
|
|
|
|
extern "C"
|
|
{
|
|
// Define missing decs so it builds.
|
|
typedef HKEY* LPHKEY;
|
|
typedef const BYTE* LPCBYTE;
|
|
#define WINCAPI
|
|
// missing
|
|
|
|
#include <setupx.h>
|
|
#include <netdi.h>
|
|
}
|
|
|
|
#ifndef _countof
|
|
#define _countof(ar) (sizeof(ar) / sizeof((ar)[0]))
|
|
#endif
|
|
|
|
|
|
extern "C" BOOL FAR PASCAL thk_ThunkConnect16(LPSTR pszDll16,
|
|
LPSTR pszDll32,
|
|
WORD hInst,
|
|
DWORD dwReason);
|
|
|
|
//
|
|
// SetupX function prototypes
|
|
//
|
|
typedef RETERR (WINAPI PASCAL FAR * PROC_DiOpenDevRegKey)(
|
|
LPDEVICE_INFO lpdi,
|
|
LPHKEY lphk,
|
|
int iFlags);
|
|
typedef DWORD (WINAPI FAR * PROC_SURegSetValueEx)(HKEY hKey,LPCSTR lpszValueName, DWORD dwReserved, DWORD dwType, LPBYTE lpszValue, DWORD dwValSize);
|
|
typedef RETERR (WINAPI FAR * PROC_DiCreateDeviceInfo)(
|
|
LPLPDEVICE_INFO lplpdi, // Ptr to ptr to dev info
|
|
LPCSTR lpszDescription, // If non-null then description string
|
|
DWORD hDevnode, // ISSUE-2002/01/16-roelfc: -- MAKE A DEVNODE
|
|
HKEY hkey, // Registry hkey for dev info
|
|
LPCSTR lpszRegsubkey, // If non-null then reg subkey string
|
|
LPCSTR lpszClassName, // If non-null then class name string
|
|
HWND hwndParent); // If non-null then hwnd of parent
|
|
typedef RETERR (WINAPI FAR * PROC_DiDestroyDeviceInfoList)(LPDEVICE_INFO lpdi);
|
|
typedef RETERR (WINAPI FAR * PROC_DiCallClassInstaller)(DI_FUNCTION diFctn, LPDEVICE_INFO lpdi);
|
|
typedef DWORD (WINAPI FAR * PROC_SURegCloseKey)(HKEY hKey);
|
|
typedef RETERR (WINAPI FAR * PROC_DiGetClassDevs)(
|
|
LPLPDEVICE_INFO lplpdi, // Ptr to ptr to dev info
|
|
LPCSTR lpszClassName, // Must be name of class
|
|
HWND hwndParent, // If non-null then hwnd of parent
|
|
int iFlags); // Options
|
|
typedef RETERR (WINAPI FAR * PROC_DiSelectDevice)( LPDEVICE_INFO lpdi );
|
|
typedef RETERR (WINAPI FAR * PROC_DiBuildCompatDrvList)(LPDEVICE_INFO lpdi);
|
|
typedef RETERR (WINAPI FAR * PASCAL PROC_DiCreateDevRegKey)(
|
|
LPDEVICE_INFO lpdi,
|
|
LPHKEY lphk,
|
|
HINF hinf,
|
|
LPCSTR lpszInfSection,
|
|
int iFlags);
|
|
typedef RETERR (WINAPI FAR * PASCAL PROC_DiDeleteDevRegKey)(LPDEVICE_INFO lpdi, int iFlags);
|
|
|
|
PROC_DiOpenDevRegKey _pfnDiOpenDevRegKey;
|
|
PROC_SURegSetValueEx _pfnSURegSetValueEx;
|
|
PROC_DiCreateDeviceInfo _pfnDiCreateDeviceInfo;
|
|
PROC_DiDestroyDeviceInfoList _pfnDiDestroyDeviceInfoList;
|
|
PROC_DiCallClassInstaller _pfnDiCallClassInstaller;
|
|
PROC_SURegCloseKey _pfnSURegCloseKey;
|
|
PROC_DiGetClassDevs _pfnDiGetClassDevs;
|
|
PROC_DiSelectDevice _pfnDiSelectDevice;
|
|
PROC_DiBuildCompatDrvList _pfnDiBuildCompatDrvList;
|
|
PROC_DiCreateDevRegKey _pfnDiCreateDevRegKey;
|
|
PROC_DiDeleteDevRegKey _pfnDiDeleteDevRegKey;
|
|
|
|
|
|
int g_cSetupxInit = 0;
|
|
HINSTANCE g_hInstSetupx = NULL;
|
|
|
|
BOOL InitSetupx()
|
|
{
|
|
if (g_hInstSetupx == NULL)
|
|
{
|
|
g_hInstSetupx = LoadLibrary("setupx.dll");
|
|
if (g_hInstSetupx < (HINSTANCE)HINSTANCE_ERROR)
|
|
return FALSE;
|
|
|
|
_pfnDiOpenDevRegKey = (PROC_DiOpenDevRegKey)GetProcAddress(g_hInstSetupx, "DiOpenDevRegKey");
|
|
_pfnSURegSetValueEx = (PROC_SURegSetValueEx)GetProcAddress(g_hInstSetupx, "SURegSetValueEx");
|
|
_pfnDiCreateDeviceInfo = (PROC_DiCreateDeviceInfo)GetProcAddress(g_hInstSetupx, "DiCreateDeviceInfo");
|
|
_pfnDiDestroyDeviceInfoList = (PROC_DiDestroyDeviceInfoList)GetProcAddress(g_hInstSetupx, "DiDestroyDeviceInfoList");
|
|
_pfnDiCallClassInstaller = (PROC_DiCallClassInstaller)GetProcAddress(g_hInstSetupx, "DiCallClassInstaller");
|
|
_pfnSURegCloseKey = (PROC_SURegCloseKey)GetProcAddress(g_hInstSetupx, "SURegCloseKey");
|
|
_pfnDiGetClassDevs = (PROC_DiGetClassDevs)GetProcAddress(g_hInstSetupx, "DiGetClassDevs");
|
|
_pfnDiSelectDevice = (PROC_DiSelectDevice)GetProcAddress(g_hInstSetupx, "DiSelectDevice");
|
|
_pfnDiBuildCompatDrvList = (PROC_DiBuildCompatDrvList)GetProcAddress(g_hInstSetupx, "DiBuildCompatDrvList");
|
|
_pfnDiCreateDevRegKey = (PROC_DiCreateDevRegKey)GetProcAddress(g_hInstSetupx, "DiCreateDevRegKey");
|
|
_pfnDiDeleteDevRegKey = (PROC_DiDeleteDevRegKey)GetProcAddress(g_hInstSetupx, "DiDeleteDevRegKey");
|
|
}
|
|
|
|
g_cSetupxInit++;
|
|
return TRUE;
|
|
}
|
|
|
|
void UninitSetupx()
|
|
{
|
|
if (g_cSetupxInit > 0)
|
|
g_cSetupxInit--;
|
|
|
|
if (g_cSetupxInit == 0)
|
|
{
|
|
if (g_hInstSetupx != NULL)
|
|
{
|
|
FreeLibrary(g_hInstSetupx);
|
|
g_hInstSetupx = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
extern "C" int FAR PASCAL LibMain(HANDLE hInstance, WORD wDataSeg, WORD wHeapSize, LPSTR lpszCmdLine)
|
|
{
|
|
if (wHeapSize != 0)
|
|
{
|
|
// UnlockData is no longer #defined to UnlockSegment(-1) in windows.h,
|
|
// so do it manually here:
|
|
//
|
|
//UnlockData(0);
|
|
UnlockSegment(-1);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
extern "C" BOOL FAR PASCAL __export DllEntryPoint(DWORD dwReason, WORD hInstance, WORD wDS, WORD wHeapSize, DWORD dwReserved1, WORD wReserved2)
|
|
{
|
|
if (!thk_ThunkConnect16(
|
|
"NCXP16.DLL",
|
|
"NCXP32.DLL",
|
|
hInstance, dwReason))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
extern "C" int CALLBACK WEP(int nExitType)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
#define ASSERT(x)
|
|
|
|
// Local function declarations
|
|
DWORD CallClassInstaller(HWND hwndParent, LPCSTR lpszClassName, LPCSTR lpszDeviceID);
|
|
DWORD BindProtocolToAdapters(HWND hwndParent, LPCSTR lpszClassName, LPCSTR lpszDeviceID);
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// String constants
|
|
// TODO: clean these up... most of them aren't used any more
|
|
|
|
// Device Manager class names
|
|
const char szClassNetCard[] = "Net";
|
|
const char szClassNetClient[] = "NetClient";
|
|
const char szClassNetProtocol[] = "NetTrans";
|
|
const char szClassModem[] = "Modem";
|
|
|
|
// Device ID string constants
|
|
const char szMSTCPIP_ID[] = "MSTCP";
|
|
const char szPPPMAC_ID[] = "*PNP8387";
|
|
const char szVREDIR_ID[] = "VREDIR";
|
|
const char szNWREDIR_ID[] = "NWREDIR";
|
|
const char szIPX_ID[] = "NWLINK";
|
|
const char szNETBEUI_ID[] = "NETBEUI";
|
|
|
|
// Registry string constants
|
|
const char szRegValSlowNet[] = "SLOWNET";
|
|
const char szRegKeyNdi[] = "Ndi";
|
|
const char szRegValDeviceID[] = "DeviceID";
|
|
const char szRegKeyBindings[] = "Bindings";
|
|
const char szRegPathOptComponents[]=REGSTR_PATH_SETUP REGSTR_KEY_SETUP "\\OptionalComponents";
|
|
const char szRegValInstalled[] = "Installed";
|
|
const char szRegPathNetwork[] = "Enum\\Network";
|
|
const char szRegPathTemp[] = "\\Temp";
|
|
const char szRegValCompatibleIDs[]= REGSTR_VAL_COMPATIBLEIDS;
|
|
const char szRegValDeviceType[] = REGSTR_VAL_DEVTYPE;
|
|
const char szRegValConfigFlags[] = REGSTR_VAL_CONFIGFLAGS;
|
|
const char szRegPathPlusSetup[] = "Software\\Microsoft\\Plus!\\Setup";
|
|
const char szRegValSourcePath[] = "SourcePath";
|
|
const char szRegValHardwareID[] = "HardwareID";
|
|
|
|
// component string constants
|
|
const char szCompRNA[] = "RNA";
|
|
const char szCompMail[] = "MAPI";
|
|
const char szCompMSN[] = "MSNetwork";
|
|
const char szCompMSN105[] = "MSNetwork105";
|
|
const char szCompInetMail[] = "InternetMail";
|
|
const char szINF[] = "INF";
|
|
const char szSection[] = "Section";
|
|
|
|
// INF string constants
|
|
const char szValSignature[] = "$CHICAGO$";
|
|
const char szKeySignature[] = "signature";
|
|
const char szSectVersion[] = "version";
|
|
|
|
// other strings
|
|
const char szNull[] = "";
|
|
const char sz1[] = "1";
|
|
const char szSlash[] = "\\";
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
// CreateTempDevRegKey
|
|
//
|
|
// Creates a temporary registry key for the device installer.
|
|
//
|
|
// History:
|
|
//
|
|
// 2/02/1999 KenSh Borrowed from ICW, which borrowed it from net setup
|
|
//
|
|
RETERR CreateTempDevRegKey(LPDEVICE_INFO lpdi,LPHKEY lphk)
|
|
{
|
|
lpdi->hRegKey = HKEY_LOCAL_MACHINE;
|
|
lstrcpy(lpdi->szRegSubkey, szRegPathNetwork);
|
|
lstrcat(lpdi->szRegSubkey, szSlash);
|
|
lstrcat(lpdi->szRegSubkey, lpdi->szClassName);
|
|
lstrcat(lpdi->szRegSubkey, szRegPathTemp);
|
|
|
|
InitSetupx();
|
|
RETERR err = (*_pfnDiCreateDevRegKey)(lpdi, lphk, NULL, NULL, DIREG_DEV);
|
|
UninitSetupx();
|
|
|
|
return err;
|
|
}
|
|
|
|
|
|
// CallClassInstaller
|
|
//
|
|
// Calls DiCallClassInstaller for the specified class to install
|
|
// the specified device ID
|
|
//
|
|
// Returns ICERR_xxx return value, defined in NetSetup.h
|
|
//
|
|
// Parameters:
|
|
//
|
|
// hwndParent - parent window handle
|
|
// lpszClassName - name of device class (e.g. "NetTrans" or "Net")
|
|
// lpszDeviceID - unique device ID to install (e.g. "MSTCP" or "PCI\VEN_10b7&DEV_5950"
|
|
//
|
|
// History:
|
|
//
|
|
// 2/02/1999 KenSh Borrowed from ICW, changed return codes for JetNet
|
|
// 3/18/1999 KenSh Cleaned up
|
|
//
|
|
extern "C" DWORD WINAPI __export CallClassInstaller16(HWND hwndParent, LPCSTR lpszClassName, LPCSTR lpszDeviceID)
|
|
{
|
|
RETERR err;
|
|
DWORD dwResult = ICERR_OK;
|
|
LPDEVICE_INFO lpdi;
|
|
HKEY hKeyTmp;
|
|
LONG uErr;
|
|
|
|
ASSERT(lpszClassName != NULL);
|
|
ASSERT(lpszDeviceID != NULL);
|
|
|
|
if (!InitSetupx())
|
|
return ICERR_DI_ERROR;
|
|
|
|
// allocate a DEVICE_INFO struct
|
|
err = (*_pfnDiCreateDeviceInfo)(&lpdi, NULL, 0, NULL, NULL, lpszClassName, hwndParent);
|
|
|
|
ASSERT(err == OK);
|
|
if (err != OK)
|
|
{
|
|
lpdi = NULL;
|
|
goto exit;
|
|
}
|
|
|
|
// since the device manager APIs are not very good, to communicate the
|
|
// device ID to it we have to create a temporary registry key and
|
|
// store the device ID there. This code borrowed from net setup
|
|
// which has to do the same thing (fill out an LPDEVICE_INFO based
|
|
// on a device ID)
|
|
err = CreateTempDevRegKey(lpdi, &hKeyTmp);
|
|
ASSERT (err == OK);
|
|
if (err != OK)
|
|
goto exit;
|
|
|
|
// set the device ID in the registry
|
|
uErr = RegSetValueEx(hKeyTmp, szRegValCompatibleIDs,
|
|
0, REG_SZ, (LPBYTE)lpszDeviceID, lstrlen(lpszDeviceID)+1);
|
|
ASSERT(uErr == ERROR_SUCCESS);
|
|
|
|
// now call device mgr API to add driver node lists and fill out structure,
|
|
// it will use the device ID we stuffed in registry.
|
|
err = (*_pfnDiBuildCompatDrvList)(lpdi);
|
|
ASSERT(err == OK);
|
|
|
|
RegCloseKey(hKeyTmp);
|
|
|
|
// need to delete temp key, set handle to null, set subkey name to
|
|
// null or else net setup thinks this device already exists and
|
|
// zany hijinks ensue
|
|
(*_pfnDiDeleteDevRegKey)(lpdi, DIREG_DEV);
|
|
lpdi->hRegKey = NULL;
|
|
lstrcpy(lpdi->szRegSubkey, szNull);
|
|
|
|
if (err == OK)
|
|
{
|
|
lpdi->lpSelectedDriver = lpdi->lpCompatDrvList;
|
|
ASSERT(lpdi->lpSelectedDriver);
|
|
|
|
err = (*_pfnDiCallClassInstaller)(DIF_INSTALLDEVICE, lpdi);
|
|
ASSERT(err == OK);
|
|
|
|
if (err == OK)
|
|
{
|
|
// if we need to reboot, set a special return code NEED_RESTART
|
|
// (which also implies success)
|
|
if (lpdi->Flags & DI_NEEDREBOOT)
|
|
{
|
|
// REVIEW: does this need to reboot, or is restart sufficient?
|
|
// err = NEED_RESTART;
|
|
dwResult = ICERR_NEED_RESTART;
|
|
}
|
|
}
|
|
}
|
|
|
|
exit:
|
|
if (lpdi != NULL)
|
|
(*_pfnDiDestroyDeviceInfoList)(lpdi);
|
|
|
|
if (err != OK)
|
|
dwResult = ICERR_DI_ERROR | (DWORD)err;
|
|
|
|
UninitSetupx();
|
|
return dwResult;
|
|
}
|
|
|
|
extern "C" HRESULT WINAPI __export FindClassDev16(HWND hwndParent, LPCSTR pszClass, LPCSTR pszDeviceID)
|
|
{
|
|
DWORD hr = S_FALSE;
|
|
LPDEVICE_INFO lpdi;
|
|
RETERR err;
|
|
|
|
if (!InitSetupx())
|
|
return E_FAIL;
|
|
|
|
if (OK != (err = (*_pfnDiGetClassDevs)(&lpdi, pszClass, NULL, DIGCF_PRESENT)))
|
|
{
|
|
UninitSetupx();
|
|
return E_FAIL;
|
|
}
|
|
|
|
// This is 16-bit code, so these pnp ids are ANSI only
|
|
//
|
|
LPSTR pszAlternateDeviceID = new char[lstrlen(pszDeviceID) + 1];
|
|
if (pszAlternateDeviceID)
|
|
{
|
|
lstrcpy(pszAlternateDeviceID, pszDeviceID);
|
|
|
|
LPCSTR szSubsysString = "SUBSYS_";
|
|
const int nSubsysIDLength = 8;
|
|
LPSTR pszSubsys = strstri(pszAlternateDeviceID, szSubsysString);
|
|
if(NULL != pszSubsys)
|
|
{
|
|
pszSubsys += _countof("SUBSYS_") - 1;
|
|
if(nSubsysIDLength <= lstrlen(pszSubsys))
|
|
{
|
|
for(int i = 0; i < nSubsysIDLength; i++)
|
|
{
|
|
pszSubsys[i] = '0';
|
|
}
|
|
}
|
|
}
|
|
|
|
for (LPDEVICE_INFO lpdiCur = lpdi; lpdiCur != NULL; lpdiCur = lpdiCur->lpNextDi)
|
|
{
|
|
char szBuf[1024];
|
|
// wsprintf(szBuf, "System\\CurrentControlSet\\Services\\Class\\%s\\%04d", pszClass, (int)lpdiCur->dnDevnode);
|
|
// MessageBox(NULL, szBuf, "RegKey", MB_OK);
|
|
|
|
HKEY hKey;
|
|
if (OK == (*_pfnDiOpenDevRegKey)(lpdiCur, &hKey, DIREG_DEV))
|
|
{
|
|
static const LPCSTR c_rgRegEntries[] = { "HardwareID", "CompatibleIDs" };
|
|
|
|
for (int i = 0; i < _countof(c_rgRegEntries); i++)
|
|
{
|
|
LONG cbBuf = sizeof(szBuf);
|
|
if (ERROR_SUCCESS == RegQueryValueEx(hKey, c_rgRegEntries[i], NULL, NULL, (LPBYTE)szBuf, &cbBuf))
|
|
{
|
|
// char szBuf2[1600];
|
|
// wsprintf(szBuf2, "Looking for: %s\n\n%s", pszDeviceID, szBuf);
|
|
// MessageBox(NULL, szBuf2, c_rgRegEntries[i], MB_OK);
|
|
|
|
// aslo check with SUBSYS 00000000 for bug 124967
|
|
if (NULL != strstri(szBuf, pszDeviceID) || NULL != strstri(szBuf, pszAlternateDeviceID))
|
|
{
|
|
hr = S_OK;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
RegCloseKey(hKey);
|
|
}
|
|
|
|
if (hr == S_OK)
|
|
break;
|
|
}
|
|
|
|
delete [] pszAlternateDeviceID;
|
|
}
|
|
|
|
(*_pfnDiDestroyDeviceInfoList)(lpdi);
|
|
|
|
UninitSetupx();
|
|
return hr;
|
|
}
|
|
|
|
extern "C" HRESULT WINAPI __export LookupDevNode16(HWND hwndParent, LPCSTR pszClass, LPCSTR pszEnumKey, DEVNODE FAR* pDevNode, DWORD FAR* pdwFreePointer)
|
|
{
|
|
DWORD hr = S_FALSE;
|
|
LPDEVICE_INFO lpdi;
|
|
RETERR err;
|
|
|
|
// MessageBox(hwndParent, "LookupDevNode16", "Debug", MB_ICONINFORMATION);
|
|
|
|
if (pDevNode == NULL || pdwFreePointer == NULL)
|
|
{
|
|
// MessageBox(hwndParent, "Returning failure 0", "Debug", 0);
|
|
return E_POINTER;
|
|
}
|
|
|
|
*pDevNode = 0;
|
|
*pdwFreePointer = 0;
|
|
|
|
if (!InitSetupx())
|
|
{
|
|
// MessageBox(hwndParent, "Returning failure 1", "Debug", 0);
|
|
return E_FAIL;
|
|
}
|
|
|
|
if (OK != (err = (*_pfnDiGetClassDevs)(&lpdi, pszClass, NULL, DIGCF_PRESENT)))
|
|
{
|
|
// MessageBox(hwndParent, "Returning failure 2", "Debug", 0);
|
|
UninitSetupx();
|
|
return E_FAIL;
|
|
}
|
|
|
|
for (LPDEVICE_INFO lpdiCur = lpdi; lpdiCur != NULL; lpdiCur = lpdiCur->lpNextDi)
|
|
{
|
|
// char szBuf[1024];
|
|
// wsprintf(szBuf, "comparing:\nlpdiCur->pszRegSubkey = \"%s\"\npszEnumKey = \"%s\"",
|
|
// (LPSTR)lpdiCur->szRegSubkey, (LPSTR)pszEnumKey);
|
|
// MessageBox(hwndParent, szBuf, "Debug", MB_ICONINFORMATION);
|
|
|
|
if (0 == lstrcmpi(lpdiCur->szRegSubkey, pszEnumKey))
|
|
{
|
|
// wsprintf(szBuf, "found devnode 0x%08lX, pvFreePointer = 0x%08lX", (DWORD)lpdiCur->dnDevnode, (DWORD)lpdi);
|
|
// MessageBox(hwndParent, szBuf, "Debug", MB_ICONINFORMATION);
|
|
|
|
*pDevNode = lpdiCur->dnDevnode;
|
|
*pdwFreePointer = (DWORD)lpdi;
|
|
return S_OK;
|
|
}
|
|
}
|
|
|
|
(*_pfnDiDestroyDeviceInfoList)(lpdi);
|
|
|
|
UninitSetupx();
|
|
|
|
// MessageBox(hwndParent, "returning failure", "Debug", MB_ICONINFORMATION);
|
|
|
|
return E_FAIL; // not found
|
|
}
|
|
|
|
extern "C" HRESULT WINAPI __export FreeDevNode16(DWORD dwFreePointer)
|
|
{
|
|
LPDEVICE_INFO lpdi = (LPDEVICE_INFO)dwFreePointer;
|
|
if (lpdi == NULL)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
if (g_hInstSetupx == NULL)
|
|
{
|
|
return E_FAIL;
|
|
}
|
|
|
|
// char szBuf[1024];
|
|
// wsprintf(szBuf, "FreeDevNode16 - freeing lpdi 0x%08lX - continue?", dwFreePointer);
|
|
// if (IDYES == MessageBox(NULL, szBuf, "Debug", MB_YESNO | MB_ICONEXCLAMATION))
|
|
{
|
|
(*_pfnDiDestroyDeviceInfoList)(lpdi);
|
|
}
|
|
|
|
UninitSetupx();
|
|
return S_OK;
|
|
}
|
|
|
|
extern "C" HRESULT WINAPI __export IcsUninstall16(void)
|
|
{
|
|
typedef void (WINAPI *RUNDLLPROC)(HWND hwndStub, HINSTANCE hAppInstance, LPSTR lpszCmdLine, int nCmdShow);
|
|
|
|
HINSTANCE hInstance = LoadLibrary("issetup.dll");
|
|
if(hInstance > 32)
|
|
{
|
|
RUNDLLPROC pExtUninstall = (RUNDLLPROC) GetProcAddress(hInstance, "ExtUninstall");
|
|
if(NULL != pExtUninstall)
|
|
{
|
|
pExtUninstall(NULL, NULL, NULL, 0);
|
|
}
|
|
|
|
FreeLibrary(hInstance);
|
|
}
|
|
return S_OK;
|
|
}
|