mirror of https://github.com/tongzx/nt5src
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.
1402 lines
35 KiB
1402 lines
35 KiB
/*++
|
|
|
|
Copyright (c) 1996 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
oemui.c
|
|
|
|
Abstract:
|
|
|
|
Support for OEM plugin UI modules
|
|
|
|
Environment:
|
|
|
|
Windows NT printer driver
|
|
|
|
Revision History:
|
|
|
|
02/13/97 -davidx-
|
|
Created it.
|
|
|
|
--*/
|
|
|
|
#include "precomp.h"
|
|
|
|
//
|
|
// User mode helper functions for OEM plugins
|
|
//
|
|
|
|
const OEMUIPROCS OemUIHelperFuncs = {
|
|
(PFN_DrvGetDriverSetting) BGetDriverSettingForOEM,
|
|
(PFN_DrvUpdateUISetting) BUpdateUISettingForOEM,
|
|
};
|
|
|
|
|
|
|
|
BOOL
|
|
BPackOemPluginItems(
|
|
PUIDATA pUiData
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Call OEM plugin UI modules to let them add their OPTITEMs
|
|
|
|
Arguments:
|
|
|
|
pUiData - Points to UIDATA structure
|
|
|
|
Return Value:
|
|
|
|
TRUE if successful, FALSE if there is an error
|
|
|
|
--*/
|
|
|
|
{
|
|
PCOMMONINFO pci;
|
|
PFN_OEMCommonUIProp pfnOEMCommonUIProp;
|
|
POEMCUIPPARAM pOemCUIPParam;
|
|
POPTITEM pOptItem;
|
|
DWORD dwOptItem;
|
|
|
|
//
|
|
// Check if we're being called for the first time.
|
|
// We assume all OEM plugin items are always packed at the end.
|
|
//
|
|
|
|
if (pUiData->pOptItem == NULL)
|
|
pUiData->dwDrvOptItem = pUiData->dwOptItem;
|
|
else if (pUiData->dwDrvOptItem != pUiData->dwOptItem)
|
|
{
|
|
RIP(("Inconsistent OPTITEM count for driver items\n"));
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// Quick exit for no OEM plugin case
|
|
//
|
|
|
|
pci = (PCOMMONINFO) pUiData;
|
|
|
|
if (pci->pOemPlugins->dwCount == 0)
|
|
return TRUE;
|
|
|
|
pOptItem = pUiData->pOptItem;
|
|
|
|
FOREACH_OEMPLUGIN_LOOP(pci)
|
|
|
|
if (!HAS_COM_INTERFACE(pOemEntry) &&
|
|
!(pfnOEMCommonUIProp = GET_OEM_ENTRYPOINT(pOemEntry, OEMCommonUIProp)))
|
|
continue;
|
|
|
|
//
|
|
// Compose the input parameter for calling OEMCommonUI
|
|
//
|
|
|
|
pOemCUIPParam = pOemEntry->pParam;
|
|
|
|
if (pOemCUIPParam == NULL)
|
|
{
|
|
//
|
|
// Allocate memory for an OEMUI_PARAM structure
|
|
// during the first pass
|
|
//
|
|
|
|
if (pOptItem != NULL)
|
|
continue;
|
|
|
|
if (! (pOemCUIPParam = HEAPALLOC(pci->hHeap, sizeof(OEMCUIPPARAM))))
|
|
{
|
|
ERR(("Memory allocation failed\n"));
|
|
return FALSE;
|
|
}
|
|
|
|
pOemEntry->pParam = pOemCUIPParam;
|
|
pOemCUIPParam->cbSize = sizeof(OEMCUIPPARAM);
|
|
pOemCUIPParam->poemuiobj = pci->pOemPlugins->pdriverobj;
|
|
pOemCUIPParam->hPrinter = pci->hPrinter;
|
|
pOemCUIPParam->pPrinterName = pci->pPrinterName;
|
|
pOemCUIPParam->hModule = pOemEntry->hInstance;
|
|
pOemCUIPParam->hOEMHeap = pci->hHeap;
|
|
pOemCUIPParam->pPublicDM = pci->pdm;
|
|
pOemCUIPParam->pOEMDM = pOemEntry->pOEMDM;
|
|
}
|
|
|
|
pOemCUIPParam->pDrvOptItems = pUiData->pDrvOptItem;
|
|
pOemCUIPParam->cDrvOptItems = pUiData->dwDrvOptItem;
|
|
pOemCUIPParam->pOEMOptItems = pOptItem;
|
|
dwOptItem = pOemCUIPParam->cOEMOptItems;
|
|
|
|
//
|
|
// Actually call OEMCommonUI entrypoint
|
|
//
|
|
|
|
if (HAS_COM_INTERFACE(pOemEntry))
|
|
{
|
|
HRESULT hr;
|
|
|
|
hr = HComOEMCommonUIProp(
|
|
pOemEntry,
|
|
(pUiData->iMode == MODE_DOCUMENT_STICKY) ? OEMCUIP_DOCPROP : OEMCUIP_PRNPROP,
|
|
pOemCUIPParam);
|
|
|
|
if (hr == E_NOTIMPL)
|
|
{
|
|
HeapFree(pci->hHeap, 0, pOemCUIPParam);
|
|
pOemEntry->pParam = NULL;
|
|
continue;
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
ERR(("OEMCommonUI failed for '%ws': %d\n",
|
|
CURRENT_OEM_MODULE_NAME(pOemEntry),
|
|
GetLastError()));
|
|
|
|
//
|
|
// OEM failure during the first pass is recoverable:
|
|
// we'll simply ignore OEM plugin items
|
|
//
|
|
|
|
if (pOptItem == NULL)
|
|
{
|
|
HeapFree(pci->hHeap, 0, pOemCUIPParam);
|
|
pOemEntry->pParam = NULL;
|
|
continue;
|
|
}
|
|
return FALSE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (!pfnOEMCommonUIProp(
|
|
(pUiData->iMode == MODE_DOCUMENT_STICKY) ? OEMCUIP_DOCPROP : OEMCUIP_PRNPROP,
|
|
pOemCUIPParam))
|
|
{
|
|
ERR(("OEMCommonUI failed for '%ws': %d\n",
|
|
CURRENT_OEM_MODULE_NAME(pOemEntry),
|
|
GetLastError()));
|
|
#if 0
|
|
(VOID) IDisplayErrorMessageBox(
|
|
NULL,
|
|
0,
|
|
IDS_OEMERR_DLGTITLE,
|
|
IDS_OEMERR_OPTITEM,
|
|
CURRENT_OEM_MODULE_NAME(pOemEntry));
|
|
#endif
|
|
//
|
|
// OEM failure during the first pass is recoverable:
|
|
// we'll simply ignore OEM plugin items
|
|
//
|
|
|
|
if (pOptItem == NULL)
|
|
{
|
|
HeapFree(pci->hHeap, 0, pOemCUIPParam);
|
|
pOemEntry->pParam = NULL;
|
|
continue;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
if (pOptItem != NULL)
|
|
{
|
|
//
|
|
// second pass - ensure the number of items is consistent
|
|
//
|
|
|
|
if (dwOptItem != pOemCUIPParam->cOEMOptItems)
|
|
{
|
|
RIP(("Inconsistent OPTITEM count reported by OEM plugin: %ws\n",
|
|
CURRENT_OEM_MODULE_NAME(pOemEntry),
|
|
GetLastError()));
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
pOptItem += pOemCUIPParam->cOEMOptItems;
|
|
pUiData->pOptItem += pOemCUIPParam->cOEMOptItems;
|
|
}
|
|
|
|
pUiData->dwOptItem += pOemCUIPParam->cOEMOptItems;
|
|
|
|
END_OEMPLUGIN_LOOP
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
|
|
LONG
|
|
LInvokeOemPluginCallbacks(
|
|
PUIDATA pUiData,
|
|
PCPSUICBPARAM pCallbackParam,
|
|
LONG lRet
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Call OEM plugin module's callback function
|
|
|
|
Arguments:
|
|
|
|
pUiData - Points to UIDATA structure
|
|
pCallbackParam - Points to callback parameter from compstui
|
|
lRet - Return value after the driver has processed the callback
|
|
|
|
Return Value:
|
|
|
|
Return value for compstui
|
|
|
|
--*/
|
|
|
|
{
|
|
PCOMMONINFO pci = (PCOMMONINFO) pUiData;
|
|
POEMCUIPPARAM pOemCUIPParam;
|
|
LONG lNewResult;
|
|
|
|
//
|
|
// Quick exit for no OEM plugin case
|
|
//
|
|
|
|
if (pci->pOemPlugins->dwCount == 0)
|
|
return lRet;
|
|
|
|
//
|
|
// Go through each OEM plugin UI module
|
|
//
|
|
|
|
FOREACH_OEMPLUGIN_LOOP(pci)
|
|
|
|
//
|
|
// Stop when anyone says don't exit
|
|
//
|
|
|
|
if (lRet == CPSUICB_ACTION_NO_APPLY_EXIT)
|
|
{
|
|
ASSERT(pCallbackParam->Reason == CPSUICB_REASON_APPLYNOW);
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Get the address of OEM callback function and call it
|
|
//
|
|
|
|
pOemCUIPParam = pOemEntry->pParam;
|
|
|
|
if (pOemCUIPParam == NULL || pOemCUIPParam->OEMCUIPCallback == NULL)
|
|
continue;
|
|
|
|
lNewResult = pOemCUIPParam->OEMCUIPCallback(pCallbackParam, pOemCUIPParam);
|
|
|
|
//
|
|
// Merge the new result with the existing result
|
|
//
|
|
|
|
switch (lNewResult)
|
|
{
|
|
case CPSUICB_ACTION_ITEMS_APPLIED:
|
|
case CPSUICB_ACTION_NO_APPLY_EXIT:
|
|
|
|
ASSERT(pCallbackParam->Reason == CPSUICB_REASON_APPLYNOW);
|
|
lRet = lNewResult;
|
|
break;
|
|
|
|
case CPSUICB_ACTION_REINIT_ITEMS:
|
|
|
|
ASSERT(pCallbackParam->Reason != CPSUICB_REASON_APPLYNOW);
|
|
lRet = lNewResult;
|
|
break;
|
|
|
|
case CPSUICB_ACTION_OPTIF_CHANGED:
|
|
|
|
ASSERT(pCallbackParam->Reason != CPSUICB_REASON_APPLYNOW);
|
|
if (lRet == CPSUICB_ACTION_NONE)
|
|
lRet = lNewResult;
|
|
break;
|
|
|
|
case CPSUICB_ACTION_NONE:
|
|
break;
|
|
|
|
default:
|
|
|
|
RIP(("Invalid return value from OEM callback: '%ws'\n",
|
|
CURRENT_OEM_MODULE_NAME(pOemEntry),
|
|
GetLastError()));
|
|
break;
|
|
}
|
|
|
|
END_OEMPLUGIN_LOOP
|
|
|
|
return lRet;
|
|
}
|
|
|
|
|
|
LRESULT
|
|
OEMDocumentPropertySheets(
|
|
PPROPSHEETUI_INFO pPSUIInfo,
|
|
LPARAM lParam
|
|
)
|
|
{
|
|
HRESULT hr;
|
|
|
|
POEM_PLUGIN_ENTRY pOemEntry;
|
|
|
|
pOemEntry = ((POEMUIPSPARAM)(pPSUIInfo->lParamInit))->pOemEntry;
|
|
|
|
hr = HComOEMDocumentPropertySheets(pOemEntry,
|
|
pPSUIInfo,
|
|
lParam);
|
|
|
|
if (SUCCEEDED(hr))
|
|
return 1;
|
|
|
|
return -1;
|
|
}
|
|
|
|
LRESULT
|
|
OEMDevicePropertySheets(
|
|
PPROPSHEETUI_INFO pPSUIInfo,
|
|
LPARAM lParam
|
|
)
|
|
{
|
|
HRESULT hr;
|
|
|
|
POEM_PLUGIN_ENTRY pOemEntry;
|
|
|
|
pOemEntry = ((POEMUIPSPARAM)(pPSUIInfo->lParamInit))->pOemEntry;
|
|
|
|
hr = HComOEMDevicePropertySheets(pOemEntry,
|
|
pPSUIInfo,
|
|
lParam);
|
|
|
|
if (SUCCEEDED(hr))
|
|
return 1;
|
|
|
|
return -1;
|
|
}
|
|
|
|
|
|
BOOL
|
|
BAddOemPluginPages(
|
|
PUIDATA pUiData,
|
|
DWORD dwFlags
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Call OEM plugin UI modules to let them add their own property sheet pages
|
|
|
|
Arguments:
|
|
|
|
pUiData - Points to UIDATA structure
|
|
dwFlags - Flags from DOCUMENTPROPERTYHEADER or DEVICEPROPERTYHEADER
|
|
|
|
Return Value:
|
|
|
|
TRUE if successful, FALSE if there is an error
|
|
|
|
--*/
|
|
|
|
{
|
|
PCOMMONINFO pci = (PCOMMONINFO) pUiData;
|
|
FARPROC pfnOEMPropertySheets;
|
|
POEMUIPSPARAM pOemUIPSParam;
|
|
|
|
//
|
|
// Quick exit for no OEM plugin case
|
|
//
|
|
|
|
if (pci->pOemPlugins->dwCount == 0)
|
|
return TRUE;
|
|
|
|
//
|
|
// Add the property sheet for each OEM plugin UI module
|
|
//
|
|
|
|
FOREACH_OEMPLUGIN_LOOP(pci)
|
|
|
|
//
|
|
// get the address of appropriate OEM entrypoint
|
|
//
|
|
|
|
|
|
if (HAS_COM_INTERFACE(pOemEntry))
|
|
{
|
|
if (pUiData->iMode == MODE_DOCUMENT_STICKY)
|
|
pfnOEMPropertySheets = (FARPROC)OEMDocumentPropertySheets;
|
|
else
|
|
pfnOEMPropertySheets = (FARPROC)OEMDevicePropertySheets;
|
|
}
|
|
else
|
|
{
|
|
if (pUiData->iMode == MODE_DOCUMENT_STICKY)
|
|
{
|
|
pfnOEMPropertySheets = (FARPROC)
|
|
GET_OEM_ENTRYPOINT(pOemEntry, OEMDocumentPropertySheets);
|
|
}
|
|
else
|
|
{
|
|
pfnOEMPropertySheets = (FARPROC)
|
|
GET_OEM_ENTRYPOINT(pOemEntry, OEMDevicePropertySheets);
|
|
}
|
|
|
|
if (pfnOEMPropertySheets == NULL)
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// Collect input parameters to be passed to OEM plugin
|
|
//
|
|
|
|
if ((pOemUIPSParam = HEAPALLOC(pci->hHeap, sizeof(OEMUIPSPARAM))) == NULL)
|
|
{
|
|
ERR(("Memory allocation failed\n"));
|
|
return FALSE;
|
|
}
|
|
|
|
pOemUIPSParam->cbSize = sizeof(OEMUIPSPARAM);
|
|
pOemUIPSParam->poemuiobj = pci->pOemPlugins->pdriverobj;
|
|
pOemUIPSParam->hPrinter = pci->hPrinter;
|
|
pOemUIPSParam->pPrinterName = pci->pPrinterName;
|
|
pOemUIPSParam->hModule = pOemEntry->hInstance;
|
|
pOemUIPSParam->hOEMHeap = pci->hHeap;
|
|
pOemUIPSParam->pPublicDM = pci->pdm;
|
|
pOemUIPSParam->pOEMDM = pOemEntry->pOEMDM;
|
|
pOemUIPSParam->dwFlags = dwFlags;
|
|
pOemUIPSParam->pOemEntry = pOemEntry;
|
|
|
|
//
|
|
// call compstui to add the OEM plugin property sheets
|
|
//
|
|
|
|
if (pUiData->pfnComPropSheet(pUiData->hComPropSheet,
|
|
CPSFUNC_ADD_PFNPROPSHEETUI,
|
|
(LPARAM) pfnOEMPropertySheets,
|
|
(LPARAM) pOemUIPSParam) <= 0)
|
|
{
|
|
VERBOSE(("Couldn't add property sheet pages for '%ws'\n",
|
|
CURRENT_OEM_MODULE_NAME(pOemEntry),
|
|
GetLastError()));
|
|
#if 0
|
|
(VOID) IDisplayErrorMessageBox(
|
|
NULL,
|
|
0,
|
|
IDS_OEMERR_DLGTITLE,
|
|
IDS_OEMERR_PROPSHEET,
|
|
CURRENT_OEM_MODULE_NAME(pOemEntry));
|
|
#endif
|
|
}
|
|
|
|
END_OEMPLUGIN_LOOP
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
|
|
BOOL
|
|
APIENTRY
|
|
BGetDriverSettingForOEM(
|
|
PCOMMONINFO pci,
|
|
PCSTR pFeatureKeyword,
|
|
PVOID pOutput,
|
|
DWORD cbSize,
|
|
PDWORD pcbNeeded,
|
|
PDWORD pdwOptionsReturned
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Provide OEM plugins access to driver private settings
|
|
|
|
Arguments:
|
|
|
|
pci - Points to basic printer information
|
|
pFeatureKeyword - Specifies the keyword the caller is interested in
|
|
pOutput - Points to output buffer
|
|
cbSize - Size of output buffer
|
|
pcbNeeded - Returns the expected size of output buffer
|
|
pdwOptionsReturned - Returns the number of options selected
|
|
|
|
Return Value:
|
|
|
|
TRUE if successful, FALSE if there is an error
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG_PTR dwIndex;
|
|
BOOL bResult;
|
|
|
|
ASSERT(pci->pvStartSign == pci);
|
|
|
|
//
|
|
// This is not very portable: If the pointer value for pFeatureKeyword
|
|
// is less than 0x10000, we assume that the pointer value actually
|
|
// specifies a predefined index.
|
|
//
|
|
|
|
//
|
|
// Following ASSERT is removed for Win64
|
|
//
|
|
// ASSERT(sizeof(pFeatureKeyword) == sizeof(DWORD));
|
|
//
|
|
|
|
dwIndex = (ULONG_PTR)pFeatureKeyword;
|
|
|
|
if (dwIndex >= OEMGDS_MIN_DOCSTICKY && dwIndex < OEMGDS_MIN_PRINTERSTICKY)
|
|
{
|
|
if (pci->pdm == NULL)
|
|
goto setting_not_available;
|
|
|
|
bResult = BGetDevmodeSettingForOEM(
|
|
pci->pdm,
|
|
(DWORD)dwIndex,
|
|
pOutput,
|
|
cbSize,
|
|
pcbNeeded);
|
|
|
|
if (bResult)
|
|
*pdwOptionsReturned = 1;
|
|
}
|
|
else if (dwIndex >= OEMGDS_MIN_PRINTERSTICKY && dwIndex < OEMGDS_MAX)
|
|
{
|
|
if (pci->pPrinterData == NULL)
|
|
goto setting_not_available;
|
|
|
|
bResult = BGetPrinterDataSettingForOEM(
|
|
pci->pPrinterData,
|
|
(DWORD)dwIndex,
|
|
pOutput,
|
|
cbSize,
|
|
pcbNeeded);
|
|
|
|
if (bResult)
|
|
*pdwOptionsReturned = 1;
|
|
}
|
|
else
|
|
{
|
|
if (pci->pCombinedOptions == NULL)
|
|
goto setting_not_available;
|
|
|
|
bResult = BGetGenericOptionSettingForOEM(
|
|
pci->pUIInfo,
|
|
pci->pCombinedOptions,
|
|
pFeatureKeyword,
|
|
pOutput,
|
|
cbSize,
|
|
pcbNeeded,
|
|
pdwOptionsReturned);
|
|
}
|
|
|
|
return bResult;
|
|
|
|
setting_not_available:
|
|
|
|
WARNING(("Requested driver setting not available: %d\n", pFeatureKeyword));
|
|
SetLastError(ERROR_NOT_SUPPORTED);
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
BOOL
|
|
BUpdateUISettingForOEM(
|
|
PCOMMONINFO pci,
|
|
PVOID pOptItem,
|
|
DWORD dwPreviousSelection,
|
|
DWORD dwMode
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Update the UI settings in optionsarray for OEM.
|
|
|
|
Arguments:
|
|
|
|
pci - Points to basic printer information
|
|
pOptItem - Points to the current OPTITEM
|
|
|
|
Return Value:
|
|
|
|
TRUE if successful, FALSE if there is an error such as conflict and
|
|
user wants to cancel.
|
|
|
|
--*/
|
|
|
|
{
|
|
POPTITEM pCurItem = pOptItem;
|
|
PUIDATA pUiData = (PUIDATA)pci;
|
|
|
|
if (ICheckConstraintsDlg(pUiData, pCurItem, 1, FALSE) == CONFLICT_CANCEL)
|
|
{
|
|
//
|
|
// If there is a conflict and the user clicked
|
|
// CANCEL to restore the original selection.
|
|
// CONFLICT_CANCEL, restore the old setting
|
|
//
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
if (dwMode == OEMCUIP_DOCPROP)
|
|
{
|
|
//
|
|
// We use FLAG_WITHIN_PLUGINCALL to indicate we are within the UI helper
|
|
// function call issued by OEM plugin. This is needed to fix bug #90923.
|
|
//
|
|
|
|
pUiData->ci.dwFlags |= FLAG_WITHIN_PLUGINCALL;
|
|
VUnpackDocumentPropertiesItems(pUiData, pCurItem, 1);
|
|
pUiData->ci.dwFlags &= ~FLAG_WITHIN_PLUGINCALL;
|
|
|
|
VPropShowConstraints(pUiData, MODE_DOCANDPRINTER_STICKY);
|
|
}
|
|
else
|
|
{
|
|
VUpdateOptionsArrayWithSelection(pUiData, pCurItem);
|
|
VPropShowConstraints(pUiData, MODE_PRINTER_STICKY);
|
|
}
|
|
|
|
//
|
|
// Record the fact that one of our OPTITEM selection has been changed by plugin's
|
|
// call of helper function DrvUpdateUISetting. This is necessary so that at the
|
|
// APPLYNOW time we know constraints could exist even though user hasn't touched
|
|
// any of our OPTITEMs.
|
|
//
|
|
|
|
pUiData->ci.dwFlags |= FLAG_PLUGIN_CHANGED_OPTITEM;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL
|
|
BUpgradeRegistrySettingForOEM(
|
|
HANDLE hPrinter,
|
|
PCSTR pFeatureKeyword,
|
|
PCSTR pOptionKeyword
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Set the Feature.Option request to our options array. OEM will only
|
|
call this function at OEMUpgradeDriver to upgrade their registry setttings
|
|
into our optionsarray saved in our PRINTERDATA
|
|
|
|
Arguments:
|
|
|
|
hPrinter - Handle of the Printer
|
|
pFeatureKeyword - Specifies the keyword the caller is interested in
|
|
pOptionKeyword - Specifies the keyword the caller is interested in
|
|
|
|
Return Value:
|
|
|
|
TRUE if successful, FALSE if there is an error
|
|
|
|
--*/
|
|
|
|
{
|
|
|
|
PFEATURE pFeature;
|
|
POPTION pOption;
|
|
DWORD dwFeatureCount, i, j;
|
|
BOOL bFeatureFound, bOptionFound, bResult = FALSE;
|
|
PCSTR pKeywordName;
|
|
POPTSELECT pOptionsArray = NULL;
|
|
PDRIVER_INFO_3 pDriverInfo3 = NULL;
|
|
PRAWBINARYDATA pRawData = NULL;
|
|
PINFOHEADER pInfoHeader = NULL;
|
|
PUIINFO pUIInfo = NULL;
|
|
PPRINTERDATA pPrinterData = NULL;
|
|
OPTSELECT DocOptions[MAX_PRINTER_OPTIONS];
|
|
|
|
//
|
|
// Get information about the printer driver
|
|
//
|
|
|
|
bResult = bFeatureFound = bOptionFound = FALSE;
|
|
|
|
if ((pDriverInfo3 = MyGetPrinterDriver(hPrinter, NULL, 3)) == NULL)
|
|
{
|
|
ERR(("Cannot get printer driver info: %d\n", GetLastError()));
|
|
goto upgrade_registry_exit;
|
|
}
|
|
|
|
// ENTER_CRITICAL_SECTION();
|
|
|
|
pRawData = LoadRawBinaryData(pDriverInfo3->pDataFile);
|
|
|
|
// LEAVE_CRITICAL_SECTION();
|
|
|
|
if (pRawData == NULL)
|
|
goto upgrade_registry_exit;
|
|
|
|
if (!(pPrinterData = MemAllocZ(sizeof(PRINTERDATA))) ||
|
|
!( BGetPrinterProperties(hPrinter, pRawData, pPrinterData)))
|
|
{
|
|
|
|
ERR(("Cannot get printer data info: %d\n", GetLastError()));
|
|
goto upgrade_registry_exit;
|
|
}
|
|
|
|
//
|
|
// Allocate memory for combined optionsarray
|
|
//
|
|
|
|
if (!(pOptionsArray = MemAllocZ(MAX_COMBINED_OPTIONS * sizeof (OPTSELECT))))
|
|
goto upgrade_registry_exit;
|
|
|
|
if (! InitDefaultOptions(pRawData,
|
|
DocOptions,
|
|
MAX_PRINTER_OPTIONS,
|
|
MODE_DOCUMENT_STICKY))
|
|
{
|
|
goto upgrade_registry_exit;
|
|
}
|
|
|
|
//
|
|
// Combine doc sticky options with printer sticky items
|
|
//
|
|
|
|
CombineOptionArray(pRawData, pOptionsArray, MAX_COMBINED_OPTIONS, DocOptions, pPrinterData->aOptions);
|
|
|
|
//
|
|
// Get an updated instance of printer description data
|
|
//
|
|
|
|
pInfoHeader = InitBinaryData(pRawData,
|
|
NULL,
|
|
pOptionsArray);
|
|
|
|
if (pInfoHeader == NULL)
|
|
{
|
|
ERR(("InitBinaryData failed\n"));
|
|
goto upgrade_registry_exit;
|
|
}
|
|
|
|
if (!(pUIInfo = OFFSET_TO_POINTER(pInfoHeader, pInfoHeader->loUIInfoOffset)))
|
|
goto upgrade_registry_exit;
|
|
|
|
//
|
|
// Look for feature.option index
|
|
//
|
|
|
|
pFeature = PGetIndexedFeature(pUIInfo, 0);
|
|
dwFeatureCount = pRawData->dwDocumentFeatures + pRawData->dwPrinterFeatures;
|
|
|
|
if (pFeature && dwFeatureCount)
|
|
{
|
|
for (i = 0; i < dwFeatureCount; i++)
|
|
{
|
|
pKeywordName = OFFSET_TO_POINTER(pUIInfo->pubResourceData, pFeature->loKeywordName);
|
|
if (strcmp(pKeywordName, pFeatureKeyword) == EQUAL_STRING)
|
|
{
|
|
bFeatureFound = TRUE;
|
|
break;
|
|
}
|
|
pFeature++;
|
|
}
|
|
}
|
|
|
|
if (bFeatureFound)
|
|
{
|
|
pOption = PGetIndexedOption(pUIInfo, pFeature, 0);
|
|
|
|
for (j = 0; j < pFeature->Options.dwCount; j++)
|
|
{
|
|
pKeywordName = OFFSET_TO_POINTER(pUIInfo->pubResourceData, pOption->loKeywordName);
|
|
if (strcmp(pKeywordName, pOptionKeyword) == EQUAL_STRING)
|
|
{
|
|
bOptionFound = TRUE;
|
|
break;
|
|
}
|
|
pOption++;
|
|
}
|
|
}
|
|
|
|
if (bFeatureFound && bOptionFound)
|
|
{
|
|
pOptionsArray[i].ubCurOptIndex = (BYTE)j;
|
|
|
|
//
|
|
// Resolve conflicts
|
|
//
|
|
|
|
if (!ResolveUIConflicts( pRawData,
|
|
pOptionsArray,
|
|
MAX_COMBINED_OPTIONS,
|
|
MODE_DOCANDPRINTER_STICKY))
|
|
{
|
|
VERBOSE(("Resolved conflicting printer feature selections.\n"));
|
|
}
|
|
|
|
|
|
SeparateOptionArray(pRawData,
|
|
pOptionsArray,
|
|
pPrinterData->aOptions,
|
|
MAX_PRINTER_OPTIONS,
|
|
MODE_PRINTER_STICKY
|
|
);
|
|
|
|
if (!BSavePrinterProperties(hPrinter, pRawData, pPrinterData, sizeof(PRINTERDATA)))
|
|
{
|
|
ERR(("BSavePrinterProperties failed\n"));
|
|
bResult = FALSE;
|
|
}
|
|
else
|
|
bResult = TRUE;
|
|
}
|
|
|
|
upgrade_registry_exit:
|
|
|
|
if (pInfoHeader)
|
|
FreeBinaryData(pInfoHeader);
|
|
|
|
if (pRawData)
|
|
UnloadRawBinaryData(pRawData);
|
|
|
|
if (pPrinterData)
|
|
MemFree(pPrinterData);
|
|
|
|
if (pDriverInfo3)
|
|
MemFree(pDriverInfo3);
|
|
|
|
if (pOptionsArray)
|
|
MemFree(pOptionsArray);
|
|
|
|
return bResult;
|
|
}
|
|
|
|
#ifdef PSCRIPT
|
|
|
|
#ifndef WINNT_40
|
|
|
|
|
|
/*++
|
|
|
|
Routine Name:
|
|
|
|
HQuerySimulationSupport
|
|
|
|
Routine Description:
|
|
|
|
In the case of UI replacement, we allows IHV to query for print processor simulation
|
|
support so they can provide simulated features on their UI.
|
|
|
|
We won't enforce hooking out QueryJobAttribute w/o UI replacement here. We will do it
|
|
at DrvQueryJobAttributes.
|
|
|
|
Arguments:
|
|
|
|
hPrinter - printer handle
|
|
dwLevel - interested level of spooler simulation capability info structure
|
|
pCaps - pointer to output buffer
|
|
cbSize - size in bytes of output buffer
|
|
pcbNeeded - buffer size in bytes needed to store the interested info structure
|
|
|
|
Return Value:
|
|
|
|
S_OK if succeeded
|
|
E_OUTOFMEMORY if output buffer is not big enough
|
|
E_NOTIMPL if the interested level is not supported
|
|
E_FAIL if encountered other internal error
|
|
|
|
Last Error:
|
|
|
|
None
|
|
|
|
--*/
|
|
HRESULT
|
|
HQuerySimulationSupport(
|
|
IN HANDLE hPrinter,
|
|
IN DWORD dwLevel,
|
|
OUT PBYTE pCaps,
|
|
IN DWORD cbSize,
|
|
OUT PDWORD pcbNeeded
|
|
)
|
|
{
|
|
PRINTPROCESSOR_CAPS_1 SplCaps;
|
|
PSIMULATE_CAPS_1 pSimCaps;
|
|
DWORD cbNeeded;
|
|
|
|
//
|
|
// currently only Level 1 is supported
|
|
//
|
|
|
|
if (dwLevel != 1)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
cbNeeded = sizeof(SIMULATE_CAPS_1);
|
|
|
|
if (pcbNeeded)
|
|
{
|
|
*pcbNeeded = cbNeeded;
|
|
}
|
|
|
|
if (!pCaps || cbSize < cbNeeded)
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
//
|
|
// Since VGetSpoolerEmfCaps doesn't return error code, we
|
|
// are using the dwLevel field to detect if the call succeeds.
|
|
// If succeeds, dwLevel should be set as 1.
|
|
//
|
|
|
|
SplCaps.dwLevel = 0;
|
|
|
|
VGetSpoolerEmfCaps(hPrinter,
|
|
NULL,
|
|
NULL,
|
|
sizeof(PRINTPROCESSOR_CAPS_1),
|
|
&SplCaps
|
|
);
|
|
|
|
if (SplCaps.dwLevel != 1)
|
|
{
|
|
ERR(("VGetSpoolerEmfCaps failed\n"));
|
|
return E_FAIL;
|
|
}
|
|
|
|
//
|
|
// BUGBUG, we should get a new PRINTPROCESSOR_CAPS level to include all
|
|
// these information instead of filling it out here. Need
|
|
// new PRINTPROCESSOR_CAPS
|
|
//
|
|
|
|
pSimCaps = (PSIMULATE_CAPS_1)pCaps;
|
|
|
|
pSimCaps->dwLevel = 1;
|
|
pSimCaps->dwPageOrderFlags = SplCaps.dwPageOrderFlags;
|
|
pSimCaps->dwNumberOfCopies = SplCaps.dwNumberOfCopies;
|
|
pSimCaps->dwNupOptions = SplCaps.dwNupOptions;
|
|
|
|
//
|
|
// PRINTPROCESSOR_CAPS_1 is designed without an explicit field for
|
|
// collate simulation. So before its CAPS_2 is introduced, we have
|
|
// to assume that if reverse printing is supported, then collate
|
|
// simulation is also supported.
|
|
//
|
|
|
|
if (SplCaps.dwPageOrderFlags & REVERSE_PRINT)
|
|
{
|
|
pSimCaps->dwCollate = 1;
|
|
}
|
|
else
|
|
{
|
|
pSimCaps->dwCollate = 0;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
#endif // !WINNT_40
|
|
|
|
|
|
/*++
|
|
|
|
Routine Name:
|
|
|
|
HEnumConstrainedOptions
|
|
|
|
Routine Description:
|
|
|
|
enumerate the constrained option keyword name list in the specified feature
|
|
|
|
Arguments:
|
|
|
|
poemuiobj - pointer to driver context object
|
|
dwFlags - flags for the enumeration operation
|
|
pszFeatureKeyword - feature keyword name
|
|
pmszConstrainedOptionList - pointer to output data buffer
|
|
cbSize - output data buffer size in bytes
|
|
pcbNeeded - buffer size in bytes needed to store the output data
|
|
|
|
Return Value:
|
|
|
|
S_OK if succeeds
|
|
E_OUTOFMEMORY if output data buffer size is not big enough
|
|
E_INVALIDARG if feature keyword name is not recognized, or the feature's
|
|
stickiness doesn't match current sticky-mode
|
|
E_FAIL if other internal failures are encountered
|
|
|
|
Last Error:
|
|
|
|
None
|
|
|
|
--*/
|
|
HRESULT
|
|
HEnumConstrainedOptions(
|
|
IN POEMUIOBJ poemuiobj,
|
|
IN DWORD dwFlags,
|
|
IN PCSTR pszFeatureKeyword,
|
|
OUT PSTR pmszConstrainedOptionList,
|
|
IN DWORD cbSize,
|
|
OUT PDWORD pcbNeeded
|
|
)
|
|
{
|
|
PCOMMONINFO pci = (PCOMMONINFO)poemuiobj;
|
|
PUIDATA pUiData;
|
|
PFEATURE pFeature;
|
|
POPTION pOption;
|
|
DWORD dwFeatureIndex, dwIndex;
|
|
PBOOL pabEnabledOptions = NULL;
|
|
PSTR pCurrentOut;
|
|
DWORD cbNeeded;
|
|
INT cbRemain;
|
|
HRESULT hr;
|
|
|
|
pUiData = (PUIDATA)pci;
|
|
|
|
if (!pszFeatureKeyword ||
|
|
(pFeature = PGetNamedFeature(pci->pUIInfo, pszFeatureKeyword, &dwFeatureIndex)) == NULL)
|
|
{
|
|
WARNING(("HEnumConstrainedOptions: invalid feature\n"));
|
|
|
|
//
|
|
// Even though we could return right here, we still use goto to maintain single exit point.
|
|
//
|
|
|
|
hr = E_INVALIDARG;
|
|
goto exit;
|
|
}
|
|
|
|
//
|
|
// pUiData->iMode can have 2 modes: MODE_DOCUMENT_STICKY and MODE_PRINTER_STICKY. See PFillUiData().
|
|
// In MODE_DOCUMENT_STICKY mode, we only support doc-sticky features.
|
|
// In MODE_PRINTER_STICKY mode, we only support printer-sticky features.
|
|
//
|
|
// This is because in function PFillUiData(), it only fills devmode in MODE_DOCUMENT_STICKY mode.
|
|
// Then in BCombineCommonInfoOptionsArray(), if devmode option array is not available, the PPD parser
|
|
// will use OPTION_INDEX_ANY for any doc-sticky features.
|
|
//
|
|
|
|
if ((pUiData->iMode == MODE_DOCUMENT_STICKY && pFeature->dwFeatureType == FEATURETYPE_PRINTERPROPERTY) ||
|
|
(pUiData->iMode == MODE_PRINTER_STICKY && pFeature->dwFeatureType != FEATURETYPE_PRINTERPROPERTY))
|
|
{
|
|
VERBOSE(("HEnumConstrainedOptions: mismatch iMode=%d, dwFeatureType=%d\n",
|
|
pUiData->iMode, pFeature->dwFeatureType)) ;
|
|
|
|
hr = E_INVALIDARG;
|
|
goto exit;
|
|
}
|
|
|
|
if (pFeature->Options.dwCount)
|
|
{
|
|
if ((pabEnabledOptions = MemAllocZ(pFeature->Options.dwCount * sizeof(BOOL))) == NULL)
|
|
{
|
|
ERR(("HEnumConstrainedOptions: memory alloc failed\n"));
|
|
hr = E_FAIL;
|
|
goto exit;
|
|
}
|
|
|
|
//
|
|
// Get the feature's enabled option list.
|
|
//
|
|
// See VPropShowConstraints() in docprop.c and prnprop.c for using different
|
|
// modes to call EnumEnabledOptions().
|
|
//
|
|
|
|
if (pUiData->iMode == MODE_DOCUMENT_STICKY)
|
|
{
|
|
EnumEnabledOptions(pci->pRawData, pci->pCombinedOptions, dwFeatureIndex,
|
|
pabEnabledOptions, MODE_DOCANDPRINTER_STICKY);
|
|
}
|
|
else
|
|
{
|
|
EnumEnabledOptions(pci->pRawData, pci->pCombinedOptions, dwFeatureIndex,
|
|
pabEnabledOptions, MODE_PRINTER_STICKY);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
RIP(("HEnumConstrainedOptions: feature %s has no options\n", pszFeatureKeyword));
|
|
|
|
//
|
|
// continue so we will output the empty string with only the NUL character
|
|
//
|
|
}
|
|
|
|
pCurrentOut = pmszConstrainedOptionList;
|
|
cbNeeded = 0;
|
|
cbRemain = (INT)cbSize;
|
|
|
|
pOption = OFFSET_TO_POINTER(pci->pInfoHeader, pFeature->Options.loOffset);
|
|
|
|
ASSERT(pOption || pFeature->Options.dwCount == 0);
|
|
|
|
if (pOption == NULL && pFeature->Options.dwCount != 0)
|
|
{
|
|
hr = E_FAIL;
|
|
goto exit;
|
|
}
|
|
|
|
for (dwIndex = 0; dwIndex < pFeature->Options.dwCount; dwIndex++)
|
|
{
|
|
if (!pabEnabledOptions[dwIndex])
|
|
{
|
|
DWORD dwNameSize;
|
|
PSTR pszKeywordName;
|
|
|
|
pszKeywordName = OFFSET_TO_POINTER(pci->pUIInfo->pubResourceData, pOption->loKeywordName);
|
|
|
|
ASSERT(pszKeywordName);
|
|
|
|
if (pszKeywordName == NULL)
|
|
{
|
|
hr = E_FAIL;
|
|
goto exit;
|
|
}
|
|
|
|
//
|
|
// count in the NUL character between constrained option keywords
|
|
//
|
|
|
|
dwNameSize = strlen(pszKeywordName) + 1;
|
|
|
|
if (pCurrentOut && cbRemain >= (INT)dwNameSize)
|
|
{
|
|
CopyMemory(pCurrentOut, pszKeywordName, dwNameSize);
|
|
pCurrentOut += dwNameSize;
|
|
}
|
|
|
|
cbRemain -= dwNameSize;
|
|
cbNeeded += dwNameSize;
|
|
}
|
|
|
|
pOption = (POPTION)((PBYTE)pOption + pFeature->dwOptionSize);
|
|
}
|
|
|
|
//
|
|
// remember the last NUL terminator for the MULTI_SZ output string
|
|
//
|
|
|
|
cbNeeded++;
|
|
|
|
if (pcbNeeded)
|
|
{
|
|
*pcbNeeded = cbNeeded;
|
|
}
|
|
|
|
if (!pCurrentOut || cbRemain < 1)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
goto exit;
|
|
}
|
|
|
|
*pCurrentOut = NUL;
|
|
|
|
//
|
|
// Succeeded
|
|
//
|
|
|
|
hr = S_OK;
|
|
|
|
exit:
|
|
|
|
MemFree(pabEnabledOptions);
|
|
return hr;
|
|
}
|
|
|
|
|
|
/*++
|
|
|
|
Routine Name:
|
|
|
|
HWhyConstrained
|
|
|
|
Routine Description:
|
|
|
|
get feature/option keyword pair that constrains the given
|
|
feature/option pair
|
|
|
|
Arguments:
|
|
|
|
poemuiobj - pointer to driver context object
|
|
dwFlags - flags for this operation
|
|
pszFeatureKeyword - feature keyword name
|
|
pszOptionKeyword - option keyword name
|
|
pmszReasonList - pointer to output data buffer
|
|
cbSize - output data buffer size in bytes
|
|
pcbNeeded - buffer size in bytes needed to store the output data
|
|
|
|
Return Value:
|
|
|
|
S_OK if succeeds
|
|
E_OUTOFMEMORY if output data buffer size is not big enough
|
|
E_INVALIDARG if the feature keyword name or option keyword name
|
|
is not recognized, or the feature's stickiness
|
|
doesn't match current sticky-mode
|
|
|
|
Last Error:
|
|
|
|
None
|
|
|
|
--*/
|
|
HRESULT
|
|
HWhyConstrained(
|
|
IN POEMUIOBJ poemuiobj,
|
|
IN DWORD dwFlags,
|
|
IN PCSTR pszFeatureKeyword,
|
|
IN PCSTR pszOptionKeyword,
|
|
OUT PSTR pmszReasonList,
|
|
IN DWORD cbSize,
|
|
OUT PDWORD pcbNeeded
|
|
)
|
|
{
|
|
PCOMMONINFO pci = (PCOMMONINFO)poemuiobj;
|
|
PUIDATA pUiData;
|
|
PFEATURE pFeature;
|
|
POPTION pOption;
|
|
DWORD dwFeatureIndex, dwOptionIndex;
|
|
CONFLICTPAIR ConflictPair;
|
|
BOOL bConflictFound;
|
|
PSTR pszConfFeatureName = NULL, pszConfOptionName = NULL;
|
|
CHAR emptyString[1] = {0};
|
|
DWORD cbConfFeatureKeySize = 0, cbConfOptionKeySize = 0;
|
|
DWORD cbNeeded = 0;
|
|
|
|
pUiData = (PUIDATA)pci;
|
|
|
|
if (!pszFeatureKeyword ||
|
|
(pFeature = PGetNamedFeature(pci->pUIInfo, pszFeatureKeyword, &dwFeatureIndex)) == NULL)
|
|
{
|
|
WARNING(("HWhyConstrained: invalid feature\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
if (!pszOptionKeyword ||
|
|
(pOption = PGetNamedOption(pci->pUIInfo, pFeature, pszOptionKeyword, &dwOptionIndex)) == NULL)
|
|
{
|
|
WARNING(("HWhyConstrained: invalid option\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
//
|
|
// See comments in HEnumConstrainedOptions() for following stickiness mode check.
|
|
//
|
|
|
|
if ((pUiData->iMode == MODE_DOCUMENT_STICKY && pFeature->dwFeatureType == FEATURETYPE_PRINTERPROPERTY) ||
|
|
(pUiData->iMode == MODE_PRINTER_STICKY && pFeature->dwFeatureType != FEATURETYPE_PRINTERPROPERTY))
|
|
{
|
|
VERBOSE(("HWhyConstrained: mismatch iMode=%d, dwFeatureType=%d\n",pUiData->iMode, pFeature->dwFeatureType));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
//
|
|
// Get the feature/option pair that constrains the feature/option pair client is querying for.
|
|
//
|
|
|
|
bConflictFound = EnumNewPickOneUIConflict(pci->pRawData,
|
|
pci->pCombinedOptions,
|
|
dwFeatureIndex,
|
|
dwOptionIndex,
|
|
&ConflictPair);
|
|
|
|
if (bConflictFound)
|
|
{
|
|
PFEATURE pConfFeature;
|
|
POPTION pConfOption;
|
|
DWORD dwConfFeatureIndex, dwConfOptionIndex;
|
|
|
|
//
|
|
// ConflictPair has the feature with higher priority as dwFeatureIndex1.
|
|
//
|
|
|
|
if (dwFeatureIndex == ConflictPair.dwFeatureIndex1)
|
|
{
|
|
dwConfFeatureIndex = ConflictPair.dwFeatureIndex2;
|
|
dwConfOptionIndex = ConflictPair.dwOptionIndex2;
|
|
}
|
|
else
|
|
{
|
|
dwConfFeatureIndex = ConflictPair.dwFeatureIndex1;
|
|
dwConfOptionIndex = ConflictPair.dwOptionIndex1;
|
|
}
|
|
|
|
pConfFeature = PGetIndexedFeature(pci->pUIInfo, dwConfFeatureIndex);
|
|
ASSERT(pConfFeature);
|
|
|
|
pConfOption = PGetIndexedOption(pci->pUIInfo, pConfFeature, dwConfOptionIndex);
|
|
|
|
//
|
|
// We don't expect pConfOption to be NULL here. Use the ASSERT to catch cases we missed.
|
|
//
|
|
|
|
ASSERT(pConfOption);
|
|
|
|
pszConfFeatureName = OFFSET_TO_POINTER(pci->pUIInfo->pubResourceData, pConfFeature->loKeywordName);
|
|
ASSERT(pszConfFeatureName);
|
|
|
|
if (pConfOption)
|
|
{
|
|
pszConfOptionName = OFFSET_TO_POINTER(pci->pUIInfo->pubResourceData, pConfOption->loKeywordName);
|
|
ASSERT(pszConfOptionName);
|
|
}
|
|
else
|
|
{
|
|
pszConfOptionName = &(emptyString[0]);
|
|
}
|
|
|
|
//
|
|
// count in the 2 NUL characters: one after feature name, one after option name.
|
|
//
|
|
|
|
cbConfFeatureKeySize = strlen(pszConfFeatureName) + 1;
|
|
cbConfOptionKeySize = strlen(pszConfOptionName) + 1;
|
|
}
|
|
|
|
//
|
|
// count in the last NUL characters at the end.
|
|
//
|
|
|
|
cbNeeded = cbConfFeatureKeySize + cbConfOptionKeySize + 1;
|
|
|
|
if (pcbNeeded)
|
|
{
|
|
*pcbNeeded = cbNeeded;
|
|
}
|
|
|
|
if (!pmszReasonList || cbSize < cbNeeded)
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
if (bConflictFound)
|
|
{
|
|
ASSERT(pszConfFeatureName && pszConfOptionName);
|
|
|
|
CopyMemory(pmszReasonList, pszConfFeatureName, cbConfFeatureKeySize);
|
|
pmszReasonList += cbConfFeatureKeySize;
|
|
|
|
CopyMemory(pmszReasonList, pszConfOptionName, cbConfOptionKeySize);
|
|
pmszReasonList += cbConfOptionKeySize;
|
|
}
|
|
|
|
//
|
|
// Now the NUL at the end to finish the MULTI_SZ output string.
|
|
//
|
|
|
|
*pmszReasonList = NUL;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
#endif // PSCRIPT
|