/* * WINSPOOL functions * * Copyright 1996 John Harvey * Copyright 1998 Andreas Mohr * Copyright 1999 Klaas van Gend * Copyright 1999, 2000 Huw D M Davies * Copyright 2001 Marcus Meissner * Copyright 2005-2010 Detlef Riekenberg * Copyright 2010 Vitaly Perov * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include "config.h" #include "wine/port.h" #include <stdarg.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <ctype.h> #include <stddef.h> #include <errno.h> #ifdef HAVE_SYS_WAIT_H #include <sys/wait.h> #endif #ifdef HAVE_UNISTD_H # include <unistd.h> #endif #include <signal.h> #ifdef HAVE_CUPS_CUPS_H # include <cups/cups.h> #endif #ifdef HAVE_APPLICATIONSERVICES_APPLICATIONSERVICES_H #define GetCurrentProcess GetCurrentProcess_Mac #define GetCurrentThread GetCurrentThread_Mac #define LoadResource LoadResource_Mac #define AnimatePalette AnimatePalette_Mac #define EqualRgn EqualRgn_Mac #define FillRgn FillRgn_Mac #define FrameRgn FrameRgn_Mac #define GetPixel GetPixel_Mac #define InvertRgn InvertRgn_Mac #define LineTo LineTo_Mac #define OffsetRgn OffsetRgn_Mac #define PaintRgn PaintRgn_Mac #define Polygon Polygon_Mac #define ResizePalette ResizePalette_Mac #define SetRectRgn SetRectRgn_Mac #define EqualRect EqualRect_Mac #define FillRect FillRect_Mac #define FrameRect FrameRect_Mac #define GetCursor GetCursor_Mac #define InvertRect InvertRect_Mac #define OffsetRect OffsetRect_Mac #define PtInRect PtInRect_Mac #define SetCursor SetCursor_Mac #define SetRect SetRect_Mac #define ShowCursor ShowCursor_Mac #define UnionRect UnionRect_Mac #include <ApplicationServices/ApplicationServices.h> #undef GetCurrentProcess #undef GetCurrentThread #undef LoadResource #undef AnimatePalette #undef EqualRgn #undef FillRgn #undef FrameRgn #undef GetPixel #undef InvertRgn #undef LineTo #undef OffsetRgn #undef PaintRgn #undef Polygon #undef ResizePalette #undef SetRectRgn #undef EqualRect #undef FillRect #undef FrameRect #undef GetCursor #undef InvertRect #undef OffsetRect #undef PtInRect #undef SetCursor #undef SetRect #undef ShowCursor #undef UnionRect #undef DPRINTF #endif #include "wine/library.h" #include "windef.h" #include "winbase.h" #include "winuser.h" #include "winerror.h" #include "winreg.h" #include "wingdi.h" #include "winspool.h" #include "winternl.h" #include "wine/windef16.h" #include "wine/unicode.h" #include "wine/debug.h" #include "wine/list.h" #include "winnls.h" #include "ddk/winsplp.h" #include "wspool.h" WINE_DEFAULT_DEBUG_CHANNEL(winspool); /* ############################### */ static CRITICAL_SECTION printer_handles_cs; static CRITICAL_SECTION_DEBUG printer_handles_cs_debug = { 0, 0, &printer_handles_cs, { &printer_handles_cs_debug.ProcessLocksList, &printer_handles_cs_debug.ProcessLocksList }, 0, 0, { (DWORD_PTR)(__FILE__ ": printer_handles_cs") } }; static CRITICAL_SECTION printer_handles_cs = { &printer_handles_cs_debug, -1, 0, 0, 0, 0 }; /* ############################### */ typedef struct { DWORD job_id; HANDLE hf; } started_doc_t; typedef struct { struct list jobs; LONG ref; } jobqueue_t; typedef struct { LPWSTR name; LPWSTR printername; HANDLE backend_printer; jobqueue_t *queue; started_doc_t *doc; DEVMODEW *devmode; } opened_printer_t; typedef struct { struct list entry; DWORD job_id; WCHAR *filename; WCHAR *portname; WCHAR *document_title; WCHAR *printer_name; LPDEVMODEW devmode; } job_t; typedef struct { LPCWSTR envname; LPCWSTR subdir; DWORD driverversion; LPCWSTR versionregpath; LPCWSTR versionsubdir; } printenv_t; /* ############################### */ static opened_printer_t **printer_handles; static UINT nb_printer_handles; static LONG next_job_id = 1; static DWORD (WINAPI *GDI_CallDeviceCapabilities16)( LPCSTR lpszDevice, LPCSTR lpszPort, WORD fwCapability, LPSTR lpszOutput, LPDEVMODEA lpdm ); static INT (WINAPI *GDI_CallExtDeviceMode16)( HWND hwnd, LPDEVMODEA lpdmOutput, LPSTR lpszDevice, LPSTR lpszPort, LPDEVMODEA lpdmInput, LPSTR lpszProfile, DWORD fwMode ); static const WCHAR DriversW[] = { 'S','y','s','t','e','m','\\', 'C','u', 'r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', 'c','o','n','t','r','o','l','\\', 'P','r','i','n','t','\\', 'E','n','v','i','r','o','n','m','e','n','t','s','\\', '%','s','\\','D','r','i','v','e','r','s','%','s',0 }; static const WCHAR PrintersW[] = {'S','y','s','t','e','m','\\', 'C','u', 'r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', 'C','o','n','t','r','o','l','\\', 'P','r','i','n','t','\\', 'P','r','i','n','t','e','r','s',0}; static const WCHAR user_default_reg_key[] = { 'S','o','f','t','w','a','r','e','\\', 'M','i','c','r','o','s','o','f','t','\\', 'W','i','n','d','o','w','s',' ','N','T','\\', 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\', 'W','i','n','d','o','w','s',0}; static const WCHAR user_printers_reg_key[] = { 'S','o','f','t','w','a','r','e','\\', 'M','i','c','r','o','s','o','f','t','\\', 'W','i','n','d','o','w','s',' ','N','T','\\', 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\', 'D','e','v','i','c','e','s',0}; static const WCHAR WinNT_CV_PrinterPortsW[] = { 'S','o','f','t','w','a','r','e','\\', 'M','i','c','r','o','s','o','f','t','\\', 'W','i','n','d','o','w','s',' ','N','T','\\', 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\', 'P','r','i','n','t','e','r','P','o','r','t','s',0}; static WCHAR envname_win40W[] = {'W','i','n','d','o','w','s',' ','4','.','0',0}; static const WCHAR envname_x64W[] = {'W','i','n','d','o','w','s',' ','x','6','4',0}; static WCHAR envname_x86W[] = {'W','i','n','d','o','w','s',' ','N','T',' ','x','8','6',0}; static const WCHAR subdir_win40W[] = {'w','i','n','4','0',0}; static const WCHAR subdir_x64W[] = {'x','6','4',0}; static const WCHAR subdir_x86W[] = {'w','3','2','x','8','6',0}; static const WCHAR Version0_RegPathW[] = {'\\','V','e','r','s','i','o','n','-','0',0}; static const WCHAR Version0_SubdirW[] = {'\\','0',0}; static const WCHAR Version3_RegPathW[] = {'\\','V','e','r','s','i','o','n','-','3',0}; static const WCHAR Version3_SubdirW[] = {'\\','3',0}; static const WCHAR AttributesW[] = {'A','t','t','r','i','b','u','t','e','s',0}; static const WCHAR backslashW[] = {'\\',0}; static const WCHAR Configuration_FileW[] = {'C','o','n','f','i','g','u','r','a','t', 'i','o','n',' ','F','i','l','e',0}; static const WCHAR DatatypeW[] = {'D','a','t','a','t','y','p','e',0}; static const WCHAR Data_FileW[] = {'D','a','t','a',' ','F','i','l','e',0}; static const WCHAR Default_DevModeW[] = {'D','e','f','a','u','l','t',' ','D','e','v','M','o','d','e',0}; static const WCHAR Default_PriorityW[] = {'D','e','f','a','u','l','t',' ','P','r','i','o','r','i','t','y',0}; static const WCHAR Dependent_FilesW[] = {'D','e','p','e','n','d','e','n','t',' ','F','i','l','e','s',0}; static const WCHAR DescriptionW[] = {'D','e','s','c','r','i','p','t','i','o','n',0}; static const WCHAR dnsTimeoutW[] = {'d','n','s','T','i','m','e','o','u','t',0}; static const WCHAR DriverW[] = {'D','r','i','v','e','r',0}; static const WCHAR HardwareIDW[] = {'H','a','r','d','w','a','r','e','I','D',0}; static const WCHAR Help_FileW[] = {'H','e','l','p',' ','F','i','l','e',0}; static const WCHAR LocationW[] = {'L','o','c','a','t','i','o','n',0}; static const WCHAR ManufacturerW[] = {'M','a','n','u','f','a','c','t','u','r','e','r',0}; static const WCHAR MonitorW[] = {'M','o','n','i','t','o','r',0}; static const WCHAR NameW[] = {'N','a','m','e',0}; static const WCHAR ObjectGUIDW[] = {'O','b','j','e','c','t','G','U','I','D',0}; static const WCHAR OEM_UrlW[] = {'O','E','M',' ','U','r','l',0}; static const WCHAR ParametersW[] = {'P','a','r','a','m','e','t','e','r','s',0}; static const WCHAR PortW[] = {'P','o','r','t',0}; static const WCHAR Previous_NamesW[] = {'P','r','e','v','i','o','u','s',' ','N','a','m','e','s',0}; static const WCHAR Print_ProcessorW[] = {'P','r','i','n','t',' ','P','r','o','c','e','s','s','o','r',0}; static const WCHAR Printer_DriverW[] = {'P','r','i','n','t','e','r',' ','D','r','i','v','e','r',0}; static const WCHAR PrinterDriverDataW[] = {'P','r','i','n','t','e','r','D','r','i','v','e','r','D','a','t','a',0}; static const WCHAR PrinterPortsW[] = {'P','r','i','n','t','e','r','P','o','r','t','s',0}; static const WCHAR PriorityW[] = {'P','r','i','o','r','i','t','y',0}; static const WCHAR ProviderW[] = {'P','r','o','v','i','d','e','r',0}; static const WCHAR Separator_FileW[] = {'S','e','p','a','r','a','t','o','r',' ','F','i','l','e',0}; static const WCHAR Share_NameW[] = {'S','h','a','r','e',' ','N','a','m','e',0}; static const WCHAR StartTimeW[] = {'S','t','a','r','t','T','i','m','e',0}; static const WCHAR StatusW[] = {'S','t','a','t','u','s',0}; static const WCHAR txTimeoutW[] = {'t','x','T','i','m','e','o','u','t',0}; static const WCHAR UntilTimeW[] = {'U','n','t','i','l','T','i','m','e',0}; static WCHAR WinPrintW[] = {'W','i','n','P','r','i','n','t',0}; static const WCHAR deviceW[] = {'d','e','v','i','c','e',0}; static const WCHAR devicesW[] = {'d','e','v','i','c','e','s',0}; static const WCHAR windowsW[] = {'w','i','n','d','o','w','s',0}; static WCHAR rawW[] = {'R','A','W',0}; static WCHAR driver_9x[] = {'w','i','n','e','p','s','1','6','.','d','r','v',0}; static WCHAR driver_nt[] = {'w','i','n','e','p','s','.','d','r','v',0}; static const WCHAR timeout_15_45[] = {',','1','5',',','4','5',0}; static const WCHAR commaW[] = {',',0}; static WCHAR emptyStringW[] = {0}; static const WCHAR May_Delete_Value[] = {'W','i','n','e','M','a','y','D','e','l','e','t','e','M','e',0}; static const WCHAR CUPS_Port[] = {'C','U','P','S',':',0}; static const WCHAR FILE_Port[] = {'F','I','L','E',':',0}; static const WCHAR LPR_Port[] = {'L','P','R',':',0}; static const WCHAR default_doc_title[] = {'L','o','c','a','l',' ','D','o','w','n','l','e','v','e','l',' ', 'D','o','c','u','m','e','n','t',0}; static const WCHAR PPD_Overrides[] = {'P','P','D',' ','O','v','e','r','r','i','d','e','s',0}; static const WCHAR DefaultPageSize[] = {'D','e','f','a','u','l','t','P','a','g','e','S','i','z','e',0}; static const DWORD di_sizeof[] = {0, sizeof(DRIVER_INFO_1W), sizeof(DRIVER_INFO_2W), sizeof(DRIVER_INFO_3W), sizeof(DRIVER_INFO_4W), sizeof(DRIVER_INFO_5W), sizeof(DRIVER_INFO_6W), 0, sizeof(DRIVER_INFO_8W)}; static const DWORD pi_sizeof[] = {0, sizeof(PRINTER_INFO_1W), sizeof(PRINTER_INFO_2W), sizeof(PRINTER_INFO_3), sizeof(PRINTER_INFO_4W), sizeof(PRINTER_INFO_5W), sizeof(PRINTER_INFO_6), sizeof(PRINTER_INFO_7W), sizeof(PRINTER_INFO_8W), sizeof(PRINTER_INFO_9W)}; static const printenv_t env_x64 = {envname_x64W, subdir_x64W, 3, Version3_RegPathW, Version3_SubdirW}; static const printenv_t env_x86 = {envname_x86W, subdir_x86W, 3, Version3_RegPathW, Version3_SubdirW}; static const printenv_t env_win40 = {envname_win40W, subdir_win40W, 0, Version0_RegPathW, Version0_SubdirW}; static const printenv_t * const all_printenv[] = {&env_x86, &env_x64, &env_win40}; /****************************************************************** * validate the user-supplied printing-environment [internal] * * PARAMS * env [I] PTR to Environment-String or NULL * * RETURNS * Failure: NULL * Success: PTR to printenv_t * * NOTES * An empty string is handled the same way as NULL. * SetLastEror(ERROR_INVALID_ENVIRONMENT) is called on Failure * */ static const printenv_t * validate_envW(LPCWSTR env) { const printenv_t *result = NULL; unsigned int i; TRACE("testing %s\n", debugstr_w(env)); if (env && env[0]) { for (i = 0; i < sizeof(all_printenv)/sizeof(all_printenv[0]); i++) { if (lstrcmpiW(env, all_printenv[i]->envname) == 0) { result = all_printenv[i]; break; } } if (result == NULL) { FIXME("unsupported Environment: %s\n", debugstr_w(env)); SetLastError(ERROR_INVALID_ENVIRONMENT); } /* on win9x, only "Windows 4.0" is allowed, but we ignore this */ } else { result = (GetVersion() & 0x80000000) ? &env_win40 : &env_x86; } TRACE("using %p: %s\n", result, debugstr_w(result ? result->envname : NULL)); return result; } /* RtlCreateUnicodeStringFromAsciiz will return an empty string in the buffer if passed a NULL string. This returns NULLs to the result. */ static inline PWSTR asciitounicode( UNICODE_STRING * usBufferPtr, LPCSTR src ) { if ( (src) ) { RtlCreateUnicodeStringFromAsciiz(usBufferPtr, src); return usBufferPtr->Buffer; } usBufferPtr->Buffer = NULL; /* so that RtlFreeUnicodeString won't barf */ return NULL; } static LPWSTR strdupW(LPCWSTR p) { LPWSTR ret; DWORD len; if(!p) return NULL; len = (strlenW(p) + 1) * sizeof(WCHAR); ret = HeapAlloc(GetProcessHeap(), 0, len); memcpy(ret, p, len); return ret; } static LPSTR strdupWtoA( LPCWSTR str ) { LPSTR ret; INT len; if (!str) return NULL; len = WideCharToMultiByte( CP_ACP, 0, str, -1, NULL, 0, NULL, NULL ); ret = HeapAlloc( GetProcessHeap(), 0, len ); if(ret) WideCharToMultiByte( CP_ACP, 0, str, -1, ret, len, NULL, NULL ); return ret; } static DEVMODEW *dup_devmode( const DEVMODEW *dm ) { DEVMODEW *ret; if (!dm) return NULL; ret = HeapAlloc( GetProcessHeap(), 0, dm->dmSize + dm->dmDriverExtra ); if (ret) memcpy( ret, dm, dm->dmSize + dm->dmDriverExtra ); return ret; } /*********************************************************** * DEVMODEdupWtoA * Creates an ansi copy of supplied devmode */ static DEVMODEA *DEVMODEdupWtoA( const DEVMODEW *dmW ) { LPDEVMODEA dmA; DWORD size; if (!dmW) return NULL; size = dmW->dmSize - CCHDEVICENAME - ((dmW->dmSize > FIELD_OFFSET( DEVMODEW, dmFormName )) ? CCHFORMNAME : 0); dmA = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, size + dmW->dmDriverExtra ); if (!dmA) return NULL; WideCharToMultiByte( CP_ACP, 0, dmW->dmDeviceName, -1, (LPSTR)dmA->dmDeviceName, CCHDEVICENAME, NULL, NULL ); if (FIELD_OFFSET( DEVMODEW, dmFormName ) >= dmW->dmSize) { memcpy( &dmA->dmSpecVersion, &dmW->dmSpecVersion, dmW->dmSize - FIELD_OFFSET( DEVMODEW, dmSpecVersion ) ); } else { memcpy( &dmA->dmSpecVersion, &dmW->dmSpecVersion, FIELD_OFFSET( DEVMODEW, dmFormName ) - FIELD_OFFSET( DEVMODEW, dmSpecVersion ) ); WideCharToMultiByte( CP_ACP, 0, dmW->dmFormName, -1, (LPSTR)dmA->dmFormName, CCHFORMNAME, NULL, NULL ); memcpy( &dmA->dmLogPixels, &dmW->dmLogPixels, dmW->dmSize - FIELD_OFFSET( DEVMODEW, dmLogPixels ) ); } dmA->dmSize = size; memcpy( (char *)dmA + dmA->dmSize, (const char *)dmW + dmW->dmSize, dmW->dmDriverExtra ); return dmA; } /****************************************************************** * verify, that the filename is a local file * */ static inline BOOL is_local_file(LPWSTR name) { return (name[0] && (name[1] == ':') && (name[2] == '\\')); } /* ################################ */ static int multi_sz_lenA(const char *str) { const char *ptr = str; if(!str) return 0; do { ptr += lstrlenA(ptr) + 1; } while(*ptr); return ptr - str + 1; } /***************************************************************************** * get_dword_from_reg * * Return DWORD associated with name from hkey. */ static DWORD get_dword_from_reg( HKEY hkey, const WCHAR *name ) { DWORD sz = sizeof(DWORD), type, value = 0; LONG ret; ret = RegQueryValueExW( hkey, name, 0, &type, (LPBYTE)&value, &sz ); if (ret != ERROR_SUCCESS) { WARN( "Got ret = %d on name %s\n", ret, debugstr_w(name) ); return 0; } if (type != REG_DWORD) { ERR( "Got type %d\n", type ); return 0; } return value; } static inline DWORD set_reg_DWORD( HKEY hkey, const WCHAR *keyname, const DWORD value ) { return RegSetValueExW( hkey, keyname, 0, REG_DWORD, (const BYTE*)&value, sizeof(value) ); } /****************************************************************** * get_opened_printer * Get the pointer to the opened printer referred by the handle */ static opened_printer_t *get_opened_printer(HANDLE hprn) { UINT_PTR idx = (UINT_PTR)hprn; opened_printer_t *ret = NULL; EnterCriticalSection(&printer_handles_cs); if ((idx > 0) && (idx <= nb_printer_handles)) { ret = printer_handles[idx - 1]; } LeaveCriticalSection(&printer_handles_cs); return ret; } /****************************************************************** * get_opened_printer_name * Get the pointer to the opened printer name referred by the handle */ static LPCWSTR get_opened_printer_name(HANDLE hprn) { opened_printer_t *printer = get_opened_printer(hprn); if(!printer) return NULL; return printer->name; } static DWORD open_printer_reg_key( const WCHAR *name, HKEY *key ) { HKEY printers; DWORD err; *key = NULL; err = RegCreateKeyW( HKEY_LOCAL_MACHINE, PrintersW, &printers ); if (err) return err; err = RegOpenKeyW( printers, name, key ); if (err) err = ERROR_INVALID_PRINTER_NAME; RegCloseKey( printers ); return err; } /****************************************************************** * WINSPOOL_GetOpenedPrinterRegKey * */ static DWORD WINSPOOL_GetOpenedPrinterRegKey(HANDLE hPrinter, HKEY *phkey) { LPCWSTR name = get_opened_printer_name(hPrinter); if(!name) return ERROR_INVALID_HANDLE; return open_printer_reg_key( name, phkey ); } static void WINSPOOL_SetDefaultPrinter(const char *devname, const char *name, BOOL force) { char qbuf[200]; /* If forcing, or no profile string entry for device yet, set the entry * * The always change entry if not WINEPS yet is discussable. */ if (force || !GetProfileStringA("windows","device","*",qbuf,sizeof(qbuf)) || !strcmp(qbuf,"*") || !strstr(qbuf,"WINEPS.DRV") ) { char *buf = HeapAlloc(GetProcessHeap(),0,strlen(name)+strlen(devname)+strlen(",WINEPS.DRV,LPR:")+1); HKEY hkey; sprintf(buf,"%s,WINEPS.DRV,LPR:%s",devname,name); WriteProfileStringA("windows","device",buf); if(RegCreateKeyW(HKEY_CURRENT_USER, user_default_reg_key, &hkey) == ERROR_SUCCESS) { RegSetValueExA(hkey, "Device", 0, REG_SZ, (LPBYTE)buf, strlen(buf) + 1); RegCloseKey(hkey); } HeapFree(GetProcessHeap(),0,buf); } } static BOOL add_printer_driver(const WCHAR *name, WCHAR *ppd) { DRIVER_INFO_3W di3; ZeroMemory(&di3, sizeof(DRIVER_INFO_3W)); di3.cVersion = 3; di3.pName = (WCHAR*)name; di3.pEnvironment = envname_x86W; di3.pDriverPath = driver_nt; di3.pDataFile = ppd; di3.pConfigFile = driver_nt; di3.pDefaultDataType = rawW; if (AddPrinterDriverExW( NULL, 3, (LPBYTE)&di3, APD_COPY_NEW_FILES | APD_COPY_FROM_DIRECTORY ) || (GetLastError() == ERROR_PRINTER_DRIVER_ALREADY_INSTALLED )) { di3.cVersion = 0; di3.pEnvironment = envname_win40W; di3.pDriverPath = driver_9x; di3.pConfigFile = driver_9x; if (AddPrinterDriverExW( NULL, 3, (LPBYTE)&di3, APD_COPY_NEW_FILES | APD_COPY_FROM_DIRECTORY ) || (GetLastError() == ERROR_PRINTER_DRIVER_ALREADY_INSTALLED )) { return TRUE; } } ERR("failed with %u for %s (%s)\n", GetLastError(), debugstr_w(di3.pDriverPath), debugstr_w(di3.pEnvironment)); return FALSE; } static inline char *expand_env_string( char *str, DWORD type ) { if (type == REG_EXPAND_SZ) { char *tmp; DWORD needed = ExpandEnvironmentStringsA( str, NULL, 0 ); tmp = HeapAlloc( GetProcessHeap(), 0, needed ); if (tmp) { ExpandEnvironmentStringsA( str, tmp, needed ); HeapFree( GetProcessHeap(), 0, str ); return tmp; } } return str; } static char *get_fallback_ppd_name( const char *printer_name ) { static const WCHAR ppds_key[] = {'S','o','f','t','w','a','r','e','\\','W','i','n','e','\\', 'P','r','i','n','t','i','n','g','\\','P','P','D',' ','F','i','l','e','s',0}; HKEY hkey; DWORD needed, type; char *ret = NULL; if (RegOpenKeyW( HKEY_CURRENT_USER, ppds_key, &hkey ) == ERROR_SUCCESS ) { const char *value_name = NULL; if (RegQueryValueExA( hkey, printer_name, 0, NULL, NULL, &needed ) == ERROR_SUCCESS) value_name = printer_name; else if (RegQueryValueExA( hkey, "generic", 0, NULL, NULL, &needed ) == ERROR_SUCCESS) value_name = "generic"; if (value_name) { ret = HeapAlloc( GetProcessHeap(), 0, needed ); if (!ret) return NULL; RegQueryValueExA( hkey, value_name, 0, &type, (BYTE *)ret, &needed ); } RegCloseKey( hkey ); if (ret) return expand_env_string( ret, type ); } return NULL; } static BOOL copy_file( const char *src, const char *dst ) { int fds[2] = {-1, -1}, num; char buf[1024]; BOOL ret = FALSE; fds[0] = open( src, O_RDONLY ); fds[1] = open( dst, O_CREAT | O_TRUNC | O_WRONLY, 0666 ); if (fds[0] == -1 || fds[1] == -1) goto fail; while ((num = read( fds[0], buf, sizeof(buf) )) != 0) { if (num == -1) goto fail; if (write( fds[1], buf, num ) != num) goto fail; } ret = TRUE; fail: if (fds[1] != -1) close( fds[1] ); if (fds[0] != -1) close( fds[0] ); return ret; } static BOOL get_internal_fallback_ppd( const WCHAR *ppd ) { static const WCHAR typeW[] = {'P','P','D','F','I','L','E',0}; char *ptr, *end; DWORD size, written; HANDLE file; BOOL ret; HRSRC res = FindResourceW( WINSPOOL_hInstance, MAKEINTRESOURCEW(1), typeW ); if (!res || !(ptr = LoadResource( WINSPOOL_hInstance, res ))) return FALSE; size = SizeofResource( WINSPOOL_hInstance, res ); end = memchr( ptr, 0, size ); /* resource file may contain additional nulls */ if (end) size = end - ptr; file = CreateFileW( ppd, GENERIC_WRITE, FILE_SHARE_READ, NULL, CREATE_ALWAYS, 0, 0 ); if (file == INVALID_HANDLE_VALUE) return FALSE; ret = WriteFile( file, ptr, size, &written, NULL ) && written == size; CloseHandle( file ); if (ret) TRACE( "using internal fallback for %s\n", debugstr_w( ppd )); else DeleteFileW( ppd ); return ret; } static BOOL get_fallback_ppd( const char *printer_name, const WCHAR *ppd ) { char *dst, *src = get_fallback_ppd_name( printer_name ); BOOL ret = FALSE; if (!src) return get_internal_fallback_ppd( ppd ); TRACE( "(%s %s) found %s\n", debugstr_a(printer_name), debugstr_w(ppd), debugstr_a(src) ); if (!(dst = wine_get_unix_file_name( ppd ))) goto fail; if (symlink( src, dst ) == -1) if (errno != ENOSYS || !copy_file( src, dst )) goto fail; ret = TRUE; fail: HeapFree( GetProcessHeap(), 0, dst ); HeapFree( GetProcessHeap(), 0, src ); return ret; } static WCHAR *get_ppd_filename( const WCHAR *dir, const WCHAR *file_name ) { static const WCHAR dot_ppd[] = {'.','p','p','d',0}; int len = (strlenW( dir ) + strlenW( file_name )) * sizeof(WCHAR) + sizeof(dot_ppd); WCHAR *ppd = HeapAlloc( GetProcessHeap(), 0, len ); if (!ppd) return NULL; strcpyW( ppd, dir ); strcatW( ppd, file_name ); strcatW( ppd, dot_ppd ); return ppd; } static WCHAR *get_ppd_dir( void ) { static const WCHAR wine_ppds[] = {'w','i','n','e','_','p','p','d','s','\\',0}; DWORD len; WCHAR *dir, tmp_path[MAX_PATH]; BOOL res; len = GetTempPathW( sizeof(tmp_path) / sizeof(tmp_path[0]), tmp_path ); if (!len) return NULL; dir = HeapAlloc( GetProcessHeap(), 0, len * sizeof(WCHAR) + sizeof(wine_ppds) ) ; if (!dir) return NULL; memcpy( dir, tmp_path, len * sizeof(WCHAR) ); memcpy( dir + len, wine_ppds, sizeof(wine_ppds) ); res = CreateDirectoryW( dir, NULL ); if (!res && GetLastError() != ERROR_ALREADY_EXISTS) { HeapFree( GetProcessHeap(), 0, dir ); dir = NULL; } TRACE( "ppd temporary dir: %s\n", debugstr_w(dir) ); return dir; } static void unlink_ppd( const WCHAR *ppd ) { char *unix_name = wine_get_unix_file_name( ppd ); unlink( unix_name ); HeapFree( GetProcessHeap(), 0, unix_name ); } #ifdef SONAME_LIBCUPS static void *cupshandle; #define CUPS_FUNCS \ DO_FUNC(cupsAddOption); \ DO_FUNC(cupsFreeDests); \ DO_FUNC(cupsFreeOptions); \ DO_FUNC(cupsGetDests); \ DO_FUNC(cupsGetOption); \ DO_FUNC(cupsGetPPD); \ DO_FUNC(cupsParseOptions); \ DO_FUNC(cupsPrintFile) #define CUPS_OPT_FUNCS \ DO_FUNC(cupsGetNamedDest); \ DO_FUNC(cupsGetPPD3) #define DO_FUNC(f) static typeof(f) *p##f CUPS_FUNCS; #undef DO_FUNC static cups_dest_t * (*pcupsGetNamedDest)(http_t *, const char *, const char *); static http_status_t (*pcupsGetPPD3)(http_t *, const char *, time_t *, char *, size_t); static http_status_t cupsGetPPD3_wrapper( http_t *http, const char *name, time_t *modtime, char *buffer, size_t bufsize ) { const char *ppd; if (pcupsGetPPD3) return pcupsGetPPD3( http, name, modtime, buffer, bufsize ); if (!pcupsGetPPD) return HTTP_NOT_FOUND; TRACE( "No cupsGetPPD3 implementation, so calling cupsGetPPD\n" ); *modtime = 0; ppd = pcupsGetPPD( name ); TRACE( "cupsGetPPD returns %s\n", debugstr_a(ppd) ); if (!ppd) return HTTP_NOT_FOUND; if (rename( ppd, buffer ) == -1) { BOOL res = copy_file( ppd, buffer ); unlink( ppd ); if (!res) return HTTP_NOT_FOUND; } return HTTP_OK; } static BOOL get_cups_ppd( const char *printer_name, const WCHAR *ppd ) { time_t modtime = 0; http_status_t http_status; char *unix_name = wine_get_unix_file_name( ppd ); TRACE( "(%s, %s)\n", debugstr_a(printer_name), debugstr_w(ppd) ); if (!unix_name) return FALSE; http_status = cupsGetPPD3_wrapper( 0, printer_name, &modtime, unix_name, strlen( unix_name ) + 1 ); if (http_status != HTTP_OK) unlink( unix_name ); HeapFree( GetProcessHeap(), 0, unix_name ); if (http_status == HTTP_OK) return TRUE; TRACE( "failed to get ppd for printer %s from cups (status %d), calling fallback\n", debugstr_a(printer_name), http_status ); return get_fallback_ppd( printer_name, ppd ); } static WCHAR *get_cups_option( const char *name, int num_options, cups_option_t *options ) { const char *value; WCHAR *ret; int len; value = pcupsGetOption( name, num_options, options ); if (!value) return NULL; len = MultiByteToWideChar( CP_UNIXCP, 0, value, -1, NULL, 0 ); ret = HeapAlloc( GetProcessHeap(), 0, len * sizeof(WCHAR) ); if (ret) MultiByteToWideChar( CP_UNIXCP, 0, value, -1, ret, len ); return ret; } static cups_ptype_t get_cups_printer_type( const cups_dest_t *dest ) { WCHAR *type = get_cups_option( "printer-type", dest->num_options, dest->options ), *end; cups_ptype_t ret = 0; if (type && *type) { ret = (cups_ptype_t)strtoulW( type, &end, 10 ); if (*end) ret = 0; } HeapFree( GetProcessHeap(), 0, type ); return ret; } static void load_cups(void) { cupshandle = wine_dlopen( SONAME_LIBCUPS, RTLD_NOW, NULL, 0 ); if (!cupshandle) return; TRACE("%p: %s loaded\n", cupshandle, SONAME_LIBCUPS); #define DO_FUNC(x) \ p##x = wine_dlsym( cupshandle, #x, NULL, 0 ); \ if (!p##x) \ { \ ERR("failed to load symbol %s\n", #x); \ cupshandle = NULL; \ return; \ } CUPS_FUNCS; #undef DO_FUNC #define DO_FUNC(x) p##x = wine_dlsym( cupshandle, #x, NULL, 0 ) CUPS_OPT_FUNCS; #undef DO_FUNC } static BOOL CUPS_LoadPrinters(void) { int i, nrofdests; BOOL hadprinter = FALSE, haddefault = FALSE; cups_dest_t *dests; PRINTER_INFO_2W pi2; WCHAR *port, *ppd_dir = NULL, *ppd; HKEY hkeyPrinter, hkeyPrinters; WCHAR nameW[MAX_PATH]; HANDLE added_printer; cups_ptype_t printer_type; if (!cupshandle) return FALSE; if(RegCreateKeyW(HKEY_LOCAL_MACHINE, PrintersW, &hkeyPrinters) != ERROR_SUCCESS) { ERR("Can't create Printers key\n"); return FALSE; } nrofdests = pcupsGetDests(&dests); TRACE("Found %d CUPS %s:\n", nrofdests, (nrofdests == 1) ? "printer" : "printers"); for (i=0;i<nrofdests;i++) { MultiByteToWideChar(CP_UNIXCP, 0, dests[i].name, -1, nameW, sizeof(nameW) / sizeof(WCHAR)); printer_type = get_cups_printer_type( dests + i ); TRACE( "Printer %d: %s. printer_type %x\n", i, debugstr_w(nameW), printer_type ); if (printer_type & 0x2000000 /* CUPS_PRINTER_SCANNER */) { TRACE( "skipping scanner-only device\n" ); continue; } port = HeapAlloc(GetProcessHeap(), 0, sizeof(CUPS_Port) + lstrlenW(nameW) * sizeof(WCHAR)); lstrcpyW(port, CUPS_Port); lstrcatW(port, nameW); if(RegOpenKeyW(hkeyPrinters, nameW, &hkeyPrinter) == ERROR_SUCCESS) { DWORD status = get_dword_from_reg( hkeyPrinter, StatusW ); /* Printer already in registry, delete the tag added in WINSPOOL_LoadSystemPrinters and continue */ TRACE("Printer already exists\n"); /* overwrite old LPR:* port */ RegSetValueExW(hkeyPrinter, PortW, 0, REG_SZ, (LPBYTE)port, (lstrlenW(port) + 1) * sizeof(WCHAR)); RegDeleteValueW(hkeyPrinter, May_Delete_Value); /* flag that the PPD file should be checked for an update */ set_reg_DWORD( hkeyPrinter, StatusW, status | PRINTER_STATUS_DRIVER_UPDATE_NEEDED ); RegCloseKey(hkeyPrinter); } else { BOOL added_driver = FALSE; if (!ppd_dir && !(ppd_dir = get_ppd_dir())) { HeapFree( GetProcessHeap(), 0, port ); break; } ppd = get_ppd_filename( ppd_dir, nameW ); if (get_cups_ppd( dests[i].name, ppd )) { added_driver = add_printer_driver( nameW, ppd ); unlink_ppd( ppd ); } HeapFree( GetProcessHeap(), 0, ppd ); if (!added_driver) { HeapFree( GetProcessHeap(), 0, port ); continue; } memset(&pi2, 0, sizeof(PRINTER_INFO_2W)); pi2.pPrinterName = nameW; pi2.pDatatype = rawW; pi2.pPrintProcessor = WinPrintW; pi2.pDriverName = nameW; pi2.pComment = get_cups_option( "printer-info", dests[i].num_options, dests[i].options ); pi2.pLocation = get_cups_option( "printer-location", dests[i].num_options, dests[i].options ); pi2.pPortName = port; pi2.pParameters = emptyStringW; pi2.pShareName = emptyStringW; pi2.pSepFile = emptyStringW; added_printer = AddPrinterW( NULL, 2, (LPBYTE)&pi2 ); if (added_printer) ClosePrinter( added_printer ); else if (GetLastError() != ERROR_PRINTER_ALREADY_EXISTS) ERR( "printer '%s' not added by AddPrinter (error %d)\n", debugstr_w(nameW), GetLastError() ); HeapFree( GetProcessHeap(), 0, pi2.pComment ); HeapFree( GetProcessHeap(), 0, pi2.pLocation ); } HeapFree( GetProcessHeap(), 0, port ); hadprinter = TRUE; if (dests[i].is_default) { SetDefaultPrinterW(nameW); haddefault = TRUE; } } if (ppd_dir) { RemoveDirectoryW( ppd_dir ); HeapFree( GetProcessHeap(), 0, ppd_dir ); } if (hadprinter && !haddefault) { MultiByteToWideChar(CP_UNIXCP, 0, dests[0].name, -1, nameW, sizeof(nameW) / sizeof(WCHAR)); SetDefaultPrinterW(nameW); } pcupsFreeDests(nrofdests, dests); RegCloseKey(hkeyPrinters); return TRUE; } #endif static char *get_queue_name( HANDLE printer, BOOL *cups ) { WCHAR *port, *name = NULL; DWORD err, needed, type; char *ret = NULL; HKEY key; *cups = FALSE; err = WINSPOOL_GetOpenedPrinterRegKey( printer, &key ); if (err) return NULL; err = RegQueryValueExW( key, PortW, 0, &type, NULL, &needed ); if (err) goto end; port = HeapAlloc( GetProcessHeap(), 0, needed ); if (!port) goto end; RegQueryValueExW( key, PortW, 0, &type, (BYTE*)port, &needed ); if (!strncmpW( port, CUPS_Port, sizeof(CUPS_Port) / sizeof(WCHAR) -1 )) { name = port + sizeof(CUPS_Port) / sizeof(WCHAR) - 1; *cups = TRUE; } else if (!strncmpW( port, LPR_Port, sizeof(LPR_Port) / sizeof(WCHAR) -1 )) name = port + sizeof(LPR_Port) / sizeof(WCHAR) - 1; if (name) { needed = WideCharToMultiByte( CP_UNIXCP, 0, name, -1, NULL, 0, NULL, NULL ); ret = HeapAlloc( GetProcessHeap(), 0, needed ); if(ret) WideCharToMultiByte( CP_UNIXCP, 0, name, -1, ret, needed, NULL, NULL ); } HeapFree( GetProcessHeap(), 0, port ); end: RegCloseKey( key ); return ret; } static void set_ppd_overrides( HANDLE printer ) { WCHAR *wstr = NULL; int size = 0; #ifdef HAVE_APPLICATIONSERVICES_APPLICATIONSERVICES_H OSStatus status; PMPrintSession session = NULL; PMPageFormat format = NULL; PMPaper paper; CFStringRef paper_name; CFRange range; status = PMCreateSession( &session ); if (status) goto end; status = PMCreatePageFormat( &format ); if (status) goto end; status = PMSessionDefaultPageFormat( session, format ); if (status) goto end; status = PMGetPageFormatPaper( format, &paper ); if (status) goto end; status = PMPaperGetPPDPaperName( paper, &paper_name ); if (status) goto end; range.location = 0; range.length = CFStringGetLength( paper_name ); size = (range.length + 1) * sizeof(WCHAR); wstr = HeapAlloc( GetProcessHeap(), 0, size ); CFStringGetCharacters( paper_name, range, (UniChar*)wstr ); wstr[range.length] = 0; end: if (format) PMRelease( format ); if (session) PMRelease( session ); #endif SetPrinterDataExW( printer, PPD_Overrides, DefaultPageSize, REG_SZ, (BYTE*)wstr, size ); HeapFree( GetProcessHeap(), 0, wstr ); } static BOOL update_driver( HANDLE printer ) { BOOL ret, is_cups; const WCHAR *name = get_opened_printer_name( printer ); WCHAR *ppd_dir, *ppd; char *queue_name; if (!name) return FALSE; queue_name = get_queue_name( printer, &is_cups ); if (!queue_name) return FALSE; if (!(ppd_dir = get_ppd_dir())) { HeapFree( GetProcessHeap(), 0, queue_name ); return FALSE; } ppd = get_ppd_filename( ppd_dir, name ); #ifdef SONAME_LIBCUPS if (is_cups) ret = get_cups_ppd( queue_name, ppd ); else #endif ret = get_fallback_ppd( queue_name, ppd ); if (ret) { TRACE( "updating driver %s\n", debugstr_w( name ) ); ret = add_printer_driver( name, ppd ); unlink_ppd( ppd ); } HeapFree( GetProcessHeap(), 0, ppd_dir ); HeapFree( GetProcessHeap(), 0, ppd ); HeapFree( GetProcessHeap(), 0, queue_name ); set_ppd_overrides( printer ); /* call into the driver to update the devmode */ DocumentPropertiesW( 0, printer, NULL, NULL, NULL, 0 ); return ret; } static BOOL PRINTCAP_ParseEntry( const char *pent, BOOL isfirst ) { PRINTER_INFO_2A pinfo2a; const char *r; size_t name_len; char *e,*s,*name,*prettyname,*devname; BOOL ret = FALSE, set_default = FALSE; char *port = NULL, *env_default; HKEY hkeyPrinter, hkeyPrinters = NULL; WCHAR devnameW[MAX_PATH], *ppd_dir = NULL, *ppd; HANDLE added_printer; while (isspace(*pent)) pent++; r = strchr(pent,':'); if (r) name_len = r - pent; else name_len = strlen(pent); name = HeapAlloc(GetProcessHeap(), 0, name_len + 1); memcpy(name, pent, name_len); name[name_len] = '\0'; if (r) pent = r; else pent = ""; TRACE("name=%s entry=%s\n",name, pent); if(ispunct(*name)) { /* a tc entry, not a real printer */ TRACE("skipping tc entry\n"); goto end; } if(strstr(pent,":server")) { /* server only version so skip */ TRACE("skipping server entry\n"); goto end; } /* Determine whether this is a postscript printer. */ ret = TRUE; env_default = getenv("PRINTER"); prettyname = name; /* Get longest name, usually the one at the right for later display. */ while((s=strchr(prettyname,'|'))) { *s = '\0'; e = s; while(isspace(*--e)) *e = '\0'; TRACE("\t%s\n", debugstr_a(prettyname)); if(env_default && !strcasecmp(prettyname, env_default)) set_default = TRUE; for(prettyname = s+1; isspace(*prettyname); prettyname++) ; } e = prettyname + strlen(prettyname); while(isspace(*--e)) *e = '\0'; TRACE("\t%s\n", debugstr_a(prettyname)); if(env_default && !strcasecmp(prettyname, env_default)) set_default = TRUE; /* prettyname must fit into the dmDeviceName member of DEVMODE struct, * if it is too long, we use it as comment below. */ devname = prettyname; if (strlen(devname)>=CCHDEVICENAME-1) devname = name; if (strlen(devname)>=CCHDEVICENAME-1) { ret = FALSE; goto end; } port = HeapAlloc(GetProcessHeap(),0,strlen("LPR:")+strlen(name)+1); sprintf(port,"LPR:%s",name); if(RegCreateKeyW(HKEY_LOCAL_MACHINE, PrintersW, &hkeyPrinters) != ERROR_SUCCESS) { ERR("Can't create Printers key\n"); ret = FALSE; goto end; } MultiByteToWideChar(CP_ACP, 0, devname, -1, devnameW, sizeof(devnameW) / sizeof(WCHAR)); if(RegOpenKeyA(hkeyPrinters, devname, &hkeyPrinter) == ERROR_SUCCESS) { DWORD status = get_dword_from_reg( hkeyPrinter, StatusW ); /* Printer already in registry, delete the tag added in WINSPOOL_LoadSystemPrinters and continue */ TRACE("Printer already exists\n"); RegDeleteValueW(hkeyPrinter, May_Delete_Value); /* flag that the PPD file should be checked for an update */ set_reg_DWORD( hkeyPrinter, StatusW, status | PRINTER_STATUS_DRIVER_UPDATE_NEEDED ); RegCloseKey(hkeyPrinter); } else { static CHAR data_type[] = "RAW", print_proc[] = "WinPrint", comment[] = "WINEPS Printer using LPR", params[] = "<parameters?>", share_name[] = "<share name?>", sep_file[] = "<sep file?>"; BOOL added_driver = FALSE; if (!ppd_dir && !(ppd_dir = get_ppd_dir())) goto end; ppd = get_ppd_filename( ppd_dir, devnameW ); if (get_fallback_ppd( devname, ppd )) { added_driver = add_printer_driver( devnameW, ppd ); unlink_ppd( ppd ); } HeapFree( GetProcessHeap(), 0, ppd ); if (!added_driver) goto end; memset(&pinfo2a,0,sizeof(pinfo2a)); pinfo2a.pPrinterName = devname; pinfo2a.pDatatype = data_type; pinfo2a.pPrintProcessor = print_proc; pinfo2a.pDriverName = devname; pinfo2a.pComment = comment; pinfo2a.pLocation = prettyname; pinfo2a.pPortName = port; pinfo2a.pParameters = params; pinfo2a.pShareName = share_name; pinfo2a.pSepFile = sep_file; added_printer = AddPrinterA( NULL, 2, (LPBYTE)&pinfo2a ); if (added_printer) ClosePrinter( added_printer ); else if (GetLastError() != ERROR_PRINTER_ALREADY_EXISTS) ERR( "printer '%s' not added by AddPrinter (error %d)\n", debugstr_a(name), GetLastError() ); } if (isfirst || set_default) WINSPOOL_SetDefaultPrinter(devname,name,TRUE); end: if (hkeyPrinters) RegCloseKey( hkeyPrinters ); if (ppd_dir) { RemoveDirectoryW( ppd_dir ); HeapFree( GetProcessHeap(), 0, ppd_dir ); } HeapFree(GetProcessHeap(), 0, port); HeapFree(GetProcessHeap(), 0, name); return ret; } static BOOL PRINTCAP_LoadPrinters(void) { BOOL hadprinter = FALSE; char buf[200]; FILE *f; char *pent = NULL; BOOL had_bash = FALSE; f = fopen("/etc/printcap","r"); if (!f) return FALSE; while(fgets(buf,sizeof(buf),f)) { char *start, *end; end=strchr(buf,'\n'); if (end) *end='\0'; start = buf; while(isspace(*start)) start++; if(*start == '#' || *start == '\0') continue; if(pent && !had_bash && *start != ':' && *start != '|') { /* start of new entry, parse the previous one */ hadprinter |= PRINTCAP_ParseEntry(pent,!hadprinter); HeapFree(GetProcessHeap(),0,pent); pent = NULL; } if (end && *--end == '\\') { *end = '\0'; had_bash = TRUE; } else had_bash = FALSE; if (pent) { pent=HeapReAlloc(GetProcessHeap(),0,pent,strlen(pent)+strlen(start)+1); strcat(pent,start); } else { pent=HeapAlloc(GetProcessHeap(),0,strlen(start)+1); strcpy(pent,start); } } if(pent) { hadprinter |= PRINTCAP_ParseEntry(pent,!hadprinter); HeapFree(GetProcessHeap(),0,pent); } fclose(f); return hadprinter; } static inline DWORD set_reg_szW(HKEY hkey, const WCHAR *keyname, const WCHAR *value) { if (value) return RegSetValueExW(hkey, keyname, 0, REG_SZ, (const BYTE*)value, (lstrlenW(value) + 1) * sizeof(WCHAR)); else return ERROR_FILE_NOT_FOUND; } static inline DWORD set_reg_devmode( HKEY key, const WCHAR *name, const DEVMODEW *dm ) { DEVMODEA *dmA = DEVMODEdupWtoA( dm ); DWORD ret = ERROR_FILE_NOT_FOUND; /* FIXME: Write DEVMODEA not DEVMODEW into reg. This is what win9x does and we support these drivers. NT writes DEVMODEW so somehow we'll need to distinguish between these when we support NT drivers */ if (dmA) { ret = RegSetValueExW( key, name, 0, REG_BINARY, (LPBYTE)dmA, dmA->dmSize + dmA->dmDriverExtra ); HeapFree( GetProcessHeap(), 0, dmA ); } return ret; } /****************************************************************** * get_servername_from_name (internal) * * for an external server, a copy of the serverpart from the full name is returned * */ static LPWSTR get_servername_from_name(LPCWSTR name) { LPWSTR server; LPWSTR ptr; WCHAR buffer[MAX_PATH]; DWORD len; if (name == NULL) return NULL; if ((name[0] != '\\') || (name[1] != '\\')) return NULL; server = strdupW(&name[2]); /* skip over both backslash */ if (server == NULL) return NULL; /* strip '\' and the printername */ ptr = strchrW(server, '\\'); if (ptr) ptr[0] = '\0'; TRACE("found %s\n", debugstr_w(server)); len = sizeof(buffer)/sizeof(buffer[0]); if (GetComputerNameW(buffer, &len)) { if (lstrcmpW(buffer, server) == 0) { /* The requested Servername is our computername */ HeapFree(GetProcessHeap(), 0, server); return NULL; } } return server; } /****************************************************************** * get_basename_from_name (internal) * * skip over the serverpart from the full name * */ static LPCWSTR get_basename_from_name(LPCWSTR name) { if (name == NULL) return NULL; if ((name[0] == '\\') && (name[1] == '\\')) { /* skip over the servername and search for the following '\' */ name = strchrW(&name[2], '\\'); if ((name) && (name[1])) { /* found a separator ('\') followed by a name: skip over the separator and return the rest */ name++; } else { /* no basename present (we found only a servername) */ return NULL; } } return name; } static void free_printer_entry( opened_printer_t *printer ) { /* the queue is shared, so don't free that here */ HeapFree( GetProcessHeap(), 0, printer->printername ); HeapFree( GetProcessHeap(), 0, printer->name ); HeapFree( GetProcessHeap(), 0, printer->devmode ); HeapFree( GetProcessHeap(), 0, printer ); } /****************************************************************** * get_opened_printer_entry * Get the first place empty in the opened printer table * * ToDo: * - pDefault is ignored */ static HANDLE get_opened_printer_entry(LPWSTR name, LPPRINTER_DEFAULTSW pDefault) { UINT_PTR handle = nb_printer_handles, i; jobqueue_t *queue = NULL; opened_printer_t *printer = NULL; LPWSTR servername; LPCWSTR printername; if ((backend == NULL) && !load_backend()) return NULL; servername = get_servername_from_name(name); if (servername) { FIXME("server %s not supported\n", debugstr_w(servername)); HeapFree(GetProcessHeap(), 0, servername); SetLastError(ERROR_INVALID_PRINTER_NAME); return NULL; } printername = get_basename_from_name(name); if (name != printername) TRACE("converted %s to %s\n", debugstr_w(name), debugstr_w(printername)); /* an empty printername is invalid */ if (printername && (!printername[0])) { SetLastError(ERROR_INVALID_PARAMETER); return NULL; } EnterCriticalSection(&printer_handles_cs); for (i = 0; i < nb_printer_handles; i++) { if (!printer_handles[i]) { if(handle == nb_printer_handles) handle = i; } else { if(!queue && (name) && !lstrcmpW(name, printer_handles[i]->name)) queue = printer_handles[i]->queue; } } if (handle >= nb_printer_handles) { opened_printer_t **new_array; if (printer_handles) new_array = HeapReAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, printer_handles, (nb_printer_handles + 16) * sizeof(*new_array) ); else new_array = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, (nb_printer_handles + 16) * sizeof(*new_array) ); if (!new_array) { handle = 0; goto end; } printer_handles = new_array; nb_printer_handles += 16; } if (!(printer = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*printer)))) { handle = 0; goto end; } /* get a printer handle from the backend */ if (! backend->fpOpenPrinter(name, &printer->backend_printer, pDefault)) { handle = 0; goto end; } /* clone the base name. This is NULL for the printserver */ printer->printername = strdupW(printername); /* clone the full name */ printer->name = strdupW(name); if (name && (!printer->name)) { handle = 0; goto end; } if (pDefault && pDefault->pDevMode) printer->devmode = dup_devmode( pDefault->pDevMode ); if(queue) printer->queue = queue; else { printer->queue = HeapAlloc(GetProcessHeap(), 0, sizeof(*queue)); if (!printer->queue) { handle = 0; goto end; } list_init(&printer->queue->jobs); printer->queue->ref = 0; } InterlockedIncrement(&printer->queue->ref); printer_handles[handle] = printer; handle++; end: LeaveCriticalSection(&printer_handles_cs); if (!handle && printer) { if (!queue) HeapFree(GetProcessHeap(), 0, printer->queue); free_printer_entry( printer ); } return (HANDLE)handle; } static void old_printer_check( BOOL delete_phase ) { PRINTER_INFO_5W* pi; DWORD needed, type, num, delete, i, size; const DWORD one = 1; HKEY key; HANDLE hprn; EnumPrintersW( PRINTER_ENUM_LOCAL, NULL, 5, NULL, 0, &needed, &num ); if (GetLastError() != ERROR_INSUFFICIENT_BUFFER) return; pi = HeapAlloc( GetProcessHeap(), 0, needed ); EnumPrintersW( PRINTER_ENUM_LOCAL, NULL, 5, (LPBYTE)pi, needed, &needed, &num ); for (i = 0; i < num; i++) { if (strncmpW( pi[i].pPortName, CUPS_Port, strlenW(CUPS_Port) ) && strncmpW( pi[i].pPortName, LPR_Port, strlenW(LPR_Port) )) continue; if (open_printer_reg_key( pi[i].pPrinterName, &key )) continue; if (!delete_phase) { RegSetValueExW( key, May_Delete_Value, 0, REG_DWORD, (LPBYTE)&one, sizeof(one) ); RegCloseKey( key ); } else { delete = 0; size = sizeof( delete ); RegQueryValueExW( key, May_Delete_Value, NULL, &type, (LPBYTE)&delete, &size ); RegCloseKey( key ); if (delete) { TRACE( "Deleting old printer %s\n", debugstr_w(pi[i].pPrinterName) ); if (OpenPrinterW( pi[i].pPrinterName, &hprn, NULL )) { DeletePrinter( hprn ); ClosePrinter( hprn ); } DeletePrinterDriverExW( NULL, NULL, pi[i].pPrinterName, 0, 0 ); } } } HeapFree(GetProcessHeap(), 0, pi); } static const WCHAR winspool_mutex_name[] = {'_','_','W','I','N','E','_','W','I','N','S','P','O','O','L','_', 'M','U','T','E','X','_','_','\0'}; static HANDLE init_mutex; void WINSPOOL_LoadSystemPrinters(void) { HKEY hkey, hkeyPrinters; DWORD needed, num, i; WCHAR PrinterName[256]; BOOL done = FALSE; #ifdef SONAME_LIBCUPS load_cups(); #endif /* FIXME: The init code should be moved to spoolsv.exe */ init_mutex = CreateMutexW( NULL, TRUE, winspool_mutex_name ); if (!init_mutex) { ERR( "Failed to create mutex\n" ); return; } if (GetLastError() == ERROR_ALREADY_EXISTS) { WaitForSingleObject( init_mutex, INFINITE ); ReleaseMutex( init_mutex ); TRACE( "Init already done\n" ); return; } /* This ensures that all printer entries have a valid Name value. If causes problems later if they don't. If one is found to be missed we create one and set it equal to the name of the key */ if(RegCreateKeyW(HKEY_LOCAL_MACHINE, PrintersW, &hkeyPrinters) == ERROR_SUCCESS) { if(RegQueryInfoKeyW(hkeyPrinters, NULL, NULL, NULL, &num, NULL, NULL, NULL, NULL, NULL, NULL, NULL) == ERROR_SUCCESS) { for(i = 0; i < num; i++) { if(RegEnumKeyW(hkeyPrinters, i, PrinterName, sizeof(PrinterName)/sizeof(PrinterName[0])) == ERROR_SUCCESS) { if(RegOpenKeyW(hkeyPrinters, PrinterName, &hkey) == ERROR_SUCCESS) { if(RegQueryValueExW(hkey, NameW, 0, 0, 0, &needed) == ERROR_FILE_NOT_FOUND) { set_reg_szW(hkey, NameW, PrinterName); } RegCloseKey(hkey); } } } } RegCloseKey(hkeyPrinters); } old_printer_check( FALSE ); #ifdef SONAME_LIBCUPS done = CUPS_LoadPrinters(); #endif if(!done) /* If we have any CUPS based printers, skip looking for printcap printers */ PRINTCAP_LoadPrinters(); old_printer_check( TRUE ); ReleaseMutex( init_mutex ); return; } /****************************************************************** * get_job * * Get the pointer to the specified job. * Should hold the printer_handles_cs before calling. */ static job_t *get_job(HANDLE hprn, DWORD JobId) { opened_printer_t *printer = get_opened_printer(hprn); job_t *job; if(!printer) return NULL; LIST_FOR_EACH_ENTRY(job, &printer->queue->jobs, job_t, entry) { if(job->job_id == JobId) return job; } return NULL; } /*********************************************************** * DEVMODEcpyAtoW */ static LPDEVMODEW DEVMODEcpyAtoW(DEVMODEW *dmW, const DEVMODEA *dmA) { BOOL Formname; ptrdiff_t off_formname = (const char *)dmA->dmFormName - (const char *)dmA; DWORD size; Formname = (dmA->dmSize > off_formname); size = dmA->dmSize + CCHDEVICENAME + (Formname ? CCHFORMNAME : 0); MultiByteToWideChar(CP_ACP, 0, (LPCSTR)dmA->dmDeviceName, -1, dmW->dmDeviceName, CCHDEVICENAME); if(!Formname) { memcpy(&dmW->dmSpecVersion, &dmA->dmSpecVersion, dmA->dmSize - CCHDEVICENAME); } else { memcpy(&dmW->dmSpecVersion, &dmA->dmSpecVersion, off_formname - CCHDEVICENAME); MultiByteToWideChar(CP_ACP, 0, (LPCSTR)dmA->dmFormName, -1, dmW->dmFormName, CCHFORMNAME); memcpy(&dmW->dmLogPixels, &dmA->dmLogPixels, dmA->dmSize - (off_formname + CCHFORMNAME)); } dmW->dmSize = size; memcpy((char *)dmW + dmW->dmSize, (const char *)dmA + dmA->dmSize, dmA->dmDriverExtra); return dmW; } /****************************************************************** * convert_printerinfo_W_to_A [internal] * */ static void convert_printerinfo_W_to_A(LPBYTE out, LPBYTE pPrintersW, DWORD level, DWORD outlen, DWORD numentries) { DWORD id = 0; LPSTR ptr; INT len; TRACE("(%p, %p, %d, %u, %u)\n", out, pPrintersW, level, outlen, numentries); len = pi_sizeof[level] * numentries; ptr = (LPSTR) out + len; outlen -= len; /* copy the numbers of all PRINTER_INFO_* first */ memcpy(out, pPrintersW, len); while (id < numentries) { switch (level) { case 1: { PRINTER_INFO_1W * piW = (PRINTER_INFO_1W *) pPrintersW; PRINTER_INFO_1A * piA = (PRINTER_INFO_1A *) out; TRACE("(%u) #%u: %s\n", level, id, debugstr_w(piW->pName)); if (piW->pDescription) { piA->pDescription = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pDescription, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pName) { piA->pName = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pName, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pComment) { piA->pComment = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pComment, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } break; } case 2: { PRINTER_INFO_2W * piW = (PRINTER_INFO_2W *) pPrintersW; PRINTER_INFO_2A * piA = (PRINTER_INFO_2A *) out; LPDEVMODEA dmA; TRACE("(%u) #%u: %s\n", level, id, debugstr_w(piW->pPrinterName)); if (piW->pServerName) { piA->pServerName = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pServerName, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pPrinterName) { piA->pPrinterName = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pPrinterName, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pShareName) { piA->pShareName = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pShareName, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pPortName) { piA->pPortName = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pPortName, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pDriverName) { piA->pDriverName = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pDriverName, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pComment) { piA->pComment = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pComment, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pLocation) { piA->pLocation = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pLocation, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } dmA = DEVMODEdupWtoA(piW->pDevMode); if (dmA) { /* align DEVMODEA to a DWORD boundary */ len = (4 - ( (DWORD_PTR) ptr & 3)) & 3; ptr += len; outlen -= len; piA->pDevMode = (LPDEVMODEA) ptr; len = dmA->dmSize + dmA->dmDriverExtra; memcpy(ptr, dmA, len); HeapFree(GetProcessHeap(), 0, dmA); ptr += len; outlen -= len; } if (piW->pSepFile) { piA->pSepFile = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pSepFile, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pPrintProcessor) { piA->pPrintProcessor = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pPrintProcessor, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pDatatype) { piA->pDatatype = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pDatatype, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pParameters) { piA->pParameters = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pParameters, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pSecurityDescriptor) { piA->pSecurityDescriptor = NULL; FIXME("pSecurityDescriptor ignored: %s\n", debugstr_w(piW->pPrinterName)); } break; } case 4: { PRINTER_INFO_4W * piW = (PRINTER_INFO_4W *) pPrintersW; PRINTER_INFO_4A * piA = (PRINTER_INFO_4A *) out; TRACE("(%u) #%u: %s\n", level, id, debugstr_w(piW->pPrinterName)); if (piW->pPrinterName) { piA->pPrinterName = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pPrinterName, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pServerName) { piA->pServerName = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pServerName, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } break; } case 5: { PRINTER_INFO_5W * piW = (PRINTER_INFO_5W *) pPrintersW; PRINTER_INFO_5A * piA = (PRINTER_INFO_5A *) out; TRACE("(%u) #%u: %s\n", level, id, debugstr_w(piW->pPrinterName)); if (piW->pPrinterName) { piA->pPrinterName = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pPrinterName, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } if (piW->pPortName) { piA->pPortName = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pPortName, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } break; } case 6: /* 6A and 6W are the same structure */ break; case 7: { PRINTER_INFO_7W * piW = (PRINTER_INFO_7W *) pPrintersW; PRINTER_INFO_7A * piA = (PRINTER_INFO_7A *) out; TRACE("(%u) #%u\n", level, id); if (piW->pszObjectGUID) { piA->pszObjectGUID = ptr; len = WideCharToMultiByte(CP_ACP, 0, piW->pszObjectGUID, -1, ptr, outlen, NULL, NULL); ptr += len; outlen -= len; } break; } case 8: case 9: { PRINTER_INFO_9W * piW = (PRINTER_INFO_9W *) pPrintersW; PRINTER_INFO_9A * piA = (PRINTER_INFO_9A *) out; LPDEVMODEA dmA; TRACE("(%u) #%u\n", level, id); dmA = DEVMODEdupWtoA(piW->pDevMode); if (dmA) { /* align DEVMODEA to a DWORD boundary */ len = (4 - ( (DWORD_PTR) ptr & 3)) & 3; ptr += len; outlen -= len; piA->pDevMode = (LPDEVMODEA) ptr; len = dmA->dmSize + dmA->dmDriverExtra; memcpy(ptr, dmA, len); HeapFree(GetProcessHeap(), 0, dmA); ptr += len; outlen -= len; } break; } default: FIXME("for level %u\n", level); } pPrintersW += pi_sizeof[level]; out += pi_sizeof[level]; id++; } } /****************************************************************** * convert_driverinfo_W_to_A [internal] * */ static void convert_driverinfo_W_to_A(LPBYTE out, LPBYTE pDriversW, DWORD level, DWORD outlen, DWORD numentries) { DWORD id = 0; LPSTR ptr; INT len; TRACE("(%p, %p, %d, %u, %u)\n", out, pDriversW, level, outlen, numentries); len = di_sizeof[level] * numentries; ptr = (LPSTR) out + len; outlen -= len; /* copy the numbers of all PRINTER_INFO_* first */ memcpy(out, pDriversW, len); #define COPY_STRING(fld) \ { if (diW->fld){ \ diA->fld = ptr; \ len = WideCharToMultiByte(CP_ACP, 0, diW->fld, -1, ptr, outlen, NULL, NULL);\ ptr += len; outlen -= len;\ }} #define COPY_MULTIZ_STRING(fld) \ { LPWSTR p = diW->fld; if (p){ \ diA->fld = ptr; \ do {\ len = WideCharToMultiByte(CP_ACP, 0, p, -1, ptr, outlen, NULL, NULL);\ ptr += len; outlen -= len; p += len;\ }\ while(len > 1 && outlen > 0); \ }} while (id < numentries) { switch (level) { case 1: { DRIVER_INFO_1W * diW = (DRIVER_INFO_1W *) pDriversW; DRIVER_INFO_1A * diA = (DRIVER_INFO_1A *) out; TRACE("(%u) #%u: %s\n", level, id, debugstr_w(diW->pName)); COPY_STRING(pName); break; } case 2: { DRIVER_INFO_2W * diW = (DRIVER_INFO_2W *) pDriversW; DRIVER_INFO_2A * diA = (DRIVER_INFO_2A *) out; TRACE("(%u) #%u: %s\n", level, id, debugstr_w(diW->pName)); COPY_STRING(pName); COPY_STRING(pEnvironment); COPY_STRING(pDriverPath); COPY_STRING(pDataFile); COPY_STRING(pConfigFile); break; } case 3: { DRIVER_INFO_3W * diW = (DRIVER_INFO_3W *) pDriversW; DRIVER_INFO_3A * diA = (DRIVER_INFO_3A *) out; TRACE("(%u) #%u: %s\n", level, id, debugstr_w(diW->pName)); COPY_STRING(pName); COPY_STRING(pEnvironment); COPY_STRING(pDriverPath); COPY_STRING(pDataFile); COPY_STRING(pConfigFile); COPY_STRING(pHelpFile); COPY_MULTIZ_STRING(pDependentFiles); COPY_STRING(pMonitorName); COPY_STRING(pDefaultDataType); break; } case 4: { DRIVER_INFO_4W * diW = (DRIVER_INFO_4W *) pDriversW; DRIVER_INFO_4A * diA = (DRIVER_INFO_4A *) out; TRACE("(%u) #%u: %s\n", level, id, debugstr_w(diW->pName)); COPY_STRING(pName); COPY_STRING(pEnvironment); COPY_STRING(pDriverPath); COPY_STRING(pDataFile); COPY_STRING(pConfigFile); COPY_STRING(pHelpFile); COPY_MULTIZ_STRING(pDependentFiles); COPY_STRING(pMonitorName); COPY_STRING(pDefaultDataType); COPY_MULTIZ_STRING(pszzPreviousNames); break; } case 5: { DRIVER_INFO_5W * diW = (DRIVER_INFO_5W *) pDriversW; DRIVER_INFO_5A * diA = (DRIVER_INFO_5A *) out; TRACE("(%u) #%u: %s\n", level, id, debugstr_w(diW->pName)); COPY_STRING(pName); COPY_STRING(pEnvironment); COPY_STRING(pDriverPath); COPY_STRING(pDataFile); COPY_STRING(pConfigFile); break; } case 6: { DRIVER_INFO_6W * diW = (DRIVER_INFO_6W *) pDriversW; DRIVER_INFO_6A * diA = (DRIVER_INFO_6A *) out; TRACE("(%u) #%u: %s\n", level, id, debugstr_w(diW->pName)); COPY_STRING(pName); COPY_STRING(pEnvironment); COPY_STRING(pDriverPath); COPY_STRING(pDataFile); COPY_STRING(pConfigFile); COPY_STRING(pHelpFile); COPY_MULTIZ_STRING(pDependentFiles); COPY_STRING(pMonitorName); COPY_STRING(pDefaultDataType); COPY_MULTIZ_STRING(pszzPreviousNames); COPY_STRING(pszMfgName); COPY_STRING(pszOEMUrl); COPY_STRING(pszHardwareID); COPY_STRING(pszProvider); break; } case 8: { DRIVER_INFO_8W * diW = (DRIVER_INFO_8W *) pDriversW; DRIVER_INFO_8A * diA = (DRIVER_INFO_8A *) out; TRACE("(%u) #%u: %s\n", level, id, debugstr_w(diW->pName)); COPY_STRING(pName); COPY_STRING(pEnvironment); COPY_STRING(pDriverPath); COPY_STRING(pDataFile); COPY_STRING(pConfigFile); COPY_STRING(pHelpFile); COPY_MULTIZ_STRING(pDependentFiles); COPY_STRING(pMonitorName); COPY_STRING(pDefaultDataType); COPY_MULTIZ_STRING(pszzPreviousNames); COPY_STRING(pszMfgName); COPY_STRING(pszOEMUrl); COPY_STRING(pszHardwareID); COPY_STRING(pszProvider); COPY_STRING(pszPrintProcessor); COPY_STRING(pszVendorSetup); COPY_MULTIZ_STRING(pszzColorProfiles); COPY_STRING(pszInfPath); COPY_MULTIZ_STRING(pszzCoreDriverDependencies); break; } default: FIXME("for level %u\n", level); } pDriversW += di_sizeof[level]; out += di_sizeof[level]; id++; } #undef COPY_STRING #undef COPY_MULTIZ_STRING } /*********************************************************** * printer_info_AtoW */ static void *printer_info_AtoW( const void *data, DWORD level ) { void *ret; UNICODE_STRING usBuffer; if (!data) return NULL; if (level < 1 || level > 9) return NULL; ret = HeapAlloc( GetProcessHeap(), 0, pi_sizeof[level] ); if (!ret) return NULL; memcpy( ret, data, pi_sizeof[level] ); /* copy everything first */ switch (level) { case 2: { const PRINTER_INFO_2A *piA = (const PRINTER_INFO_2A *)data; PRINTER_INFO_2W *piW = (PRINTER_INFO_2W *)ret; piW->pServerName = asciitounicode( &usBuffer, piA->pServerName ); piW->pPrinterName = asciitounicode( &usBuffer, piA->pPrinterName ); piW->pShareName = asciitounicode( &usBuffer, piA->pShareName ); piW->pPortName = asciitounicode( &usBuffer, piA->pPortName ); piW->pDriverName = asciitounicode( &usBuffer, piA->pDriverName ); piW->pComment = asciitounicode( &usBuffer, piA->pComment ); piW->pLocation = asciitounicode( &usBuffer, piA->pLocation ); piW->pDevMode = piA->pDevMode ? GdiConvertToDevmodeW( piA->pDevMode ) : NULL; piW->pSepFile = asciitounicode( &usBuffer, piA->pSepFile ); piW->pPrintProcessor = asciitounicode( &usBuffer, piA->pPrintProcessor ); piW->pDatatype = asciitounicode( &usBuffer, piA->pDatatype ); piW->pParameters = asciitounicode( &usBuffer, piA->pParameters ); break; } case 8: case 9: { const PRINTER_INFO_9A *piA = (const PRINTER_INFO_9A *)data; PRINTER_INFO_9W *piW = (PRINTER_INFO_9W *)ret; piW->pDevMode = piA->pDevMode ? GdiConvertToDevmodeW( piA->pDevMode ) : NULL; break; } default: FIXME( "Unhandled level %d\n", level ); HeapFree( GetProcessHeap(), 0, ret ); return NULL; } return ret; } /*********************************************************** * free_printer_info */ static void free_printer_info( void *data, DWORD level ) { if (!data) return; switch (level) { case 2: { PRINTER_INFO_2W *piW = (PRINTER_INFO_2W *)data; HeapFree( GetProcessHeap(), 0, piW->pServerName ); HeapFree( GetProcessHeap(), 0, piW->pPrinterName ); HeapFree( GetProcessHeap(), 0, piW->pShareName ); HeapFree( GetProcessHeap(), 0, piW->pPortName ); HeapFree( GetProcessHeap(), 0, piW->pDriverName ); HeapFree( GetProcessHeap(), 0, piW->pComment ); HeapFree( GetProcessHeap(), 0, piW->pLocation ); HeapFree( GetProcessHeap(), 0, piW->pDevMode ); HeapFree( GetProcessHeap(), 0, piW->pSepFile ); HeapFree( GetProcessHeap(), 0, piW->pPrintProcessor ); HeapFree( GetProcessHeap(), 0, piW->pDatatype ); HeapFree( GetProcessHeap(), 0, piW->pParameters ); break; } case 8: case 9: { PRINTER_INFO_9W *piW = (PRINTER_INFO_9W *)data; HeapFree( GetProcessHeap(), 0, piW->pDevMode ); break; } default: FIXME( "Unhandled level %d\n", level ); } HeapFree( GetProcessHeap(), 0, data ); return; } /****************************************************************** * DeviceCapabilities [WINSPOOL.@] * DeviceCapabilitiesA [WINSPOOL.@] * */ INT WINAPI DeviceCapabilitiesA(LPCSTR pDevice,LPCSTR pPort, WORD cap, LPSTR pOutput, LPDEVMODEA lpdm) { INT ret; TRACE("%s,%s,%u,%p,%p\n", debugstr_a(pDevice), debugstr_a(pPort), cap, pOutput, lpdm); if (!GDI_CallDeviceCapabilities16) { GDI_CallDeviceCapabilities16 = (void*)GetProcAddress( GetModuleHandleA("gdi32"), (LPCSTR)104 ); if (!GDI_CallDeviceCapabilities16) return -1; } ret = GDI_CallDeviceCapabilities16(pDevice, pPort, cap, pOutput, lpdm); /* If DC_PAPERSIZE map POINT16s to POINTs */ if(ret != -1 && cap == DC_PAPERSIZE && pOutput) { POINT16 *tmp = HeapAlloc( GetProcessHeap(), 0, ret * sizeof(POINT16) ); POINT *pt = (POINT *)pOutput; INT i; memcpy(tmp, pOutput, ret * sizeof(POINT16)); for(i = 0; i < ret; i++, pt++) { pt->x = tmp[i].x; pt->y = tmp[i].y; } HeapFree( GetProcessHeap(), 0, tmp ); } return ret; } /***************************************************************************** * DeviceCapabilitiesW [WINSPOOL.@] * * Call DeviceCapabilitiesA since we later call 16bit stuff anyway * */ INT WINAPI DeviceCapabilitiesW(LPCWSTR pDevice, LPCWSTR pPort, WORD fwCapability, LPWSTR pOutput, const DEVMODEW *pDevMode) { LPDEVMODEA dmA = DEVMODEdupWtoA(pDevMode); LPSTR pDeviceA = strdupWtoA(pDevice); LPSTR pPortA = strdupWtoA(pPort); INT ret; TRACE("%s,%s,%u,%p,%p\n", debugstr_w(pDevice), debugstr_w(pPort), fwCapability, pOutput, pDevMode); if(pOutput && (fwCapability == DC_BINNAMES || fwCapability == DC_FILEDEPENDENCIES || fwCapability == DC_PAPERNAMES)) { /* These need A -> W translation */ INT size = 0, i; LPSTR pOutputA; ret = DeviceCapabilitiesA(pDeviceA, pPortA, fwCapability, NULL, dmA); if(ret == -1) return ret; switch(fwCapability) { case DC_BINNAMES: size = 24; break; case DC_PAPERNAMES: case DC_FILEDEPENDENCIES: size = 64; break; } pOutputA = HeapAlloc(GetProcessHeap(), 0, size * ret); ret = DeviceCapabilitiesA(pDeviceA, pPortA, fwCapability, pOutputA, dmA); for(i = 0; i < ret; i++) MultiByteToWideChar(CP_ACP, 0, pOutputA + (i * size), -1, pOutput + (i * size), size); HeapFree(GetProcessHeap(), 0, pOutputA); } else { ret = DeviceCapabilitiesA(pDeviceA, pPortA, fwCapability, (LPSTR)pOutput, dmA); } HeapFree(GetProcessHeap(),0,pPortA); HeapFree(GetProcessHeap(),0,pDeviceA); HeapFree(GetProcessHeap(),0,dmA); return ret; } /****************************************************************** * DocumentPropertiesA [WINSPOOL.@] * * FIXME: implement DocumentPropertiesA via DocumentPropertiesW, not vice versa */ LONG WINAPI DocumentPropertiesA(HWND hWnd,HANDLE hPrinter, LPSTR pDeviceName, LPDEVMODEA pDevModeOutput, LPDEVMODEA pDevModeInput,DWORD fMode ) { LPSTR lpName = pDeviceName, dupname = NULL; static CHAR port[] = "LPT1:"; LONG ret; TRACE("(%p,%p,%s,%p,%p,%d)\n", hWnd,hPrinter,pDeviceName,pDevModeOutput,pDevModeInput,fMode ); if(!pDeviceName || !*pDeviceName) { LPCWSTR lpNameW = get_opened_printer_name(hPrinter); if(!lpNameW) { ERR("no name from hPrinter?\n"); SetLastError(ERROR_INVALID_HANDLE); return -1; } lpName = dupname = strdupWtoA(lpNameW); } if (!GDI_CallExtDeviceMode16) { GDI_CallExtDeviceMode16 = (void*)GetProcAddress( GetModuleHandleA("gdi32"), (LPCSTR)102 ); if (!GDI_CallExtDeviceMode16) { ERR("No CallExtDeviceMode16?\n"); ret = -1; goto end; } } ret = GDI_CallExtDeviceMode16(hWnd, pDevModeOutput, lpName, port, pDevModeInput, NULL, fMode); end: HeapFree(GetProcessHeap(), 0, dupname); return ret; } /***************************************************************************** * DocumentPropertiesW (WINSPOOL.@) * * FIXME: implement DocumentPropertiesA via DocumentPropertiesW, not vice versa */ LONG WINAPI DocumentPropertiesW(HWND hWnd, HANDLE hPrinter, LPWSTR pDeviceName, LPDEVMODEW pDevModeOutput, LPDEVMODEW pDevModeInput, DWORD fMode) { LPSTR pDeviceNameA = strdupWtoA(pDeviceName); LPDEVMODEA pDevModeInputA; LPDEVMODEA pDevModeOutputA = NULL; LONG ret; TRACE("(%p,%p,%s,%p,%p,%d)\n", hWnd,hPrinter,debugstr_w(pDeviceName),pDevModeOutput,pDevModeInput, fMode); if(pDevModeOutput) { ret = DocumentPropertiesA(hWnd, hPrinter, pDeviceNameA, NULL, NULL, 0); if(ret < 0) return ret; pDevModeOutputA = HeapAlloc(GetProcessHeap(), 0, ret); } pDevModeInputA = (fMode & DM_IN_BUFFER) ? DEVMODEdupWtoA(pDevModeInput) : NULL; ret = DocumentPropertiesA(hWnd, hPrinter, pDeviceNameA, pDevModeOutputA, pDevModeInputA, fMode); if(pDevModeOutput) { DEVMODEcpyAtoW(pDevModeOutput, pDevModeOutputA); HeapFree(GetProcessHeap(),0,pDevModeOutputA); } if(fMode == 0 && ret > 0) ret += (CCHDEVICENAME + CCHFORMNAME); HeapFree(GetProcessHeap(),0,pDevModeInputA); HeapFree(GetProcessHeap(),0,pDeviceNameA); return ret; } /***************************************************************************** * IsValidDevmodeA [WINSPOOL.@] * * Validate a DEVMODE structure and fix errors if possible. * */ BOOL WINAPI IsValidDevmodeA(PDEVMODEA *pDevMode, SIZE_T size) { FIXME("(%p,%ld): stub\n", pDevMode, size); if(!pDevMode) return FALSE; return TRUE; } /***************************************************************************** * IsValidDevmodeW [WINSPOOL.@] * * Validate a DEVMODE structure and fix errors if possible. * */ BOOL WINAPI IsValidDevmodeW(PDEVMODEW *pDevMode, SIZE_T size) { FIXME("(%p,%ld): stub\n", pDevMode, size); if(!pDevMode) return FALSE; return TRUE; } /****************************************************************** * OpenPrinterA [WINSPOOL.@] * * See OpenPrinterW. * */ BOOL WINAPI OpenPrinterA(LPSTR lpPrinterName,HANDLE *phPrinter, LPPRINTER_DEFAULTSA pDefault) { UNICODE_STRING lpPrinterNameW; UNICODE_STRING usBuffer; PRINTER_DEFAULTSW DefaultW, *pDefaultW = NULL; PWSTR pwstrPrinterNameW; BOOL ret; TRACE("%s,%p,%p\n", debugstr_a(lpPrinterName), phPrinter, pDefault); pwstrPrinterNameW = asciitounicode(&lpPrinterNameW,lpPrinterName); if(pDefault) { DefaultW.pDatatype = asciitounicode(&usBuffer,pDefault->pDatatype); DefaultW.pDevMode = pDefault->pDevMode ? GdiConvertToDevmodeW(pDefault->pDevMode) : NULL; DefaultW.DesiredAccess = pDefault->DesiredAccess; pDefaultW = &DefaultW; } ret = OpenPrinterW(pwstrPrinterNameW, phPrinter, pDefaultW); if(pDefault) { RtlFreeUnicodeString(&usBuffer); HeapFree(GetProcessHeap(), 0, DefaultW.pDevMode); } RtlFreeUnicodeString(&lpPrinterNameW); return ret; } /****************************************************************** * OpenPrinterW [WINSPOOL.@] * * Open a Printer / Printserver or a Printer-Object * * PARAMS * lpPrinterName [I] Name of Printserver, Printer, or Printer-Object * phPrinter [O] The resulting Handle is stored here * pDefault [I] PTR to Default Printer Settings or NULL * * RETURNS * Success: TRUE * Failure: FALSE * * NOTES * lpPrinterName is one of: *| Printserver (NT only): "Servername" or NULL for the local Printserver *| Printer: "PrinterName" *| Printer-Object: "PrinterName,Job xxx" *| XcvMonitor: "Servername,XcvMonitor MonitorName" *| XcvPort: "Servername,XcvPort PortName" * * BUGS *| Printer-Object not supported *| pDefaults is ignored * */ BOOL WINAPI OpenPrinterW(LPWSTR lpPrinterName,HANDLE *phPrinter, LPPRINTER_DEFAULTSW pDefault) { HKEY key; TRACE("(%s, %p, %p)\n", debugstr_w(lpPrinterName), phPrinter, pDefault); if(!phPrinter) { /* NT: FALSE with ERROR_INVALID_PARAMETER, 9x: TRUE */ SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } /* Get the unique handle of the printer or Printserver */ *phPrinter = get_opened_printer_entry(lpPrinterName, pDefault); if (*phPrinter && WINSPOOL_GetOpenedPrinterRegKey( *phPrinter, &key ) == ERROR_SUCCESS) { DWORD deleting = 0, size = sizeof( deleting ), type; DWORD status; RegQueryValueExW( key, May_Delete_Value, NULL, &type, (LPBYTE)&deleting, &size ); WaitForSingleObject( init_mutex, INFINITE ); status = get_dword_from_reg( key, StatusW ); set_reg_DWORD( key, StatusW, status & ~PRINTER_STATUS_DRIVER_UPDATE_NEEDED ); ReleaseMutex( init_mutex ); if (!deleting && (status & PRINTER_STATUS_DRIVER_UPDATE_NEEDED)) update_driver( *phPrinter ); RegCloseKey( key ); } TRACE("returning %d with %u and %p\n", *phPrinter != NULL, GetLastError(), *phPrinter); return (*phPrinter != 0); } /****************************************************************** * AddMonitorA [WINSPOOL.@] * * See AddMonitorW. * */ BOOL WINAPI AddMonitorA(LPSTR pName, DWORD Level, LPBYTE pMonitors) { LPWSTR nameW = NULL; INT len; BOOL res; LPMONITOR_INFO_2A mi2a; MONITOR_INFO_2W mi2w; mi2a = (LPMONITOR_INFO_2A) pMonitors; TRACE("(%s, %d, %p) : %s %s %s\n", debugstr_a(pName), Level, pMonitors, debugstr_a(mi2a ? mi2a->pName : NULL), debugstr_a(mi2a ? mi2a->pEnvironment : NULL), debugstr_a(mi2a ? mi2a->pDLLName : NULL)); if (Level != 2) { SetLastError(ERROR_INVALID_LEVEL); return FALSE; } /* XP: unchanged, win9x: ERROR_INVALID_ENVIRONMENT */ if (mi2a == NULL) { return FALSE; } if (pName) { len = MultiByteToWideChar(CP_ACP, 0, pName, -1, NULL, 0); nameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pName, -1, nameW, len); } memset(&mi2w, 0, sizeof(MONITOR_INFO_2W)); if (mi2a->pName) { len = MultiByteToWideChar(CP_ACP, 0, mi2a->pName, -1, NULL, 0); mi2w.pName = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, mi2a->pName, -1, mi2w.pName, len); } if (mi2a->pEnvironment) { len = MultiByteToWideChar(CP_ACP, 0, mi2a->pEnvironment, -1, NULL, 0); mi2w.pEnvironment = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, mi2a->pEnvironment, -1, mi2w.pEnvironment, len); } if (mi2a->pDLLName) { len = MultiByteToWideChar(CP_ACP, 0, mi2a->pDLLName, -1, NULL, 0); mi2w.pDLLName = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, mi2a->pDLLName, -1, mi2w.pDLLName, len); } res = AddMonitorW(nameW, Level, (LPBYTE) &mi2w); HeapFree(GetProcessHeap(), 0, mi2w.pName); HeapFree(GetProcessHeap(), 0, mi2w.pEnvironment); HeapFree(GetProcessHeap(), 0, mi2w.pDLLName); HeapFree(GetProcessHeap(), 0, nameW); return (res); } /****************************************************************************** * AddMonitorW [WINSPOOL.@] * * Install a Printmonitor * * PARAMS * pName [I] Servername or NULL (local Computer) * Level [I] Structure-Level (Must be 2) * pMonitors [I] PTR to MONITOR_INFO_2 * * RETURNS * Success: TRUE * Failure: FALSE * * NOTES * All Files for the Monitor must already be copied to %winsysdir% ("%SystemRoot%\system32") * */ BOOL WINAPI AddMonitorW(LPWSTR pName, DWORD Level, LPBYTE pMonitors) { LPMONITOR_INFO_2W mi2w; mi2w = (LPMONITOR_INFO_2W) pMonitors; TRACE("(%s, %d, %p) : %s %s %s\n", debugstr_w(pName), Level, pMonitors, debugstr_w(mi2w ? mi2w->pName : NULL), debugstr_w(mi2w ? mi2w->pEnvironment : NULL), debugstr_w(mi2w ? mi2w->pDLLName : NULL)); if ((backend == NULL) && !load_backend()) return FALSE; if (Level != 2) { SetLastError(ERROR_INVALID_LEVEL); return FALSE; } /* XP: unchanged, win9x: ERROR_INVALID_ENVIRONMENT */ if (mi2w == NULL) { return FALSE; } return backend->fpAddMonitor(pName, Level, pMonitors); } /****************************************************************** * DeletePrinterDriverA [WINSPOOL.@] * */ BOOL WINAPI DeletePrinterDriverA (LPSTR pName, LPSTR pEnvironment, LPSTR pDriverName) { return DeletePrinterDriverExA(pName, pEnvironment, pDriverName, 0, 0); } /****************************************************************** * DeletePrinterDriverW [WINSPOOL.@] * */ BOOL WINAPI DeletePrinterDriverW (LPWSTR pName, LPWSTR pEnvironment, LPWSTR pDriverName) { return DeletePrinterDriverExW(pName, pEnvironment, pDriverName, 0, 0); } /****************************************************************** * DeleteMonitorA [WINSPOOL.@] * * See DeleteMonitorW. * */ BOOL WINAPI DeleteMonitorA (LPSTR pName, LPSTR pEnvironment, LPSTR pMonitorName) { LPWSTR nameW = NULL; LPWSTR EnvironmentW = NULL; LPWSTR MonitorNameW = NULL; BOOL res; INT len; if (pName) { len = MultiByteToWideChar(CP_ACP, 0, pName, -1, NULL, 0); nameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pName, -1, nameW, len); } if (pEnvironment) { len = MultiByteToWideChar(CP_ACP, 0, pEnvironment, -1, NULL, 0); EnvironmentW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pEnvironment, -1, EnvironmentW, len); } if (pMonitorName) { len = MultiByteToWideChar(CP_ACP, 0, pMonitorName, -1, NULL, 0); MonitorNameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pMonitorName, -1, MonitorNameW, len); } res = DeleteMonitorW(nameW, EnvironmentW, MonitorNameW); HeapFree(GetProcessHeap(), 0, MonitorNameW); HeapFree(GetProcessHeap(), 0, EnvironmentW); HeapFree(GetProcessHeap(), 0, nameW); return (res); } /****************************************************************** * DeleteMonitorW [WINSPOOL.@] * * Delete a specific Printmonitor from a Printing-Environment * * PARAMS * pName [I] Servername or NULL (local Computer) * pEnvironment [I] Printing-Environment of the Monitor or NULL (Default) * pMonitorName [I] Name of the Monitor, that should be deleted * * RETURNS * Success: TRUE * Failure: FALSE * * NOTES * pEnvironment is ignored in Windows for the local Computer. * */ BOOL WINAPI DeleteMonitorW(LPWSTR pName, LPWSTR pEnvironment, LPWSTR pMonitorName) { TRACE("(%s, %s, %s)\n",debugstr_w(pName),debugstr_w(pEnvironment), debugstr_w(pMonitorName)); if ((backend == NULL) && !load_backend()) return FALSE; return backend->fpDeleteMonitor(pName, pEnvironment, pMonitorName); } /****************************************************************** * DeletePortA [WINSPOOL.@] * * See DeletePortW. * */ BOOL WINAPI DeletePortA (LPSTR pName, HWND hWnd, LPSTR pPortName) { LPWSTR nameW = NULL; LPWSTR portW = NULL; INT len; DWORD res; TRACE("(%s, %p, %s)\n", debugstr_a(pName), hWnd, debugstr_a(pPortName)); /* convert servername to unicode */ if (pName) { len = MultiByteToWideChar(CP_ACP, 0, pName, -1, NULL, 0); nameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pName, -1, nameW, len); } /* convert portname to unicode */ if (pPortName) { len = MultiByteToWideChar(CP_ACP, 0, pPortName, -1, NULL, 0); portW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pPortName, -1, portW, len); } res = DeletePortW(nameW, hWnd, portW); HeapFree(GetProcessHeap(), 0, nameW); HeapFree(GetProcessHeap(), 0, portW); return res; } /****************************************************************** * DeletePortW [WINSPOOL.@] * * Delete a specific Port * * PARAMS * pName [I] Servername or NULL (local Computer) * hWnd [I] Handle to parent Window for the Dialog-Box * pPortName [I] Name of the Port, that should be deleted * * RETURNS * Success: TRUE * Failure: FALSE * */ BOOL WINAPI DeletePortW (LPWSTR pName, HWND hWnd, LPWSTR pPortName) { TRACE("(%s, %p, %s)\n", debugstr_w(pName), hWnd, debugstr_w(pPortName)); if ((backend == NULL) && !load_backend()) return FALSE; if (!pPortName) { SetLastError(RPC_X_NULL_REF_POINTER); return FALSE; } return backend->fpDeletePort(pName, hWnd, pPortName); } /****************************************************************************** * WritePrinter [WINSPOOL.@] */ BOOL WINAPI WritePrinter(HANDLE hPrinter, LPVOID pBuf, DWORD cbBuf, LPDWORD pcWritten) { opened_printer_t *printer; BOOL ret = FALSE; TRACE("(%p, %p, %d, %p)\n", hPrinter, pBuf, cbBuf, pcWritten); EnterCriticalSection(&printer_handles_cs); printer = get_opened_printer(hPrinter); if(!printer) { SetLastError(ERROR_INVALID_HANDLE); goto end; } if(!printer->doc) { SetLastError(ERROR_SPL_NO_STARTDOC); goto end; } ret = WriteFile(printer->doc->hf, pBuf, cbBuf, pcWritten, NULL); end: LeaveCriticalSection(&printer_handles_cs); return ret; } /***************************************************************************** * AddFormA [WINSPOOL.@] */ BOOL WINAPI AddFormA(HANDLE hPrinter, DWORD Level, LPBYTE pForm) { FIXME("(%p,%d,%p): stub\n", hPrinter, Level, pForm); return TRUE; } /***************************************************************************** * AddFormW [WINSPOOL.@] */ BOOL WINAPI AddFormW(HANDLE hPrinter, DWORD Level, LPBYTE pForm) { FIXME("(%p,%d,%p): stub\n", hPrinter, Level, pForm); return TRUE; } /***************************************************************************** * AddJobA [WINSPOOL.@] */ BOOL WINAPI AddJobA(HANDLE hPrinter, DWORD Level, LPBYTE pData, DWORD cbBuf, LPDWORD pcbNeeded) { BOOL ret; BYTE buf[MAX_PATH * sizeof(WCHAR) + sizeof(ADDJOB_INFO_1W)]; DWORD needed; if(Level != 1) { SetLastError(ERROR_INVALID_LEVEL); return FALSE; } ret = AddJobW(hPrinter, Level, buf, sizeof(buf), &needed); if(ret) { ADDJOB_INFO_1W *addjobW = (ADDJOB_INFO_1W*)buf; DWORD len = WideCharToMultiByte(CP_ACP, 0, addjobW->Path, -1, NULL, 0, NULL, NULL); *pcbNeeded = len + sizeof(ADDJOB_INFO_1A); if(*pcbNeeded > cbBuf) { SetLastError(ERROR_INSUFFICIENT_BUFFER); ret = FALSE; } else { ADDJOB_INFO_1A *addjobA = (ADDJOB_INFO_1A*)pData; addjobA->JobId = addjobW->JobId; addjobA->Path = (char *)(addjobA + 1); WideCharToMultiByte(CP_ACP, 0, addjobW->Path, -1, addjobA->Path, len, NULL, NULL); } } return ret; } /***************************************************************************** * AddJobW [WINSPOOL.@] */ BOOL WINAPI AddJobW(HANDLE hPrinter, DWORD Level, LPBYTE pData, DWORD cbBuf, LPDWORD pcbNeeded) { opened_printer_t *printer; job_t *job; BOOL ret = FALSE; static const WCHAR spool_path[] = {'s','p','o','o','l','\\','P','R','I','N','T','E','R','S','\\',0}; static const WCHAR fmtW[] = {'%','s','%','0','5','d','.','S','P','L',0}; WCHAR path[MAX_PATH], filename[MAX_PATH]; DWORD len; ADDJOB_INFO_1W *addjob; TRACE("(%p,%d,%p,%d,%p)\n", hPrinter, Level, pData, cbBuf, pcbNeeded); EnterCriticalSection(&printer_handles_cs); printer = get_opened_printer(hPrinter); if(!printer) { SetLastError(ERROR_INVALID_HANDLE); goto end; } if(Level != 1) { SetLastError(ERROR_INVALID_LEVEL); goto end; } job = HeapAlloc(GetProcessHeap(), 0, sizeof(*job)); if(!job) goto end; job->job_id = InterlockedIncrement(&next_job_id); len = GetSystemDirectoryW(path, sizeof(path) / sizeof(WCHAR)); if(path[len - 1] != '\\') path[len++] = '\\'; memcpy(path + len, spool_path, sizeof(spool_path)); sprintfW(filename, fmtW, path, job->job_id); len = strlenW(filename); job->filename = HeapAlloc(GetProcessHeap(), 0, (len + 1) * sizeof(WCHAR)); memcpy(job->filename, filename, (len + 1) * sizeof(WCHAR)); job->portname = NULL; job->document_title = strdupW(default_doc_title); job->printer_name = strdupW(printer->name); job->devmode = dup_devmode( printer->devmode ); list_add_tail(&printer->queue->jobs, &job->entry); *pcbNeeded = (len + 1) * sizeof(WCHAR) + sizeof(*addjob); if(*pcbNeeded <= cbBuf) { addjob = (ADDJOB_INFO_1W*)pData; addjob->JobId = job->job_id; addjob->Path = (WCHAR *)(addjob + 1); memcpy(addjob->Path, filename, (len + 1) * sizeof(WCHAR)); ret = TRUE; } else SetLastError(ERROR_INSUFFICIENT_BUFFER); end: LeaveCriticalSection(&printer_handles_cs); return ret; } /***************************************************************************** * GetPrintProcessorDirectoryA [WINSPOOL.@] * * Return the PATH for the Print-Processors * * See GetPrintProcessorDirectoryW. * * */ BOOL WINAPI GetPrintProcessorDirectoryA(LPSTR server, LPSTR env, DWORD level, LPBYTE Info, DWORD cbBuf, LPDWORD pcbNeeded) { LPWSTR serverW = NULL; LPWSTR envW = NULL; BOOL ret; INT len; TRACE("(%s, %s, %d, %p, %d, %p)\n", debugstr_a(server), debugstr_a(env), level, Info, cbBuf, pcbNeeded); if (server) { len = MultiByteToWideChar(CP_ACP, 0, server, -1, NULL, 0); serverW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, server, -1, serverW, len); } if (env) { len = MultiByteToWideChar(CP_ACP, 0, env, -1, NULL, 0); envW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, env, -1, envW, len); } /* NT requires the buffersize from GetPrintProcessorDirectoryW also for GetPrintProcessorDirectoryA and WC2MB is done in-place. */ ret = GetPrintProcessorDirectoryW(serverW, envW, level, Info, cbBuf, pcbNeeded); if (ret) ret = WideCharToMultiByte(CP_ACP, 0, (LPWSTR)Info, -1, (LPSTR)Info, cbBuf, NULL, NULL) > 0; TRACE(" required: 0x%x/%d\n", pcbNeeded ? *pcbNeeded : 0, pcbNeeded ? *pcbNeeded : 0); HeapFree(GetProcessHeap(), 0, envW); HeapFree(GetProcessHeap(), 0, serverW); return ret; } /***************************************************************************** * GetPrintProcessorDirectoryW [WINSPOOL.@] * * Return the PATH for the Print-Processors * * PARAMS * server [I] Servername (NT only) or NULL (local Computer) * env [I] Printing-Environment (see below) or NULL (Default) * level [I] Structure-Level (must be 1) * Info [O] PTR to Buffer that receives the Result * cbBuf [I] Size of Buffer at "Info" * pcbNeeded [O] PTR to DWORD that receives the size in Bytes used / * required for the Buffer at "Info" * * RETURNS * Success: TRUE and in pcbNeeded the Bytes used in Info * Failure: FALSE and in pcbNeeded the Bytes required for Info, * if cbBuf is too small * * Native Values returned in Info on Success: *| NT(Windows NT x86): "%winsysdir%\\spool\\PRTPROCS\\w32x86" *| NT(Windows 4.0): "%winsysdir%\\spool\\PRTPROCS\\win40" *| win9x(Windows 4.0): "%winsysdir%" * * "%winsysdir%" is the Value from GetSystemDirectoryW() * * BUGS * Only NULL or "" is supported for server * */ BOOL WINAPI GetPrintProcessorDirectoryW(LPWSTR server, LPWSTR env, DWORD level, LPBYTE Info, DWORD cbBuf, LPDWORD pcbNeeded) { TRACE("(%s, %s, %d, %p, %d, %p)\n", debugstr_w(server), debugstr_w(env), level, Info, cbBuf, pcbNeeded); if ((backend == NULL) && !load_backend()) return FALSE; if (level != 1) { /* (Level != 1) is ignored in win9x */ SetLastError(ERROR_INVALID_LEVEL); return FALSE; } if (pcbNeeded == NULL) { /* (pcbNeeded == NULL) is ignored in win9x */ SetLastError(RPC_X_NULL_REF_POINTER); return FALSE; } return backend->fpGetPrintProcessorDirectory(server, env, level, Info, cbBuf, pcbNeeded); } /***************************************************************************** * WINSPOOL_OpenDriverReg [internal] * * opens the registry for the printer drivers depending on the given input * variable pEnvironment * * RETURNS: * the opened hkey on success * NULL on error */ static HKEY WINSPOOL_OpenDriverReg( LPCVOID pEnvironment) { HKEY retval = NULL; LPWSTR buffer; const printenv_t * env; TRACE("(%s)\n", debugstr_w(pEnvironment)); env = validate_envW(pEnvironment); if (!env) return NULL; buffer = HeapAlloc( GetProcessHeap(), 0, (strlenW(DriversW) + strlenW(env->envname) + strlenW(env->versionregpath) + 1) * sizeof(WCHAR)); if(buffer) { wsprintfW(buffer, DriversW, env->envname, env->versionregpath); RegCreateKeyW(HKEY_LOCAL_MACHINE, buffer, &retval); HeapFree(GetProcessHeap(), 0, buffer); } return retval; } /***************************************************************************** * set_devices_and_printerports [internal] * * set the [Devices] and [PrinterPorts] entries for a printer. * */ static void set_devices_and_printerports(PRINTER_INFO_2W *pi) { DWORD portlen = lstrlenW(pi->pPortName) * sizeof(WCHAR); WCHAR *devline; HKEY hkey; TRACE("(%p) %s\n", pi, debugstr_w(pi->pPrinterName)); /* FIXME: the driver must change to "winspool" */ devline = HeapAlloc(GetProcessHeap(), 0, sizeof(driver_nt) + portlen + sizeof(timeout_15_45)); if (devline) { lstrcpyW(devline, driver_nt); lstrcatW(devline, commaW); lstrcatW(devline, pi->pPortName); TRACE("using %s\n", debugstr_w(devline)); WriteProfileStringW(devicesW, pi->pPrinterName, devline); if (!RegCreateKeyW(HKEY_CURRENT_USER, user_printers_reg_key, &hkey)) { RegSetValueExW(hkey, pi->pPrinterName, 0, REG_SZ, (LPBYTE)devline, (lstrlenW(devline) + 1) * sizeof(WCHAR)); RegCloseKey(hkey); } lstrcatW(devline, timeout_15_45); WriteProfileStringW(PrinterPortsW, pi->pPrinterName, devline); if (!RegCreateKeyW(HKEY_CURRENT_USER, WinNT_CV_PrinterPortsW, &hkey)) { RegSetValueExW(hkey, pi->pPrinterName, 0, REG_SZ, (LPBYTE)devline, (lstrlenW(devline) + 1) * sizeof(WCHAR)); RegCloseKey(hkey); } HeapFree(GetProcessHeap(), 0, devline); } } /***************************************************************************** * AddPrinterW [WINSPOOL.@] */ HANDLE WINAPI AddPrinterW(LPWSTR pName, DWORD Level, LPBYTE pPrinter) { PRINTER_INFO_2W *pi = (PRINTER_INFO_2W *) pPrinter; LPDEVMODEW dm; HANDLE retval; HKEY hkeyPrinter, hkeyPrinters, hkeyDriver, hkeyDrivers; LONG size; TRACE("(%s,%d,%p)\n", debugstr_w(pName), Level, pPrinter); if(pName && *pName) { ERR("pName = %s - unsupported\n", debugstr_w(pName)); SetLastError(ERROR_INVALID_PARAMETER); return 0; } if(Level != 2) { ERR("Level = %d, unsupported!\n", Level); SetLastError(ERROR_INVALID_LEVEL); return 0; } if(!pPrinter) { SetLastError(ERROR_INVALID_PARAMETER); return 0; } if(RegCreateKeyW(HKEY_LOCAL_MACHINE, PrintersW, &hkeyPrinters) != ERROR_SUCCESS) { ERR("Can't create Printers key\n"); return 0; } if(!RegOpenKeyW(hkeyPrinters, pi->pPrinterName, &hkeyPrinter)) { if (!RegQueryValueW(hkeyPrinter, AttributesW, NULL, NULL)) { SetLastError(ERROR_PRINTER_ALREADY_EXISTS); RegCloseKey(hkeyPrinter); RegCloseKey(hkeyPrinters); return 0; } RegCloseKey(hkeyPrinter); } hkeyDrivers = WINSPOOL_OpenDriverReg(NULL); if(!hkeyDrivers) { ERR("Can't create Drivers key\n"); RegCloseKey(hkeyPrinters); return 0; } if(RegOpenKeyW(hkeyDrivers, pi->pDriverName, &hkeyDriver) != ERROR_SUCCESS) { WARN("Can't find driver %s\n", debugstr_w(pi->pDriverName)); RegCloseKey(hkeyPrinters); RegCloseKey(hkeyDrivers); SetLastError(ERROR_UNKNOWN_PRINTER_DRIVER); return 0; } RegCloseKey(hkeyDriver); RegCloseKey(hkeyDrivers); if(lstrcmpiW(pi->pPrintProcessor, WinPrintW)) { /* FIXME */ FIXME("Can't find processor %s\n", debugstr_w(pi->pPrintProcessor)); SetLastError(ERROR_UNKNOWN_PRINTPROCESSOR); RegCloseKey(hkeyPrinters); return 0; } if(RegCreateKeyW(hkeyPrinters, pi->pPrinterName, &hkeyPrinter) != ERROR_SUCCESS) { FIXME("Can't create printer %s\n", debugstr_w(pi->pPrinterName)); SetLastError(ERROR_INVALID_PRINTER_NAME); RegCloseKey(hkeyPrinters); return 0; } set_devices_and_printerports(pi); set_reg_DWORD(hkeyPrinter, AttributesW, pi->Attributes); set_reg_szW(hkeyPrinter, DatatypeW, pi->pDatatype); set_reg_DWORD(hkeyPrinter, Default_PriorityW, pi->DefaultPriority); set_reg_szW(hkeyPrinter, DescriptionW, pi->pComment); set_reg_DWORD(hkeyPrinter, dnsTimeoutW, 0); set_reg_szW(hkeyPrinter, LocationW, pi->pLocation); set_reg_szW(hkeyPrinter, NameW, pi->pPrinterName); set_reg_szW(hkeyPrinter, ParametersW, pi->pParameters); set_reg_szW(hkeyPrinter, PortW, pi->pPortName); set_reg_szW(hkeyPrinter, Print_ProcessorW, pi->pPrintProcessor); set_reg_szW(hkeyPrinter, Printer_DriverW, pi->pDriverName); set_reg_DWORD(hkeyPrinter, PriorityW, pi->Priority); set_reg_szW(hkeyPrinter, Separator_FileW, pi->pSepFile); set_reg_szW(hkeyPrinter, Share_NameW, pi->pShareName); set_reg_DWORD(hkeyPrinter, StartTimeW, pi->StartTime); set_reg_DWORD(hkeyPrinter, StatusW, pi->Status); set_reg_DWORD(hkeyPrinter, txTimeoutW, 0); set_reg_DWORD(hkeyPrinter, UntilTimeW, pi->UntilTime); size = DocumentPropertiesW(0, 0, pi->pPrinterName, NULL, NULL, 0); if (size < 0) { FIXME("DocumentPropertiesW on printer %s fails\n", debugstr_w(pi->pPrinterName)); size = sizeof(DEVMODEW); } if(pi->pDevMode) dm = pi->pDevMode; else { dm = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, size ); dm->dmSize = size; if (DocumentPropertiesW(0, 0, pi->pPrinterName, dm, NULL, DM_OUT_BUFFER) < 0) { WARN("DocumentPropertiesW on printer %s failed!\n", debugstr_w(pi->pPrinterName)); HeapFree( GetProcessHeap(), 0, dm ); dm = NULL; } else { /* set devmode to printer name */ lstrcpynW( dm->dmDeviceName, pi->pPrinterName, CCHDEVICENAME ); } } set_reg_devmode( hkeyPrinter, Default_DevModeW, dm ); if (!pi->pDevMode) HeapFree( GetProcessHeap(), 0, dm ); RegCloseKey(hkeyPrinter); RegCloseKey(hkeyPrinters); if(!OpenPrinterW(pi->pPrinterName, &retval, NULL)) { ERR("OpenPrinter failing\n"); return 0; } return retval; } /***************************************************************************** * AddPrinterA [WINSPOOL.@] */ HANDLE WINAPI AddPrinterA(LPSTR pName, DWORD Level, LPBYTE pPrinter) { UNICODE_STRING pNameW; PWSTR pwstrNameW; PRINTER_INFO_2W *piW; PRINTER_INFO_2A *piA = (PRINTER_INFO_2A*)pPrinter; HANDLE ret; TRACE("(%s, %d, %p)\n", debugstr_a(pName), Level, pPrinter); if(Level != 2) { ERR("Level = %d, unsupported!\n", Level); SetLastError(ERROR_INVALID_LEVEL); return 0; } pwstrNameW = asciitounicode(&pNameW,pName); piW = printer_info_AtoW( piA, Level ); ret = AddPrinterW(pwstrNameW, Level, (LPBYTE)piW); free_printer_info( piW, Level ); RtlFreeUnicodeString(&pNameW); return ret; } /***************************************************************************** * ClosePrinter [WINSPOOL.@] */ BOOL WINAPI ClosePrinter(HANDLE hPrinter) { UINT_PTR i = (UINT_PTR)hPrinter; opened_printer_t *printer = NULL; BOOL ret = FALSE; TRACE("(%p)\n", hPrinter); EnterCriticalSection(&printer_handles_cs); if ((i > 0) && (i <= nb_printer_handles)) printer = printer_handles[i - 1]; if(printer) { struct list *cursor, *cursor2; TRACE("closing %s (doc: %p)\n", debugstr_w(printer->name), printer->doc); if (printer->backend_printer) { backend->fpClosePrinter(printer->backend_printer); } if(printer->doc) EndDocPrinter(hPrinter); if(InterlockedDecrement(&printer->queue->ref) == 0) { LIST_FOR_EACH_SAFE(cursor, cursor2, &printer->queue->jobs) { job_t *job = LIST_ENTRY(cursor, job_t, entry); ScheduleJob(hPrinter, job->job_id); } HeapFree(GetProcessHeap(), 0, printer->queue); } free_printer_entry( printer ); printer_handles[i - 1] = NULL; ret = TRUE; } LeaveCriticalSection(&printer_handles_cs); return ret; } /***************************************************************************** * DeleteFormA [WINSPOOL.@] */ BOOL WINAPI DeleteFormA(HANDLE hPrinter, LPSTR pFormName) { FIXME("(%p,%s): stub\n", hPrinter, pFormName); return TRUE; } /***************************************************************************** * DeleteFormW [WINSPOOL.@] */ BOOL WINAPI DeleteFormW(HANDLE hPrinter, LPWSTR pFormName) { FIXME("(%p,%s): stub\n", hPrinter, debugstr_w(pFormName)); return TRUE; } /***************************************************************************** * DeletePrinter [WINSPOOL.@] */ BOOL WINAPI DeletePrinter(HANDLE hPrinter) { LPCWSTR lpNameW = get_opened_printer_name(hPrinter); HKEY hkeyPrinters, hkey; WCHAR def[MAX_PATH]; DWORD size = sizeof( def ) / sizeof( def[0] ); if(!lpNameW) { SetLastError(ERROR_INVALID_HANDLE); return FALSE; } if(RegOpenKeyW(HKEY_LOCAL_MACHINE, PrintersW, &hkeyPrinters) == ERROR_SUCCESS) { RegDeleteTreeW(hkeyPrinters, lpNameW); RegCloseKey(hkeyPrinters); } WriteProfileStringW(devicesW, lpNameW, NULL); WriteProfileStringW(PrinterPortsW, lpNameW, NULL); if(RegCreateKeyW(HKEY_CURRENT_USER, user_printers_reg_key, &hkey) == ERROR_SUCCESS) { RegDeleteValueW(hkey, lpNameW); RegCloseKey(hkey); } if(RegCreateKeyW(HKEY_CURRENT_USER, WinNT_CV_PrinterPortsW, &hkey) == ERROR_SUCCESS) { RegDeleteValueW(hkey, lpNameW); RegCloseKey(hkey); } if (GetDefaultPrinterW( def, &size ) && !strcmpW( def, lpNameW )) { WriteProfileStringW( windowsW, deviceW, NULL ); if (!RegCreateKeyW( HKEY_CURRENT_USER, user_default_reg_key, &hkey )) { RegDeleteValueW( hkey, deviceW ); RegCloseKey( hkey ); } SetDefaultPrinterW( NULL ); } return TRUE; } /***************************************************************************** * SetPrinterA [WINSPOOL.@] */ BOOL WINAPI SetPrinterA( HANDLE printer, DWORD level, LPBYTE data, DWORD command ) { BYTE *dataW = data; BOOL ret; if (level != 0) { dataW = printer_info_AtoW( data, level ); if (!dataW) return FALSE; } ret = SetPrinterW( printer, level, dataW, command ); if (dataW != data) free_printer_info( dataW, level ); return ret; } static void set_printer_2( HKEY key, const PRINTER_INFO_2W *pi ) { set_reg_szW( key, NameW, pi->pPrinterName ); set_reg_szW( key, Share_NameW, pi->pShareName ); set_reg_szW( key, PortW, pi->pPortName ); set_reg_szW( key, Printer_DriverW, pi->pDriverName ); set_reg_szW( key, DescriptionW, pi->pComment ); set_reg_szW( key, LocationW, pi->pLocation ); if (pi->pDevMode) set_reg_devmode( key, Default_DevModeW, pi->pDevMode ); set_reg_szW( key, Separator_FileW, pi->pSepFile ); set_reg_szW( key, Print_ProcessorW, pi->pPrintProcessor ); set_reg_szW( key, DatatypeW, pi->pDatatype ); set_reg_szW( key, ParametersW, pi->pParameters ); set_reg_DWORD( key, AttributesW, pi->Attributes ); set_reg_DWORD( key, PriorityW, pi->Priority ); set_reg_DWORD( key, Default_PriorityW, pi->DefaultPriority ); set_reg_DWORD( key, StartTimeW, pi->StartTime ); set_reg_DWORD( key, UntilTimeW, pi->UntilTime ); } static BOOL set_printer_9( HKEY key, const PRINTER_INFO_9W *pi ) { if (!pi->pDevMode) return FALSE; set_reg_devmode( key, Default_DevModeW, pi->pDevMode ); return TRUE; } /****************************************************************************** * SetPrinterW [WINSPOOL.@] */ BOOL WINAPI SetPrinterW( HANDLE printer, DWORD level, LPBYTE data, DWORD command ) { HKEY key; BOOL ret = FALSE; TRACE( "(%p, %d, %p, %d)\n", printer, level, data, command ); if (command != 0) FIXME( "Ignoring command %d\n", command ); if (WINSPOOL_GetOpenedPrinterRegKey( printer, &key )) return FALSE; switch (level) { case 2: { PRINTER_INFO_2W *pi2 = (PRINTER_INFO_2W *)data; set_printer_2( key, pi2 ); ret = TRUE; break; } case 9: { PRINTER_INFO_9W *pi = (PRINTER_INFO_9W *)data; ret = set_printer_9( key, pi ); break; } default: FIXME( "Unimplemented level %d\n", level ); SetLastError( ERROR_INVALID_LEVEL ); } RegCloseKey( key ); return ret; } /***************************************************************************** * SetJobA [WINSPOOL.@] */ BOOL WINAPI SetJobA(HANDLE hPrinter, DWORD JobId, DWORD Level, LPBYTE pJob, DWORD Command) { BOOL ret; LPBYTE JobW; UNICODE_STRING usBuffer; TRACE("(%p, %d, %d, %p, %d)\n",hPrinter, JobId, Level, pJob, Command); /* JobId, pPrinterName, pMachineName, pDriverName, Size, Submitted, Time and TotalPages are all ignored by SetJob, so we don't bother copying them */ switch(Level) { case 0: JobW = NULL; break; case 1: { JOB_INFO_1W *info1W = HeapAlloc(GetProcessHeap(), 0, sizeof(*info1W)); JOB_INFO_1A *info1A = (JOB_INFO_1A*)pJob; JobW = (LPBYTE)info1W; info1W->pUserName = asciitounicode(&usBuffer, info1A->pUserName); info1W->pDocument = asciitounicode(&usBuffer, info1A->pDocument); info1W->pDatatype = asciitounicode(&usBuffer, info1A->pDatatype); info1W->pStatus = asciitounicode(&usBuffer, info1A->pStatus); info1W->Status = info1A->Status; info1W->Priority = info1A->Priority; info1W->Position = info1A->Position; info1W->PagesPrinted = info1A->PagesPrinted; break; } case 2: { JOB_INFO_2W *info2W = HeapAlloc(GetProcessHeap(), 0, sizeof(*info2W)); JOB_INFO_2A *info2A = (JOB_INFO_2A*)pJob; JobW = (LPBYTE)info2W; info2W->pUserName = asciitounicode(&usBuffer, info2A->pUserName); info2W->pDocument = asciitounicode(&usBuffer, info2A->pDocument); info2W->pNotifyName = asciitounicode(&usBuffer, info2A->pNotifyName); info2W->pDatatype = asciitounicode(&usBuffer, info2A->pDatatype); info2W->pPrintProcessor = asciitounicode(&usBuffer, info2A->pPrintProcessor); info2W->pParameters = asciitounicode(&usBuffer, info2A->pParameters); info2W->pDevMode = info2A->pDevMode ? GdiConvertToDevmodeW(info2A->pDevMode) : NULL; info2W->pStatus = asciitounicode(&usBuffer, info2A->pStatus); info2W->pSecurityDescriptor = info2A->pSecurityDescriptor; info2W->Status = info2A->Status; info2W->Priority = info2A->Priority; info2W->Position = info2A->Position; info2W->StartTime = info2A->StartTime; info2W->UntilTime = info2A->UntilTime; info2W->PagesPrinted = info2A->PagesPrinted; break; } case 3: JobW = HeapAlloc(GetProcessHeap(), 0, sizeof(JOB_INFO_3)); memcpy(JobW, pJob, sizeof(JOB_INFO_3)); break; default: SetLastError(ERROR_INVALID_LEVEL); return FALSE; } ret = SetJobW(hPrinter, JobId, Level, JobW, Command); switch(Level) { case 1: { JOB_INFO_1W *info1W = (JOB_INFO_1W*)JobW; HeapFree(GetProcessHeap(), 0, info1W->pUserName); HeapFree(GetProcessHeap(), 0, info1W->pDocument); HeapFree(GetProcessHeap(), 0, info1W->pDatatype); HeapFree(GetProcessHeap(), 0, info1W->pStatus); break; } case 2: { JOB_INFO_2W *info2W = (JOB_INFO_2W*)JobW; HeapFree(GetProcessHeap(), 0, info2W->pUserName); HeapFree(GetProcessHeap(), 0, info2W->pDocument); HeapFree(GetProcessHeap(), 0, info2W->pNotifyName); HeapFree(GetProcessHeap(), 0, info2W->pDatatype); HeapFree(GetProcessHeap(), 0, info2W->pPrintProcessor); HeapFree(GetProcessHeap(), 0, info2W->pParameters); HeapFree(GetProcessHeap(), 0, info2W->pDevMode); HeapFree(GetProcessHeap(), 0, info2W->pStatus); break; } } HeapFree(GetProcessHeap(), 0, JobW); return ret; } /***************************************************************************** * SetJobW [WINSPOOL.@] */ BOOL WINAPI SetJobW(HANDLE hPrinter, DWORD JobId, DWORD Level, LPBYTE pJob, DWORD Command) { BOOL ret = FALSE; job_t *job; TRACE("(%p, %d, %d, %p, %d)\n", hPrinter, JobId, Level, pJob, Command); FIXME("Ignoring everything other than document title\n"); EnterCriticalSection(&printer_handles_cs); job = get_job(hPrinter, JobId); if(!job) goto end; switch(Level) { case 0: break; case 1: { JOB_INFO_1W *info1 = (JOB_INFO_1W*)pJob; HeapFree(GetProcessHeap(), 0, job->document_title); job->document_title = strdupW(info1->pDocument); break; } case 2: { JOB_INFO_2W *info2 = (JOB_INFO_2W*)pJob; HeapFree(GetProcessHeap(), 0, job->document_title); job->document_title = strdupW(info2->pDocument); HeapFree(GetProcessHeap(), 0, job->devmode); job->devmode = dup_devmode( info2->pDevMode ); break; } case 3: break; default: SetLastError(ERROR_INVALID_LEVEL); goto end; } ret = TRUE; end: LeaveCriticalSection(&printer_handles_cs); return ret; } /***************************************************************************** * EndDocPrinter [WINSPOOL.@] */ BOOL WINAPI EndDocPrinter(HANDLE hPrinter) { opened_printer_t *printer; BOOL ret = FALSE; TRACE("(%p)\n", hPrinter); EnterCriticalSection(&printer_handles_cs); printer = get_opened_printer(hPrinter); if(!printer) { SetLastError(ERROR_INVALID_HANDLE); goto end; } if(!printer->doc) { SetLastError(ERROR_SPL_NO_STARTDOC); goto end; } CloseHandle(printer->doc->hf); ScheduleJob(hPrinter, printer->doc->job_id); HeapFree(GetProcessHeap(), 0, printer->doc); printer->doc = NULL; ret = TRUE; end: LeaveCriticalSection(&printer_handles_cs); return ret; } /***************************************************************************** * EndPagePrinter [WINSPOOL.@] */ BOOL WINAPI EndPagePrinter(HANDLE hPrinter) { FIXME("(%p): stub\n", hPrinter); return TRUE; } /***************************************************************************** * StartDocPrinterA [WINSPOOL.@] */ DWORD WINAPI StartDocPrinterA(HANDLE hPrinter, DWORD Level, LPBYTE pDocInfo) { UNICODE_STRING usBuffer; DOC_INFO_2W doc2W; DOC_INFO_2A *doc2 = (DOC_INFO_2A*)pDocInfo; DWORD ret; /* DOC_INFO_1, 2 and 3 all have the strings in the same place with either two (DOC_INFO_2) or one (DOC_INFO_3) extra DWORDs */ switch(Level) { case 2: doc2W.JobId = doc2->JobId; /* fall through */ case 3: doc2W.dwMode = doc2->dwMode; /* fall through */ case 1: doc2W.pDocName = asciitounicode(&usBuffer, doc2->pDocName); doc2W.pOutputFile = asciitounicode(&usBuffer, doc2->pOutputFile); doc2W.pDatatype = asciitounicode(&usBuffer, doc2->pDatatype); break; default: SetLastError(ERROR_INVALID_LEVEL); return FALSE; } ret = StartDocPrinterW(hPrinter, Level, (LPBYTE)&doc2W); HeapFree(GetProcessHeap(), 0, doc2W.pDatatype); HeapFree(GetProcessHeap(), 0, doc2W.pOutputFile); HeapFree(GetProcessHeap(), 0, doc2W.pDocName); return ret; } /***************************************************************************** * StartDocPrinterW [WINSPOOL.@] */ DWORD WINAPI StartDocPrinterW(HANDLE hPrinter, DWORD Level, LPBYTE pDocInfo) { DOC_INFO_2W *doc = (DOC_INFO_2W *)pDocInfo; opened_printer_t *printer; BYTE addjob_buf[MAX_PATH * sizeof(WCHAR) + sizeof(ADDJOB_INFO_1W)]; ADDJOB_INFO_1W *addjob = (ADDJOB_INFO_1W*) addjob_buf; JOB_INFO_1W job_info; DWORD needed, ret = 0; HANDLE hf; WCHAR *filename; job_t *job; TRACE("(hPrinter = %p, Level = %d, pDocInfo = %p {pDocName = %s, pOutputFile = %s, pDatatype = %s}):\n", hPrinter, Level, doc, debugstr_w(doc->pDocName), debugstr_w(doc->pOutputFile), debugstr_w(doc->pDatatype)); if(Level < 1 || Level > 3) { SetLastError(ERROR_INVALID_LEVEL); return 0; } EnterCriticalSection(&printer_handles_cs); printer = get_opened_printer(hPrinter); if(!printer) { SetLastError(ERROR_INVALID_HANDLE); goto end; } if(printer->doc) { SetLastError(ERROR_INVALID_PRINTER_STATE); goto end; } /* Even if we're printing to a file we still add a print job, we'll just ignore the spool file name */ if(!AddJobW(hPrinter, 1, addjob_buf, sizeof(addjob_buf), &needed)) { ERR("AddJob failed gle %u\n", GetLastError()); goto end; } /* use pOutputFile only, when it is a real filename */ if ((doc->pOutputFile) && is_local_file(doc->pOutputFile)) filename = doc->pOutputFile; else filename = addjob->Path; hf = CreateFileW(filename, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); if(hf == INVALID_HANDLE_VALUE) goto end; memset(&job_info, 0, sizeof(job_info)); job_info.pDocument = doc->pDocName; SetJobW(hPrinter, addjob->JobId, 1, (LPBYTE)&job_info, 0); printer->doc = HeapAlloc(GetProcessHeap(), 0, sizeof(*printer->doc)); printer->doc->hf = hf; ret = printer->doc->job_id = addjob->JobId; job = get_job(hPrinter, ret); job->portname = strdupW(doc->pOutputFile); end: LeaveCriticalSection(&printer_handles_cs); return ret; } /***************************************************************************** * StartPagePrinter [WINSPOOL.@] */ BOOL WINAPI StartPagePrinter(HANDLE hPrinter) { FIXME("(%p): stub\n", hPrinter); return TRUE; } /***************************************************************************** * GetFormA [WINSPOOL.@] */ BOOL WINAPI GetFormA(HANDLE hPrinter, LPSTR pFormName, DWORD Level, LPBYTE pForm, DWORD cbBuf, LPDWORD pcbNeeded) { FIXME("(%p,%s,%d,%p,%d,%p): stub\n",hPrinter,pFormName, Level,pForm,cbBuf,pcbNeeded); return FALSE; } /***************************************************************************** * GetFormW [WINSPOOL.@] */ BOOL WINAPI GetFormW(HANDLE hPrinter, LPWSTR pFormName, DWORD Level, LPBYTE pForm, DWORD cbBuf, LPDWORD pcbNeeded) { FIXME("(%p,%s,%d,%p,%d,%p): stub\n",hPrinter, debugstr_w(pFormName),Level,pForm,cbBuf,pcbNeeded); return FALSE; } /***************************************************************************** * SetFormA [WINSPOOL.@] */ BOOL WINAPI SetFormA(HANDLE hPrinter, LPSTR pFormName, DWORD Level, LPBYTE pForm) { FIXME("(%p,%s,%d,%p): stub\n",hPrinter,pFormName,Level,pForm); return FALSE; } /***************************************************************************** * SetFormW [WINSPOOL.@] */ BOOL WINAPI SetFormW(HANDLE hPrinter, LPWSTR pFormName, DWORD Level, LPBYTE pForm) { FIXME("(%p,%p,%d,%p): stub\n",hPrinter,pFormName,Level,pForm); return FALSE; } /***************************************************************************** * ReadPrinter [WINSPOOL.@] */ BOOL WINAPI ReadPrinter(HANDLE hPrinter, LPVOID pBuf, DWORD cbBuf, LPDWORD pNoBytesRead) { FIXME("(%p,%p,%d,%p): stub\n",hPrinter,pBuf,cbBuf,pNoBytesRead); return FALSE; } /***************************************************************************** * ResetPrinterA [WINSPOOL.@] */ BOOL WINAPI ResetPrinterA(HANDLE hPrinter, LPPRINTER_DEFAULTSA pDefault) { FIXME("(%p, %p): stub\n", hPrinter, pDefault); return FALSE; } /***************************************************************************** * ResetPrinterW [WINSPOOL.@] */ BOOL WINAPI ResetPrinterW(HANDLE hPrinter, LPPRINTER_DEFAULTSW pDefault) { FIXME("(%p, %p): stub\n", hPrinter, pDefault); return FALSE; } /***************************************************************************** * get_filename_from_reg [internal] * * Get ValueName from hkey storing result in out * when the Value in the registry has only a filename, use driverdir as prefix * outlen is space left in out * String is stored either as unicode or ascii * */ static BOOL get_filename_from_reg(HKEY hkey, LPCWSTR driverdir, DWORD dirlen, LPCWSTR ValueName, LPBYTE out, DWORD outlen, LPDWORD needed) { WCHAR filename[MAX_PATH]; DWORD size; DWORD type; LONG ret; LPWSTR buffer = filename; LPWSTR ptr; *needed = 0; size = sizeof(filename); buffer[0] = '\0'; ret = RegQueryValueExW(hkey, ValueName, NULL, &type, (LPBYTE) buffer, &size); if (ret == ERROR_MORE_DATA) { TRACE("need dynamic buffer: %u\n", size); buffer = HeapAlloc(GetProcessHeap(), 0, size); if (!buffer) { /* No Memory is bad */ return FALSE; } buffer[0] = '\0'; ret = RegQueryValueExW(hkey, ValueName, NULL, &type, (LPBYTE) buffer, &size); } if ((ret != ERROR_SUCCESS) || (!buffer[0])) { if (buffer != filename) HeapFree(GetProcessHeap(), 0, buffer); return FALSE; } ptr = buffer; while (ptr) { /* do we have a full path ? */ ret = (((buffer[0] == '\\') && (buffer[1] == '\\')) || (buffer[0] && (buffer[1] == ':') && (buffer[2] == '\\')) ); if (!ret) { /* we must build the full Path */ *needed += dirlen; if ((out) && (outlen > dirlen)) { lstrcpyW((LPWSTR)out, driverdir); out += dirlen; outlen -= dirlen; } else out = NULL; } /* write the filename */ size = (lstrlenW(ptr) + 1) * sizeof(WCHAR); if ((out) && (outlen >= size)) { lstrcpyW((LPWSTR)out, ptr); out += size; outlen -= size; } else out = NULL; *needed += size; ptr += lstrlenW(ptr)+1; if ((type != REG_MULTI_SZ) || (!ptr[0])) ptr = NULL; } if (buffer != filename) HeapFree(GetProcessHeap(), 0, buffer); /* write the multisz-termination */ if (type == REG_MULTI_SZ) { size = sizeof(WCHAR); *needed += size; if (out && (outlen >= size)) { memset (out, 0, size); } } return TRUE; } /***************************************************************************** * WINSPOOL_GetStringFromReg * * Get ValueName from hkey storing result in ptr. buflen is space left in ptr * String is stored as unicode. */ static BOOL WINSPOOL_GetStringFromReg(HKEY hkey, LPCWSTR ValueName, LPBYTE ptr, DWORD buflen, DWORD *needed) { DWORD sz = buflen, type; LONG ret; ret = RegQueryValueExW(hkey, ValueName, 0, &type, ptr, &sz); if(ret != ERROR_SUCCESS && ret != ERROR_MORE_DATA) { WARN("Got ret = %d\n", ret); *needed = 0; return FALSE; } /* add space for terminating '\0' */ sz += sizeof(WCHAR); *needed = sz; if (ptr) TRACE("%s: %s\n", debugstr_w(ValueName), debugstr_w((LPCWSTR)ptr)); return TRUE; } /***************************************************************************** * WINSPOOL_GetDefaultDevMode * * Get a default DevMode values for wineps. */ static void WINSPOOL_GetDefaultDevMode(LPBYTE ptr, DWORD buflen, DWORD *needed) { static const WCHAR winepsW[] = { 'w','i','n','e','p','s','.','d','r','v',0 }; if (buflen >= sizeof(DEVMODEW)) { DEVMODEW *dm = (DEVMODEW *)ptr; /* the driver will update registry with real values */ memset(dm, 0, sizeof(*dm)); dm->dmSize = sizeof(*dm); lstrcpyW(dm->dmDeviceName, winepsW); } *needed = sizeof(DEVMODEW); } /***************************************************************************** * WINSPOOL_GetDevModeFromReg * * Get ValueName from hkey storing result in ptr. buflen is space left in ptr * DevMode is stored either as unicode or ascii. */ static BOOL WINSPOOL_GetDevModeFromReg(HKEY hkey, LPCWSTR ValueName, LPBYTE ptr, DWORD buflen, DWORD *needed) { DWORD sz = buflen, type; LONG ret; if (ptr && buflen>=sizeof(DEVMODEA)) memset(ptr, 0, sizeof(DEVMODEA)); ret = RegQueryValueExW(hkey, ValueName, 0, &type, ptr, &sz); if ((ret != ERROR_SUCCESS && ret != ERROR_MORE_DATA)) sz = 0; if (sz < sizeof(DEVMODEA)) { TRACE("corrupted registry for %s ( size %d)\n",debugstr_w(ValueName),sz); return FALSE; } /* ensures that dmSize is not erratically bogus if registry is invalid */ if (ptr && ((DEVMODEA*)ptr)->dmSize < sizeof(DEVMODEA)) ((DEVMODEA*)ptr)->dmSize = sizeof(DEVMODEA); sz += (CCHDEVICENAME + CCHFORMNAME); if (ptr && (buflen >= sz)) { DEVMODEW *dmW = GdiConvertToDevmodeW((DEVMODEA*)ptr); memcpy(ptr, dmW, sz); HeapFree(GetProcessHeap(),0,dmW); } *needed = sz; return TRUE; } /********************************************************************* * WINSPOOL_GetPrinter_1 * * Fills out a PRINTER_INFO_1W struct storing the strings in buf. */ static BOOL WINSPOOL_GetPrinter_1(HKEY hkeyPrinter, PRINTER_INFO_1W *pi1, LPBYTE buf, DWORD cbBuf, LPDWORD pcbNeeded) { DWORD size, left = cbBuf; BOOL space = (cbBuf > 0); LPBYTE ptr = buf; *pcbNeeded = 0; if(WINSPOOL_GetStringFromReg(hkeyPrinter, NameW, ptr, left, &size)) { if(space && size <= left) { pi1->pName = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } /* FIXME: pDescription should be something like "Name,Driver_Name,". */ if(WINSPOOL_GetStringFromReg(hkeyPrinter, NameW, ptr, left, &size)) { if(space && size <= left) { pi1->pDescription = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(WINSPOOL_GetStringFromReg(hkeyPrinter, DescriptionW, ptr, left, &size)) { if(space && size <= left) { pi1->pComment = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(pi1) pi1->Flags = PRINTER_ENUM_ICON8; /* We're a printer */ if(!space && pi1) /* zero out pi1 if we can't completely fill buf */ memset(pi1, 0, sizeof(*pi1)); return space; } /********************************************************************* * WINSPOOL_GetPrinter_2 * * Fills out a PRINTER_INFO_2W struct storing the strings in buf. */ static BOOL WINSPOOL_GetPrinter_2(HKEY hkeyPrinter, PRINTER_INFO_2W *pi2, LPBYTE buf, DWORD cbBuf, LPDWORD pcbNeeded) { DWORD size, left = cbBuf; BOOL space = (cbBuf > 0); LPBYTE ptr = buf; *pcbNeeded = 0; if(WINSPOOL_GetStringFromReg(hkeyPrinter, NameW, ptr, left, &size)) { if(space && size <= left) { pi2->pPrinterName = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(WINSPOOL_GetStringFromReg(hkeyPrinter, Share_NameW, ptr, left, &size)) { if(space && size <= left) { pi2->pShareName = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(WINSPOOL_GetStringFromReg(hkeyPrinter, PortW, ptr, left, &size)) { if(space && size <= left) { pi2->pPortName = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(WINSPOOL_GetStringFromReg(hkeyPrinter, Printer_DriverW, ptr, left, &size)) { if(space && size <= left) { pi2->pDriverName = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(WINSPOOL_GetStringFromReg(hkeyPrinter, DescriptionW, ptr, left, &size)) { if(space && size <= left) { pi2->pComment = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(WINSPOOL_GetStringFromReg(hkeyPrinter, LocationW, ptr, left, &size)) { if(space && size <= left) { pi2->pLocation = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(WINSPOOL_GetDevModeFromReg(hkeyPrinter, Default_DevModeW, ptr, left, &size)) { if(space && size <= left) { pi2->pDevMode = (LPDEVMODEW)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } else { WINSPOOL_GetDefaultDevMode(ptr, left, &size); if(space && size <= left) { pi2->pDevMode = (LPDEVMODEW)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(WINSPOOL_GetStringFromReg(hkeyPrinter, Separator_FileW, ptr, left, &size)) { if(space && size <= left) { pi2->pSepFile = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(WINSPOOL_GetStringFromReg(hkeyPrinter, Print_ProcessorW, ptr, left, &size)) { if(space && size <= left) { pi2->pPrintProcessor = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(WINSPOOL_GetStringFromReg(hkeyPrinter, DatatypeW, ptr, left, &size)) { if(space && size <= left) { pi2->pDatatype = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(WINSPOOL_GetStringFromReg(hkeyPrinter, ParametersW, ptr, left, &size)) { if(space && size <= left) { pi2->pParameters = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(pi2) { pi2->Attributes = get_dword_from_reg( hkeyPrinter, AttributesW ); pi2->Priority = get_dword_from_reg( hkeyPrinter, PriorityW ); pi2->DefaultPriority = get_dword_from_reg( hkeyPrinter, Default_PriorityW ); pi2->StartTime = get_dword_from_reg( hkeyPrinter, StartTimeW ); pi2->UntilTime = get_dword_from_reg( hkeyPrinter, UntilTimeW ); } if(!space && pi2) /* zero out pi2 if we can't completely fill buf */ memset(pi2, 0, sizeof(*pi2)); return space; } /********************************************************************* * WINSPOOL_GetPrinter_4 * * Fills out a PRINTER_INFO_4 struct storing the strings in buf. */ static BOOL WINSPOOL_GetPrinter_4(HKEY hkeyPrinter, PRINTER_INFO_4W *pi4, LPBYTE buf, DWORD cbBuf, LPDWORD pcbNeeded) { DWORD size, left = cbBuf; BOOL space = (cbBuf > 0); LPBYTE ptr = buf; *pcbNeeded = 0; if(WINSPOOL_GetStringFromReg(hkeyPrinter, NameW, ptr, left, &size)) { if(space && size <= left) { pi4->pPrinterName = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(pi4) { pi4->Attributes = get_dword_from_reg( hkeyPrinter, AttributesW ); } if(!space && pi4) /* zero out pi4 if we can't completely fill buf */ memset(pi4, 0, sizeof(*pi4)); return space; } /********************************************************************* * WINSPOOL_GetPrinter_5 * * Fills out a PRINTER_INFO_5 struct storing the strings in buf. */ static BOOL WINSPOOL_GetPrinter_5(HKEY hkeyPrinter, PRINTER_INFO_5W *pi5, LPBYTE buf, DWORD cbBuf, LPDWORD pcbNeeded) { DWORD size, left = cbBuf; BOOL space = (cbBuf > 0); LPBYTE ptr = buf; *pcbNeeded = 0; if(WINSPOOL_GetStringFromReg(hkeyPrinter, NameW, ptr, left, &size)) { if(space && size <= left) { pi5->pPrinterName = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(WINSPOOL_GetStringFromReg(hkeyPrinter, PortW, ptr, left, &size)) { if(space && size <= left) { pi5->pPortName = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if(pi5) { pi5->Attributes = get_dword_from_reg( hkeyPrinter, AttributesW ); pi5->DeviceNotSelectedTimeout = get_dword_from_reg( hkeyPrinter, dnsTimeoutW ); pi5->TransmissionRetryTimeout = get_dword_from_reg( hkeyPrinter, txTimeoutW ); } if(!space && pi5) /* zero out pi5 if we can't completely fill buf */ memset(pi5, 0, sizeof(*pi5)); return space; } /********************************************************************* * WINSPOOL_GetPrinter_7 * * Fills out a PRINTER_INFO_7 struct storing the strings in buf. */ static BOOL WINSPOOL_GetPrinter_7(HKEY hkeyPrinter, PRINTER_INFO_7W *pi7, LPBYTE buf, DWORD cbBuf, LPDWORD pcbNeeded) { DWORD size, left = cbBuf; BOOL space = (cbBuf > 0); LPBYTE ptr = buf; *pcbNeeded = 0; if (! WINSPOOL_GetStringFromReg(hkeyPrinter, ObjectGUIDW, ptr, left, &size)) { ptr = NULL; size = sizeof(pi7->pszObjectGUID); } if (space && size <= left) { pi7->pszObjectGUID = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; if (pi7) { /* We do not have a Directory Service */ pi7->dwAction = DSPRINT_UNPUBLISH; } if (!space && pi7) /* zero out pi7 if we can't completely fill buf */ memset(pi7, 0, sizeof(*pi7)); return space; } /********************************************************************* * WINSPOOL_GetPrinter_9 * * Fills out a PRINTER_INFO_9AW struct storing the strings in buf. */ static BOOL WINSPOOL_GetPrinter_9(HKEY hkeyPrinter, PRINTER_INFO_9W *pi9, LPBYTE buf, DWORD cbBuf, LPDWORD pcbNeeded) { DWORD size; BOOL space = (cbBuf > 0); *pcbNeeded = 0; if(WINSPOOL_GetDevModeFromReg(hkeyPrinter, Default_DevModeW, buf, cbBuf, &size)) { if(space && size <= cbBuf) { pi9->pDevMode = (LPDEVMODEW)buf; } else space = FALSE; *pcbNeeded += size; } else { WINSPOOL_GetDefaultDevMode(buf, cbBuf, &size); if(space && size <= cbBuf) { pi9->pDevMode = (LPDEVMODEW)buf; } else space = FALSE; *pcbNeeded += size; } if(!space && pi9) /* zero out pi9 if we can't completely fill buf */ memset(pi9, 0, sizeof(*pi9)); return space; } /***************************************************************************** * GetPrinterW [WINSPOOL.@] */ BOOL WINAPI GetPrinterW(HANDLE hPrinter, DWORD Level, LPBYTE pPrinter, DWORD cbBuf, LPDWORD pcbNeeded) { DWORD size, needed = 0, err; LPBYTE ptr = NULL; HKEY hkeyPrinter; BOOL ret; TRACE("(%p,%d,%p,%d,%p)\n",hPrinter,Level,pPrinter,cbBuf, pcbNeeded); err = WINSPOOL_GetOpenedPrinterRegKey( hPrinter, &hkeyPrinter ); if (err) { SetLastError( err ); return FALSE; } switch(Level) { case 2: { PRINTER_INFO_2W *pi2 = (PRINTER_INFO_2W *)pPrinter; size = sizeof(PRINTER_INFO_2W); if(size <= cbBuf) { ptr = pPrinter + size; cbBuf -= size; memset(pPrinter, 0, size); } else { pi2 = NULL; cbBuf = 0; } ret = WINSPOOL_GetPrinter_2(hkeyPrinter, pi2, ptr, cbBuf, &needed); needed += size; break; } case 4: { PRINTER_INFO_4W *pi4 = (PRINTER_INFO_4W *)pPrinter; size = sizeof(PRINTER_INFO_4W); if(size <= cbBuf) { ptr = pPrinter + size; cbBuf -= size; memset(pPrinter, 0, size); } else { pi4 = NULL; cbBuf = 0; } ret = WINSPOOL_GetPrinter_4(hkeyPrinter, pi4, ptr, cbBuf, &needed); needed += size; break; } case 5: { PRINTER_INFO_5W *pi5 = (PRINTER_INFO_5W *)pPrinter; size = sizeof(PRINTER_INFO_5W); if(size <= cbBuf) { ptr = pPrinter + size; cbBuf -= size; memset(pPrinter, 0, size); } else { pi5 = NULL; cbBuf = 0; } ret = WINSPOOL_GetPrinter_5(hkeyPrinter, pi5, ptr, cbBuf, &needed); needed += size; break; } case 6: { PRINTER_INFO_6 *pi6 = (PRINTER_INFO_6 *) pPrinter; size = sizeof(PRINTER_INFO_6); if (size <= cbBuf) { /* FIXME: We do not update the status yet */ pi6->dwStatus = get_dword_from_reg( hkeyPrinter, StatusW ); ret = TRUE; } else { ret = FALSE; } needed += size; break; } case 7: { PRINTER_INFO_7W *pi7 = (PRINTER_INFO_7W *) pPrinter; size = sizeof(PRINTER_INFO_7W); if (size <= cbBuf) { ptr = pPrinter + size; cbBuf -= size; memset(pPrinter, 0, size); } else { pi7 = NULL; cbBuf = 0; } ret = WINSPOOL_GetPrinter_7(hkeyPrinter, pi7, ptr, cbBuf, &needed); needed += size; break; } case 8: /* 8 is the global default printer info and 9 already gets it instead of the per-user one */ /* still, PRINTER_INFO_8W is the same as PRINTER_INFO_9W */ /* fall through */ case 9: { PRINTER_INFO_9W *pi9 = (PRINTER_INFO_9W *)pPrinter; size = sizeof(PRINTER_INFO_9W); if(size <= cbBuf) { ptr = pPrinter + size; cbBuf -= size; memset(pPrinter, 0, size); } else { pi9 = NULL; cbBuf = 0; } ret = WINSPOOL_GetPrinter_9(hkeyPrinter, pi9, ptr, cbBuf, &needed); needed += size; break; } default: FIXME("Unimplemented level %d\n", Level); SetLastError(ERROR_INVALID_LEVEL); RegCloseKey(hkeyPrinter); return FALSE; } RegCloseKey(hkeyPrinter); TRACE("returning %d needed = %d\n", ret, needed); if(pcbNeeded) *pcbNeeded = needed; if(!ret) SetLastError(ERROR_INSUFFICIENT_BUFFER); return ret; } /***************************************************************************** * GetPrinterA [WINSPOOL.@] */ BOOL WINAPI GetPrinterA(HANDLE hPrinter, DWORD Level, LPBYTE pPrinter, DWORD cbBuf, LPDWORD pcbNeeded) { BOOL ret; LPBYTE buf = NULL; if (cbBuf) buf = HeapAlloc(GetProcessHeap(), 0, cbBuf); ret = GetPrinterW(hPrinter, Level, buf, cbBuf, pcbNeeded); if (ret) convert_printerinfo_W_to_A(pPrinter, buf, Level, cbBuf, 1); HeapFree(GetProcessHeap(), 0, buf); return ret; } /***************************************************************************** * WINSPOOL_EnumPrintersW * * Implementation of EnumPrintersW */ static BOOL WINSPOOL_EnumPrintersW(DWORD dwType, LPWSTR lpszName, DWORD dwLevel, LPBYTE lpbPrinters, DWORD cbBuf, LPDWORD lpdwNeeded, LPDWORD lpdwReturned) { HKEY hkeyPrinters, hkeyPrinter; WCHAR PrinterName[255]; DWORD needed = 0, number = 0; DWORD used, i, left; PBYTE pi, buf; if(lpbPrinters) memset(lpbPrinters, 0, cbBuf); if(lpdwReturned) *lpdwReturned = 0; if(lpdwNeeded) *lpdwNeeded = 0; /* PRINTER_ENUM_DEFAULT is only supported under win9x, we behave like NT */ if(dwType == PRINTER_ENUM_DEFAULT) return TRUE; if (dwType & PRINTER_ENUM_CONNECTIONS) { TRACE("ignoring PRINTER_ENUM_CONNECTIONS\n"); dwType &= ~PRINTER_ENUM_CONNECTIONS; /* we don't handle that */ if (!dwType) { FIXME("We don't handle PRINTER_ENUM_CONNECTIONS\n"); return TRUE; } } if (!((dwType & PRINTER_ENUM_LOCAL) || (dwType & PRINTER_ENUM_NAME))) { FIXME("dwType = %08x\n", dwType); SetLastError(ERROR_INVALID_FLAGS); return FALSE; } if(RegCreateKeyW(HKEY_LOCAL_MACHINE, PrintersW, &hkeyPrinters) != ERROR_SUCCESS) { ERR("Can't create Printers key\n"); return FALSE; } if(RegQueryInfoKeyA(hkeyPrinters, NULL, NULL, NULL, &number, NULL, NULL, NULL, NULL, NULL, NULL, NULL) != ERROR_SUCCESS) { RegCloseKey(hkeyPrinters); ERR("Can't query Printers key\n"); return FALSE; } TRACE("Found %d printers\n", number); switch(dwLevel) { case 1: used = number * sizeof(PRINTER_INFO_1W); break; case 2: used = number * sizeof(PRINTER_INFO_2W); break; case 4: used = number * sizeof(PRINTER_INFO_4W); break; case 5: used = number * sizeof(PRINTER_INFO_5W); break; default: SetLastError(ERROR_INVALID_LEVEL); RegCloseKey(hkeyPrinters); return FALSE; } pi = (used <= cbBuf) ? lpbPrinters : NULL; for(i = 0; i < number; i++) { if(RegEnumKeyW(hkeyPrinters, i, PrinterName, sizeof(PrinterName)/sizeof(PrinterName[0])) != ERROR_SUCCESS) { ERR("Can't enum key number %d\n", i); RegCloseKey(hkeyPrinters); return FALSE; } TRACE("Printer %d is %s\n", i, debugstr_w(PrinterName)); if(RegOpenKeyW(hkeyPrinters, PrinterName, &hkeyPrinter) != ERROR_SUCCESS) { ERR("Can't open key %s\n", debugstr_w(PrinterName)); RegCloseKey(hkeyPrinters); return FALSE; } if(cbBuf > used) { buf = lpbPrinters + used; left = cbBuf - used; } else { buf = NULL; left = 0; } switch(dwLevel) { case 1: WINSPOOL_GetPrinter_1(hkeyPrinter, (PRINTER_INFO_1W *)pi, buf, left, &needed); used += needed; if(pi) pi += sizeof(PRINTER_INFO_1W); break; case 2: WINSPOOL_GetPrinter_2(hkeyPrinter, (PRINTER_INFO_2W *)pi, buf, left, &needed); used += needed; if(pi) pi += sizeof(PRINTER_INFO_2W); break; case 4: WINSPOOL_GetPrinter_4(hkeyPrinter, (PRINTER_INFO_4W *)pi, buf, left, &needed); used += needed; if(pi) pi += sizeof(PRINTER_INFO_4W); break; case 5: WINSPOOL_GetPrinter_5(hkeyPrinter, (PRINTER_INFO_5W *)pi, buf, left, &needed); used += needed; if(pi) pi += sizeof(PRINTER_INFO_5W); break; default: ERR("Shouldn't be here!\n"); RegCloseKey(hkeyPrinter); RegCloseKey(hkeyPrinters); return FALSE; } RegCloseKey(hkeyPrinter); } RegCloseKey(hkeyPrinters); if(lpdwNeeded) *lpdwNeeded = used; if(used > cbBuf) { if(lpbPrinters) memset(lpbPrinters, 0, cbBuf); SetLastError(ERROR_INSUFFICIENT_BUFFER); return FALSE; } if(lpdwReturned) *lpdwReturned = number; SetLastError(ERROR_SUCCESS); return TRUE; } /****************************************************************** * EnumPrintersW [WINSPOOL.@] * * Enumerates the available printers, print servers and print * providers, depending on the specified flags, name and level. * * RETURNS: * * If level is set to 1: * Returns an array of PRINTER_INFO_1 data structures in the * lpbPrinters buffer. * * If level is set to 2: * Possible flags: PRINTER_ENUM_CONNECTIONS, PRINTER_ENUM_LOCAL. * Returns an array of PRINTER_INFO_2 data structures in the * lpbPrinters buffer. Note that according to MSDN also an * OpenPrinter should be performed on every remote printer. * * If level is set to 4 (officially WinNT only): * Possible flags: PRINTER_ENUM_CONNECTIONS, PRINTER_ENUM_LOCAL. * Fast: Only the registry is queried to retrieve printer names, * no connection to the driver is made. * Returns an array of PRINTER_INFO_4 data structures in the * lpbPrinters buffer. * * If level is set to 5 (officially WinNT4/Win9x only): * Fast: Only the registry is queried to retrieve printer names, * no connection to the driver is made. * Returns an array of PRINTER_INFO_5 data structures in the * lpbPrinters buffer. * * If level set to 3 or 6+: * returns zero (failure!) * * Returns nonzero (TRUE) on success, or zero on failure, use GetLastError * for information. * * BUGS: * - Only PRINTER_ENUM_LOCAL and PRINTER_ENUM_NAME are implemented. * - Only levels 2, 4 and 5 are implemented at the moment. * - 16-bit printer drivers are not enumerated. * - Returned amount of bytes used/needed does not match the real Windoze * implementation (as in this implementation, all strings are part * of the buffer, whereas Win32 keeps them somewhere else) * - At level 2, EnumPrinters should also call OpenPrinter for remote printers. * * NOTE: * - In a regular Wine installation, no registry settings for printers * exist, which makes this function return an empty list. */ BOOL WINAPI EnumPrintersW( DWORD dwType, /* [in] Types of print objects to enumerate */ LPWSTR lpszName, /* [in] name of objects to enumerate */ DWORD dwLevel, /* [in] type of printer info structure */ LPBYTE lpbPrinters, /* [out] buffer which receives info */ DWORD cbBuf, /* [in] max size of buffer in bytes */ LPDWORD lpdwNeeded, /* [out] pointer to var: # bytes used/needed */ LPDWORD lpdwReturned /* [out] number of entries returned */ ) { return WINSPOOL_EnumPrintersW(dwType, lpszName, dwLevel, lpbPrinters, cbBuf, lpdwNeeded, lpdwReturned); } /****************************************************************** * EnumPrintersA [WINSPOOL.@] * * See EnumPrintersW * */ BOOL WINAPI EnumPrintersA(DWORD flags, LPSTR pName, DWORD level, LPBYTE pPrinters, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { BOOL ret; UNICODE_STRING pNameU; LPWSTR pNameW; LPBYTE pPrintersW; TRACE("(0x%x, %s, %u, %p, %d, %p, %p)\n", flags, debugstr_a(pName), level, pPrinters, cbBuf, pcbNeeded, pcReturned); pNameW = asciitounicode(&pNameU, pName); /* Request a buffer with a size, that is big enough for EnumPrintersW. MS Office need this */ pPrintersW = (pPrinters && cbBuf) ? HeapAlloc(GetProcessHeap(), 0, cbBuf) : NULL; ret = EnumPrintersW(flags, pNameW, level, pPrintersW, cbBuf, pcbNeeded, pcReturned); RtlFreeUnicodeString(&pNameU); if (ret) { convert_printerinfo_W_to_A(pPrinters, pPrintersW, level, *pcbNeeded, *pcReturned); } HeapFree(GetProcessHeap(), 0, pPrintersW); return ret; } /***************************************************************************** * WINSPOOL_GetDriverInfoFromReg [internal] * * Enters the information from the registry into the DRIVER_INFO struct * * RETURNS * zero if the printer driver does not exist in the registry * (only if Level > 1) otherwise nonzero */ static BOOL WINSPOOL_GetDriverInfoFromReg( HKEY hkeyDrivers, LPWSTR DriverName, const printenv_t * env, DWORD Level, LPBYTE ptr, /* DRIVER_INFO */ LPBYTE pDriverStrings, /* strings buffer */ DWORD cbBuf, /* size of string buffer */ LPDWORD pcbNeeded) /* space needed for str. */ { DWORD size, tmp; HKEY hkeyDriver; WCHAR driverdir[MAX_PATH]; DWORD dirlen; LPBYTE strPtr = pDriverStrings; LPDRIVER_INFO_8W di = (LPDRIVER_INFO_8W) ptr; TRACE("(%p, %s, %p, %d, %p, %p, %d)\n", hkeyDrivers, debugstr_w(DriverName), env, Level, di, pDriverStrings, cbBuf); if (di) ZeroMemory(di, di_sizeof[Level]); *pcbNeeded = (lstrlenW(DriverName) + 1) * sizeof(WCHAR); if (*pcbNeeded <= cbBuf) strcpyW((LPWSTR)strPtr, DriverName); /* pName for level 1 has a different offset! */ if (Level == 1) { if (di) ((LPDRIVER_INFO_1W) di)->pName = (LPWSTR) strPtr; return TRUE; } /* .cVersion and .pName for level > 1 */ if (di) { di->cVersion = env->driverversion; di->pName = (LPWSTR) strPtr; strPtr = (pDriverStrings) ? (pDriverStrings + (*pcbNeeded)) : NULL; } /* Reserve Space for the largest subdir and a Backslash*/ size = sizeof(driverdir) - sizeof(Version3_SubdirW) - sizeof(WCHAR); if (!GetPrinterDriverDirectoryW(NULL, (LPWSTR) env->envname, 1, (LPBYTE) driverdir, size, &size)) { /* Should never Fail */ return FALSE; } lstrcatW(driverdir, env->versionsubdir); lstrcatW(driverdir, backslashW); /* dirlen must not include the terminating zero */ dirlen = lstrlenW(driverdir) * sizeof(WCHAR); if (!DriverName[0] || RegOpenKeyW(hkeyDrivers, DriverName, &hkeyDriver) != ERROR_SUCCESS) { ERR("Can't find driver %s in registry\n", debugstr_w(DriverName)); SetLastError(ERROR_UNKNOWN_PRINTER_DRIVER); /* ? */ return FALSE; } /* pEnvironment */ size = (lstrlenW(env->envname) + 1) * sizeof(WCHAR); *pcbNeeded += size; if (*pcbNeeded <= cbBuf) { lstrcpyW((LPWSTR)strPtr, env->envname); if (di) di->pEnvironment = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? (pDriverStrings + (*pcbNeeded)) : NULL; } /* .pDriverPath is the Graphics rendering engine. The full Path is required to avoid a crash in some apps */ if (get_filename_from_reg(hkeyDriver, driverdir, dirlen, DriverW, strPtr, 0, &size)) { *pcbNeeded += size; if (*pcbNeeded <= cbBuf) get_filename_from_reg(hkeyDriver, driverdir, dirlen, DriverW, strPtr, size, &tmp); if (di) di->pDriverPath = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? (pDriverStrings + (*pcbNeeded)) : NULL; } /* .pDataFile: For postscript-drivers, this is the ppd-file */ if (get_filename_from_reg(hkeyDriver, driverdir, dirlen, Data_FileW, strPtr, 0, &size)) { *pcbNeeded += size; if (*pcbNeeded <= cbBuf) get_filename_from_reg(hkeyDriver, driverdir, dirlen, Data_FileW, strPtr, size, &size); if (di) di->pDataFile = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? pDriverStrings + (*pcbNeeded) : NULL; } /* .pConfigFile is the Driver user Interface */ if (get_filename_from_reg(hkeyDriver, driverdir, dirlen, Configuration_FileW, strPtr, 0, &size)) { *pcbNeeded += size; if (*pcbNeeded <= cbBuf) get_filename_from_reg(hkeyDriver, driverdir, dirlen, Configuration_FileW, strPtr, size, &size); if (di) di->pConfigFile = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? pDriverStrings + (*pcbNeeded) : NULL; } if (Level == 2 ) { RegCloseKey(hkeyDriver); TRACE("buffer space %d required %d\n", cbBuf, *pcbNeeded); return TRUE; } if (Level == 5 ) { RegCloseKey(hkeyDriver); FIXME("level 5: incomplete\n"); return TRUE; } /* .pHelpFile */ if (get_filename_from_reg(hkeyDriver, driverdir, dirlen, Help_FileW, strPtr, 0, &size)) { *pcbNeeded += size; if (*pcbNeeded <= cbBuf) get_filename_from_reg(hkeyDriver, driverdir, dirlen, Help_FileW, strPtr, size, &size); if (di) di->pHelpFile = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? pDriverStrings + (*pcbNeeded) : NULL; } /* .pDependentFiles */ if (get_filename_from_reg(hkeyDriver, driverdir, dirlen, Dependent_FilesW, strPtr, 0, &size)) { *pcbNeeded += size; if (*pcbNeeded <= cbBuf) get_filename_from_reg(hkeyDriver, driverdir, dirlen, Dependent_FilesW, strPtr, size, &size); if (di) di->pDependentFiles = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? pDriverStrings + (*pcbNeeded) : NULL; } else if (GetVersion() & 0x80000000) { /* PowerPoint XP expects that pDependentFiles is always valid on win9x */ size = 2 * sizeof(WCHAR); *pcbNeeded += size; if ((*pcbNeeded <= cbBuf) && strPtr) ZeroMemory(strPtr, size); if (di) di->pDependentFiles = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? pDriverStrings + (*pcbNeeded) : NULL; } /* .pMonitorName is the optional Language Monitor */ if (WINSPOOL_GetStringFromReg(hkeyDriver, MonitorW, strPtr, 0, &size)) { *pcbNeeded += size; if (*pcbNeeded <= cbBuf) WINSPOOL_GetStringFromReg(hkeyDriver, MonitorW, strPtr, size, &size); if (di) di->pMonitorName = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? pDriverStrings + (*pcbNeeded) : NULL; } /* .pDefaultDataType */ if (WINSPOOL_GetStringFromReg(hkeyDriver, DatatypeW, strPtr, 0, &size)) { *pcbNeeded += size; if(*pcbNeeded <= cbBuf) WINSPOOL_GetStringFromReg(hkeyDriver, DatatypeW, strPtr, size, &size); if (di) di->pDefaultDataType = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? pDriverStrings + (*pcbNeeded) : NULL; } if (Level == 3 ) { RegCloseKey(hkeyDriver); TRACE("buffer space %d required %d\n", cbBuf, *pcbNeeded); return TRUE; } /* .pszzPreviousNames */ if (WINSPOOL_GetStringFromReg(hkeyDriver, Previous_NamesW, strPtr, 0, &size)) { *pcbNeeded += size; if(*pcbNeeded <= cbBuf) WINSPOOL_GetStringFromReg(hkeyDriver, Previous_NamesW, strPtr, size, &size); if (di) di->pszzPreviousNames = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? pDriverStrings + (*pcbNeeded) : NULL; } if (Level == 4 ) { RegCloseKey(hkeyDriver); TRACE("buffer space %d required %d\n", cbBuf, *pcbNeeded); return TRUE; } /* support is missing, but not important enough for a FIXME */ TRACE("%s: DriverDate + DriverVersion not supported\n", debugstr_w(DriverName)); /* .pszMfgName */ if (WINSPOOL_GetStringFromReg(hkeyDriver, ManufacturerW, strPtr, 0, &size)) { *pcbNeeded += size; if(*pcbNeeded <= cbBuf) WINSPOOL_GetStringFromReg(hkeyDriver, ManufacturerW, strPtr, size, &size); if (di) di->pszMfgName = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? pDriverStrings + (*pcbNeeded) : NULL; } /* .pszOEMUrl */ if (WINSPOOL_GetStringFromReg(hkeyDriver, OEM_UrlW, strPtr, 0, &size)) { *pcbNeeded += size; if(*pcbNeeded <= cbBuf) WINSPOOL_GetStringFromReg(hkeyDriver, OEM_UrlW, strPtr, size, &size); if (di) di->pszOEMUrl = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? pDriverStrings + (*pcbNeeded) : NULL; } /* .pszHardwareID */ if (WINSPOOL_GetStringFromReg(hkeyDriver, HardwareIDW, strPtr, 0, &size)) { *pcbNeeded += size; if(*pcbNeeded <= cbBuf) WINSPOOL_GetStringFromReg(hkeyDriver, HardwareIDW, strPtr, size, &size); if (di) di->pszHardwareID = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? pDriverStrings + (*pcbNeeded) : NULL; } /* .pszProvider */ if (WINSPOOL_GetStringFromReg(hkeyDriver, ProviderW, strPtr, 0, &size)) { *pcbNeeded += size; if(*pcbNeeded <= cbBuf) WINSPOOL_GetStringFromReg(hkeyDriver, ProviderW, strPtr, size, &size); if (di) di->pszProvider = (LPWSTR)strPtr; strPtr = (pDriverStrings) ? pDriverStrings + (*pcbNeeded) : NULL; } if (Level == 6 ) { RegCloseKey(hkeyDriver); return TRUE; } /* support is missing, but not important enough for a FIXME */ TRACE("level 8: incomplete\n"); TRACE("buffer space %d required %d\n", cbBuf, *pcbNeeded); RegCloseKey(hkeyDriver); return TRUE; } /***************************************************************************** * GetPrinterDriverW [WINSPOOL.@] */ BOOL WINAPI GetPrinterDriverW(HANDLE hPrinter, LPWSTR pEnvironment, DWORD Level, LPBYTE pDriverInfo, DWORD cbBuf, LPDWORD pcbNeeded) { LPCWSTR name; WCHAR DriverName[100]; DWORD ret, type, size, needed = 0; LPBYTE ptr = NULL; HKEY hkeyPrinter, hkeyDrivers; const printenv_t * env; TRACE("(%p,%s,%d,%p,%d,%p)\n",hPrinter,debugstr_w(pEnvironment), Level,pDriverInfo,cbBuf, pcbNeeded); if (cbBuf > 0) ZeroMemory(pDriverInfo, cbBuf); if (!(name = get_opened_printer_name(hPrinter))) { SetLastError(ERROR_INVALID_HANDLE); return FALSE; } if (Level < 1 || Level == 7 || Level > 8) { SetLastError(ERROR_INVALID_LEVEL); return FALSE; } env = validate_envW(pEnvironment); if (!env) return FALSE; /* SetLastError() is in validate_envW */ ret = open_printer_reg_key( name, &hkeyPrinter ); if (ret) { ERR( "Can't find opened printer %s in registry\n", debugstr_w(name) ); SetLastError( ret ); return FALSE; } size = sizeof(DriverName); DriverName[0] = 0; ret = RegQueryValueExW(hkeyPrinter, Printer_DriverW, 0, &type, (LPBYTE)DriverName, &size); RegCloseKey(hkeyPrinter); if(ret != ERROR_SUCCESS) { ERR("Can't get DriverName for printer %s\n", debugstr_w(name)); return FALSE; } hkeyDrivers = WINSPOOL_OpenDriverReg(pEnvironment); if(!hkeyDrivers) { ERR("Can't create Drivers key\n"); return FALSE; } size = di_sizeof[Level]; if ((size <= cbBuf) && pDriverInfo) ptr = pDriverInfo + size; if(!WINSPOOL_GetDriverInfoFromReg(hkeyDrivers, DriverName, env, Level, pDriverInfo, ptr, (cbBuf < size) ? 0 : cbBuf - size, &needed)) { RegCloseKey(hkeyDrivers); return FALSE; } RegCloseKey(hkeyDrivers); if(pcbNeeded) *pcbNeeded = size + needed; TRACE("buffer space %d required %d\n", cbBuf, size + needed); if(cbBuf >= size + needed) return TRUE; SetLastError(ERROR_INSUFFICIENT_BUFFER); return FALSE; } /***************************************************************************** * GetPrinterDriverA [WINSPOOL.@] */ BOOL WINAPI GetPrinterDriverA(HANDLE hPrinter, LPSTR pEnvironment, DWORD Level, LPBYTE pDriverInfo, DWORD cbBuf, LPDWORD pcbNeeded) { BOOL ret; UNICODE_STRING pEnvW; PWSTR pwstrEnvW; LPBYTE buf = NULL; if (cbBuf) { ZeroMemory(pDriverInfo, cbBuf); buf = HeapAlloc(GetProcessHeap(), 0, cbBuf); } pwstrEnvW = asciitounicode(&pEnvW, pEnvironment); ret = GetPrinterDriverW(hPrinter, pwstrEnvW, Level, buf, cbBuf, pcbNeeded); if (ret) convert_driverinfo_W_to_A(pDriverInfo, buf, Level, cbBuf, 1); HeapFree(GetProcessHeap(), 0, buf); RtlFreeUnicodeString(&pEnvW); return ret; } /***************************************************************************** * GetPrinterDriverDirectoryW [WINSPOOL.@] * * Return the PATH for the Printer-Drivers (UNICODE) * * PARAMS * pName [I] Servername (NT only) or NULL (local Computer) * pEnvironment [I] Printing-Environment (see below) or NULL (Default) * Level [I] Structure-Level (must be 1) * pDriverDirectory [O] PTR to Buffer that receives the Result * cbBuf [I] Size of Buffer at pDriverDirectory * pcbNeeded [O] PTR to DWORD that receives the size in Bytes used / * required for pDriverDirectory * * RETURNS * Success: TRUE and in pcbNeeded the Bytes used in pDriverDirectory * Failure: FALSE and in pcbNeeded the Bytes required for pDriverDirectory, * if cbBuf is too small * * Native Values returned in pDriverDirectory on Success: *| NT(Windows NT x86): "%winsysdir%\\spool\\DRIVERS\\w32x86" *| NT(Windows 4.0): "%winsysdir%\\spool\\DRIVERS\\win40" *| win9x(Windows 4.0): "%winsysdir%" * * "%winsysdir%" is the Value from GetSystemDirectoryW() * * FIXME *- Only NULL or "" is supported for pName * */ BOOL WINAPI GetPrinterDriverDirectoryW(LPWSTR pName, LPWSTR pEnvironment, DWORD Level, LPBYTE pDriverDirectory, DWORD cbBuf, LPDWORD pcbNeeded) { TRACE("(%s, %s, %d, %p, %d, %p)\n", debugstr_w(pName), debugstr_w(pEnvironment), Level, pDriverDirectory, cbBuf, pcbNeeded); if ((backend == NULL) && !load_backend()) return FALSE; if (Level != 1) { /* (Level != 1) is ignored in win9x */ SetLastError(ERROR_INVALID_LEVEL); return FALSE; } if (pcbNeeded == NULL) { /* (pcbNeeded == NULL) is ignored in win9x */ SetLastError(RPC_X_NULL_REF_POINTER); return FALSE; } return backend->fpGetPrinterDriverDirectory(pName, pEnvironment, Level, pDriverDirectory, cbBuf, pcbNeeded); } /***************************************************************************** * GetPrinterDriverDirectoryA [WINSPOOL.@] * * Return the PATH for the Printer-Drivers (ANSI) * * See GetPrinterDriverDirectoryW. * * NOTES * On NT, pDriverDirectory need the same Size as the Unicode-Version * */ BOOL WINAPI GetPrinterDriverDirectoryA(LPSTR pName, LPSTR pEnvironment, DWORD Level, LPBYTE pDriverDirectory, DWORD cbBuf, LPDWORD pcbNeeded) { UNICODE_STRING nameW, environmentW; BOOL ret; DWORD pcbNeededW; INT len = cbBuf * sizeof(WCHAR)/sizeof(CHAR); WCHAR *driverDirectoryW = NULL; TRACE("(%s, %s, %d, %p, %d, %p)\n", debugstr_a(pName), debugstr_a(pEnvironment), Level, pDriverDirectory, cbBuf, pcbNeeded); if (len) driverDirectoryW = HeapAlloc( GetProcessHeap(), 0, len ); if(pName) RtlCreateUnicodeStringFromAsciiz(&nameW, pName); else nameW.Buffer = NULL; if(pEnvironment) RtlCreateUnicodeStringFromAsciiz(&environmentW, pEnvironment); else environmentW.Buffer = NULL; ret = GetPrinterDriverDirectoryW( nameW.Buffer, environmentW.Buffer, Level, (LPBYTE)driverDirectoryW, len, &pcbNeededW ); if (ret) { DWORD needed; needed = WideCharToMultiByte( CP_ACP, 0, driverDirectoryW, -1, (LPSTR)pDriverDirectory, cbBuf, NULL, NULL); if(pcbNeeded) *pcbNeeded = needed; ret = needed <= cbBuf; } else if(pcbNeeded) *pcbNeeded = pcbNeededW * sizeof(CHAR)/sizeof(WCHAR); TRACE("required: 0x%x/%d\n", pcbNeeded ? *pcbNeeded : 0, pcbNeeded ? *pcbNeeded : 0); HeapFree( GetProcessHeap(), 0, driverDirectoryW ); RtlFreeUnicodeString(&environmentW); RtlFreeUnicodeString(&nameW); return ret; } /***************************************************************************** * AddPrinterDriverA [WINSPOOL.@] * * See AddPrinterDriverW. * */ BOOL WINAPI AddPrinterDriverA(LPSTR pName, DWORD level, LPBYTE pDriverInfo) { TRACE("(%s, %d, %p)\n", debugstr_a(pName), level, pDriverInfo); return AddPrinterDriverExA(pName, level, pDriverInfo, APD_COPY_NEW_FILES); } /****************************************************************************** * AddPrinterDriverW (WINSPOOL.@) * * Install a Printer Driver * * PARAMS * pName [I] Servername or NULL (local Computer) * level [I] Level for the supplied DRIVER_INFO_*W struct * pDriverInfo [I] PTR to DRIVER_INFO_*W struct with the Driver Parameter * * RESULTS * Success: TRUE * Failure: FALSE * */ BOOL WINAPI AddPrinterDriverW(LPWSTR pName, DWORD level, LPBYTE pDriverInfo) { TRACE("(%s, %d, %p)\n", debugstr_w(pName), level, pDriverInfo); return AddPrinterDriverExW(pName, level, pDriverInfo, APD_COPY_NEW_FILES); } /***************************************************************************** * AddPrintProcessorA [WINSPOOL.@] */ BOOL WINAPI AddPrintProcessorA(LPSTR pName, LPSTR pEnvironment, LPSTR pPathName, LPSTR pPrintProcessorName) { FIXME("(%s,%s,%s,%s): stub\n", debugstr_a(pName), debugstr_a(pEnvironment), debugstr_a(pPathName), debugstr_a(pPrintProcessorName)); return FALSE; } /***************************************************************************** * AddPrintProcessorW [WINSPOOL.@] */ BOOL WINAPI AddPrintProcessorW(LPWSTR pName, LPWSTR pEnvironment, LPWSTR pPathName, LPWSTR pPrintProcessorName) { FIXME("(%s,%s,%s,%s): stub\n", debugstr_w(pName), debugstr_w(pEnvironment), debugstr_w(pPathName), debugstr_w(pPrintProcessorName)); return TRUE; } /***************************************************************************** * AddPrintProvidorA [WINSPOOL.@] */ BOOL WINAPI AddPrintProvidorA(LPSTR pName, DWORD Level, LPBYTE pProviderInfo) { FIXME("(%s,0x%08x,%p): stub\n", debugstr_a(pName), Level, pProviderInfo); return FALSE; } /***************************************************************************** * AddPrintProvidorW [WINSPOOL.@] */ BOOL WINAPI AddPrintProvidorW(LPWSTR pName, DWORD Level, LPBYTE pProviderInfo) { FIXME("(%s,0x%08x,%p): stub\n", debugstr_w(pName), Level, pProviderInfo); return FALSE; } /***************************************************************************** * AdvancedDocumentPropertiesA [WINSPOOL.@] */ LONG WINAPI AdvancedDocumentPropertiesA(HWND hWnd, HANDLE hPrinter, LPSTR pDeviceName, PDEVMODEA pDevModeOutput, PDEVMODEA pDevModeInput) { FIXME("(%p,%p,%s,%p,%p): stub\n", hWnd, hPrinter, debugstr_a(pDeviceName), pDevModeOutput, pDevModeInput); return 0; } /***************************************************************************** * AdvancedDocumentPropertiesW [WINSPOOL.@] */ LONG WINAPI AdvancedDocumentPropertiesW(HWND hWnd, HANDLE hPrinter, LPWSTR pDeviceName, PDEVMODEW pDevModeOutput, PDEVMODEW pDevModeInput) { FIXME("(%p,%p,%s,%p,%p): stub\n", hWnd, hPrinter, debugstr_w(pDeviceName), pDevModeOutput, pDevModeInput); return 0; } /***************************************************************************** * PrinterProperties [WINSPOOL.@] * * Displays a dialog to set the properties of the printer. * * RETURNS * nonzero on success or zero on failure * * BUGS * implemented as stub only */ BOOL WINAPI PrinterProperties(HWND hWnd, /* [in] handle to parent window */ HANDLE hPrinter /* [in] handle to printer object */ ){ FIXME("(%p,%p): stub\n", hWnd, hPrinter); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return FALSE; } /***************************************************************************** * EnumJobsA [WINSPOOL.@] * */ BOOL WINAPI EnumJobsA(HANDLE hPrinter, DWORD FirstJob, DWORD NoJobs, DWORD Level, LPBYTE pJob, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { FIXME("(%p,first=%d,no=%d,level=%d,job=%p,cb=%d,%p,%p), stub!\n", hPrinter, FirstJob, NoJobs, Level, pJob, cbBuf, pcbNeeded, pcReturned ); if(pcbNeeded) *pcbNeeded = 0; if(pcReturned) *pcReturned = 0; return FALSE; } /***************************************************************************** * EnumJobsW [WINSPOOL.@] * */ BOOL WINAPI EnumJobsW(HANDLE hPrinter, DWORD FirstJob, DWORD NoJobs, DWORD Level, LPBYTE pJob, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { FIXME("(%p,first=%d,no=%d,level=%d,job=%p,cb=%d,%p,%p), stub!\n", hPrinter, FirstJob, NoJobs, Level, pJob, cbBuf, pcbNeeded, pcReturned ); if(pcbNeeded) *pcbNeeded = 0; if(pcReturned) *pcReturned = 0; return FALSE; } /***************************************************************************** * WINSPOOL_EnumPrinterDrivers [internal] * * Delivers information about all printer drivers installed on the * localhost or a given server * * RETURNS * nonzero on success or zero on failure. If the buffer for the returned * information is too small the function will return an error * * BUGS * - only implemented for localhost, foreign hosts will return an error */ static BOOL WINSPOOL_EnumPrinterDrivers(LPWSTR pName, LPCWSTR pEnvironment, DWORD Level, LPBYTE pDriverInfo, DWORD driver_index, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcFound, DWORD data_offset) { HKEY hkeyDrivers; DWORD i, size = 0; const printenv_t * env; TRACE("%s,%s,%d,%p,%d,%d,%d\n", debugstr_w(pName), debugstr_w(pEnvironment), Level, pDriverInfo, driver_index, cbBuf, data_offset); env = validate_envW(pEnvironment); if (!env) return FALSE; /* SetLastError() is in validate_envW */ *pcFound = 0; hkeyDrivers = WINSPOOL_OpenDriverReg(pEnvironment); if(!hkeyDrivers) { ERR("Can't open Drivers key\n"); return FALSE; } if(RegQueryInfoKeyA(hkeyDrivers, NULL, NULL, NULL, pcFound, NULL, NULL, NULL, NULL, NULL, NULL, NULL) != ERROR_SUCCESS) { RegCloseKey(hkeyDrivers); ERR("Can't query Drivers key\n"); return FALSE; } TRACE("Found %d Drivers\n", *pcFound); /* get size of single struct * unicode and ascii structure have the same size */ size = di_sizeof[Level]; if (data_offset == 0) data_offset = size * (*pcFound); *pcbNeeded = data_offset; for( i = 0; i < *pcFound; i++) { WCHAR DriverNameW[255]; PBYTE table_ptr = NULL; PBYTE data_ptr = NULL; DWORD needed = 0; if(RegEnumKeyW(hkeyDrivers, i, DriverNameW, sizeof(DriverNameW)/sizeof(DriverNameW[0])) != ERROR_SUCCESS) { ERR("Can't enum key number %d\n", i); RegCloseKey(hkeyDrivers); return FALSE; } if (pDriverInfo && ((driver_index + i + 1) * size) <= cbBuf) table_ptr = pDriverInfo + (driver_index + i) * size; if (pDriverInfo && *pcbNeeded <= cbBuf) data_ptr = pDriverInfo + *pcbNeeded; if(!WINSPOOL_GetDriverInfoFromReg(hkeyDrivers, DriverNameW, env, Level, table_ptr, data_ptr, (cbBuf < *pcbNeeded) ? 0 : cbBuf - *pcbNeeded, &needed)) { RegCloseKey(hkeyDrivers); return FALSE; } *pcbNeeded += needed; } RegCloseKey(hkeyDrivers); if(cbBuf < *pcbNeeded){ SetLastError(ERROR_INSUFFICIENT_BUFFER); return FALSE; } return TRUE; } /***************************************************************************** * EnumPrinterDriversW [WINSPOOL.@] * * see function EnumPrinterDrivers for RETURNS, BUGS */ BOOL WINAPI EnumPrinterDriversW(LPWSTR pName, LPWSTR pEnvironment, DWORD Level, LPBYTE pDriverInfo, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { static const WCHAR allW[] = {'a','l','l',0}; BOOL ret; DWORD found; if ((pcbNeeded == NULL) || (pcReturned == NULL)) { SetLastError(RPC_X_NULL_REF_POINTER); return FALSE; } /* check for local drivers */ if((pName) && (pName[0])) { FIXME("remote drivers (%s) not supported!\n", debugstr_w(pName)); SetLastError(ERROR_ACCESS_DENIED); return FALSE; } /* check input parameter */ if ((Level < 1) || (Level == 7) || (Level > 8)) { SetLastError(ERROR_INVALID_LEVEL); return FALSE; } if(pDriverInfo && cbBuf > 0) memset( pDriverInfo, 0, cbBuf); /* Exception: pull all printers */ if (pEnvironment && !strcmpW(pEnvironment, allW)) { DWORD i, needed, bufsize = cbBuf; DWORD total_found = 0; DWORD data_offset; /* Precompute the overall total; we need this to know where pointers end and data begins (i.e. data_offset) */ for (i = 0; i < sizeof(all_printenv)/sizeof(all_printenv[0]); i++) { needed = found = 0; ret = WINSPOOL_EnumPrinterDrivers(pName, all_printenv[i]->envname, Level, NULL, 0, 0, &needed, &found, 0); if (!ret && GetLastError() != ERROR_INSUFFICIENT_BUFFER) return FALSE; total_found += found; } data_offset = di_sizeof[Level] * total_found; *pcReturned = 0; *pcbNeeded = 0; total_found = 0; for (i = 0; i < sizeof(all_printenv)/sizeof(all_printenv[0]); i++) { needed = found = 0; ret = WINSPOOL_EnumPrinterDrivers(pName, all_printenv[i]->envname, Level, pDriverInfo, total_found, bufsize, &needed, &found, data_offset); if (!ret && GetLastError() != ERROR_INSUFFICIENT_BUFFER) return FALSE; else if (ret) *pcReturned += found; *pcbNeeded = needed; data_offset = needed; total_found += found; } return ret; } /* Normal behavior */ ret = WINSPOOL_EnumPrinterDrivers(pName, pEnvironment, Level, pDriverInfo, 0, cbBuf, pcbNeeded, &found, 0); if (ret) *pcReturned = found; return ret; } /***************************************************************************** * EnumPrinterDriversA [WINSPOOL.@] * * see function EnumPrinterDrivers for RETURNS, BUGS */ BOOL WINAPI EnumPrinterDriversA(LPSTR pName, LPSTR pEnvironment, DWORD Level, LPBYTE pDriverInfo, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { BOOL ret; UNICODE_STRING pNameW, pEnvironmentW; PWSTR pwstrNameW, pwstrEnvironmentW; LPBYTE buf = NULL; if (cbBuf) buf = HeapAlloc(GetProcessHeap(), 0, cbBuf); pwstrNameW = asciitounicode(&pNameW, pName); pwstrEnvironmentW = asciitounicode(&pEnvironmentW, pEnvironment); ret = EnumPrinterDriversW(pwstrNameW, pwstrEnvironmentW, Level, buf, cbBuf, pcbNeeded, pcReturned); if (ret) convert_driverinfo_W_to_A(pDriverInfo, buf, Level, cbBuf, *pcReturned); HeapFree(GetProcessHeap(), 0, buf); RtlFreeUnicodeString(&pNameW); RtlFreeUnicodeString(&pEnvironmentW); return ret; } /****************************************************************************** * EnumPortsA (WINSPOOL.@) * * See EnumPortsW. * */ BOOL WINAPI EnumPortsA( LPSTR pName, DWORD Level, LPBYTE pPorts, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { BOOL res; LPBYTE bufferW = NULL; LPWSTR nameW = NULL; DWORD needed = 0; DWORD numentries = 0; INT len; TRACE("(%s, %d, %p, %d, %p, %p)\n", debugstr_a(pName), Level, pPorts, cbBuf, pcbNeeded, pcReturned); /* convert servername to unicode */ if (pName) { len = MultiByteToWideChar(CP_ACP, 0, pName, -1, NULL, 0); nameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pName, -1, nameW, len); } /* alloc (userbuffersize*sizeof(WCHAR) and try to enum the Ports */ needed = cbBuf * sizeof(WCHAR); if (needed) bufferW = HeapAlloc(GetProcessHeap(), 0, needed); res = EnumPortsW(nameW, Level, bufferW, needed, pcbNeeded, pcReturned); if(!res && (GetLastError() == ERROR_INSUFFICIENT_BUFFER)) { if (pcbNeeded) needed = *pcbNeeded; /* HeapReAlloc return NULL, when bufferW was NULL */ bufferW = (bufferW) ? HeapReAlloc(GetProcessHeap(), 0, bufferW, needed) : HeapAlloc(GetProcessHeap(), 0, needed); /* Try again with the large Buffer */ res = EnumPortsW(nameW, Level, bufferW, needed, pcbNeeded, pcReturned); } needed = pcbNeeded ? *pcbNeeded : 0; numentries = pcReturned ? *pcReturned : 0; /* W2k require the buffersize from EnumPortsW also for EnumPortsA. We use the smaller Ansi-Size to avoid conflicts with fixed Buffers of old Apps. */ if (res) { /* EnumPortsW collected all Data. Parse them to calculate ANSI-Size */ DWORD entrysize = 0; DWORD index; LPSTR ptr; LPPORT_INFO_2W pi2w; LPPORT_INFO_2A pi2a; needed = 0; entrysize = (Level == 1) ? sizeof(PORT_INFO_1A) : sizeof(PORT_INFO_2A); /* First pass: calculate the size for all Entries */ pi2w = (LPPORT_INFO_2W) bufferW; pi2a = (LPPORT_INFO_2A) pPorts; index = 0; while (index < numentries) { index++; needed += entrysize; /* PORT_INFO_?A */ TRACE("%p: parsing #%d (%s)\n", pi2w, index, debugstr_w(pi2w->pPortName)); needed += WideCharToMultiByte(CP_ACP, 0, pi2w->pPortName, -1, NULL, 0, NULL, NULL); if (Level > 1) { needed += WideCharToMultiByte(CP_ACP, 0, pi2w->pMonitorName, -1, NULL, 0, NULL, NULL); needed += WideCharToMultiByte(CP_ACP, 0, pi2w->pDescription, -1, NULL, 0, NULL, NULL); } /* use LPBYTE with entrysize to avoid double code (PORT_INFO_1 + PORT_INFO_2) */ pi2w = (LPPORT_INFO_2W) (((LPBYTE)pi2w) + entrysize); pi2a = (LPPORT_INFO_2A) (((LPBYTE)pi2a) + entrysize); } /* check for errors and quit on failure */ if (cbBuf < needed) { SetLastError(ERROR_INSUFFICIENT_BUFFER); res = FALSE; goto cleanup; } len = entrysize * numentries; /* room for all PORT_INFO_?A */ ptr = (LPSTR) &pPorts[len]; /* room for strings */ cbBuf -= len ; /* free Bytes in the user-Buffer */ pi2w = (LPPORT_INFO_2W) bufferW; pi2a = (LPPORT_INFO_2A) pPorts; index = 0; /* Second Pass: Fill the User Buffer (if we have one) */ while ((index < numentries) && pPorts) { index++; TRACE("%p: writing PORT_INFO_%dA #%d\n", pi2a, Level, index); pi2a->pPortName = ptr; len = WideCharToMultiByte(CP_ACP, 0, pi2w->pPortName, -1, ptr, cbBuf , NULL, NULL); ptr += len; cbBuf -= len; if (Level > 1) { pi2a->pMonitorName = ptr; len = WideCharToMultiByte(CP_ACP, 0, pi2w->pMonitorName, -1, ptr, cbBuf, NULL, NULL); ptr += len; cbBuf -= len; pi2a->pDescription = ptr; len = WideCharToMultiByte(CP_ACP, 0, pi2w->pDescription, -1, ptr, cbBuf, NULL, NULL); ptr += len; cbBuf -= len; pi2a->fPortType = pi2w->fPortType; pi2a->Reserved = 0; /* documented: "must be zero" */ } /* use LPBYTE with entrysize to avoid double code (PORT_INFO_1 + PORT_INFO_2) */ pi2w = (LPPORT_INFO_2W) (((LPBYTE)pi2w) + entrysize); pi2a = (LPPORT_INFO_2A) (((LPBYTE)pi2a) + entrysize); } } cleanup: if (pcbNeeded) *pcbNeeded = needed; if (pcReturned) *pcReturned = (res) ? numentries : 0; HeapFree(GetProcessHeap(), 0, nameW); HeapFree(GetProcessHeap(), 0, bufferW); TRACE("returning %d with %d (%d byte for %d of %d entries)\n", (res), GetLastError(), needed, (res)? numentries : 0, numentries); return (res); } /****************************************************************************** * EnumPortsW (WINSPOOL.@) * * Enumerate available Ports * * PARAMS * pName [I] Servername or NULL (local Computer) * Level [I] Structure-Level (1 or 2) * pPorts [O] PTR to Buffer that receives the Result * cbBuf [I] Size of Buffer at pPorts * pcbNeeded [O] PTR to DWORD that receives the size in Bytes used / required for pPorts * pcReturned [O] PTR to DWORD that receives the number of Ports in pPorts * * RETURNS * Success: TRUE * Failure: FALSE and in pcbNeeded the Bytes required for pPorts, if cbBuf is too small * */ BOOL WINAPI EnumPortsW(LPWSTR pName, DWORD Level, LPBYTE pPorts, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { TRACE("(%s, %d, %p, %d, %p, %p)\n", debugstr_w(pName), Level, pPorts, cbBuf, pcbNeeded, pcReturned); if ((backend == NULL) && !load_backend()) return FALSE; /* Level is not checked in win9x */ if (!Level || (Level > 2)) { WARN("level (%d) is ignored in win9x\n", Level); SetLastError(ERROR_INVALID_LEVEL); return FALSE; } if (!pcbNeeded || (!pPorts && (cbBuf > 0))) { SetLastError(RPC_X_NULL_REF_POINTER); return FALSE; } return backend->fpEnumPorts(pName, Level, pPorts, cbBuf, pcbNeeded, pcReturned); } /****************************************************************************** * GetDefaultPrinterW (WINSPOOL.@) * * FIXME * This function must read the value from data 'device' of key * HCU\\Software\\Microsoft\\Windows NT\\CurrentVersion\\Windows */ BOOL WINAPI GetDefaultPrinterW(LPWSTR name, LPDWORD namesize) { BOOL retval = TRUE; DWORD insize, len; WCHAR *buffer, *ptr; if (!namesize) { SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } /* make the buffer big enough for the stuff from the profile/registry, * the content must fit into the local buffer to compute the correct * size even if the extern buffer is too small or not given. * (20 for ,driver,port) */ insize = *namesize; len = max(100, (insize + 20)); buffer = HeapAlloc( GetProcessHeap(), 0, len * sizeof(WCHAR)); if (!GetProfileStringW(windowsW, deviceW, emptyStringW, buffer, len)) { SetLastError (ERROR_FILE_NOT_FOUND); retval = FALSE; goto end; } TRACE("%s\n", debugstr_w(buffer)); if ((ptr = strchrW(buffer, ',')) == NULL) { SetLastError(ERROR_INVALID_NAME); retval = FALSE; goto end; } *ptr = 0; *namesize = strlenW(buffer) + 1; if(!name || (*namesize > insize)) { SetLastError(ERROR_INSUFFICIENT_BUFFER); retval = FALSE; goto end; } strcpyW(name, buffer); end: HeapFree( GetProcessHeap(), 0, buffer); return retval; } /****************************************************************************** * GetDefaultPrinterA (WINSPOOL.@) */ BOOL WINAPI GetDefaultPrinterA(LPSTR name, LPDWORD namesize) { BOOL retval = TRUE; DWORD insize = 0; WCHAR *bufferW = NULL; if (!namesize) { SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } if(name && *namesize) { insize = *namesize; bufferW = HeapAlloc( GetProcessHeap(), 0, insize * sizeof(WCHAR)); } if(!GetDefaultPrinterW( bufferW, namesize)) { retval = FALSE; goto end; } *namesize = WideCharToMultiByte(CP_ACP, 0, bufferW, -1, name, insize, NULL, NULL); if (!*namesize) { *namesize = WideCharToMultiByte(CP_ACP, 0, bufferW, -1, NULL, 0, NULL, NULL); retval = FALSE; } TRACE("0x%08x/0x%08x:%s\n", *namesize, insize, debugstr_w(bufferW)); end: HeapFree( GetProcessHeap(), 0, bufferW); return retval; } /****************************************************************************** * SetDefaultPrinterW (WINSPOOL.204) * * Set the Name of the Default Printer * * PARAMS * pszPrinter [I] Name of the Printer or NULL * * RETURNS * Success: True * Failure: FALSE * * NOTES * When the Parameter is NULL or points to an Empty String and * a Default Printer was already present, then this Function changes nothing. * Without a Default Printer and NULL (or an Empty String) as Parameter, * the First enumerated local Printer is used. * */ BOOL WINAPI SetDefaultPrinterW(LPCWSTR pszPrinter) { WCHAR default_printer[MAX_PATH]; LPWSTR buffer = NULL; HKEY hreg; DWORD size; DWORD namelen; LONG lres; TRACE("(%s)\n", debugstr_w(pszPrinter)); if ((pszPrinter == NULL) || (pszPrinter[0] == '\0')) { default_printer[0] = '\0'; size = sizeof(default_printer)/sizeof(WCHAR); /* if we have a default Printer, do nothing. */ if (GetDefaultPrinterW(default_printer, &size)) return TRUE; pszPrinter = NULL; /* we have no default Printer: search local Printers and use the first */ if (!RegOpenKeyExW(HKEY_LOCAL_MACHINE, PrintersW, 0, KEY_READ, &hreg)) { default_printer[0] = '\0'; size = sizeof(default_printer)/sizeof(WCHAR); if (!RegEnumKeyExW(hreg, 0, default_printer, &size, NULL, NULL, NULL, NULL)) { pszPrinter = default_printer; TRACE("using %s\n", debugstr_w(pszPrinter)); } RegCloseKey(hreg); } if (pszPrinter == NULL) { TRACE("no local printer found\n"); SetLastError(ERROR_FILE_NOT_FOUND); return FALSE; } } /* "pszPrinter" is never empty or NULL here. */ namelen = lstrlenW(pszPrinter); size = namelen + (MAX_PATH * 2) + 3; /* printer,driver,port and a 0 */ buffer = HeapAlloc(GetProcessHeap(), 0, size * sizeof(WCHAR)); if (!buffer || (RegOpenKeyExW(HKEY_CURRENT_USER, user_printers_reg_key, 0, KEY_READ, &hreg) != ERROR_SUCCESS)) { HeapFree(GetProcessHeap(), 0, buffer); SetLastError(ERROR_FILE_NOT_FOUND); return FALSE; } /* read the devices entry for the printer (driver,port) to build the string for the default device entry (printer,driver,port) */ memcpy(buffer, pszPrinter, namelen * sizeof(WCHAR)); buffer[namelen] = ','; namelen++; /* move index to the start of the driver */ size = ((MAX_PATH * 2) + 2) * sizeof(WCHAR); /* driver,port and a 0 */ lres = RegQueryValueExW(hreg, pszPrinter, NULL, NULL, (LPBYTE) (&buffer[namelen]), &size); if (!lres) { TRACE("set device to %s\n", debugstr_w(buffer)); if (!WriteProfileStringW(windowsW, deviceW, buffer)) { TRACE("failed to set the device entry: %d\n", GetLastError()); lres = ERROR_INVALID_PRINTER_NAME; } /* remove the next section, when INIFileMapping is implemented */ { HKEY hdev; if (!RegCreateKeyW(HKEY_CURRENT_USER, user_default_reg_key, &hdev)) { RegSetValueExW(hdev, deviceW, 0, REG_SZ, (LPBYTE)buffer, (lstrlenW(buffer) + 1) * sizeof(WCHAR)); RegCloseKey(hdev); } } } else { if (lres != ERROR_FILE_NOT_FOUND) FIXME("RegQueryValueExW failed with %d for %s\n", lres, debugstr_w(pszPrinter)); SetLastError(ERROR_INVALID_PRINTER_NAME); } RegCloseKey(hreg); HeapFree(GetProcessHeap(), 0, buffer); return (lres == ERROR_SUCCESS); } /****************************************************************************** * SetDefaultPrinterA (WINSPOOL.202) * * See SetDefaultPrinterW. * */ BOOL WINAPI SetDefaultPrinterA(LPCSTR pszPrinter) { LPWSTR bufferW = NULL; BOOL res; TRACE("(%s)\n", debugstr_a(pszPrinter)); if(pszPrinter) { INT len = MultiByteToWideChar(CP_ACP, 0, pszPrinter, -1, NULL, 0); bufferW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); if (bufferW) MultiByteToWideChar(CP_ACP, 0, pszPrinter, -1, bufferW, len); } res = SetDefaultPrinterW(bufferW); HeapFree(GetProcessHeap(), 0, bufferW); return res; } /****************************************************************************** * SetPrinterDataExA (WINSPOOL.@) */ DWORD WINAPI SetPrinterDataExA(HANDLE hPrinter, LPCSTR pKeyName, LPCSTR pValueName, DWORD Type, LPBYTE pData, DWORD cbData) { HKEY hkeyPrinter, hkeySubkey; DWORD ret; TRACE("(%p, %s, %s %08x, %p, %08x)\n", hPrinter, debugstr_a(pKeyName), debugstr_a(pValueName), Type, pData, cbData); if((ret = WINSPOOL_GetOpenedPrinterRegKey(hPrinter, &hkeyPrinter)) != ERROR_SUCCESS) return ret; if((ret = RegCreateKeyA(hkeyPrinter, pKeyName, &hkeySubkey)) != ERROR_SUCCESS) { ERR("Can't create subkey %s\n", debugstr_a(pKeyName)); RegCloseKey(hkeyPrinter); return ret; } ret = RegSetValueExA(hkeySubkey, pValueName, 0, Type, pData, cbData); RegCloseKey(hkeySubkey); RegCloseKey(hkeyPrinter); return ret; } /****************************************************************************** * SetPrinterDataExW (WINSPOOL.@) */ DWORD WINAPI SetPrinterDataExW(HANDLE hPrinter, LPCWSTR pKeyName, LPCWSTR pValueName, DWORD Type, LPBYTE pData, DWORD cbData) { HKEY hkeyPrinter, hkeySubkey; DWORD ret; TRACE("(%p, %s, %s %08x, %p, %08x)\n", hPrinter, debugstr_w(pKeyName), debugstr_w(pValueName), Type, pData, cbData); if((ret = WINSPOOL_GetOpenedPrinterRegKey(hPrinter, &hkeyPrinter)) != ERROR_SUCCESS) return ret; if((ret = RegCreateKeyW(hkeyPrinter, pKeyName, &hkeySubkey)) != ERROR_SUCCESS) { ERR("Can't create subkey %s\n", debugstr_w(pKeyName)); RegCloseKey(hkeyPrinter); return ret; } ret = RegSetValueExW(hkeySubkey, pValueName, 0, Type, pData, cbData); RegCloseKey(hkeySubkey); RegCloseKey(hkeyPrinter); return ret; } /****************************************************************************** * SetPrinterDataA (WINSPOOL.@) */ DWORD WINAPI SetPrinterDataA(HANDLE hPrinter, LPSTR pValueName, DWORD Type, LPBYTE pData, DWORD cbData) { return SetPrinterDataExA(hPrinter, "PrinterDriverData", pValueName, Type, pData, cbData); } /****************************************************************************** * SetPrinterDataW (WINSPOOL.@) */ DWORD WINAPI SetPrinterDataW(HANDLE hPrinter, LPWSTR pValueName, DWORD Type, LPBYTE pData, DWORD cbData) { return SetPrinterDataExW(hPrinter, PrinterDriverDataW, pValueName, Type, pData, cbData); } /****************************************************************************** * GetPrinterDataExA (WINSPOOL.@) */ DWORD WINAPI GetPrinterDataExA(HANDLE hPrinter, LPCSTR pKeyName, LPCSTR pValueName, LPDWORD pType, LPBYTE pData, DWORD nSize, LPDWORD pcbNeeded) { opened_printer_t *printer; HKEY hkeyPrinters, hkeyPrinter = 0, hkeySubkey = 0; DWORD ret; TRACE("(%p, %s, %s, %p, %p, %u, %p)\n", hPrinter, debugstr_a(pKeyName), debugstr_a(pValueName), pType, pData, nSize, pcbNeeded); printer = get_opened_printer(hPrinter); if(!printer) return ERROR_INVALID_HANDLE; ret = RegOpenKeyW(HKEY_LOCAL_MACHINE, PrintersW, &hkeyPrinters); if (ret) return ret; TRACE("printer->name: %s\n", debugstr_w(printer->name)); if (printer->name) { ret = RegOpenKeyW(hkeyPrinters, printer->name, &hkeyPrinter); if (ret) { RegCloseKey(hkeyPrinters); return ret; } if((ret = RegOpenKeyA(hkeyPrinter, pKeyName, &hkeySubkey)) != ERROR_SUCCESS) { WARN("Can't open subkey %s: %d\n", debugstr_a(pKeyName), ret); RegCloseKey(hkeyPrinter); RegCloseKey(hkeyPrinters); return ret; } } *pcbNeeded = nSize; ret = RegQueryValueExA(printer->name ? hkeySubkey : hkeyPrinters, pValueName, 0, pType, pData, pcbNeeded); if (!ret && !pData) ret = ERROR_MORE_DATA; RegCloseKey(hkeySubkey); RegCloseKey(hkeyPrinter); RegCloseKey(hkeyPrinters); TRACE("--> %d\n", ret); return ret; } /****************************************************************************** * GetPrinterDataExW (WINSPOOL.@) */ DWORD WINAPI GetPrinterDataExW(HANDLE hPrinter, LPCWSTR pKeyName, LPCWSTR pValueName, LPDWORD pType, LPBYTE pData, DWORD nSize, LPDWORD pcbNeeded) { opened_printer_t *printer; HKEY hkeyPrinters, hkeyPrinter = 0, hkeySubkey = 0; DWORD ret; TRACE("(%p, %s, %s, %p, %p, %u, %p)\n", hPrinter, debugstr_w(pKeyName), debugstr_w(pValueName), pType, pData, nSize, pcbNeeded); printer = get_opened_printer(hPrinter); if(!printer) return ERROR_INVALID_HANDLE; ret = RegOpenKeyW(HKEY_LOCAL_MACHINE, PrintersW, &hkeyPrinters); if (ret) return ret; TRACE("printer->name: %s\n", debugstr_w(printer->name)); if (printer->name) { ret = RegOpenKeyW(hkeyPrinters, printer->name, &hkeyPrinter); if (ret) { RegCloseKey(hkeyPrinters); return ret; } if((ret = RegOpenKeyW(hkeyPrinter, pKeyName, &hkeySubkey)) != ERROR_SUCCESS) { WARN("Can't open subkey %s: %d\n", debugstr_w(pKeyName), ret); RegCloseKey(hkeyPrinter); RegCloseKey(hkeyPrinters); return ret; } } *pcbNeeded = nSize; ret = RegQueryValueExW(printer->name ? hkeySubkey : hkeyPrinters, pValueName, 0, pType, pData, pcbNeeded); if (!ret && !pData) ret = ERROR_MORE_DATA; RegCloseKey(hkeySubkey); RegCloseKey(hkeyPrinter); RegCloseKey(hkeyPrinters); TRACE("--> %d\n", ret); return ret; } /****************************************************************************** * GetPrinterDataA (WINSPOOL.@) */ DWORD WINAPI GetPrinterDataA(HANDLE hPrinter, LPSTR pValueName, LPDWORD pType, LPBYTE pData, DWORD nSize, LPDWORD pcbNeeded) { return GetPrinterDataExA(hPrinter, "PrinterDriverData", pValueName, pType, pData, nSize, pcbNeeded); } /****************************************************************************** * GetPrinterDataW (WINSPOOL.@) */ DWORD WINAPI GetPrinterDataW(HANDLE hPrinter, LPWSTR pValueName, LPDWORD pType, LPBYTE pData, DWORD nSize, LPDWORD pcbNeeded) { return GetPrinterDataExW(hPrinter, PrinterDriverDataW, pValueName, pType, pData, nSize, pcbNeeded); } /******************************************************************************* * EnumPrinterDataExW [WINSPOOL.@] */ DWORD WINAPI EnumPrinterDataExW(HANDLE hPrinter, LPCWSTR pKeyName, LPBYTE pEnumValues, DWORD cbEnumValues, LPDWORD pcbEnumValues, LPDWORD pnEnumValues) { HKEY hkPrinter, hkSubKey; DWORD r, ret, dwIndex, cValues, cbMaxValueNameLen, cbValueNameLen, cbMaxValueLen, cbValueLen, cbBufSize, dwType; LPWSTR lpValueName; HANDLE hHeap; PBYTE lpValue; PPRINTER_ENUM_VALUESW ppev; TRACE ("%p %s\n", hPrinter, debugstr_w (pKeyName)); if (pKeyName == NULL || *pKeyName == 0) return ERROR_INVALID_PARAMETER; ret = WINSPOOL_GetOpenedPrinterRegKey (hPrinter, &hkPrinter); if (ret != ERROR_SUCCESS) { TRACE ("WINSPOOL_GetOpenedPrinterRegKey (%p) returned %i\n", hPrinter, ret); return ret; } ret = RegOpenKeyExW (hkPrinter, pKeyName, 0, KEY_READ, &hkSubKey); if (ret != ERROR_SUCCESS) { r = RegCloseKey (hkPrinter); if (r != ERROR_SUCCESS) WARN ("RegCloseKey returned %i\n", r); TRACE ("RegOpenKeyExW (%p, %s) returned %i\n", hPrinter, debugstr_w (pKeyName), ret); return ret; } ret = RegCloseKey (hkPrinter); if (ret != ERROR_SUCCESS) { ERR ("RegCloseKey returned %i\n", ret); r = RegCloseKey (hkSubKey); if (r != ERROR_SUCCESS) WARN ("RegCloseKey returned %i\n", r); return ret; } ret = RegQueryInfoKeyW (hkSubKey, NULL, NULL, NULL, NULL, NULL, NULL, &cValues, &cbMaxValueNameLen, &cbMaxValueLen, NULL, NULL); if (ret != ERROR_SUCCESS) { r = RegCloseKey (hkSubKey); if (r != ERROR_SUCCESS) WARN ("RegCloseKey returned %i\n", r); TRACE ("RegQueryInfoKeyW (%p) returned %i\n", hkSubKey, ret); return ret; } TRACE ("RegQueryInfoKeyW returned cValues = %i, cbMaxValueNameLen = %i, " "cbMaxValueLen = %i\n", cValues, cbMaxValueNameLen, cbMaxValueLen); if (cValues == 0) /* empty key */ { r = RegCloseKey (hkSubKey); if (r != ERROR_SUCCESS) WARN ("RegCloseKey returned %i\n", r); *pcbEnumValues = *pnEnumValues = 0; return ERROR_SUCCESS; } ++cbMaxValueNameLen; /* allow for trailing '\0' */ hHeap = GetProcessHeap (); if (hHeap == NULL) { ERR ("GetProcessHeap failed\n"); r = RegCloseKey (hkSubKey); if (r != ERROR_SUCCESS) WARN ("RegCloseKey returned %i\n", r); return ERROR_OUTOFMEMORY; } lpValueName = HeapAlloc (hHeap, 0, cbMaxValueNameLen * sizeof (WCHAR)); if (lpValueName == NULL) { ERR ("Failed to allocate %i WCHARs from process heap\n", cbMaxValueNameLen); r = RegCloseKey (hkSubKey); if (r != ERROR_SUCCESS) WARN ("RegCloseKey returned %i\n", r); return ERROR_OUTOFMEMORY; } lpValue = HeapAlloc (hHeap, 0, cbMaxValueLen); if (lpValue == NULL) { ERR ("Failed to allocate %i bytes from process heap\n", cbMaxValueLen); if (HeapFree (hHeap, 0, lpValueName) == 0) WARN ("HeapFree failed with code %i\n", GetLastError ()); r = RegCloseKey (hkSubKey); if (r != ERROR_SUCCESS) WARN ("RegCloseKey returned %i\n", r); return ERROR_OUTOFMEMORY; } TRACE ("pass 1: calculating buffer required for all names and values\n"); cbBufSize = cValues * sizeof (PRINTER_ENUM_VALUESW); TRACE ("%i bytes required for %i headers\n", cbBufSize, cValues); for (dwIndex = 0; dwIndex < cValues; ++dwIndex) { cbValueNameLen = cbMaxValueNameLen; cbValueLen = cbMaxValueLen; ret = RegEnumValueW (hkSubKey, dwIndex, lpValueName, &cbValueNameLen, NULL, NULL, lpValue, &cbValueLen); if (ret != ERROR_SUCCESS) { if (HeapFree (hHeap, 0, lpValue) == 0) WARN ("HeapFree failed with code %i\n", GetLastError ()); if (HeapFree (hHeap, 0, lpValueName) == 0) WARN ("HeapFree failed with code %i\n", GetLastError ()); r = RegCloseKey (hkSubKey); if (r != ERROR_SUCCESS) WARN ("RegCloseKey returned %i\n", r); TRACE ("RegEnumValueW (%i) returned %i\n", dwIndex, ret); return ret; } TRACE ("%s [%i]: name needs %i WCHARs, data needs %i bytes\n", debugstr_w (lpValueName), dwIndex, cbValueNameLen + 1, cbValueLen); cbBufSize += (cbValueNameLen + 1) * sizeof (WCHAR); cbBufSize += cbValueLen; } TRACE ("%i bytes required for all %i values\n", cbBufSize, cValues); *pcbEnumValues = cbBufSize; *pnEnumValues = cValues; if (cbEnumValues < cbBufSize) /* buffer too small */ { if (HeapFree (hHeap, 0, lpValue) == 0) WARN ("HeapFree failed with code %i\n", GetLastError ()); if (HeapFree (hHeap, 0, lpValueName) == 0) WARN ("HeapFree failed with code %i\n", GetLastError ()); r = RegCloseKey (hkSubKey); if (r != ERROR_SUCCESS) WARN ("RegCloseKey returned %i\n", r); TRACE ("%i byte buffer is not large enough\n", cbEnumValues); return ERROR_MORE_DATA; } TRACE ("pass 2: copying all names and values to buffer\n"); ppev = (PPRINTER_ENUM_VALUESW) pEnumValues; /* array of structs */ pEnumValues += cValues * sizeof (PRINTER_ENUM_VALUESW); for (dwIndex = 0; dwIndex < cValues; ++dwIndex) { cbValueNameLen = cbMaxValueNameLen; cbValueLen = cbMaxValueLen; ret = RegEnumValueW (hkSubKey, dwIndex, lpValueName, &cbValueNameLen, NULL, &dwType, lpValue, &cbValueLen); if (ret != ERROR_SUCCESS) { if (HeapFree (hHeap, 0, lpValue) == 0) WARN ("HeapFree failed with code %i\n", GetLastError ()); if (HeapFree (hHeap, 0, lpValueName) == 0) WARN ("HeapFree failed with code %i\n", GetLastError ()); r = RegCloseKey (hkSubKey); if (r != ERROR_SUCCESS) WARN ("RegCloseKey returned %i\n", r); TRACE ("RegEnumValueW (%i) returned %i\n", dwIndex, ret); return ret; } cbValueNameLen = (cbValueNameLen + 1) * sizeof (WCHAR); memcpy (pEnumValues, lpValueName, cbValueNameLen); ppev[dwIndex].pValueName = (LPWSTR) pEnumValues; pEnumValues += cbValueNameLen; /* return # of *bytes* (including trailing \0), not # of chars */ ppev[dwIndex].cbValueName = cbValueNameLen; ppev[dwIndex].dwType = dwType; memcpy (pEnumValues, lpValue, cbValueLen); ppev[dwIndex].pData = pEnumValues; pEnumValues += cbValueLen; ppev[dwIndex].cbData = cbValueLen; TRACE ("%s [%i]: copied name (%i bytes) and data (%i bytes)\n", debugstr_w (lpValueName), dwIndex, cbValueNameLen, cbValueLen); } if (HeapFree (hHeap, 0, lpValue) == 0) { ret = GetLastError (); ERR ("HeapFree failed with code %i\n", ret); if (HeapFree (hHeap, 0, lpValueName) == 0) WARN ("HeapFree failed with code %i\n", GetLastError ()); r = RegCloseKey (hkSubKey); if (r != ERROR_SUCCESS) WARN ("RegCloseKey returned %i\n", r); return ret; } if (HeapFree (hHeap, 0, lpValueName) == 0) { ret = GetLastError (); ERR ("HeapFree failed with code %i\n", ret); r = RegCloseKey (hkSubKey); if (r != ERROR_SUCCESS) WARN ("RegCloseKey returned %i\n", r); return ret; } ret = RegCloseKey (hkSubKey); if (ret != ERROR_SUCCESS) { ERR ("RegCloseKey returned %i\n", ret); return ret; } return ERROR_SUCCESS; } /******************************************************************************* * EnumPrinterDataExA [WINSPOOL.@] * * This functions returns value names and REG_SZ, REG_EXPAND_SZ, and * REG_MULTI_SZ values as ASCII strings in Unicode-sized buffers. This is * what Windows 2000 SP1 does. * */ DWORD WINAPI EnumPrinterDataExA(HANDLE hPrinter, LPCSTR pKeyName, LPBYTE pEnumValues, DWORD cbEnumValues, LPDWORD pcbEnumValues, LPDWORD pnEnumValues) { INT len; LPWSTR pKeyNameW; DWORD ret, dwIndex, dwBufSize; HANDLE hHeap; LPSTR pBuffer; TRACE ("%p %s\n", hPrinter, pKeyName); if (pKeyName == NULL || *pKeyName == 0) return ERROR_INVALID_PARAMETER; len = MultiByteToWideChar (CP_ACP, 0, pKeyName, -1, NULL, 0); if (len == 0) { ret = GetLastError (); ERR ("MultiByteToWideChar failed with code %i\n", ret); return ret; } hHeap = GetProcessHeap (); if (hHeap == NULL) { ERR ("GetProcessHeap failed\n"); return ERROR_OUTOFMEMORY; } pKeyNameW = HeapAlloc (hHeap, 0, len * sizeof (WCHAR)); if (pKeyNameW == NULL) { ERR ("Failed to allocate %i bytes from process heap\n", (LONG)(len * sizeof (WCHAR))); return ERROR_OUTOFMEMORY; } if (MultiByteToWideChar (CP_ACP, 0, pKeyName, -1, pKeyNameW, len) == 0) { ret = GetLastError (); ERR ("MultiByteToWideChar failed with code %i\n", ret); if (HeapFree (hHeap, 0, pKeyNameW) == 0) WARN ("HeapFree failed with code %i\n", GetLastError ()); return ret; } ret = EnumPrinterDataExW (hPrinter, pKeyNameW, pEnumValues, cbEnumValues, pcbEnumValues, pnEnumValues); if (ret != ERROR_SUCCESS) { if (HeapFree (hHeap, 0, pKeyNameW) == 0) WARN ("HeapFree failed with code %i\n", GetLastError ()); TRACE ("EnumPrinterDataExW returned %i\n", ret); return ret; } if (HeapFree (hHeap, 0, pKeyNameW) == 0) { ret = GetLastError (); ERR ("HeapFree failed with code %i\n", ret); return ret; } if (*pnEnumValues == 0) /* empty key */ return ERROR_SUCCESS; dwBufSize = 0; for (dwIndex = 0; dwIndex < *pnEnumValues; ++dwIndex) { PPRINTER_ENUM_VALUESW ppev = &((PPRINTER_ENUM_VALUESW) pEnumValues)[dwIndex]; if (dwBufSize < ppev->cbValueName) dwBufSize = ppev->cbValueName; if (dwBufSize < ppev->cbData && (ppev->dwType == REG_SZ || ppev->dwType == REG_EXPAND_SZ || ppev->dwType == REG_MULTI_SZ)) dwBufSize = ppev->cbData; } TRACE ("Largest Unicode name or value is %i bytes\n", dwBufSize); pBuffer = HeapAlloc (hHeap, 0, dwBufSize); if (pBuffer == NULL) { ERR ("Failed to allocate %i bytes from process heap\n", dwBufSize); return ERROR_OUTOFMEMORY; } for (dwIndex = 0; dwIndex < *pnEnumValues; ++dwIndex) { PPRINTER_ENUM_VALUESW ppev = &((PPRINTER_ENUM_VALUESW) pEnumValues)[dwIndex]; len = WideCharToMultiByte (CP_ACP, 0, ppev->pValueName, ppev->cbValueName / sizeof (WCHAR), pBuffer, dwBufSize, NULL, NULL); if (len == 0) { ret = GetLastError (); ERR ("WideCharToMultiByte failed with code %i\n", ret); if (HeapFree (hHeap, 0, pBuffer) == 0) WARN ("HeapFree failed with code %i\n", GetLastError ()); return ret; } memcpy (ppev->pValueName, pBuffer, len); TRACE ("Converted '%s' from Unicode to ASCII\n", pBuffer); if (ppev->dwType != REG_SZ && ppev->dwType != REG_EXPAND_SZ && ppev->dwType != REG_MULTI_SZ) continue; len = WideCharToMultiByte (CP_ACP, 0, (LPWSTR) ppev->pData, ppev->cbData / sizeof (WCHAR), pBuffer, dwBufSize, NULL, NULL); if (len == 0) { ret = GetLastError (); ERR ("WideCharToMultiByte failed with code %i\n", ret); if (HeapFree (hHeap, 0, pBuffer) == 0) WARN ("HeapFree failed with code %i\n", GetLastError ()); return ret; } memcpy (ppev->pData, pBuffer, len); TRACE ("Converted '%s' from Unicode to ASCII\n", pBuffer); TRACE (" (only first string of REG_MULTI_SZ printed)\n"); } if (HeapFree (hHeap, 0, pBuffer) == 0) { ret = GetLastError (); ERR ("HeapFree failed with code %i\n", ret); return ret; } return ERROR_SUCCESS; } /****************************************************************************** * AbortPrinter (WINSPOOL.@) */ BOOL WINAPI AbortPrinter( HANDLE hPrinter ) { FIXME("(%p), stub!\n", hPrinter); return TRUE; } /****************************************************************************** * AddPortA (WINSPOOL.@) * * See AddPortW. * */ BOOL WINAPI AddPortA(LPSTR pName, HWND hWnd, LPSTR pMonitorName) { LPWSTR nameW = NULL; LPWSTR monitorW = NULL; DWORD len; BOOL res; TRACE("(%s, %p, %s)\n",debugstr_a(pName), hWnd, debugstr_a(pMonitorName)); if (pName) { len = MultiByteToWideChar(CP_ACP, 0, pName, -1, NULL, 0); nameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pName, -1, nameW, len); } if (pMonitorName) { len = MultiByteToWideChar(CP_ACP, 0, pMonitorName, -1, NULL, 0); monitorW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pMonitorName, -1, monitorW, len); } res = AddPortW(nameW, hWnd, monitorW); HeapFree(GetProcessHeap(), 0, nameW); HeapFree(GetProcessHeap(), 0, monitorW); return res; } /****************************************************************************** * AddPortW (WINSPOOL.@) * * Add a Port for a specific Monitor * * PARAMS * pName [I] Servername or NULL (local Computer) * hWnd [I] Handle to parent Window for the Dialog-Box * pMonitorName [I] Name of the Monitor that manage the Port * * RETURNS * Success: TRUE * Failure: FALSE * */ BOOL WINAPI AddPortW(LPWSTR pName, HWND hWnd, LPWSTR pMonitorName) { TRACE("(%s, %p, %s)\n", debugstr_w(pName), hWnd, debugstr_w(pMonitorName)); if ((backend == NULL) && !load_backend()) return FALSE; if (!pMonitorName) { SetLastError(RPC_X_NULL_REF_POINTER); return FALSE; } return backend->fpAddPort(pName, hWnd, pMonitorName); } /****************************************************************************** * AddPortExA (WINSPOOL.@) * * See AddPortExW. * */ BOOL WINAPI AddPortExA(LPSTR pName, DWORD level, LPBYTE pBuffer, LPSTR pMonitorName) { PORT_INFO_2W pi2W; PORT_INFO_2A * pi2A; LPWSTR nameW = NULL; LPWSTR monitorW = NULL; DWORD len; BOOL res; pi2A = (PORT_INFO_2A *) pBuffer; TRACE("(%s, %d, %p, %s): %s\n", debugstr_a(pName), level, pBuffer, debugstr_a(pMonitorName), debugstr_a(pi2A ? pi2A->pPortName : NULL)); if ((level < 1) || (level > 2)) { SetLastError(ERROR_INVALID_LEVEL); return FALSE; } if (!pi2A) { SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } if (pName) { len = MultiByteToWideChar(CP_ACP, 0, pName, -1, NULL, 0); nameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pName, -1, nameW, len); } if (pMonitorName) { len = MultiByteToWideChar(CP_ACP, 0, pMonitorName, -1, NULL, 0); monitorW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pMonitorName, -1, monitorW, len); } ZeroMemory(&pi2W, sizeof(PORT_INFO_2W)); if (pi2A->pPortName) { len = MultiByteToWideChar(CP_ACP, 0, pi2A->pPortName, -1, NULL, 0); pi2W.pPortName = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pi2A->pPortName, -1, pi2W.pPortName, len); } if (level > 1) { if (pi2A->pMonitorName) { len = MultiByteToWideChar(CP_ACP, 0, pi2A->pMonitorName, -1, NULL, 0); pi2W.pMonitorName = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pi2A->pMonitorName, -1, pi2W.pMonitorName, len); } if (pi2A->pDescription) { len = MultiByteToWideChar(CP_ACP, 0, pi2A->pDescription, -1, NULL, 0); pi2W.pDescription = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pi2A->pDescription, -1, pi2W.pDescription, len); } pi2W.fPortType = pi2A->fPortType; pi2W.Reserved = pi2A->Reserved; } res = AddPortExW(nameW, level, (LPBYTE) &pi2W, monitorW); HeapFree(GetProcessHeap(), 0, nameW); HeapFree(GetProcessHeap(), 0, monitorW); HeapFree(GetProcessHeap(), 0, pi2W.pPortName); HeapFree(GetProcessHeap(), 0, pi2W.pMonitorName); HeapFree(GetProcessHeap(), 0, pi2W.pDescription); return res; } /****************************************************************************** * AddPortExW (WINSPOOL.@) * * Add a Port for a specific Monitor, without presenting a user interface * * PARAMS * pName [I] Servername or NULL (local Computer) * level [I] Structure-Level (1 or 2) for pBuffer * pBuffer [I] PTR to: PORT_INFO_1 or PORT_INFO_2 * pMonitorName [I] Name of the Monitor that manage the Port * * RETURNS * Success: TRUE * Failure: FALSE * */ BOOL WINAPI AddPortExW(LPWSTR pName, DWORD level, LPBYTE pBuffer, LPWSTR pMonitorName) { PORT_INFO_2W * pi2; pi2 = (PORT_INFO_2W *) pBuffer; TRACE("(%s, %d, %p, %s): %s %s %s\n", debugstr_w(pName), level, pBuffer, debugstr_w(pMonitorName), debugstr_w(pi2 ? pi2->pPortName : NULL), debugstr_w(((level > 1) && pi2) ? pi2->pMonitorName : NULL), debugstr_w(((level > 1) && pi2) ? pi2->pDescription : NULL)); if ((backend == NULL) && !load_backend()) return FALSE; if ((!pi2) || (!pMonitorName) || (!pMonitorName[0])) { SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } return backend->fpAddPortEx(pName, level, pBuffer, pMonitorName); } /****************************************************************************** * AddPrinterConnectionA (WINSPOOL.@) */ BOOL WINAPI AddPrinterConnectionA( LPSTR pName ) { FIXME("%s\n", debugstr_a(pName)); return FALSE; } /****************************************************************************** * AddPrinterConnectionW (WINSPOOL.@) */ BOOL WINAPI AddPrinterConnectionW( LPWSTR pName ) { FIXME("%s\n", debugstr_w(pName)); return FALSE; } /****************************************************************************** * AddPrinterDriverExW (WINSPOOL.@) * * Install a Printer Driver with the Option to upgrade / downgrade the Files * * PARAMS * pName [I] Servername or NULL (local Computer) * level [I] Level for the supplied DRIVER_INFO_*W struct * pDriverInfo [I] PTR to DRIVER_INFO_*W struct with the Driver Parameter * dwFileCopyFlags [I] How to Copy / Upgrade / Downgrade the needed Files * * RESULTS * Success: TRUE * Failure: FALSE * */ BOOL WINAPI AddPrinterDriverExW( LPWSTR pName, DWORD level, LPBYTE pDriverInfo, DWORD dwFileCopyFlags) { TRACE("(%s, %d, %p, 0x%x)\n", debugstr_w(pName), level, pDriverInfo, dwFileCopyFlags); if ((backend == NULL) && !load_backend()) return FALSE; if (level < 2 || level == 5 || level == 7 || level > 8) { SetLastError(ERROR_INVALID_LEVEL); return FALSE; } if (!pDriverInfo) { SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } return backend->fpAddPrinterDriverEx(pName, level, pDriverInfo, dwFileCopyFlags); } /****************************************************************************** * AddPrinterDriverExA (WINSPOOL.@) * * See AddPrinterDriverExW. * */ BOOL WINAPI AddPrinterDriverExA(LPSTR pName, DWORD Level, LPBYTE pDriverInfo, DWORD dwFileCopyFlags) { DRIVER_INFO_8A *diA; DRIVER_INFO_8W diW; LPWSTR nameW = NULL; DWORD lenA; DWORD len; BOOL res = FALSE; TRACE("(%s, %d, %p, 0x%x)\n", debugstr_a(pName), Level, pDriverInfo, dwFileCopyFlags); diA = (DRIVER_INFO_8A *) pDriverInfo; ZeroMemory(&diW, sizeof(diW)); if (Level < 2 || Level == 5 || Level == 7 || Level > 8) { SetLastError(ERROR_INVALID_LEVEL); return FALSE; } if (diA == NULL) { SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } /* convert servername to unicode */ if (pName) { len = MultiByteToWideChar(CP_ACP, 0, pName, -1, NULL, 0); nameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pName, -1, nameW, len); } /* common fields */ diW.cVersion = diA->cVersion; if (diA->pName) { len = MultiByteToWideChar(CP_ACP, 0, diA->pName, -1, NULL, 0); diW.pName = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, diA->pName, -1, diW.pName, len); } if (diA->pEnvironment) { len = MultiByteToWideChar(CP_ACP, 0, diA->pEnvironment, -1, NULL, 0); diW.pEnvironment = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, diA->pEnvironment, -1, diW.pEnvironment, len); } if (diA->pDriverPath) { len = MultiByteToWideChar(CP_ACP, 0, diA->pDriverPath, -1, NULL, 0); diW.pDriverPath = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, diA->pDriverPath, -1, diW.pDriverPath, len); } if (diA->pDataFile) { len = MultiByteToWideChar(CP_ACP, 0, diA->pDataFile, -1, NULL, 0); diW.pDataFile = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, diA->pDataFile, -1, diW.pDataFile, len); } if (diA->pConfigFile) { len = MultiByteToWideChar(CP_ACP, 0, diA->pConfigFile, -1, NULL, 0); diW.pConfigFile = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, diA->pConfigFile, -1, diW.pConfigFile, len); } if ((Level > 2) && diA->pDependentFiles) { lenA = multi_sz_lenA(diA->pDependentFiles); len = MultiByteToWideChar(CP_ACP, 0, diA->pDependentFiles, lenA, NULL, 0); diW.pDependentFiles = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, diA->pDependentFiles, lenA, diW.pDependentFiles, len); } if ((Level > 2) && diA->pMonitorName) { len = MultiByteToWideChar(CP_ACP, 0, diA->pMonitorName, -1, NULL, 0); diW.pMonitorName = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, diA->pMonitorName, -1, diW.pMonitorName, len); } if ((Level > 3) && diA->pDefaultDataType) { len = MultiByteToWideChar(CP_ACP, 0, diA->pDefaultDataType, -1, NULL, 0); diW.pDefaultDataType = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, diA->pDefaultDataType, -1, diW.pDefaultDataType, len); } if ((Level > 3) && diA->pszzPreviousNames) { lenA = multi_sz_lenA(diA->pszzPreviousNames); len = MultiByteToWideChar(CP_ACP, 0, diA->pszzPreviousNames, lenA, NULL, 0); diW.pszzPreviousNames = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, diA->pszzPreviousNames, lenA, diW.pszzPreviousNames, len); } if ((Level > 5) && diA->pszMfgName) { len = MultiByteToWideChar(CP_ACP, 0, diA->pszMfgName, -1, NULL, 0); diW.pszMfgName = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, diA->pszMfgName, -1, diW.pszMfgName, len); } if ((Level > 5) && diA->pszOEMUrl) { len = MultiByteToWideChar(CP_ACP, 0, diA->pszOEMUrl, -1, NULL, 0); diW.pszOEMUrl = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, diA->pszOEMUrl, -1, diW.pszOEMUrl, len); } if ((Level > 5) && diA->pszHardwareID) { len = MultiByteToWideChar(CP_ACP, 0, diA->pszHardwareID, -1, NULL, 0); diW.pszHardwareID = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, diA->pszHardwareID, -1, diW.pszHardwareID, len); } if ((Level > 5) && diA->pszProvider) { len = MultiByteToWideChar(CP_ACP, 0, diA->pszProvider, -1, NULL, 0); diW.pszProvider = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, diA->pszProvider, -1, diW.pszProvider, len); } if (Level > 7) { FIXME("level %u is incomplete\n", Level); } res = AddPrinterDriverExW(nameW, Level, (LPBYTE) &diW, dwFileCopyFlags); TRACE("got %u with %u\n", res, GetLastError()); HeapFree(GetProcessHeap(), 0, nameW); HeapFree(GetProcessHeap(), 0, diW.pName); HeapFree(GetProcessHeap(), 0, diW.pEnvironment); HeapFree(GetProcessHeap(), 0, diW.pDriverPath); HeapFree(GetProcessHeap(), 0, diW.pDataFile); HeapFree(GetProcessHeap(), 0, diW.pConfigFile); HeapFree(GetProcessHeap(), 0, diW.pDependentFiles); HeapFree(GetProcessHeap(), 0, diW.pMonitorName); HeapFree(GetProcessHeap(), 0, diW.pDefaultDataType); HeapFree(GetProcessHeap(), 0, diW.pszzPreviousNames); HeapFree(GetProcessHeap(), 0, diW.pszMfgName); HeapFree(GetProcessHeap(), 0, diW.pszOEMUrl); HeapFree(GetProcessHeap(), 0, diW.pszHardwareID); HeapFree(GetProcessHeap(), 0, diW.pszProvider); TRACE("=> %u with %u\n", res, GetLastError()); return res; } /****************************************************************************** * ConfigurePortA (WINSPOOL.@) * * See ConfigurePortW. * */ BOOL WINAPI ConfigurePortA(LPSTR pName, HWND hWnd, LPSTR pPortName) { LPWSTR nameW = NULL; LPWSTR portW = NULL; INT len; DWORD res; TRACE("(%s, %p, %s)\n", debugstr_a(pName), hWnd, debugstr_a(pPortName)); /* convert servername to unicode */ if (pName) { len = MultiByteToWideChar(CP_ACP, 0, pName, -1, NULL, 0); nameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pName, -1, nameW, len); } /* convert portname to unicode */ if (pPortName) { len = MultiByteToWideChar(CP_ACP, 0, pPortName, -1, NULL, 0); portW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pPortName, -1, portW, len); } res = ConfigurePortW(nameW, hWnd, portW); HeapFree(GetProcessHeap(), 0, nameW); HeapFree(GetProcessHeap(), 0, portW); return res; } /****************************************************************************** * ConfigurePortW (WINSPOOL.@) * * Display the Configuration-Dialog for a specific Port * * PARAMS * pName [I] Servername or NULL (local Computer) * hWnd [I] Handle to parent Window for the Dialog-Box * pPortName [I] Name of the Port, that should be configured * * RETURNS * Success: TRUE * Failure: FALSE * */ BOOL WINAPI ConfigurePortW(LPWSTR pName, HWND hWnd, LPWSTR pPortName) { TRACE("(%s, %p, %s)\n", debugstr_w(pName), hWnd, debugstr_w(pPortName)); if ((backend == NULL) && !load_backend()) return FALSE; if (!pPortName) { SetLastError(RPC_X_NULL_REF_POINTER); return FALSE; } return backend->fpConfigurePort(pName, hWnd, pPortName); } /****************************************************************************** * ConnectToPrinterDlg (WINSPOOL.@) */ HANDLE WINAPI ConnectToPrinterDlg( HWND hWnd, DWORD Flags ) { FIXME("%p %x\n", hWnd, Flags); return NULL; } /****************************************************************************** * DeletePrinterConnectionA (WINSPOOL.@) */ BOOL WINAPI DeletePrinterConnectionA( LPSTR pName ) { FIXME("%s\n", debugstr_a(pName)); return TRUE; } /****************************************************************************** * DeletePrinterConnectionW (WINSPOOL.@) */ BOOL WINAPI DeletePrinterConnectionW( LPWSTR pName ) { FIXME("%s\n", debugstr_w(pName)); return TRUE; } /****************************************************************************** * DeletePrinterDriverExW (WINSPOOL.@) */ BOOL WINAPI DeletePrinterDriverExW( LPWSTR pName, LPWSTR pEnvironment, LPWSTR pDriverName, DWORD dwDeleteFlag, DWORD dwVersionFlag) { HKEY hkey_drivers; BOOL ret = FALSE; TRACE("%s %s %s %x %x\n", debugstr_w(pName), debugstr_w(pEnvironment), debugstr_w(pDriverName), dwDeleteFlag, dwVersionFlag); if(pName && pName[0]) { FIXME("pName = %s - unsupported\n", debugstr_w(pName)); SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } if(dwDeleteFlag) { FIXME("dwDeleteFlag = %x - unsupported\n", dwDeleteFlag); SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } hkey_drivers = WINSPOOL_OpenDriverReg(pEnvironment); if(!hkey_drivers) { ERR("Can't open drivers key\n"); return FALSE; } if(RegDeleteTreeW(hkey_drivers, pDriverName) == ERROR_SUCCESS) ret = TRUE; RegCloseKey(hkey_drivers); return ret; } /****************************************************************************** * DeletePrinterDriverExA (WINSPOOL.@) */ BOOL WINAPI DeletePrinterDriverExA( LPSTR pName, LPSTR pEnvironment, LPSTR pDriverName, DWORD dwDeleteFlag, DWORD dwVersionFlag) { UNICODE_STRING NameW, EnvW, DriverW; BOOL ret; asciitounicode(&NameW, pName); asciitounicode(&EnvW, pEnvironment); asciitounicode(&DriverW, pDriverName); ret = DeletePrinterDriverExW(NameW.Buffer, EnvW.Buffer, DriverW.Buffer, dwDeleteFlag, dwVersionFlag); RtlFreeUnicodeString(&DriverW); RtlFreeUnicodeString(&EnvW); RtlFreeUnicodeString(&NameW); return ret; } /****************************************************************************** * DeletePrinterDataExW (WINSPOOL.@) */ DWORD WINAPI DeletePrinterDataExW( HANDLE hPrinter, LPCWSTR pKeyName, LPCWSTR pValueName) { FIXME("%p %s %s\n", hPrinter, debugstr_w(pKeyName), debugstr_w(pValueName)); return ERROR_INVALID_PARAMETER; } /****************************************************************************** * DeletePrinterDataExA (WINSPOOL.@) */ DWORD WINAPI DeletePrinterDataExA( HANDLE hPrinter, LPCSTR pKeyName, LPCSTR pValueName) { FIXME("%p %s %s\n", hPrinter, debugstr_a(pKeyName), debugstr_a(pValueName)); return ERROR_INVALID_PARAMETER; } /****************************************************************************** * DeletePrintProcessorA (WINSPOOL.@) */ BOOL WINAPI DeletePrintProcessorA(LPSTR pName, LPSTR pEnvironment, LPSTR pPrintProcessorName) { FIXME("%s %s %s\n", debugstr_a(pName), debugstr_a(pEnvironment), debugstr_a(pPrintProcessorName)); return TRUE; } /****************************************************************************** * DeletePrintProcessorW (WINSPOOL.@) */ BOOL WINAPI DeletePrintProcessorW(LPWSTR pName, LPWSTR pEnvironment, LPWSTR pPrintProcessorName) { FIXME("%s %s %s\n", debugstr_w(pName), debugstr_w(pEnvironment), debugstr_w(pPrintProcessorName)); return TRUE; } /****************************************************************************** * DeletePrintProvidorA (WINSPOOL.@) */ BOOL WINAPI DeletePrintProvidorA(LPSTR pName, LPSTR pEnvironment, LPSTR pPrintProviderName) { FIXME("%s %s %s\n", debugstr_a(pName), debugstr_a(pEnvironment), debugstr_a(pPrintProviderName)); return TRUE; } /****************************************************************************** * DeletePrintProvidorW (WINSPOOL.@) */ BOOL WINAPI DeletePrintProvidorW(LPWSTR pName, LPWSTR pEnvironment, LPWSTR pPrintProviderName) { FIXME("%s %s %s\n", debugstr_w(pName), debugstr_w(pEnvironment), debugstr_w(pPrintProviderName)); return TRUE; } /****************************************************************************** * EnumFormsA (WINSPOOL.@) */ BOOL WINAPI EnumFormsA( HANDLE hPrinter, DWORD Level, LPBYTE pForm, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned ) { FIXME("%p %x %p %x %p %p\n", hPrinter, Level, pForm, cbBuf, pcbNeeded, pcReturned); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return FALSE; } /****************************************************************************** * EnumFormsW (WINSPOOL.@) */ BOOL WINAPI EnumFormsW( HANDLE hPrinter, DWORD Level, LPBYTE pForm, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned ) { FIXME("%p %x %p %x %p %p\n", hPrinter, Level, pForm, cbBuf, pcbNeeded, pcReturned); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return FALSE; } /***************************************************************************** * EnumMonitorsA [WINSPOOL.@] * * See EnumMonitorsW. * */ BOOL WINAPI EnumMonitorsA(LPSTR pName, DWORD Level, LPBYTE pMonitors, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { BOOL res; LPBYTE bufferW = NULL; LPWSTR nameW = NULL; DWORD needed = 0; DWORD numentries = 0; INT len; TRACE("(%s, %d, %p, %d, %p, %p)\n", debugstr_a(pName), Level, pMonitors, cbBuf, pcbNeeded, pcReturned); /* convert servername to unicode */ if (pName) { len = MultiByteToWideChar(CP_ACP, 0, pName, -1, NULL, 0); nameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pName, -1, nameW, len); } /* alloc (userbuffersize*sizeof(WCHAR) and try to enum the monitors */ needed = cbBuf * sizeof(WCHAR); if (needed) bufferW = HeapAlloc(GetProcessHeap(), 0, needed); res = EnumMonitorsW(nameW, Level, bufferW, needed, pcbNeeded, pcReturned); if(!res && (GetLastError() == ERROR_INSUFFICIENT_BUFFER)) { if (pcbNeeded) needed = *pcbNeeded; /* HeapReAlloc return NULL, when bufferW was NULL */ bufferW = (bufferW) ? HeapReAlloc(GetProcessHeap(), 0, bufferW, needed) : HeapAlloc(GetProcessHeap(), 0, needed); /* Try again with the large Buffer */ res = EnumMonitorsW(nameW, Level, bufferW, needed, pcbNeeded, pcReturned); } numentries = pcReturned ? *pcReturned : 0; needed = 0; /* W2k require the buffersize from EnumMonitorsW also for EnumMonitorsA. We use the smaller Ansi-Size to avoid conflicts with fixed Buffers of old Apps. */ if (res) { /* EnumMonitorsW collected all Data. Parse them to calculate ANSI-Size */ DWORD entrysize = 0; DWORD index; LPSTR ptr; LPMONITOR_INFO_2W mi2w; LPMONITOR_INFO_2A mi2a; /* MONITOR_INFO_*W and MONITOR_INFO_*A have the same size */ entrysize = (Level == 1) ? sizeof(MONITOR_INFO_1A) : sizeof(MONITOR_INFO_2A); /* First pass: calculate the size for all Entries */ mi2w = (LPMONITOR_INFO_2W) bufferW; mi2a = (LPMONITOR_INFO_2A) pMonitors; index = 0; while (index < numentries) { index++; needed += entrysize; /* MONITOR_INFO_?A */ TRACE("%p: parsing #%d (%s)\n", mi2w, index, debugstr_w(mi2w->pName)); needed += WideCharToMultiByte(CP_ACP, 0, mi2w->pName, -1, NULL, 0, NULL, NULL); if (Level > 1) { needed += WideCharToMultiByte(CP_ACP, 0, mi2w->pEnvironment, -1, NULL, 0, NULL, NULL); needed += WideCharToMultiByte(CP_ACP, 0, mi2w->pDLLName, -1, NULL, 0, NULL, NULL); } /* use LPBYTE with entrysize to avoid double code (MONITOR_INFO_1 + MONITOR_INFO_2) */ mi2w = (LPMONITOR_INFO_2W) (((LPBYTE)mi2w) + entrysize); mi2a = (LPMONITOR_INFO_2A) (((LPBYTE)mi2a) + entrysize); } /* check for errors and quit on failure */ if (cbBuf < needed) { SetLastError(ERROR_INSUFFICIENT_BUFFER); res = FALSE; goto emA_cleanup; } len = entrysize * numentries; /* room for all MONITOR_INFO_?A */ ptr = (LPSTR) &pMonitors[len]; /* room for strings */ cbBuf -= len ; /* free Bytes in the user-Buffer */ mi2w = (LPMONITOR_INFO_2W) bufferW; mi2a = (LPMONITOR_INFO_2A) pMonitors; index = 0; /* Second Pass: Fill the User Buffer (if we have one) */ while ((index < numentries) && pMonitors) { index++; TRACE("%p: writing MONITOR_INFO_%dA #%d\n", mi2a, Level, index); mi2a->pName = ptr; len = WideCharToMultiByte(CP_ACP, 0, mi2w->pName, -1, ptr, cbBuf , NULL, NULL); ptr += len; cbBuf -= len; if (Level > 1) { mi2a->pEnvironment = ptr; len = WideCharToMultiByte(CP_ACP, 0, mi2w->pEnvironment, -1, ptr, cbBuf, NULL, NULL); ptr += len; cbBuf -= len; mi2a->pDLLName = ptr; len = WideCharToMultiByte(CP_ACP, 0, mi2w->pDLLName, -1, ptr, cbBuf, NULL, NULL); ptr += len; cbBuf -= len; } /* use LPBYTE with entrysize to avoid double code (MONITOR_INFO_1 + MONITOR_INFO_2) */ mi2w = (LPMONITOR_INFO_2W) (((LPBYTE)mi2w) + entrysize); mi2a = (LPMONITOR_INFO_2A) (((LPBYTE)mi2a) + entrysize); } } emA_cleanup: if (pcbNeeded) *pcbNeeded = needed; if (pcReturned) *pcReturned = (res) ? numentries : 0; HeapFree(GetProcessHeap(), 0, nameW); HeapFree(GetProcessHeap(), 0, bufferW); TRACE("returning %d with %d (%d byte for %d entries)\n", (res), GetLastError(), needed, numentries); return (res); } /***************************************************************************** * EnumMonitorsW [WINSPOOL.@] * * Enumerate available Port-Monitors * * PARAMS * pName [I] Servername or NULL (local Computer) * Level [I] Structure-Level (1:Win9x+NT or 2:NT only) * pMonitors [O] PTR to Buffer that receives the Result * cbBuf [I] Size of Buffer at pMonitors * pcbNeeded [O] PTR to DWORD that receives the size in Bytes used / required for pMonitors * pcReturned [O] PTR to DWORD that receives the number of Monitors in pMonitors * * RETURNS * Success: TRUE * Failure: FALSE and in pcbNeeded the Bytes required for buffer, if cbBuf is too small * */ BOOL WINAPI EnumMonitorsW(LPWSTR pName, DWORD Level, LPBYTE pMonitors, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { TRACE("(%s, %d, %p, %d, %p, %p)\n", debugstr_w(pName), Level, pMonitors, cbBuf, pcbNeeded, pcReturned); if ((backend == NULL) && !load_backend()) return FALSE; if (!pcbNeeded || !pcReturned || (!pMonitors && (cbBuf > 0))) { SetLastError(RPC_X_NULL_REF_POINTER); return FALSE; } return backend->fpEnumMonitors(pName, Level, pMonitors, cbBuf, pcbNeeded, pcReturned); } /****************************************************************************** * SpoolerInit (WINSPOOL.@) * * Initialize the Spooler * * RETURNS * Success: TRUE * Failure: FALSE * * NOTES * The function fails on windows, when the spooler service is not running * */ BOOL WINAPI SpoolerInit(void) { if ((backend == NULL) && !load_backend()) return FALSE; return TRUE; } /****************************************************************************** * XcvDataW (WINSPOOL.@) * * Execute commands in the Printmonitor DLL * * PARAMS * hXcv [i] Handle from OpenPrinter (with XcvMonitor or XcvPort) * pszDataName [i] Name of the command to execute * pInputData [i] Buffer for extra Input Data (needed only for some commands) * cbInputData [i] Size in Bytes of Buffer at pInputData * pOutputData [o] Buffer to receive additional Data (needed only for some commands) * cbOutputData [i] Size in Bytes of Buffer at pOutputData * pcbOutputNeeded [o] PTR to receive the minimal Size in Bytes of the Buffer at pOutputData * pdwStatus [o] PTR to receive the win32 error code from the Printmonitor DLL * * RETURNS * Success: TRUE * Failure: FALSE * * NOTES * Returning "TRUE" does mean, that the Printmonitor DLL was called successful. * The execution of the command can still fail (check pdwStatus for ERROR_SUCCESS). * * Minimal List of commands, that a Printmonitor DLL should support: * *| "MonitorUI" : Return the Name of the Userinterface-DLL as WSTR in pOutputData *| "AddPort" : Add a Port *| "DeletePort": Delete a Port * * Many Printmonitors support additional commands. Examples for localspl.dll: * "GetDefaultCommConfig", "SetDefaultCommConfig", * "GetTransmissionRetryTimeout", "ConfigureLPTPortCommandOK" * */ BOOL WINAPI XcvDataW( HANDLE hXcv, LPCWSTR pszDataName, PBYTE pInputData, DWORD cbInputData, PBYTE pOutputData, DWORD cbOutputData, PDWORD pcbOutputNeeded, PDWORD pdwStatus) { opened_printer_t *printer; TRACE("(%p, %s, %p, %d, %p, %d, %p, %p)\n", hXcv, debugstr_w(pszDataName), pInputData, cbInputData, pOutputData, cbOutputData, pcbOutputNeeded, pdwStatus); if ((backend == NULL) && !load_backend()) return FALSE; printer = get_opened_printer(hXcv); if (!printer || (!printer->backend_printer)) { SetLastError(ERROR_INVALID_HANDLE); return FALSE; } if (!pcbOutputNeeded) { SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } if (!pszDataName || !pdwStatus || (!pOutputData && (cbOutputData > 0))) { SetLastError(RPC_X_NULL_REF_POINTER); return FALSE; } *pcbOutputNeeded = 0; return backend->fpXcvData(printer->backend_printer, pszDataName, pInputData, cbInputData, pOutputData, cbOutputData, pcbOutputNeeded, pdwStatus); } /***************************************************************************** * EnumPrinterDataA [WINSPOOL.@] * */ DWORD WINAPI EnumPrinterDataA( HANDLE hPrinter, DWORD dwIndex, LPSTR pValueName, DWORD cbValueName, LPDWORD pcbValueName, LPDWORD pType, LPBYTE pData, DWORD cbData, LPDWORD pcbData ) { FIXME("%p %x %p %x %p %p %p %x %p\n", hPrinter, dwIndex, pValueName, cbValueName, pcbValueName, pType, pData, cbData, pcbData); return ERROR_NO_MORE_ITEMS; } /***************************************************************************** * EnumPrinterDataW [WINSPOOL.@] * */ DWORD WINAPI EnumPrinterDataW( HANDLE hPrinter, DWORD dwIndex, LPWSTR pValueName, DWORD cbValueName, LPDWORD pcbValueName, LPDWORD pType, LPBYTE pData, DWORD cbData, LPDWORD pcbData ) { FIXME("%p %x %p %x %p %p %p %x %p\n", hPrinter, dwIndex, pValueName, cbValueName, pcbValueName, pType, pData, cbData, pcbData); return ERROR_NO_MORE_ITEMS; } /***************************************************************************** * EnumPrintProcessorDatatypesA [WINSPOOL.@] * */ BOOL WINAPI EnumPrintProcessorDatatypesA(LPSTR pName, LPSTR pPrintProcessorName, DWORD Level, LPBYTE pDatatypes, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { FIXME("Stub: %s %s %d %p %d %p %p\n", debugstr_a(pName), debugstr_a(pPrintProcessorName), Level, pDatatypes, cbBuf, pcbNeeded, pcReturned); return FALSE; } /***************************************************************************** * EnumPrintProcessorDatatypesW [WINSPOOL.@] * */ BOOL WINAPI EnumPrintProcessorDatatypesW(LPWSTR pName, LPWSTR pPrintProcessorName, DWORD Level, LPBYTE pDatatypes, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { FIXME("Stub: %s %s %d %p %d %p %p\n", debugstr_w(pName), debugstr_w(pPrintProcessorName), Level, pDatatypes, cbBuf, pcbNeeded, pcReturned); return FALSE; } /***************************************************************************** * EnumPrintProcessorsA [WINSPOOL.@] * * See EnumPrintProcessorsW. * */ BOOL WINAPI EnumPrintProcessorsA(LPSTR pName, LPSTR pEnvironment, DWORD Level, LPBYTE pPPInfo, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { BOOL res; LPBYTE bufferW = NULL; LPWSTR nameW = NULL; LPWSTR envW = NULL; DWORD needed = 0; DWORD numentries = 0; INT len; TRACE("(%s, %s, %d, %p, %d, %p, %p)\n", debugstr_a(pName), debugstr_a(pEnvironment), Level, pPPInfo, cbBuf, pcbNeeded, pcReturned); /* convert names to unicode */ if (pName) { len = MultiByteToWideChar(CP_ACP, 0, pName, -1, NULL, 0); nameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pName, -1, nameW, len); } if (pEnvironment) { len = MultiByteToWideChar(CP_ACP, 0, pEnvironment, -1, NULL, 0); envW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, pEnvironment, -1, envW, len); } /* alloc (userbuffersize*sizeof(WCHAR) and try to enum the monitors */ needed = cbBuf * sizeof(WCHAR); if (needed) bufferW = HeapAlloc(GetProcessHeap(), 0, needed); res = EnumPrintProcessorsW(nameW, envW, Level, bufferW, needed, pcbNeeded, pcReturned); if(!res && (GetLastError() == ERROR_INSUFFICIENT_BUFFER)) { if (pcbNeeded) needed = *pcbNeeded; /* HeapReAlloc return NULL, when bufferW was NULL */ bufferW = (bufferW) ? HeapReAlloc(GetProcessHeap(), 0, bufferW, needed) : HeapAlloc(GetProcessHeap(), 0, needed); /* Try again with the large Buffer */ res = EnumPrintProcessorsW(nameW, envW, Level, bufferW, needed, pcbNeeded, pcReturned); } numentries = pcReturned ? *pcReturned : 0; needed = 0; if (res) { /* EnumPrintProcessorsW collected all Data. Parse them to calculate ANSI-Size */ DWORD index; LPSTR ptr; PPRINTPROCESSOR_INFO_1W ppiw; PPRINTPROCESSOR_INFO_1A ppia; /* First pass: calculate the size for all Entries */ ppiw = (PPRINTPROCESSOR_INFO_1W) bufferW; ppia = (PPRINTPROCESSOR_INFO_1A) pPPInfo; index = 0; while (index < numentries) { index++; needed += sizeof(PRINTPROCESSOR_INFO_1A); TRACE("%p: parsing #%d (%s)\n", ppiw, index, debugstr_w(ppiw->pName)); needed += WideCharToMultiByte(CP_ACP, 0, ppiw->pName, -1, NULL, 0, NULL, NULL); ppiw = (PPRINTPROCESSOR_INFO_1W) (((LPBYTE)ppiw) + sizeof(PRINTPROCESSOR_INFO_1W)); ppia = (PPRINTPROCESSOR_INFO_1A) (((LPBYTE)ppia) + sizeof(PRINTPROCESSOR_INFO_1A)); } /* check for errors and quit on failure */ if (cbBuf < needed) { SetLastError(ERROR_INSUFFICIENT_BUFFER); res = FALSE; goto epp_cleanup; } len = numentries * sizeof(PRINTPROCESSOR_INFO_1A); /* room for structs */ ptr = (LPSTR) &pPPInfo[len]; /* start of strings */ cbBuf -= len ; /* free Bytes in the user-Buffer */ ppiw = (PPRINTPROCESSOR_INFO_1W) bufferW; ppia = (PPRINTPROCESSOR_INFO_1A) pPPInfo; index = 0; /* Second Pass: Fill the User Buffer (if we have one) */ while ((index < numentries) && pPPInfo) { index++; TRACE("%p: writing PRINTPROCESSOR_INFO_1A #%d\n", ppia, index); ppia->pName = ptr; len = WideCharToMultiByte(CP_ACP, 0, ppiw->pName, -1, ptr, cbBuf , NULL, NULL); ptr += len; cbBuf -= len; ppiw = (PPRINTPROCESSOR_INFO_1W) (((LPBYTE)ppiw) + sizeof(PRINTPROCESSOR_INFO_1W)); ppia = (PPRINTPROCESSOR_INFO_1A) (((LPBYTE)ppia) + sizeof(PRINTPROCESSOR_INFO_1A)); } } epp_cleanup: if (pcbNeeded) *pcbNeeded = needed; if (pcReturned) *pcReturned = (res) ? numentries : 0; HeapFree(GetProcessHeap(), 0, nameW); HeapFree(GetProcessHeap(), 0, envW); HeapFree(GetProcessHeap(), 0, bufferW); TRACE("returning %d with %d (%d byte for %d entries)\n", (res), GetLastError(), needed, numentries); return (res); } /***************************************************************************** * EnumPrintProcessorsW [WINSPOOL.@] * * Enumerate available Print Processors * * PARAMS * pName [I] Servername or NULL (local Computer) * pEnvironment [I] Printing-Environment or NULL (Default) * Level [I] Structure-Level (Only 1 is allowed) * pPPInfo [O] PTR to Buffer that receives the Result * cbBuf [I] Size of Buffer at pPPInfo * pcbNeeded [O] PTR to DWORD that receives the size in Bytes used / required for pPPInfo * pcReturned [O] PTR to DWORD that receives the number of Print Processors in pPPInfo * * RETURNS * Success: TRUE * Failure: FALSE and in pcbNeeded the Bytes required for pPPInfo, if cbBuf is too small * */ BOOL WINAPI EnumPrintProcessorsW(LPWSTR pName, LPWSTR pEnvironment, DWORD Level, LPBYTE pPPInfo, DWORD cbBuf, LPDWORD pcbNeeded, LPDWORD pcReturned) { TRACE("(%s, %s, %d, %p, %d, %p, %p)\n", debugstr_w(pName), debugstr_w(pEnvironment), Level, pPPInfo, cbBuf, pcbNeeded, pcReturned); if ((backend == NULL) && !load_backend()) return FALSE; if (!pcbNeeded || !pcReturned) { SetLastError(RPC_X_NULL_REF_POINTER); return FALSE; } if (!pPPInfo && (cbBuf > 0)) { SetLastError(ERROR_INVALID_USER_BUFFER); return FALSE; } return backend->fpEnumPrintProcessors(pName, pEnvironment, Level, pPPInfo, cbBuf, pcbNeeded, pcReturned); } /***************************************************************************** * ExtDeviceMode [WINSPOOL.@] * */ LONG WINAPI ExtDeviceMode( HWND hWnd, HANDLE hInst, LPDEVMODEA pDevModeOutput, LPSTR pDeviceName, LPSTR pPort, LPDEVMODEA pDevModeInput, LPSTR pProfile, DWORD fMode) { FIXME("Stub: %p %p %p %s %s %p %s %x\n", hWnd, hInst, pDevModeOutput, debugstr_a(pDeviceName), debugstr_a(pPort), pDevModeInput, debugstr_a(pProfile), fMode); return -1; } /***************************************************************************** * FindClosePrinterChangeNotification [WINSPOOL.@] * */ BOOL WINAPI FindClosePrinterChangeNotification( HANDLE hChange ) { FIXME("Stub: %p\n", hChange); return TRUE; } /***************************************************************************** * FindFirstPrinterChangeNotification [WINSPOOL.@] * */ HANDLE WINAPI FindFirstPrinterChangeNotification( HANDLE hPrinter, DWORD fdwFlags, DWORD fdwOptions, LPVOID pPrinterNotifyOptions ) { FIXME("Stub: %p %x %x %p\n", hPrinter, fdwFlags, fdwOptions, pPrinterNotifyOptions); return INVALID_HANDLE_VALUE; } /***************************************************************************** * FindNextPrinterChangeNotification [WINSPOOL.@] * */ BOOL WINAPI FindNextPrinterChangeNotification( HANDLE hChange, PDWORD pdwChange, LPVOID pPrinterNotifyOptions, LPVOID *ppPrinterNotifyInfo ) { FIXME("Stub: %p %p %p %p\n", hChange, pdwChange, pPrinterNotifyOptions, ppPrinterNotifyInfo); return FALSE; } /***************************************************************************** * FreePrinterNotifyInfo [WINSPOOL.@] * */ BOOL WINAPI FreePrinterNotifyInfo( PPRINTER_NOTIFY_INFO pPrinterNotifyInfo ) { FIXME("Stub: %p\n", pPrinterNotifyInfo); return TRUE; } /***************************************************************************** * string_to_buf * * Copies a unicode string into a buffer. The buffer will either contain unicode or * ansi depending on the unicode parameter. */ static BOOL string_to_buf(LPCWSTR str, LPBYTE ptr, DWORD cb, DWORD *size, BOOL unicode) { if(!str) { *size = 0; return TRUE; } if(unicode) { *size = (strlenW(str) + 1) * sizeof(WCHAR); if(*size <= cb) { memcpy(ptr, str, *size); return TRUE; } return FALSE; } else { *size = WideCharToMultiByte(CP_ACP, 0, str, -1, NULL, 0, NULL, NULL); if(*size <= cb) { WideCharToMultiByte(CP_ACP, 0, str, -1, (LPSTR)ptr, *size, NULL, NULL); return TRUE; } return FALSE; } } /***************************************************************************** * get_job_info_1 */ static BOOL get_job_info_1(job_t *job, JOB_INFO_1W *ji1, LPBYTE buf, DWORD cbBuf, LPDWORD pcbNeeded, BOOL unicode) { DWORD size, left = cbBuf; BOOL space = (cbBuf > 0); LPBYTE ptr = buf; *pcbNeeded = 0; if(space) { ji1->JobId = job->job_id; } string_to_buf(job->document_title, ptr, left, &size, unicode); if(space && size <= left) { ji1->pDocument = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; if (job->printer_name) { string_to_buf(job->printer_name, ptr, left, &size, unicode); if(space && size <= left) { ji1->pPrinterName = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } return space; } /***************************************************************************** * get_job_info_2 */ static BOOL get_job_info_2(job_t *job, JOB_INFO_2W *ji2, LPBYTE buf, DWORD cbBuf, LPDWORD pcbNeeded, BOOL unicode) { DWORD size, left = cbBuf; DWORD shift; BOOL space = (cbBuf > 0); LPBYTE ptr = buf; LPDEVMODEA dmA = NULL; LPDEVMODEW devmode; *pcbNeeded = 0; if(space) { ji2->JobId = job->job_id; } string_to_buf(job->document_title, ptr, left, &size, unicode); if(space && size <= left) { ji2->pDocument = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; if (job->printer_name) { string_to_buf(job->printer_name, ptr, left, &size, unicode); if(space && size <= left) { ji2->pPrinterName = (LPWSTR)ptr; ptr += size; left -= size; } else space = FALSE; *pcbNeeded += size; } if (job->devmode) { if (!unicode) { dmA = DEVMODEdupWtoA(job->devmode); devmode = (LPDEVMODEW) dmA; if (dmA) size = dmA->dmSize + dmA->dmDriverExtra; } else { devmode = job->devmode; size = devmode->dmSize + devmode->dmDriverExtra; } if (!devmode) FIXME("Can't convert DEVMODE W to A\n"); else { /* align DEVMODE to a DWORD boundary */ shift = (4 - (*pcbNeeded & 3)) & 3; size += shift; if (size <= left) { ptr += shift; memcpy(ptr, devmode, size-shift); ji2->pDevMode = (LPDEVMODEW)ptr; if (!unicode) HeapFree(GetProcessHeap(), 0, dmA); ptr += size-shift; left -= size; } else space = FALSE; *pcbNeeded +=size; } } return space; } /***************************************************************************** * get_job_info */ static BOOL get_job_info(HANDLE hPrinter, DWORD JobId, DWORD Level, LPBYTE pJob, DWORD cbBuf, LPDWORD pcbNeeded, BOOL unicode) { BOOL ret = FALSE; DWORD needed = 0, size; job_t *job; LPBYTE ptr = pJob; TRACE("%p %d %d %p %d %p\n", hPrinter, JobId, Level, pJob, cbBuf, pcbNeeded); EnterCriticalSection(&printer_handles_cs); job = get_job(hPrinter, JobId); if(!job) goto end; switch(Level) { case 1: size = sizeof(JOB_INFO_1W); if(cbBuf >= size) { cbBuf -= size; ptr += size; memset(pJob, 0, size); } else cbBuf = 0; ret = get_job_info_1(job, (JOB_INFO_1W *)pJob, ptr, cbBuf, &needed, unicode); needed += size; break; case 2: size = sizeof(JOB_INFO_2W); if(cbBuf >= size) { cbBuf -= size; ptr += size; memset(pJob, 0, size); } else cbBuf = 0; ret = get_job_info_2(job, (JOB_INFO_2W *)pJob, ptr, cbBuf, &needed, unicode); needed += size; break; case 3: size = sizeof(JOB_INFO_3); if(cbBuf >= size) { cbBuf -= size; memset(pJob, 0, size); ret = TRUE; } else cbBuf = 0; needed = size; break; default: SetLastError(ERROR_INVALID_LEVEL); goto end; } if(pcbNeeded) *pcbNeeded = needed; end: LeaveCriticalSection(&printer_handles_cs); return ret; } /***************************************************************************** * GetJobA [WINSPOOL.@] * */ BOOL WINAPI GetJobA(HANDLE hPrinter, DWORD JobId, DWORD Level, LPBYTE pJob, DWORD cbBuf, LPDWORD pcbNeeded) { return get_job_info(hPrinter, JobId, Level, pJob, cbBuf, pcbNeeded, FALSE); } /***************************************************************************** * GetJobW [WINSPOOL.@] * */ BOOL WINAPI GetJobW(HANDLE hPrinter, DWORD JobId, DWORD Level, LPBYTE pJob, DWORD cbBuf, LPDWORD pcbNeeded) { return get_job_info(hPrinter, JobId, Level, pJob, cbBuf, pcbNeeded, TRUE); } /***************************************************************************** * schedule_pipe */ static BOOL schedule_pipe(LPCWSTR cmd, LPCWSTR filename) { #ifdef HAVE_FORK char *unixname, *cmdA; DWORD len; int fds[2] = {-1, -1}, file_fd = -1, no_read; BOOL ret = FALSE; char buf[1024]; pid_t pid, wret; int status; if(!(unixname = wine_get_unix_file_name(filename))) return FALSE; len = WideCharToMultiByte(CP_UNIXCP, 0, cmd, -1, NULL, 0, NULL, NULL); cmdA = HeapAlloc(GetProcessHeap(), 0, len); WideCharToMultiByte(CP_UNIXCP, 0, cmd, -1, cmdA, len, NULL, NULL); TRACE("printing with: %s\n", cmdA); if((file_fd = open(unixname, O_RDONLY)) == -1) goto end; if (pipe(fds)) { ERR("pipe() failed!\n"); goto end; } if ((pid = fork()) == 0) { close(0); dup2(fds[0], 0); close(fds[1]); /* reset signals that we previously set to SIG_IGN */ signal(SIGPIPE, SIG_DFL); execl("/bin/sh", "/bin/sh", "-c", cmdA, NULL); _exit(1); } else if (pid == -1) { ERR("fork() failed!\n"); goto end; } close(fds[0]); fds[0] = -1; while((no_read = read(file_fd, buf, sizeof(buf))) > 0) write(fds[1], buf, no_read); close(fds[1]); fds[1] = -1; /* reap child */ do { wret = waitpid(pid, &status, 0); } while (wret < 0 && errno == EINTR); if (wret < 0) { ERR("waitpid() failed!\n"); goto end; } if (!WIFEXITED(status) || WEXITSTATUS(status)) { ERR("child process failed! %d\n", status); goto end; } ret = TRUE; end: if(file_fd != -1) close(file_fd); if(fds[0] != -1) close(fds[0]); if(fds[1] != -1) close(fds[1]); HeapFree(GetProcessHeap(), 0, cmdA); HeapFree(GetProcessHeap(), 0, unixname); return ret; #else return FALSE; #endif } /***************************************************************************** * schedule_lpr */ static BOOL schedule_lpr(LPCWSTR printer_name, LPCWSTR filename) { WCHAR *cmd; const WCHAR fmtW[] = {'l','p','r',' ','-','P','\'','%','s','\'',0}; BOOL r; cmd = HeapAlloc(GetProcessHeap(), 0, strlenW(printer_name) * sizeof(WCHAR) + sizeof(fmtW)); sprintfW(cmd, fmtW, printer_name); r = schedule_pipe(cmd, filename); HeapFree(GetProcessHeap(), 0, cmd); return r; } #ifdef SONAME_LIBCUPS /***************************************************************************** * get_cups_jobs_ticket_options * * Explicitly set CUPS options based on any %cupsJobTicket lines. * The CUPS scheduler only looks for these in Print-File requests, and since * cupsPrintFile uses Create-Job / Send-Document, the ticket lines don't get * parsed. */ static int get_cups_job_ticket_options( const char *file, int num_options, cups_option_t **options ) { FILE *fp = fopen( file, "r" ); char buf[257]; /* DSC max of 256 + '\0' */ const char *ps_adobe = "%!PS-Adobe-"; const char *cups_job = "%cupsJobTicket:"; if (!fp) return num_options; if (!fgets( buf, sizeof(buf), fp )) goto end; if (strncmp( buf, ps_adobe, strlen( ps_adobe ) )) goto end; while (fgets( buf, sizeof(buf), fp )) { if (strncmp( buf, cups_job, strlen( cups_job ) )) break; num_options = pcupsParseOptions( buf + strlen( cups_job ), num_options, options ); } end: fclose( fp ); return num_options; } static int get_cups_default_options( const char *printer, int num_options, cups_option_t **options ) { cups_dest_t *dest; int i; if (!pcupsGetNamedDest) return num_options; dest = pcupsGetNamedDest( NULL, printer, NULL ); if (!dest) return num_options; for (i = 0; i < dest->num_options; i++) { if (!pcupsGetOption( dest->options[i].name, num_options, *options )) num_options = pcupsAddOption( dest->options[i].name, dest->options[i].value, num_options, options ); } pcupsFreeDests( 1, dest ); return num_options; } #endif /***************************************************************************** * schedule_cups */ static BOOL schedule_cups(LPCWSTR printer_name, LPCWSTR filename, LPCWSTR document_title) { #ifdef SONAME_LIBCUPS if(pcupsPrintFile) { char *unixname, *queue, *unix_doc_title; DWORD len; BOOL ret; int num_options = 0, i; cups_option_t *options = NULL; if(!(unixname = wine_get_unix_file_name(filename))) return FALSE; len = WideCharToMultiByte(CP_UNIXCP, 0, printer_name, -1, NULL, 0, NULL, NULL); queue = HeapAlloc(GetProcessHeap(), 0, len); WideCharToMultiByte(CP_UNIXCP, 0, printer_name, -1, queue, len, NULL, NULL); len = WideCharToMultiByte(CP_UNIXCP, 0, document_title, -1, NULL, 0, NULL, NULL); unix_doc_title = HeapAlloc(GetProcessHeap(), 0, len); WideCharToMultiByte(CP_UNIXCP, 0, document_title, -1, unix_doc_title, len, NULL, NULL); num_options = get_cups_job_ticket_options( unixname, num_options, &options ); num_options = get_cups_default_options( queue, num_options, &options ); TRACE( "printing via cups with options:\n" ); for (i = 0; i < num_options; i++) TRACE( "\t%d: %s = %s\n", i, options[i].name, options[i].value ); ret = pcupsPrintFile( queue, unixname, unix_doc_title, num_options, options ); pcupsFreeOptions( num_options, options ); HeapFree(GetProcessHeap(), 0, unix_doc_title); HeapFree(GetProcessHeap(), 0, queue); HeapFree(GetProcessHeap(), 0, unixname); return ret; } else #endif { return schedule_lpr(printer_name, filename); } } static INT_PTR CALLBACK file_dlg_proc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) { LPWSTR filename; switch(msg) { case WM_INITDIALOG: SetWindowLongPtrW(hwnd, DWLP_USER, lparam); return TRUE; case WM_COMMAND: if(HIWORD(wparam) == BN_CLICKED) { if(LOWORD(wparam) == IDOK) { HANDLE hf; DWORD len = SendDlgItemMessageW(hwnd, EDITBOX, WM_GETTEXTLENGTH, 0, 0); LPWSTR *output; filename = HeapAlloc(GetProcessHeap(), 0, (len + 1) * sizeof(WCHAR)); GetDlgItemTextW(hwnd, EDITBOX, filename, len + 1); if(GetFileAttributesW(filename) != INVALID_FILE_ATTRIBUTES) { WCHAR caption[200], message[200]; int mb_ret; LoadStringW(WINSPOOL_hInstance, IDS_CAPTION, caption, sizeof(caption) / sizeof(WCHAR)); LoadStringW(WINSPOOL_hInstance, IDS_FILE_EXISTS, message, sizeof(message) / sizeof(WCHAR)); mb_ret = MessageBoxW(hwnd, message, caption, MB_OKCANCEL | MB_ICONEXCLAMATION); if(mb_ret == IDCANCEL) { HeapFree(GetProcessHeap(), 0, filename); return TRUE; } } hf = CreateFileW(filename, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); if(hf == INVALID_HANDLE_VALUE) { WCHAR caption[200], message[200]; LoadStringW(WINSPOOL_hInstance, IDS_CAPTION, caption, sizeof(caption) / sizeof(WCHAR)); LoadStringW(WINSPOOL_hInstance, IDS_CANNOT_OPEN, message, sizeof(message) / sizeof(WCHAR)); MessageBoxW(hwnd, message, caption, MB_OK | MB_ICONEXCLAMATION); HeapFree(GetProcessHeap(), 0, filename); return TRUE; } CloseHandle(hf); DeleteFileW(filename); output = (LPWSTR *)GetWindowLongPtrW(hwnd, DWLP_USER); *output = filename; EndDialog(hwnd, IDOK); return TRUE; } if(LOWORD(wparam) == IDCANCEL) { EndDialog(hwnd, IDCANCEL); return TRUE; } } return FALSE; } return FALSE; } /***************************************************************************** * get_filename */ static BOOL get_filename(LPWSTR *filename) { return DialogBoxParamW(WINSPOOL_hInstance, MAKEINTRESOURCEW(FILENAME_DIALOG), GetForegroundWindow(), file_dlg_proc, (LPARAM)filename) == IDOK; } /***************************************************************************** * schedule_file */ static BOOL schedule_file(LPCWSTR filename) { LPWSTR output = NULL; if(get_filename(&output)) { BOOL r; TRACE("copy to %s\n", debugstr_w(output)); r = CopyFileW(filename, output, FALSE); HeapFree(GetProcessHeap(), 0, output); return r; } return FALSE; } /***************************************************************************** * schedule_unixfile */ static BOOL schedule_unixfile(LPCWSTR output, LPCWSTR filename) { int in_fd, out_fd, no_read; char buf[1024]; BOOL ret = FALSE; char *unixname, *outputA; DWORD len; if(!(unixname = wine_get_unix_file_name(filename))) return FALSE; len = WideCharToMultiByte(CP_UNIXCP, 0, output, -1, NULL, 0, NULL, NULL); outputA = HeapAlloc(GetProcessHeap(), 0, len); WideCharToMultiByte(CP_UNIXCP, 0, output, -1, outputA, len, NULL, NULL); out_fd = open(outputA, O_CREAT | O_TRUNC | O_WRONLY, 0666); in_fd = open(unixname, O_RDONLY); if(out_fd == -1 || in_fd == -1) goto end; while((no_read = read(in_fd, buf, sizeof(buf))) > 0) write(out_fd, buf, no_read); ret = TRUE; end: if(in_fd != -1) close(in_fd); if(out_fd != -1) close(out_fd); HeapFree(GetProcessHeap(), 0, outputA); HeapFree(GetProcessHeap(), 0, unixname); return ret; } /***************************************************************************** * ScheduleJob [WINSPOOL.@] * */ BOOL WINAPI ScheduleJob( HANDLE hPrinter, DWORD dwJobID ) { opened_printer_t *printer; BOOL ret = FALSE; struct list *cursor, *cursor2; TRACE("(%p, %x)\n", hPrinter, dwJobID); EnterCriticalSection(&printer_handles_cs); printer = get_opened_printer(hPrinter); if(!printer) goto end; LIST_FOR_EACH_SAFE(cursor, cursor2, &printer->queue->jobs) { job_t *job = LIST_ENTRY(cursor, job_t, entry); HANDLE hf; if(job->job_id != dwJobID) continue; hf = CreateFileW(job->filename, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL); if(hf != INVALID_HANDLE_VALUE) { PRINTER_INFO_5W *pi5 = NULL; LPWSTR portname = job->portname; DWORD needed; HKEY hkey; WCHAR output[1024]; static const WCHAR spooler_key[] = {'S','o','f','t','w','a','r','e','\\','W','i','n','e','\\', 'P','r','i','n','t','i','n','g','\\','S','p','o','o','l','e','r',0}; if (!portname) { GetPrinterW(hPrinter, 5, NULL, 0, &needed); pi5 = HeapAlloc(GetProcessHeap(), 0, needed); GetPrinterW(hPrinter, 5, (LPBYTE)pi5, needed, &needed); portname = pi5->pPortName; } TRACE("need to schedule job %d filename %s to port %s\n", job->job_id, debugstr_w(job->filename), debugstr_w(portname)); output[0] = 0; /* @@ Wine registry key: HKCU\Software\Wine\Printing\Spooler */ if(RegOpenKeyW(HKEY_CURRENT_USER, spooler_key, &hkey) == ERROR_SUCCESS) { DWORD type, count = sizeof(output); RegQueryValueExW(hkey, portname, NULL, &type, (LPBYTE)output, &count); RegCloseKey(hkey); } if(output[0] == '|') { ret = schedule_pipe(output + 1, job->filename); } else if(output[0]) { ret = schedule_unixfile(output, job->filename); } else if(!strncmpW(portname, LPR_Port, strlenW(LPR_Port))) { ret = schedule_lpr(portname + strlenW(LPR_Port), job->filename); } else if(!strncmpW(portname, CUPS_Port, strlenW(CUPS_Port))) { ret = schedule_cups(portname + strlenW(CUPS_Port), job->filename, job->document_title); } else if(!strncmpW(portname, FILE_Port, strlenW(FILE_Port))) { ret = schedule_file(job->filename); } else if(isalpha(portname[0]) && portname[1] == ':') { TRACE("copying to %s\n", debugstr_w(portname)); ret = CopyFileW(job->filename, portname, FALSE); } else { FIXME("can't schedule to port %s\n", debugstr_w(portname)); } HeapFree(GetProcessHeap(), 0, pi5); CloseHandle(hf); DeleteFileW(job->filename); } list_remove(cursor); HeapFree(GetProcessHeap(), 0, job->document_title); HeapFree(GetProcessHeap(), 0, job->printer_name); HeapFree(GetProcessHeap(), 0, job->portname); HeapFree(GetProcessHeap(), 0, job->filename); HeapFree(GetProcessHeap(), 0, job->devmode); HeapFree(GetProcessHeap(), 0, job); break; } end: LeaveCriticalSection(&printer_handles_cs); return ret; } /***************************************************************************** * StartDocDlgA [WINSPOOL.@] */ LPSTR WINAPI StartDocDlgA( HANDLE hPrinter, DOCINFOA *doc ) { UNICODE_STRING usBuffer; DOCINFOW docW; LPWSTR retW; LPWSTR docnameW = NULL, outputW = NULL, datatypeW = NULL; LPSTR ret = NULL; docW.cbSize = sizeof(docW); if (doc->lpszDocName) { docnameW = asciitounicode(&usBuffer, doc->lpszDocName); if (!(docW.lpszDocName = docnameW)) return NULL; } if (doc->lpszOutput) { outputW = asciitounicode(&usBuffer, doc->lpszOutput); if (!(docW.lpszOutput = outputW)) return NULL; } if (doc->lpszDatatype) { datatypeW = asciitounicode(&usBuffer, doc->lpszDatatype); if (!(docW.lpszDatatype = datatypeW)) return NULL; } docW.fwType = doc->fwType; retW = StartDocDlgW(hPrinter, &docW); if(retW) { DWORD len = WideCharToMultiByte(CP_ACP, 0, retW, -1, NULL, 0, NULL, NULL); ret = HeapAlloc(GetProcessHeap(), 0, len); WideCharToMultiByte(CP_ACP, 0, retW, -1, ret, len, NULL, NULL); HeapFree(GetProcessHeap(), 0, retW); } HeapFree(GetProcessHeap(), 0, datatypeW); HeapFree(GetProcessHeap(), 0, outputW); HeapFree(GetProcessHeap(), 0, docnameW); return ret; } /***************************************************************************** * StartDocDlgW [WINSPOOL.@] * * Undocumented: Apparently used by gdi32:StartDocW() to popup the file dialog * when lpszOutput is "FILE:" or if lpszOutput is NULL and the default printer * port is "FILE:". Also returns the full path if passed a relative path. * * The caller should free the returned string from the process heap. */ LPWSTR WINAPI StartDocDlgW( HANDLE hPrinter, DOCINFOW *doc ) { LPWSTR ret = NULL; DWORD len, attr; if(doc->lpszOutput == NULL) /* Check whether default port is FILE: */ { PRINTER_INFO_5W *pi5; GetPrinterW(hPrinter, 5, NULL, 0, &len); if(GetLastError() != ERROR_INSUFFICIENT_BUFFER) return NULL; pi5 = HeapAlloc(GetProcessHeap(), 0, len); GetPrinterW(hPrinter, 5, (LPBYTE)pi5, len, &len); if(!pi5->pPortName || strcmpW(pi5->pPortName, FILE_Port)) { HeapFree(GetProcessHeap(), 0, pi5); return NULL; } HeapFree(GetProcessHeap(), 0, pi5); } if(doc->lpszOutput == NULL || !strcmpW(doc->lpszOutput, FILE_Port)) { LPWSTR name; if (get_filename(&name)) { if(!(len = GetFullPathNameW(name, 0, NULL, NULL))) { HeapFree(GetProcessHeap(), 0, name); return NULL; } ret = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); GetFullPathNameW(name, len, ret, NULL); HeapFree(GetProcessHeap(), 0, name); } return ret; } if(!(len = GetFullPathNameW(doc->lpszOutput, 0, NULL, NULL))) return NULL; ret = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); GetFullPathNameW(doc->lpszOutput, len, ret, NULL); attr = GetFileAttributesW(ret); if(attr != INVALID_FILE_ATTRIBUTES && (attr & FILE_ATTRIBUTE_DIRECTORY)) { HeapFree(GetProcessHeap(), 0, ret); ret = NULL; } return ret; } /***************************************************************************** * UploadPrinterDriverPackageA [WINSPOOL.@] */ HRESULT WINAPI UploadPrinterDriverPackageA( LPCSTR server, LPCSTR path, LPCSTR env, DWORD flags, HWND hwnd, LPSTR dst, PULONG dstlen ) { FIXME("%s, %s, %s, %x, %p, %p, %p\n", debugstr_a(server), debugstr_a(path), debugstr_a(env), flags, hwnd, dst, dstlen); return E_NOTIMPL; } /***************************************************************************** * UploadPrinterDriverPackageW [WINSPOOL.@] */ HRESULT WINAPI UploadPrinterDriverPackageW( LPCWSTR server, LPCWSTR path, LPCWSTR env, DWORD flags, HWND hwnd, LPWSTR dst, PULONG dstlen ) { FIXME("%s, %s, %s, %x, %p, %p, %p\n", debugstr_w(server), debugstr_w(path), debugstr_w(env), flags, hwnd, dst, dstlen); return E_NOTIMPL; }