|
|
/*****************************************************************************\
* MODULE: inetpp.c * * The module contains routines for handling the INETPP functionality. Use * of these routines require the locking/unlocking of a critical-section * to maninpulate the INIMONPORT list. All internal routines assume the * crit-sect is locked prior to executing. CheckMonCrit() is a debugging * call to verify the monitor-crit-sect is locked. * * NOTE: Each of the Inetmon* calls must be protected by the global-crit-sect. * If a new routine is added to this module which is to be called from * another module, be sure to include the call to (semCheckCrit), so * that the debug-code can catch unprotected access. * * Copyright (C) 1996-1997 Microsoft Corporation * Copyright (C) 1996-1997 Hewlett Packard * * History: * 07-Oct-1996 HWP-Guys Initiated port from win95 to winNT * 14-Nov-1997 ChrisWil Added local-spooling functionality. * 10-Jul-1998 WeihaiC Change Authentication Dialog Code * \*****************************************************************************/
#include "precomp.h"
#include "priv.h"
CInetMon::CInetMon () { // Initialize the monitor-port-list.
//
m_pPortList = (PINIMONPORTLIST)memAlloc(sizeof(INIMONPORTLIST));
m_bValid = (m_pPortList != NULL); }
CInetMon::~CInetMon () { }
/*****************************************************************************\
* _inet_validate_portname (Local Routine) * * Validate the portname. * * NOTE: If this check becomes more rigorous, it must account for the (%) * character (escape). * \*****************************************************************************/ BOOL CInetMon::_inet_validate_portname( LPCTSTR lpszPortName) { if (lpszPortName && (*lpszPortName)) return TRUE;
DBG_MSG(DBG_LEV_ERROR, (TEXT("Err : _inet_validate_portname : Invalid Name"))); SetLastError(ERROR_INVALID_NAME);
return FALSE; }
/*****************************************************************************\
* _inet_find_port (Local Routine) * * Locates the entry in the list where the port-name resides. Return the * full entry-type (INIMONPORT) for the location. * \*****************************************************************************/ PCINETMONPORT CInetMon::_inet_find_port( LPCTSTR lpszPortName) { PCINETMONPORT pIniPort;
pIniPort = m_pPortList->pIniFirstPort;
while (pIniPort && lstrcmpi(lpszPortName, pIniPort->m_lpszName)) pIniPort = pIniPort->m_pNext;
return pIniPort; }
/*****************************************************************************\
* _inet_create_port (Local Routine) * * Create a new port data structure and link into the list. This routine * assumes that the Crit is held. * * This routine also assumes the pszPortName is a valid http:// format.
* \*****************************************************************************/ PCINETMONPORT CInetMon::_inet_create_port( LPCTSTR lpszPortName, PCPORTMGR pPortMgr) { PCINETMONPORT pIniPort; PCINETMONPORT pPort; BOOL bRet = FALSE;
if ((pIniPort = new CInetMonPort (lpszPortName, g_szLocalPort, pPortMgr)) && pIniPort->bValid ()) {
// Set the link. If this is the first port added, the
// we set the global variable.
//
if (pPort = m_pPortList->pIniFirstPort) {
while (pPort->GetNext()) pPort = pPort->GetNext();
pPort->SetNext (pIniPort);
} else {
m_pPortList->pIniFirstPort = pIniPort; }
bRet = TRUE; }
if (!bRet) {
if (pIniPort) { delete pIniPort; pIniPort = NULL; } }
return pIniPort; }
/*****************************************************************************\
* _inet_delete_port (Local Routine) * * Free a port data structure that is no longer needed. * \*****************************************************************************/ BOOL CInetMon::_inet_delete_port( LPCTSTR lpszPortName) { PCINETMONPORT pIniPort; PCINETMONPORT pPrvPort; BOOL bRet = FALSE;
// Keep track of our previous/current entries, so that
// we can remove the specified port.
//
pIniPort = m_pPortList->pIniFirstPort;
while (pIniPort && lstrcmpi(pIniPort->m_lpszName, lpszPortName)) {
pPrvPort = pIniPort; pIniPort = pIniPort->GetNext (); }
// If the PortName is found, then delete it.
//
if (pIniPort) {
if (pIniPort->m_cRef > 0 && pIniPort->m_hTerminateEvent) {
// To tell spooling thread to terminate
//
SetEvent (pIniPort->m_hTerminateEvent);
semSafeLeaveCrit(pIniPort); //
// Leave the critical section so that the spooling thread can decrease the ref count
//
Sleep (250);
semSafeEnterCrit (pIniPort);
}
// Only allow this port to be deleted if the reference count
// is at zero.
//
if (pIniPort->m_cRef == 0) {
pIniPort->m_bDeletePending = TRUE;
//
// Remove the pointer from the list
//
// If this is our first-port then we need to handle
// differently as we keep a global-pointer to this.
//
if (pIniPort == m_pPortList->pIniFirstPort) {
m_pPortList->pIniFirstPort = pIniPort->GetNext();
} else {
pPrvPort->SetNext(pIniPort->GetNext()); }
// We only free the memeory if there are no open handles
//
if (pIniPort->m_cPrinterRef == 0) {
DBG_MSG(DBG_LEV_INFO, (TEXT("Info: pIniPort Freed %p."), pIniPort));
delete pIniPort; }
bRet = TRUE;
} else {
DBG_MSG(DBG_LEV_WARN, (TEXT("Warn: _inet_delete_port: Port in use: Name(%s)"), lpszPortName)); SetLastError(ERROR_BUSY); }
} else {
DBG_MSG(DBG_LEV_WARN, (TEXT("Warn: _inet_delete_port: Unrecognized PortName: Name(%s)"), lpszPortName)); SetLastError(ERROR_UNKNOWN_PORT); }
return bRet; }
/*****************************************************************************\
* _inet_is_xcv_open * * Check whether it is a XCV Open * \*****************************************************************************/ BOOL CInetMon::_inet_is_xcv_open ( LPCTSTR lpszPortName, LPTSTR *ppszServerName, LPTSTR *ppszRealPortName, LPBOOL pbXcv) { static CONST TCHAR cchSlash = _T ('\\'); static CONST TCHAR cszXcvPort[] = _T (",XcvPort "); static CONST DWORD cdwXcvPortLen = (sizeof (cszXcvPort) / sizeof (TCHAR)) - 1; BOOL bRet = TRUE; LPCTSTR pServerStart = NULL; LPTSTR pServerEnd = NULL; LPTSTR pszServerName = NULL; LPTSTR pszRealPortName = NULL;
// "\\Server\,XcvPort Object_" */
DWORD dwLen = lstrlen (lpszPortName); *pbXcv = FALSE;
if (dwLen > cdwXcvPortLen + 1) {
if (lpszPortName[0] == _T (',')) { // No Server
pServerEnd = (LPTSTR) lpszPortName; } else if (lpszPortName[0] == cchSlash && lpszPortName[1] == cchSlash) {
pServerStart = lpszPortName + 2; if (pServerEnd = _tcschr (pServerStart, cchSlash)) pServerEnd ++; } else { return bRet; }
if (pServerEnd && ! _tcsncmp (pServerEnd, cszXcvPort, cdwXcvPortLen)) {
LPCTSTR pPortStart = pServerEnd + cdwXcvPortLen;
if (pServerStart) { pszServerName = new TCHAR [(pServerEnd - 1) - pServerStart + 1]; if (!pszServerName) { bRet = FALSE; } else { StringCchCopy(pszServerName, pServerEnd - 1 - pServerStart, pServerStart); pszServerName[pServerEnd - 1 - pServerStart] = 0; } }
if (bRet) { size_t uSize = lstrlen (pPortStart) + 1;
pszRealPortName = new TCHAR [uSize];
if (!pszRealPortName) bRet = FALSE; else { if (SUCCEEDED(StringCchCopy(pszRealPortName, uSize, pPortStart))) *pbXcv = TRUE; else bRet = FALSE; } }
if (!bRet) { if (pszServerName) { delete [] pszServerName; pszServerName = NULL; }
if (pszRealPortName) { delete [] pszRealPortName; pszRealPortName = NULL; } }
*ppszServerName = pszServerName; *ppszRealPortName = pszRealPortName;
}
}
return bRet; }
PCINETMONPORT CInetMon::InetmonOpenPort( LPCTSTR lpszPortName, PBOOL pbXcv) { PCINETMONPORT pIniPort = NULL; PCPORTMGR pPortMgr = NULL; DWORD dwLE;
*pbXcv = FALSE;
semCheckCrit();
if (_inet_validate_portname(lpszPortName)) {
// Let's first look to see if it is a XCV call
LPTSTR pszServerName = NULL; LPTSTR pszRealPortName = NULL; BOOL bXcv;
if (_inet_is_xcv_open (lpszPortName, &pszServerName, &pszRealPortName, &bXcv) && bXcv) {
if (pIniPort = _inet_find_port(pszRealPortName)) {
if (!pIniPort->m_bDeletePending) { // The refrernce to the iniport is not changed, since the client may open a XCV handle
// to delete the port
//
*pbXcv = TRUE; } else { DBG_MSG(DBG_LEV_ERROR, (TEXT("Err : InetmonOpenPort (XCV) : Open deleted port: Port(%s)"), lpszPortName)); SetLastError(ERROR_INVALID_NAME); } }
// This is a valid XcvOpen
if (pszServerName) { delete [] pszServerName; } if (pszRealPortName) { delete [] pszRealPortName; }
if (pIniPort) { return pIniPort; } }
// Let's first look to see if this port is already in the
// list. If not, then add it to the list and continue with
// the open.
//
if ((pIniPort = _inet_find_port(lpszPortName)) == NULL) {
semLeaveCrit(); // Leave the critical section becuase the following call will hit the network
pPortMgr = new CPortMgr;
if (pPortMgr != NULL) { if (! pPortMgr->Create (lpszPortName)) { delete (pPortMgr); pPortMgr = NULL; } }
dwLE = GetLastError (); semEnterCrit();
if (! (pPortMgr)) { // The connection is invalid
DBG_MSG(DBG_LEV_ERROR, (TEXT("Err: InetmonOpenPort : PortMgrCreate Failed: LastError(%d)"), GetLastError()));
if (dwLE != ERROR_ACCESS_DENIED && dwLE != ERROR_INVALID_PRINTER_NAME && dwLE != ERROR_INVALID_NAME) {
dwLE = ERROR_PRINTER_NOT_FOUND; }
SetLastError ( dwLE );
goto exit_openport; }
if (_inet_create_port(lpszPortName, pPortMgr) == FALSE) {
DBG_MSG(DBG_LEV_ERROR, (TEXT("Err: InetmonOpenPort : Add Port Failed: LastError(%d)"), GetLastError())); SetLastError(ERROR_INVALID_NAME);
goto exit_openport; } }
if (pIniPort || (pIniPort = _inet_find_port(lpszPortName))) {
if (!pIniPort->m_bDeletePending) { //
// We stop increasing the ref count since the open handles
// won't have affect on the port deletion, i.e. the port
// may be deleted even when there are open handles.
//
// pIniPort->m_cRef ++;
//
// Increase the printer open handle ref count. This
// count is used to manage the life time of the PCINETMONPORT
// data structure. i.e. if the port is deleted when there are
// open handles, PCINETMONPORT is not freed until
//
// pIniPort->m_cPrinterRef == 0
//
pIniPort->IncPrinterRef ();
} else { DBG_MSG(DBG_LEV_ERROR, (TEXT("Err : InetmonOpenPort : Open deleted port: Port(%s)"), lpszPortName)); SetLastError(ERROR_INVALID_NAME); }
} else {
DBG_MSG(DBG_LEV_ERROR, (TEXT("Err : InetmonOpenPort : Invalid Name: Port(%s)"), lpszPortName)); SetLastError(ERROR_INVALID_NAME); } }
exit_openport:
return pIniPort; }
BOOL CInetMon::InetmonReleasePort( PCINETMONPORT pIniPort) { DBG_MSG(DBG_LEV_CALLTREE, (TEXT("Call: InetmonReleasePort: pIniPort(%08lX)"), pIniPort));
semCheckCrit();
// Validate the port and proceed to close the port.
//
pIniPort->DecPrinterRef ();
if (pIniPort->m_bDeletePending && pIniPort->m_cPrinterRef == 0) { //
// There is no open handles, free PCINETMONPORT
//
DBG_MSG(DBG_LEV_INFO, (TEXT("Info: InetmonReleasePort free pIniPort %p."), pIniPort));
delete pIniPort; }
return TRUE; }
/*****************************************************************************\
* InetmonClosePort * * Close the internet connection. * \*****************************************************************************/ BOOL CInetMon::InetmonClosePort( PCINETMONPORT pIniPort, HANDLE hPrinter) {
DBG_MSG(DBG_LEV_CALLTREE, (TEXT("Call: InetmonClosePort: pIniPort(%08lX)"), pIniPort));
semCheckCrit();
pIniPort->ClosePort (hPrinter);
if (pIniPort->m_bDeletePending && pIniPort->m_cPrinterRef == 0) { //
// There is no open handles, free PCINETMONPORT
//
DBG_MSG(DBG_LEV_INFO, (TEXT("Info: pIniPort Freed %p."), pIniPort));
delete pIniPort; }
return TRUE; }
/*****************************************************************************\
* InetmonEnumPorts * * Enumerate the ports registered in our list. * \*****************************************************************************/ BOOL CInetMon::InetmonEnumPorts( LPTSTR lpszServerName, DWORD dwLevel, LPBYTE pPorts, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { PCINETMONPORT pIniPort; DWORD cb; LPBYTE pEnd; DWORD dwError = ERROR_SUCCESS;
semCheckCrit();
// Traverse the list to build the size of the entire list.
//
cb = 0;
pIniPort = m_pPortList->pIniFirstPort;
while (pIniPort) {
cb += pIniPort->_inet_size_entry(dwLevel);
pIniPort = pIniPort->m_pNext; }
// Store the size of the list (This is the size needed).
//
*pcbNeeded = cb;
// If the size of the list is capable of being stored in the buffer
// passed in, then we can return the entries.
//
if (cb <= cbBuf) {
pEnd = pPorts + cbBuf;
*pcReturned = 0;
pIniPort = m_pPortList->pIniFirstPort;
while (pIniPort) {
pEnd = pIniPort->_inet_copy_entry(dwLevel, pPorts, pEnd);
switch (dwLevel) {
case PRINT_LEVEL_1: pPorts += sizeof(PORT_INFO_1); break;
case PRINT_LEVEL_2: pPorts += sizeof(PORT_INFO_2); break; }
pIniPort = pIniPort->m_pNext;
(*pcReturned)++; }
} else {
dwError = ERROR_INSUFFICIENT_BUFFER; }
if (dwError != ERROR_SUCCESS) { SetLastError(dwError); return FALSE; }
return TRUE; }
/*****************************************************************************\
* InetmonDeletePort * * Deletes a port from the INIMONPORT list. * \*****************************************************************************/ BOOL CInetMon::InetmonDeletePort( LPCTSTR lpszPortName, HWND hWnd, LPCTSTR lpszMonitorName) { BOOL bRet = FALSE;
semCheckCrit();
if (_inet_validate_portname(lpszPortName)) bRet = _inet_delete_port(lpszPortName);
return bRet; }
/*****************************************************************************\
* InetmonAddPort * * Adds a port to the INIMONPORT list. * \*****************************************************************************/ BOOL CInetMon::InetmonAddPort( LPCTSTR lpszPortName, LPCTSTR lpszMonitorName) { BOOL bRet = FALSE; PCPORTMGR pPortMgr = NULL;
semCheckCrit();
// If the port is not-found, then we can add it. Otherwise,
// the port already exists.
//
if (_inet_validate_portname(lpszPortName)) {
if (_inet_find_port(lpszPortName) == NULL) {
pPortMgr = new CPortMgr;
if (pPortMgr != NULL) { if (!pPortMgr->Init (lpszPortName)) { delete pPortMgr; pPortMgr = NULL; }
if (pPortMgr) { bRet = (_inet_create_port(lpszPortName, pPortMgr) != NULL); } } } }
return bRet; }
/*****************************************************************************\
* InetmonFindPort * * Looks for port in INIMONPORT list. * \*****************************************************************************/ PCINETMONPORT CInetMon::InetmonFindPort( LPCTSTR lpszPortName) { PCINETMONPORT hPort = NULL;
semCheckCrit();
if (_inet_validate_portname(lpszPortName)) hPort = _inet_find_port(lpszPortName) ;
return hPort; }
/********************************************************************************
** End of FIle (inetpp.c) ********************************************************************************/
|