mirror of https://github.com/lianthony/NT4.0
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.
711 lines
16 KiB
711 lines
16 KiB
#include "stdafx.h"
|
|
|
|
#define DLL_BASED __declspec(dllexport)
|
|
|
|
#include "fscfg.h"
|
|
#include "fservic.h"
|
|
#include "fmessage.h"
|
|
#include "ftpdir.h"
|
|
|
|
// ===========================================================================
|
|
// Standard configuration Information
|
|
// ===========================================================================
|
|
|
|
//
|
|
// Is this server discoverable by INETSLOC?
|
|
//
|
|
#define INETSLOC_DISCOVERY TRUE
|
|
|
|
#if (INETSLOC_DISCOVERY) && !defined(_SVCLOC_)
|
|
#error You must include svcloc.h.
|
|
#endif
|
|
|
|
//
|
|
// If INETSLOC_DISCOVERY == TRUE, define the discovery MASK here.
|
|
//
|
|
#if (INETSLOC_DISCOVERY)
|
|
#define INETSLOC_MASK INET_FTP_SERVICE
|
|
#else // (!INETSLOC_DISCOVERY)
|
|
#define INETSLOC_MASK (ULONGLONG)(0x00000000)
|
|
#endif // (INETSLOC_DISCOVERY)
|
|
|
|
#ifdef NO_SERVICE_CONTROLLER
|
|
#define CAN_CHANGE_SERVICE_STATE FALSE
|
|
#define CAN_PAUSE_SERVICE FALSE
|
|
#else
|
|
//
|
|
// Can we change the service state (start/pause/continue)?
|
|
//
|
|
#define CAN_CHANGE_SERVICE_STATE TRUE
|
|
|
|
//
|
|
// Can we pause this service?
|
|
//
|
|
#define CAN_PAUSE_SERVICE TRUE
|
|
#endif // NO_SERVICE_CONTROLLER
|
|
|
|
//
|
|
// Name used for this service by the service controller manager.
|
|
//
|
|
#define SERVICE_SC_NAME _T("MSFTPSVC")
|
|
|
|
//
|
|
// Short descriptive name of the service. This
|
|
// is what will show up as the name of the service
|
|
// in the internet manager tool.
|
|
//
|
|
// Issue: I'm assuming here that this name does NOT
|
|
// require localisation.
|
|
//
|
|
#define SERVICE_SHORT_NAME _T("FTP")
|
|
|
|
//
|
|
// Longer name. This is the text that shows up in
|
|
// the tooltips text on the internet manager
|
|
// tool. This probably should be localised.
|
|
//
|
|
|
|
#define SERVICE_LONG_NAME _T("FTP Service")
|
|
|
|
//
|
|
// Use normal colour mapping.
|
|
//
|
|
#define NORMAL_TB_MAPPING TRUE
|
|
|
|
//
|
|
// Toolbar button background mask. This is
|
|
// the colour that gets masked out in
|
|
// the bitmap file and replaced with the
|
|
// actual button background. This setting
|
|
// is automatically assumed to be lt. gray
|
|
// if NORMAL_TB_MAPPING (above) is TRUE
|
|
//
|
|
#define BUTTON_BMP_BACKGROUND RGB(192, 192, 192) // Lt. Gray
|
|
|
|
//
|
|
// Resource ID of the toolbar button bitmap.
|
|
//
|
|
// The bitmap must be 17x17
|
|
//
|
|
#define BUTTON_BMP_ID IDB_FTP
|
|
|
|
//
|
|
// Similar to BUTTON_BMP_BACKGROUND, this is the
|
|
// background mask for the service ID
|
|
//
|
|
#define SERVICE_BMP_BACKGROUND RGB(255,0,255)
|
|
|
|
//
|
|
// Bitmap id which is used in the service view
|
|
// of the service manager. This may be the same
|
|
// bitmap as BUTTON_BMP_BACKGROUND.
|
|
//
|
|
// The bitmap must be 17x17.
|
|
//
|
|
#define SERVICE_BMP_ID IDB_FTPVIEW
|
|
|
|
// ===========================================================================
|
|
// End Of Standard configuration Information
|
|
// ===========================================================================
|
|
|
|
//
|
|
// FtpGetAdminInformation API wrapper
|
|
//
|
|
CFtpConfigInfo::CFtpConfigInfo(
|
|
CStringList * pServerList,
|
|
CWnd * pParent
|
|
)
|
|
: CInetConfig(INET_FTP, pServerList, pParent)
|
|
{
|
|
}
|
|
|
|
void
|
|
CFtpConfigInfo::Initialize()
|
|
{
|
|
CInetConfig::Initialize();
|
|
|
|
m_pInfo = new FTP_CONFIG_INFO;
|
|
::memset(m_pInfo, 0, sizeof(FTP_CONFIG_INFO));
|
|
}
|
|
|
|
void
|
|
CFtpConfigInfo::SetValues(
|
|
BOOL fAllowAnonymous,
|
|
BOOL fOnlyAnonymous
|
|
)
|
|
{
|
|
TRACEEOLID(_T("Ftp Service SetValues() called"));
|
|
|
|
Initialize();
|
|
ASSERT(m_pInfo != NULL);
|
|
|
|
LPFTP_CONFIG_INFO lp = (LPFTP_CONFIG_INFO)m_pInfo;
|
|
|
|
SetField( lp->FieldControl,
|
|
FC_FTP_ALLOW_ANONYMOUS |
|
|
FC_FTP_ANONYMOUS_ONLY
|
|
);
|
|
|
|
lp->fAllowAnonymous = fAllowAnonymous;
|
|
lp->fAnonymousOnly = fOnlyAnonymous;
|
|
}
|
|
|
|
void
|
|
CFtpConfigInfo::SetValues(
|
|
LPWSTR lpszExitMessage,
|
|
LPWSTR lpszGreetingMessage,
|
|
LPWSTR lpszMaxClientsMessage
|
|
)
|
|
{
|
|
TRACEEOLID(_T("Ftp Service SetValues() called"));
|
|
|
|
Initialize();
|
|
ASSERT(m_pInfo != NULL);
|
|
|
|
LPFTP_CONFIG_INFO lp = (LPFTP_CONFIG_INFO)m_pInfo;
|
|
|
|
SetField( lp->FieldControl,
|
|
FC_FTP_EXIT_MESSAGE |
|
|
FC_FTP_GREETING_MESSAGE |
|
|
FC_FTP_MAX_CLIENTS_MESSAGE
|
|
);
|
|
|
|
lp->lpszExitMessage = lpszExitMessage;
|
|
lp->lpszGreetingMessage = lpszGreetingMessage;
|
|
lp->lpszMaxClientsMessage = lpszMaxClientsMessage;
|
|
}
|
|
|
|
void
|
|
CFtpConfigInfo::SetValues(
|
|
int nUnixDos
|
|
)
|
|
{
|
|
TRACEEOLID(_T("Ftp Service SetValues() called"));
|
|
|
|
Initialize();
|
|
ASSERT(m_pInfo != NULL);
|
|
|
|
LPFTP_CONFIG_INFO lp = (LPFTP_CONFIG_INFO)m_pInfo;
|
|
|
|
SetField( lp->FieldControl, FC_FTP_MSDOS_DIR_OUTPUT);
|
|
|
|
lp->fMsdosDirOutput = (nUnixDos == 1);
|
|
}
|
|
|
|
NET_API_STATUS
|
|
CFtpConfigInfo::GetApiStructure(
|
|
LPCTSTR lpstrServer
|
|
)
|
|
{
|
|
ASSERT(m_pInfo == NULL); // Should be clean
|
|
|
|
NET_API_STATUS err;
|
|
|
|
TRY
|
|
{
|
|
err = ::FtpGetAdminInformation( TWSTRREF(lpstrServer),
|
|
(LPFTP_CONFIG_INFO *)&m_pInfo );
|
|
TRACEEOLID(_T("FtpGetAdminInformation returned ") << err);
|
|
}
|
|
CATCH_ALL(e)
|
|
{
|
|
err = ::GetLastError();
|
|
}
|
|
END_CATCH_ALL
|
|
|
|
return err;
|
|
}
|
|
|
|
//
|
|
// Call the set api.
|
|
//
|
|
NET_API_STATUS
|
|
CFtpConfigInfo::SetApiStructure(
|
|
LPCTSTR lpstrServer,
|
|
BOOL fCommon // Ignored
|
|
)
|
|
{
|
|
ASSERT(m_pInfo != NULL);
|
|
NET_API_STATUS err;
|
|
|
|
TRY
|
|
{
|
|
err = ::FtpSetAdminInformation( TWSTRREF(lpstrServer),
|
|
(LPFTP_CONFIG_INFO)m_pInfo );
|
|
TRACEEOLID(_T("FtpSetAdminInformation returned ") << err);
|
|
}
|
|
CATCH_ALL(e)
|
|
{
|
|
err = ::GetLastError();
|
|
}
|
|
END_CATCH_ALL
|
|
|
|
return err;
|
|
}
|
|
|
|
//
|
|
// Ftp Property sheet
|
|
//
|
|
BEGIN_MESSAGE_MAP(CFtpSheet, INetPropertySheet)
|
|
//{{AFX_MSG_MAP(InetPropertySheet)
|
|
ON_COMMAND(ID_HELP, OnHelp)
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
CFtpSheet::CFtpSheet(
|
|
UINT nIDCaption,
|
|
DWORD dwServiceMask,
|
|
CStringList *pServerList,
|
|
CWnd* pParentWnd,
|
|
UINT iSelectPage
|
|
)
|
|
: INetPropertySheet(nIDCaption, dwServiceMask,
|
|
pServerList, pParentWnd, iSelectPage),
|
|
m_ftpConfig(pServerList, pParentWnd)
|
|
{
|
|
}
|
|
|
|
CFtpSheet::CFtpSheet(
|
|
LPCTSTR pszCaption,
|
|
DWORD dwServiceMask,
|
|
CStringList *pServerList,
|
|
CWnd* pParentWnd,
|
|
UINT iSelectPage
|
|
)
|
|
: INetPropertySheet(pszCaption, dwServiceMask,
|
|
pServerList, pParentWnd, iSelectPage),
|
|
m_ftpConfig(pServerList, pParentWnd)
|
|
{
|
|
}
|
|
|
|
void
|
|
CFtpSheet::Initialize()
|
|
{
|
|
INetPropertySheet::Initialize();
|
|
|
|
if (QueryError() == NO_ERROR)
|
|
{
|
|
m_ftpConfig.GetInfo();
|
|
}
|
|
}
|
|
|
|
void
|
|
CFtpSheet::OnHelp()
|
|
{
|
|
//
|
|
// this could have been provided by CPropertySheet::OnCmdMsg
|
|
// but we can't add virtual in a point release
|
|
//
|
|
CWnd* pOwner = GetWindow(GW_OWNER);
|
|
ASSERT(pOwner != this);
|
|
|
|
if (pOwner == NULL || !pOwner->OnCmdMsg(ID_HELP, CN_COMMAND, NULL, NULL))
|
|
{
|
|
//
|
|
// last crack goes to the current CWinThread object
|
|
//
|
|
CWinThread* pThread = AfxGetThread();
|
|
if (pThread != NULL)
|
|
{
|
|
pThread->OnCmdMsg(ID_HELP, CN_COMMAND, NULL, NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Global DLL instance
|
|
//
|
|
HINSTANCE hInstance;
|
|
|
|
// ============================================================================
|
|
// ISM API Functions
|
|
// ============================================================================
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Return service-specific information back to
|
|
// to the application. This function is called
|
|
// by the service manager immediately after
|
|
// LoadLibary(); The size element must be
|
|
// set prior to calling this API.
|
|
//
|
|
DLL_BASED DWORD APIENTRY
|
|
ISMQueryServiceInfo(
|
|
ISMSERVICEINFO * psi // Service information returned.
|
|
)
|
|
{
|
|
if ( psi == NULL
|
|
|| psi->dwSize < ISMSERVICEINFO_SIZE
|
|
)
|
|
{
|
|
TRACEEOLID(_T("ISMQueryServiceInfo: ISMSERVICEINFO invalid"));
|
|
ASSERT(0);
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
#ifdef _DEBUG
|
|
|
|
if (psi->dwSize != ISMSERVICEINFO_SIZE)
|
|
{
|
|
TRACEEOLID(_T("Warning: internet manager is newer than DLL"));
|
|
}
|
|
|
|
#endif // _DEBUG
|
|
|
|
psi->dwSize = ISMSERVICEINFO_SIZE;
|
|
psi->dwVersion = ISM_VERSION;
|
|
|
|
psi->flServiceInfoFlags = 0
|
|
#if (INETSLOC_DISCOVERY)
|
|
| ISMI_INETSLOCDISCOVER
|
|
#endif
|
|
#if (CAN_CHANGE_SERVICE_STATE)
|
|
| ISMI_CANCONTROLSERVICE
|
|
#endif
|
|
#if (CAN_PAUSE_SERVICE)
|
|
| ISMI_CANPAUSESERVICE
|
|
#endif
|
|
#if (NORMAL_TB_MAPPING)
|
|
| ISMI_NORMALTBMAPPING
|
|
#endif
|
|
; /**/
|
|
|
|
ASSERT(::lstrlen(SERVICE_LONG_NAME) <= MAX_LNLEN);
|
|
ASSERT(::lstrlen(SERVICE_SHORT_NAME) <= MAX_SNLEN);
|
|
|
|
psi->ullDiscoveryMask = INETSLOC_MASK;
|
|
psi->rgbButtonBkMask = BUTTON_BMP_BACKGROUND;
|
|
psi->nButtonBitmapID = BUTTON_BMP_ID;
|
|
psi->rgbServiceBkMask = SERVICE_BMP_BACKGROUND;
|
|
psi->nServiceBitmapID = SERVICE_BMP_ID;
|
|
::lstrcpy(psi->atchShortName, SERVICE_SHORT_NAME);
|
|
::lstrcpy(psi->atchLongName, SERVICE_LONG_NAME);
|
|
|
|
return ERROR_SUCCESS;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Discover machines running this service. This is
|
|
// only necessary for services not discovered with
|
|
// inetscloc (which don't give a mask)
|
|
//
|
|
DLL_BASED DWORD APIENTRY
|
|
ISMDiscoverServers(
|
|
ISMSERVERINFO * psi, // Server info buffer.
|
|
DWORD * pdwBufferSize, // Size required/available.
|
|
int * cServers // Number of servers in buffer.
|
|
)
|
|
{
|
|
#if (INETSLOC_DISCOVERY)
|
|
*cServers = 0;
|
|
*pdwBufferSize = 0L;
|
|
|
|
TRACEEOLID(_T("Warning: service manager called bogus ISMDiscoverServers"));
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
#else
|
|
|
|
#error Service specific discovery function must be provided.
|
|
|
|
#endif // (INETSLOC_DISCOVERY)
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Get information about a specific server with
|
|
// regards to this service.
|
|
//
|
|
DLL_BASED DWORD APIENTRY
|
|
ISMQueryServerInfo(
|
|
LPCTSTR lpstrServerName, // Name of server.
|
|
ISMSERVERINFO * psi // Server information returned.
|
|
)
|
|
{
|
|
if ( psi == NULL
|
|
|| psi->dwSize < ISMSERVERINFO_SIZE
|
|
|| ::lstrlen(lpstrServerName) > MAX_SERVERNAME_LEN
|
|
)
|
|
{
|
|
TRACEEOLID(_T("ISMQueryServerInfo: bad parameters"));
|
|
ASSERT(0);
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
#ifdef _DEBUG
|
|
|
|
if (psi->dwSize != ISMSERVERINFO_SIZE)
|
|
{
|
|
TRACEEOLID(_T("Warning internet manager is newer than DLL"));
|
|
}
|
|
|
|
#endif // _DEBUG
|
|
|
|
psi->dwSize = ISMSERVERINFO_SIZE;
|
|
::lstrcpy(psi->atchServerName, lpstrServerName);
|
|
|
|
//
|
|
// Start with NULL comment
|
|
//
|
|
*psi->atchComment = _T('\0');
|
|
|
|
DWORD err = ::QueryInetServiceStatus(psi->atchServerName,
|
|
SERVICE_SC_NAME, &(psi->nState));
|
|
|
|
if (err == ERROR_SUCCESS)
|
|
{
|
|
//
|
|
// Try to get comment
|
|
//
|
|
::GetInetComment(psi->atchServerName, INET_FTP,
|
|
STRSIZE(psi->atchComment), psi->atchComment);
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Change the service state of the servers (to paused/continue, started,
|
|
// stopped, etc)
|
|
//
|
|
DLL_BASED DWORD APIENTRY
|
|
ISMChangeServiceState(
|
|
int nNewState, // INetService* definition.
|
|
int * pnCurrentState, // Ptr to current state (will be changed
|
|
DWORD dwReserved, // Reserved: must be 0
|
|
LPCTSTR lpstrServers // Double NULL terminated list of servers.
|
|
)
|
|
{
|
|
if ( dwReserved != 0L
|
|
|| nNewState < INetServiceStopped
|
|
|| nNewState > INetServicePaused
|
|
)
|
|
{
|
|
TRACEEOLID(_T("ISMChangeServiceState: Invalid information passed"));
|
|
ASSERT(0);
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// BUGBUG: SINGLE SELECTION ONLY!
|
|
//
|
|
return ChangeInetServiceState(lpstrServers,
|
|
SERVICE_SC_NAME, nNewState, pnCurrentState);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Display configuration property sheet.
|
|
//
|
|
#define CDELETE(x) {if (x != NULL) delete x;}
|
|
|
|
DLL_BASED DWORD APIENTRY
|
|
ISMConfigureServers(
|
|
HWND hWnd, // Main app window handle
|
|
DWORD dwReserved, // Reserved: must be 0
|
|
LPCTSTR lpstrServers // Double NULL terminated list of servers
|
|
)
|
|
{
|
|
DWORD err;
|
|
|
|
if ( dwReserved != 0L )
|
|
{
|
|
TRACEEOLID(_T("ISMConfigureServers: Invalid information passed"));
|
|
ASSERT(0);
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// convert the list of servers to a
|
|
// more manageable CStringList.
|
|
//
|
|
LPCTSTR pBuf = lpstrServers;
|
|
CStringList strlServers;
|
|
while ( *pBuf != TCHAR('\0' ))
|
|
{
|
|
CString strTmp = pBuf;
|
|
strlServers.AddTail( strTmp );
|
|
pBuf += strTmp.GetLength() + 1;
|
|
}
|
|
|
|
if ( strlServers.GetCount() == 0 )
|
|
{
|
|
TRACEEOLID(_T("Error: strlServers Count == 0."));
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// Save the resource handle.
|
|
//
|
|
HINSTANCE hOldInstance = ::AfxGetResourceHandle();
|
|
ASSERT( hInstance != NULL );
|
|
|
|
//
|
|
// Load resources out of our own resource segment
|
|
//
|
|
::AfxSetResourceHandle( hInstance );
|
|
|
|
CString strCaption;
|
|
|
|
if (strlServers.GetCount() == 1)
|
|
{
|
|
CString str;
|
|
LPCTSTR lpCaption = *lpstrServers == _T('\\')
|
|
? lpstrServers + 2
|
|
: lpstrServers;
|
|
|
|
VERIFY(str.LoadString(IDS_CAPTION));
|
|
strCaption.Format(str, lpCaption);
|
|
}
|
|
else // Multiple server caption
|
|
{
|
|
VERIFY(strCaption.LoadString(IDS_CAPTION_MULTIPLE));
|
|
}
|
|
|
|
CFtpSheet Sheet( strCaption, INET_FTP, &strlServers,
|
|
CWnd::FromHandle(hWnd) );
|
|
|
|
//
|
|
// Call the APIs and build the property pages
|
|
//
|
|
Sheet.Initialize();
|
|
err = Sheet.QueryError();
|
|
|
|
if (err == NO_ERROR)
|
|
{
|
|
CFServicePage * pservpageF = NULL;
|
|
CFTPMessagePage * pmesspageF = NULL;
|
|
LoggingPage * plogpage = NULL;
|
|
SiteSecurityPage * psecpage = NULL;
|
|
CFTPDirectoryPage * pdirpage = NULL;
|
|
|
|
TRY
|
|
{
|
|
//
|
|
// Add private pages
|
|
//
|
|
Sheet.AddPage( pservpageF = new CFServicePage (&Sheet));
|
|
Sheet.AddPage( pmesspageF = new CFTPMessagePage (&Sheet));
|
|
|
|
//
|
|
// Add common pages
|
|
//
|
|
#ifndef _COMSTATIC
|
|
HINSTANCE hComprop = ::GetModuleHandle (COMPROP_DLL_NAME);
|
|
ASSERT(hComprop != NULL);
|
|
if (hComprop != NULL)
|
|
{
|
|
::AfxSetResourceHandle( hComprop );
|
|
#endif // _COMSTATIC
|
|
|
|
Sheet.AddPage( pdirpage = new CFTPDirectoryPage(&Sheet));
|
|
Sheet.AddPage( plogpage = new LoggingPage(&Sheet));
|
|
|
|
if (Sheet.IsAdvancedEnabled())
|
|
{
|
|
Sheet.AddPage( psecpage = new SiteSecurityPage(&Sheet));
|
|
}
|
|
|
|
#ifndef _COMSTATIC
|
|
::AfxSetResourceHandle( hInstance );
|
|
}
|
|
#endif // _COMSTATIC
|
|
}
|
|
CATCH_ALL(e)
|
|
{
|
|
TRACEEOLID(_T("Aborting due to exception"));
|
|
err = ::GetLastError();
|
|
}
|
|
END_CATCH_ALL
|
|
|
|
if ( err == ERROR_SUCCESS )
|
|
{
|
|
if ( Sheet.DoModal() == IDOK )
|
|
{
|
|
//
|
|
// Save each dirty page, but don't
|
|
// bother with the DDV stuff, as that's
|
|
// already done at this point.
|
|
//
|
|
err = Sheet.SavePages(FALSE);
|
|
}
|
|
}
|
|
|
|
CDELETE(pservpageF);
|
|
CDELETE(pmesspageF);
|
|
CDELETE(plogpage);
|
|
CDELETE(psecpage);
|
|
CDELETE(pdirpage);
|
|
}
|
|
|
|
//
|
|
// reset the resource handle to the application's
|
|
// resource segment.
|
|
//
|
|
::AfxSetResourceHandle( hOldInstance );
|
|
|
|
return err;
|
|
}
|
|
|
|
// ============================================================================
|
|
// End of ISM API Functions
|
|
// ============================================================================
|
|
|
|
//
|
|
// Perform additional initialisation as necessary
|
|
//
|
|
void
|
|
InitializeDLL()
|
|
{
|
|
#ifdef _DEBUG
|
|
afxMemDF |= checkAlwaysMemDF;
|
|
#endif // _DEBUG
|
|
|
|
#ifdef UNICODE
|
|
|
|
TRACEEOLID(_T("Loading UNICODE fscfg.dll"));
|
|
|
|
#else
|
|
|
|
TRACEEOLID(_T("Loading ANSI fscfg.dll"));
|
|
|
|
#endif UNICODE
|
|
|
|
CWndIpAddress::CreateWindowClass( hInstance );
|
|
}
|
|
|
|
//
|
|
// DLL Main entry point
|
|
//
|
|
DLL_BASED BOOL WINAPI
|
|
LibMain(
|
|
HINSTANCE hDll,
|
|
DWORD dwReason,
|
|
LPVOID lpReserved
|
|
)
|
|
{
|
|
BOOL bResult = TRUE ;
|
|
|
|
switch ( dwReason )
|
|
{
|
|
case DLL_PROCESS_ATTACH:
|
|
case DLL_THREAD_ATTACH:
|
|
ASSERT( hDll != NULL );
|
|
hInstance = hDll;
|
|
InitializeDLL();
|
|
break ;
|
|
|
|
case DLL_THREAD_DETACH:
|
|
case DLL_PROCESS_DETACH:
|
|
//
|
|
// termination
|
|
//
|
|
break ;
|
|
}
|
|
|
|
ASSERT( hInstance != NULL );
|
|
|
|
return bResult ;
|
|
}
|