|
|
/*++
Copyright (c) 1996-2003 Microsoft Corporation
Module Name:
comoem.cpp
Abstract:
Implementation of OEMGetInfo and OEMDevMode. Shared by all Unidrv OEM test dll's.
Environment:
Windows 2000, Windows XP, Windows Server 2003 Unidrv driver
Revision History:
Created it.
--*/
#ifndef _WIN32_WINNT
#define _WIN32_WINNT 0x0500
#endif
#include <tty.h>
#include <printoem.h>
#include <prntfont.h>
#include "..\inc\name.h"
#include <initguid.h>
#include <prcomoem.h>
#include <assert.h>
#include "oemcom.h"
#include "ttyui.h"
#include "debug.h"
#include <strsafe.h>
// Globals
static HMODULE g_hModule = NULL ; // DLL module handle
static long g_cComponents = 0 ; // Count of active components
static long g_cServerLocks = 0 ; // Count of locks
////////////////////////////////////////////////////////////////////////////////
//
// IOemCB body
//
HRESULT __stdcall IOemCB::QueryInterface(const IID& iid, void** ppv) { VERBOSE(DLLTEXT("IOemCB: QueryInterface entry\n")); if (iid == IID_IUnknown) { *ppv = static_cast<IUnknown*>(this); VERBOSE(DLLTEXT("IOemCB:Return pointer to IUnknown.\n")) ; } else if (iid == IID_IPrintOemUni2) { *ppv = static_cast<IPrintOemUni2*>(this) ; VERBOSE(DLLTEXT("IOemCB:Return pointer to IPrintOemUni.\n")) ; } else { *ppv = NULL ; WARNING(DLLTEXT("IOemCB:Return NULL.\n")) ; return E_NOINTERFACE ; } reinterpret_cast<IUnknown*>(*ppv)->AddRef() ; return S_OK ; }
ULONG __stdcall IOemCB::AddRef() { VERBOSE(DLLTEXT("IOemCB::AddRef() entry.\r\n")); return InterlockedIncrement(&m_cRef) ; }
ULONG __stdcall IOemCB::Release() { VERBOSE(DLLTEXT("IOemCB::Release() entry.\r\n")); ASSERT( 0 != m_cRef); ULONG cRef = InterlockedDecrement(&m_cRef); if (0 == cRef) { delete this; } return cRef; }
IOemCB::~IOemCB() { //
// Make sure that driver's helper function interface is released.
//
if(NULL != pOEMHelp) { pOEMHelp->Release(); pOEMHelp = NULL; }
//
// If this instance of the object is being deleted, then the reference
// count should be zero.
//
assert(0 == m_cRef) ; }
LONG __stdcall IOemCB::PublishDriverInterface( IUnknown *pIUnknown) { VERBOSE(DLLTEXT("IOemCB::PublishDriverInterface() entry.\r\n"));
//
// Need to store pointer to Driver Helper functions, if we already haven't.
//
if (this->pOEMHelp == NULL) { HRESULT hResult;
//
// Get Interface to Helper Functions.
//
hResult = pIUnknown->QueryInterface(IID_IPrintOemDriverUni, (void** ) &(this->pOEMHelp));
if(!SUCCEEDED(hResult)) { //
// Make sure that interface pointer reflects interface query failure.
//
this->pOEMHelp = NULL; } }
if (this->pOEMHelp) return S_OK; else return E_FAIL; }
LONG __stdcall IOemCB::EnableDriver(DWORD dwDriverVersion, DWORD cbSize, PDRVENABLEDATA pded) { VERBOSE(DLLTEXT("IOemCB::EnableDriver() entry.\r\n"));
if(OEMEnableDriver(dwDriverVersion, cbSize, pded) ) return S_OK; return E_FAIL; }
LONG __stdcall IOemCB::DisableDriver(VOID) { VERBOSE(DLLTEXT("IOemCB::DisaleDriver() entry.\r\n"));
if (this->pOEMHelp) { this->pOEMHelp->Release(); this->pOEMHelp = NULL ; } return S_OK; }
// the following OEM function is called by Unidrv to provide
// the OEM dll with an interface it can use to access Unidrv's
// helper functions like WriteSpoolBuf etc.
// warning: calling this more per IOemCB object than once will result
// in mismatched reference counts.
LONG __stdcall IOemCB::EnablePDEV( PDEVOBJ pdevobj, PWSTR pPrinterName, ULONG cPatterns, HSURF *phsurfPatterns, ULONG cjGdiInfo, GDIINFO *pGdiInfo, ULONG cjDevInfo, DEVINFO *pDevInfo, DRVENABLEDATA *pded, OUT PDEVOEM *ppDevOem) { VERBOSE(DLLTEXT("IOemCB::EnablePDEV() entry.\r\n"));
if(*ppDevOem = OEMEnablePDEV(pdevobj, pPrinterName, cPatterns, phsurfPatterns, cjGdiInfo, pGdiInfo, cjDevInfo, pDevInfo, pded) ) return S_OK; return E_FAIL;
}
LONG __stdcall IOemCB::ResetPDEV( PDEVOBJ pdevobjOld, PDEVOBJ pdevobjNew) {
return E_NOTIMPL; }
LONG __stdcall IOemCB::DisablePDEV( PDEVOBJ pdevobj) { VERBOSE(DLLTEXT("IOemCB::DisablePDEV() entry.\r\n"));
OEMDisablePDEV(pdevobj) ; return S_OK; };
LONG __stdcall IOemCB::GetInfo ( DWORD dwMode, PVOID pBuffer, DWORD cbSize, PDWORD pcbNeeded) { VERBOSE(DLLTEXT("IOemCB::GetInfo() entry.\r\n"));
if (OEMGetInfo(dwMode, pBuffer, cbSize, pcbNeeded)) return S_OK; else return E_FAIL; }
LONG __stdcall IOemCB::GetImplementedMethod( PSTR pMethodName) {
LONG lReturn = FALSE; VERBOSE(DLLTEXT("IOemCB::GetImplementedMethod() entry.\r\n")); VERBOSE(DLLTEXT(" Function:%s:"),pMethodName);
if (pMethodName == NULL) { lReturn = FALSE; } else { switch (*pMethodName) {
case (WCHAR)'C': if (!strcmp(pstrCommandCallback, pMethodName)) lReturn = TRUE; #if 0
else if (!strcmp(pstrCompression, pMethodName)) lReturn = TRUE; #endif
break;
case (WCHAR)'D': if (!strcmp(pstrDisableDriver, pMethodName)) lReturn = TRUE; else if (!strcmp(pstrDisablePDEV, pMethodName)) lReturn = TRUE; #if 0
else if (!strcmp(pstrDriverDMS, pMethodName)) lReturn = TRUE; #endif
#if 0
else if (!strcmp(pstrDevMode, pMethodName)) lReturn = TRUE; #endif
#if 0
else if (!strcmp(pstrDownloadFontHeader, pMethodName)) lReturn = TRUE; #endif
#if 0
else if (!strcmp(pstrDownloadCharGlyph, pMethodName)) lReturn = TRUE; #endif
break;
case (WCHAR)'E': if (!strcmp(pstrEnableDriver, pMethodName)) lReturn = TRUE; else if (!strcmp(pstrEnablePDEV, pMethodName)) lReturn = TRUE; break;
case (WCHAR)'F': #if 0
if (!strcmp(pstrFilterGraphics, pMethodName)) lReturn = TRUE; #endif
break;
case (WCHAR)'G': if (!strcmp(pstrGetInfo, pMethodName)) lReturn = TRUE; break;
case (WCHAR)'H': #if 0
if (!strcmp(pstrHalftonePattern, pMethodName)) lReturn = TRUE; #endif
break;
case (WCHAR)'I': #if 0
if (!strcmp(pstrImageProcessing, pMethodName)) lReturn = TRUE; #endif
break;
case (WCHAR)'M': #if 0
if (!strcmp(pstrMemoryUsage, pMethodName)) lReturn = TRUE; #endif
break;
case (WCHAR)'O':
if (!strcmp(pstrOutputCharStr, pMethodName)) lReturn = TRUE;
break;
case (WCHAR)'R': #if 0
if (!strcmp(pstrResetPDEV, pMethodName)) lReturn = TRUE; #endif
break;
case (WCHAR)'S':
if (!strcmp(pstrSendFontCmd, pMethodName)) lReturn = TRUE;
break;
case (WCHAR)'T': if (!strcmp(pstrTTYGetInfo, pMethodName)) lReturn = TRUE; #if 0
else if (!strcmp(pstrTextOutAsBitmap, pMethodName)) lReturn = TRUE; #endif
#if 0
else if (!strcmp(pstrTTDownloadMethod, pMethodName)) lReturn = TRUE; #endif
break; #if 0
case (WCHAR)'W': if (!strcmp(pstrWritePrinter,pMethodName) ) lReturn = TRUE; #endif
break; } }
if (lReturn) { VERBOSE(__TEXT("Supported\r\n")); return S_OK; } else { VERBOSE(__TEXT("NOT supported\r\n")); return S_FALSE; } }
LONG __stdcall IOemCB::DevMode( DWORD dwMode, POEMDMPARAM pOemDMParam) { VERBOSE(DLLTEXT("IOemCB::DevMode() entry.\r\n"));
return E_NOTIMPL;
}
LONG __stdcall IOemCB::CommandCallback( PDEVOBJ pdevobj, DWORD dwCallbackID, DWORD dwCount, PDWORD pdwParams, OUT INT *piResult) { VERBOSE(DLLTEXT("IOemCB::CommandCallback() entry.\r\n")); VERBOSE(DLLTEXT(" dwCallbackID = %d\r\n"), dwCallbackID); VERBOSE(DLLTEXT(" dwCount = %d\r\n"), dwCount);
DWORD dwResult; PREGSTRUCT pMyStuff; // sebset of pGlobals
PCMDSTR pSelectCmd = NULL ; // points to one of the command structures
// in pMyStuff.
*piResult = 0 ; // for all non movement commands
pMyStuff = (PREGSTRUCT)pdevobj->pdevOEM ;
switch(dwCallbackID) { case TTY_CB_BEGINJOB: pSelectCmd = &pMyStuff->BeginJob ; break; case TTY_CB_ENDJOB: pSelectCmd = &pMyStuff->EndJob ; break; case TTY_CB_PAPERSELECT: pSelectCmd = &pMyStuff->PaperSelect ; break; case TTY_CB_FEEDSELECT: pSelectCmd = &pMyStuff->FeedSelect ; break; case TTY_CB_BOLD_ON: pSelectCmd = &pMyStuff->Bold_ON ; break; case TTY_CB_BOLD_OFF: pSelectCmd = &pMyStuff->Bold_OFF ; break; case TTY_CB_UNDERLINE_ON: pSelectCmd = &pMyStuff->Underline_ON ; break; case TTY_CB_UNDERLINE_OFF: pSelectCmd = &pMyStuff->Underline_OFF ; break;
default: return S_OK; }
if(pSelectCmd) pOEMHelp->DrvWriteSpoolBuf(pdevobj, pSelectCmd->strCmd, pSelectCmd->dwLen, &dwResult);
#if 0
// this code is just to test new Unidrv helper functions.
pOEMHelp->DrvWriteAbortBuf(pdevobj, pSelectCmd->strCmd, pSelectCmd->dwLen, 3000);
// temp variables:
DWORD dwcursorPosX, dwcursorPosY, cbNeeded ; PBYTE pBuffer;
pOEMHelp->DrvGetStandardVariable( pdevobj, SVI_CURSORORIGINX, &dwcursorPosX, sizeof(DWORD), &cbNeeded);
pOEMHelp->DrvGetStandardVariable( pdevobj, SVI_CURSORORIGINY, &dwcursorPosY, sizeof(DWORD), &cbNeeded);
pOEMHelp->DrvGetGPDData( pdevobj, GPD_OEMCUSTOMDATA, // Type of the data
NULL, // reserved. Should be set to 0
NULL, // Caller allocated Buffer to be copied
0, // Size of the buffer
&cbNeeded // New Size of the buffer if needed.
) ;
pBuffer = (PBYTE)MemAlloc(cbNeeded);
pOEMHelp->DrvGetGPDData( pdevobj, GPD_OEMCUSTOMDATA, // Type of the data
NULL, // reserved. Should be set to 0
pBuffer, // Caller allocated Buffer to be copied
cbNeeded, // Size of the buffer
&cbNeeded // New Size of the buffer if needed.
) ;
pOEMHelp->DrvWriteSpoolBuf(pdevobj, pBuffer, cbNeeded, &dwResult);
MemFree(pBuffer);
#endif
return S_OK; }
LONG __stdcall IOemCB::ImageProcessing( PDEVOBJ pdevobj, PBYTE pSrcBitmap, PBITMAPINFOHEADER pBitmapInfoHeader, PBYTE pColorTable, DWORD dwCallbackID, PIPPARAMS pIPParams, OUT PBYTE *ppbResult) { VERBOSE(DLLTEXT("IOemCB::ImageProcessing() entry.\r\n")); return E_NOTIMPL ; /* if (*ppbResult = OEMImageProcessing(pdevobj, pSrcBitmap,
pBitmapInfoHeader, pColorTable, dwCallbackID, pIPParams)) return S_OK; else return E_FAIL; */ }
LONG __stdcall IOemCB::FilterGraphics( PDEVOBJ pdevobj, PBYTE pBuf, DWORD dwLen) { // DWORD dwResult;
VERBOSE(DLLTEXT("IOemCB::FilterGraphis() entry.\r\n")); return E_NOTIMPL ;
/* if (OEMFilterGraphics( pdevobj, pBuf, dwLen))
return S_OK; else return E_FAIL; */ }
LONG __stdcall IOemCB::Compression( PDEVOBJ pdevobj, PBYTE pInBuf, PBYTE pOutBuf, DWORD dwInLen, DWORD dwOutLen, OUT INT *piResult) { VERBOSE(DLLTEXT("IOemCB::Compression() entry.\r\n"));
return E_NOTIMPL; }
LONG __stdcall IOemCB::HalftonePattern( PDEVOBJ pdevobj, PBYTE pHTPattern, DWORD dwHTPatternX, DWORD dwHTPatternY, DWORD dwHTNumPatterns, DWORD dwCallbackID, PBYTE pResource, DWORD dwResourceSize) { VERBOSE(DLLTEXT("IOemCB::HalftonePattern() entry.\r\n"));
return E_NOTIMPL; }
LONG __stdcall IOemCB::MemoryUsage( PDEVOBJ pdevobj, POEMMEMORYUSAGE pMemoryUsage) { VERBOSE(DLLTEXT("IOemCB::MemoryUsage() entry.\r\n"));
return E_NOTIMPL; }
LONG __stdcall IOemCB::DownloadFontHeader( PDEVOBJ pdevobj, PUNIFONTOBJ pUFObj, OUT DWORD *pdwResult) { VERBOSE(DLLTEXT("IOemCB::DownloadFontHeader() entry.\r\n"));
return E_NOTIMPL; }
LONG __stdcall IOemCB::DownloadCharGlyph( PDEVOBJ pdevobj, PUNIFONTOBJ pUFObj, HGLYPH hGlyph, PDWORD pdwWidth, OUT DWORD *pdwResult) { VERBOSE(DLLTEXT("IOemCB::DownloadCharGlyph() entry.\r\n"));
return E_NOTIMPL; }
LONG __stdcall IOemCB::TTDownloadMethod( PDEVOBJ pdevobj, PUNIFONTOBJ pUFObj, OUT DWORD *pdwResult) { VERBOSE(DLLTEXT("IOemCB::TTDownloadMethod() entry.\r\n"));
return E_NOTIMPL; }
LONG __stdcall IOemCB::OutputCharStr( PDEVOBJ pdevobj, PUNIFONTOBJ pUFObj, DWORD dwType, DWORD dwCount, PVOID pGlyph) { GETINFO_GLYPHSTRING GStr; PTRANSDATA pTrans; DWORD dwI, dwGetInfo, dwResult, dwGlyphBufSiz, dwSpoolBufSiz, dwDst; PREGSTRUCT pMyStuff;
VERBOSE(DLLTEXT("OEMOutputCharStr() entry.\r\n"));
pMyStuff = (PREGSTRUCT)pdevobj->pdevOEM ;
// is the glyph buffers cache large enough?
dwGlyphBufSiz = dwCount * sizeof(TRANSDATA);
if(dwGlyphBufSiz > pMyStuff->dwGlyphBufSiz) // need to realloc mem.
{ if(pMyStuff->aubGlyphBuf) { MemFree(pMyStuff->aubGlyphBuf) ; pMyStuff->dwGlyphBufSiz = 0 ; } if(pMyStuff->aubGlyphBuf = (PBYTE)MemAlloc(dwGlyphBufSiz)) pMyStuff->dwGlyphBufSiz = dwGlyphBufSiz ; else return(E_FAIL); // unable to alloc needed buffers.
}
dwSpoolBufSiz = dwCount * sizeof(WORD); // assume worst case.
if(dwSpoolBufSiz > pMyStuff->dwSpoolBufSiz) // need to realloc mem.
{ if(pMyStuff->aubSpoolBuf) { MemFree(pMyStuff->aubSpoolBuf) ; pMyStuff->dwSpoolBufSiz = 0 ; } if(pMyStuff->aubSpoolBuf = (PBYTE)MemAlloc(dwSpoolBufSiz)) pMyStuff->dwSpoolBufSiz = dwSpoolBufSiz ; else return(E_FAIL); // unable to alloc needed buffers.
}
if(dwType != TYPE_GLYPHHANDLE) return(E_FAIL); // E_FAIL can only handle device fonts.
GStr.dwSize = sizeof(GETINFO_GLYPHSTRING); GStr.dwCount = dwCount; GStr.dwTypeIn = dwType; GStr.pGlyphIn = pGlyph; GStr.dwTypeOut = TYPE_TRANSDATA; GStr.pGlyphOut = pMyStuff->aubGlyphBuf ; GStr.dwGlyphOutSize = dwGlyphBufSiz; // instead of pMyStuff->dwGlyphBufSiz, which cause AV
dwGetInfo = GStr.dwSize; if (!pUFObj->pfnGetInfo(pUFObj, UFO_GETINFO_GLYPHSTRING, &GStr, dwGetInfo, &dwGetInfo)) { WARNING(DLLTEXT("UNIFONTOBJ_GetInfo:UFO_GETINFO_GLYPHSTRING failed.\r\n")); return(E_FAIL); }
pTrans = (PTRANSDATA)pMyStuff->aubGlyphBuf;
for (dwDst = dwI = 0 ; dwI < dwCount; dwI++, pTrans++) { VERBOSE(DLLTEXT("TYPE_TRANSDATA:ubCodePageID:0x%x\n"),pTrans->ubCodePageID); VERBOSE(DLLTEXT("TYPE_TRANSDATA:ubType:0x%x\n"),pTrans->ubType); switch (pTrans->ubType & MTYPE_FORMAT_MASK) { case MTYPE_DIRECT: pMyStuff->aubSpoolBuf[dwDst++] = pTrans->uCode.ubCode ; break; case MTYPE_PAIRED: VERBOSE(DLLTEXT("TYPE_TRANSDATA:ubPairs:0x%x\n"),*(PWORD)(pTrans->uCode.ubPairs)); pMyStuff->aubSpoolBuf[dwDst++] = pTrans->uCode.ubPairs[0] ; pMyStuff->aubSpoolBuf[dwDst++] = pTrans->uCode.ubPairs[1] ; break; default: return (E_FAIL); // no other MTYPE is supported.
break; } }
// send dst bytes to printer.
pOEMHelp->DrvWriteSpoolBuf(pdevobj, pMyStuff->aubSpoolBuf, dwDst, &dwResult);
if(dwResult == dwDst) return S_OK; return E_FAIL ; }
LONG __stdcall IOemCB::SendFontCmd( PDEVOBJ pdevobj, PUNIFONTOBJ pUFObj, PFINVOCATION pFInv) { VERBOSE(DLLTEXT("IOemCB::SendFontCmd() entry.\r\n"));
PREGSTRUCT pMyStuff; // sebset of pGlobals
PCMDSTR pSelectCmd ; // points to one of the font selection
// command structures in pMyStuff.
DWORD dwResult;
// I expect pFInv to contain either "10", "12" or "17"
pMyStuff = (PREGSTRUCT)pdevobj->pdevOEM ;
if(pFInv->dwCount == 2 && pFInv->pubCommand[0] == '1') { if (pFInv->pubCommand[1] == '2') pSelectCmd = &pMyStuff->Sel_12_cpi ; else if (pFInv->pubCommand[1] == '7') pSelectCmd = &pMyStuff->Sel_17_cpi ; else if (pFInv->pubCommand[1] == '0') pSelectCmd = &pMyStuff->Sel_10_cpi ; else return S_OK; // maybe asking to deselect a font. Don't do anything!
} else return S_OK; // maybe asking to deselect a font. Don't do anything!
pOEMHelp->DrvWriteSpoolBuf(pdevobj, pSelectCmd->strCmd, pSelectCmd->dwLen, &dwResult);
return S_OK; }
LONG __stdcall IOemCB::DriverDMS( PVOID pDevObj, PVOID pBuffer, DWORD cbSize, PDWORD pcbNeeded) { VERBOSE(DLLTEXT("IOemCB::DriverDMS() entry.\r\n"));
return E_NOTIMPL; }
LONG __stdcall IOemCB::TextOutAsBitmap( SURFOBJ *pso, STROBJ *pstro, FONTOBJ *pfo, CLIPOBJ *pco, RECTL *prclExtra, RECTL *prclOpaque, BRUSHOBJ *pboFore, BRUSHOBJ *pboOpaque, POINTL *pptlOrg, MIX mix) { VERBOSE(DLLTEXT("IOemCB::TextOutAsBitmap() entry.\r\n"));
return E_NOTIMPL; }
LONG __stdcall IOemCB::TTYGetInfo( PDEVOBJ pdevobj, DWORD dwInfoIndex, PVOID pOutputBuf, DWORD dwSize, DWORD *pcbcNeeded ) { VERBOSE(DLLTEXT("IOemCB::TTYGetInfo() entry.\r\n"));
if(OEMTTYGetInfo( pdevobj, dwInfoIndex, pOutputBuf, dwSize, pcbcNeeded) ) return S_OK; return E_FAIL ; }
LONG __stdcall IOemCB:: WritePrinter ( PDEVOBJ pdevobj, PVOID pBuf, DWORD cbBuffer, PDWORD pcbWritten) { VERBOSE(DLLTEXT("IOemCB::WritePrinter() entry.\r\n"));
return E_NOTIMPL; }
////////////////////////////////////////////////////////////////////////////////
//
// oem class factory
//
class IOemCF : public IClassFactory { public: // *** IUnknown methods ***
STDMETHOD(QueryInterface) (THIS_ REFIID riid, LPVOID FAR* ppvObj); STDMETHOD_(ULONG,AddRef) (THIS); STDMETHOD_(ULONG,Release) (THIS);
// *** IClassFactory methods ***
STDMETHOD(CreateInstance) (THIS_ LPUNKNOWN pUnkOuter, REFIID riid, LPVOID FAR* ppvObject); STDMETHOD(LockServer) (THIS_ BOOL bLock);
// Constructor
IOemCF(): m_cRef(1) { }; ~IOemCF() { };
protected: LONG m_cRef;
};
///////////////////////////////////////////////////////////
//
// Class factory body
//
HRESULT __stdcall IOemCF::QueryInterface(const IID& iid, void** ppv) { if ((iid == IID_IUnknown) || (iid == IID_IClassFactory)) { *ppv = static_cast<IOemCF*>(this) ; } else { *ppv = NULL ; return E_NOINTERFACE ; } reinterpret_cast<IUnknown*>(*ppv)->AddRef() ; return S_OK ; }
ULONG __stdcall IOemCF::AddRef() { return InterlockedIncrement(&m_cRef) ; }
ULONG __stdcall IOemCF::Release() { ASSERT( 0 != m_cRef); ULONG cRef = InterlockedDecrement(&m_cRef); if (0 == cRef) { delete this; } return cRef; }
// IClassFactory implementation
HRESULT __stdcall IOemCF::CreateInstance(IUnknown* pUnknownOuter, const IID& iid, void** ppv) { VERBOSE(DLLTEXT("Class factory:\t\tCreate component.")) ;
// Cannot aggregate.
if (pUnknownOuter != NULL) { return CLASS_E_NOAGGREGATION ; }
// Create component.
IOemCB* pOemCB = new IOemCB ; if (pOemCB == NULL) { return E_OUTOFMEMORY ; }
// Get the requested interface.
HRESULT hr = pOemCB->QueryInterface(iid, ppv) ;
// Release the IUnknown pointer.
// (If QueryInterface failed, component will delete itself.)
pOemCB->Release() ; return hr ; }
// LockServer
HRESULT __stdcall IOemCF::LockServer(BOOL bLock) { if (bLock) { InterlockedIncrement(&g_cServerLocks) ; } else { InterlockedDecrement(&g_cServerLocks) ; } return S_OK ; }
///////////////////////////////////////////////////////////
//
// Registration functions
// Testing purpose
//
//
// Can DLL unload now?
//
STDAPI DllCanUnloadNow() { if ((g_cComponents == 0) && (g_cServerLocks == 0)) { return S_OK ; } else { return S_FALSE ; } }
//
// Get class factory
//
STDAPI DllGetClassObject(const CLSID& clsid, const IID& iid, void** ppv) { VERBOSE(DLLTEXT("DllGetClassObject:\tCreate class factory.")) ;
// Can we create this component?
if (clsid != CLSID_OEMRENDER) { return CLASS_E_CLASSNOTAVAILABLE ; }
// Create class factory.
IOemCF* pFontCF = new IOemCF ; // Reference count set to 1
// in constructor
if (pFontCF == NULL) { return E_OUTOFMEMORY ; }
// Get requested interface.
HRESULT hr = pFontCF->QueryInterface(iid, ppv) ; pFontCF->Release() ;
return hr ; }
///////////////////////////////////////////////////////////
//
// DLL module information
//
BOOL APIENTRY DllMain(HANDLE hModule, DWORD dwReason, void* lpReserved) { if (dwReason == DLL_PROCESS_ATTACH) { g_hModule = (HMODULE)hModule ; } g_hModule = (HMODULE)hModule ;
return SUCCEEDED(S_OK); }
|